美文网首页
Java 设置两个JScrollPane中的JList同步显示

Java 设置两个JScrollPane中的JList同步显示

作者: 向祥祥 | 来源:发表于2020-03-30 16:36 被阅读0次

1.为什么需要两个JScrollPane中的JList同步显示

有些情况下,我们会遇到这种情况:有两列一一对应的数据,需要其在两个JList中分别显示,如下图



如果需要显示的数据如左图,有清晰的对应关系,我们可以轻易的得知某一个数据A和与这个数据对应的数据B。虽然在这种情况下我们能得到对应关系,但是在我们拖动某一个滚动条的时候,另一个位置不会作相应的改动,会出现两个JList显示的数据不是同一段数据。在大多数情况下我们是不能从两个JList中清楚的看出对应关系的,如右图。
基于上面的情况我们有这样的需求:
1.如果其中一个JScrollPane的滚动条的位置变化,另一个JScrollPane的滚动条的位置也作相应的变化;
2.在其中一个JList中选中一个数据后,另一个JList中与之对应的数据也被选中。

2.JScrollPane滚动条(JScrollBar)位置的同步变化实现方法

参考论坛贴,实现JScrollPane滚动条位置同步变化的方法有:
1.将两个JScrollPane的JScrollBar设成同一个JScrollBar。这种方法的好处是两个JScrollBar在不需要控制和计算的条件下即可完美的联动;缺陷是只能看到其中一个JScrollBar。
2.注册两个JScrollBar的事件监听AdjustmentListener,在事件处理中设置需要改变位置的JScrollBar的value值。这种方法的好处是看得到两个滚动条;缺陷是不能在两个JScrollPane大小不同时实现完美的联动,此时可能需要较多的运算。

3.JList中对应数据同步显示实现方法

注册两个JList的时间监听ListSelectionListener,在事件处理中设置另一个JList中对应数据被选中。

4.实现两个JScrollPane中JScrollBar同步滚动案例(未实现JList同步选中)

package com.jscrollpane;
import java.awt.GridLayout;
import javax.swing.JFrame;
import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
public class SynJList  extends JFrame
{
    JPanel jpanel;
    JScrollPane jscrollpane1;
    JScrollPane jscrollpane2;
    JList<String> jlist1;
    JList<String> jlist2;
    public static void main(String[] args)
    {
        SynJList sjl=new SynJList();
    }
    public SynJList()
    {
        jpanel=new JPanel();
        jlist1=new JList<String>();
        jlist2=new JList<String>();
        jscrollpane1=new JScrollPane(jlist1);
        jscrollpane2=new JScrollPane(jlist2);
        //将两个JScrollPane的JScrollBar设成同一个JScrollBar
        this.jscrollpane2.setVerticalScrollBar(this.jscrollpane1.getVerticalScrollBar());
        jlist1.setListData(this.NewStringArray(100));
        jlist2.setListData(this.NewStringArray(100));
        jpanel.setLayout(new GridLayout(1, 2));
        jpanel.add(jscrollpane1);
        jpanel.add(jscrollpane2);
        this.add(jpanel);
        this.setTitle("JScrollPane中JList同步显示");
        this.setSize(300, 300);
        this.setVisible(true);
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    }
    //用于创建String数组的方法
    public String[] NewStringArray(int count)
    {
        String result[]=new String[count];
        for(int i=0;i<count;i++)
        {
            result[i]=Integer.toString((int)(Math.random()*100));
        }
        return result;
    }
}

5.两个JScrollPane中的JList同步显示案例(第二种方法实现JScrollBar同步滚动)

package com.test;
import java.awt.GridLayout;
import java.awt.event.AdjustmentEvent;
import java.awt.event.AdjustmentListener;
import javax.swing.JFrame;
import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.ListSelectionListener;
public class SynJList  extends JFrame implements AdjustmentListener,ListSelectionListener
{
    JPanel jpanel;
    JScrollPane jscrollpane1;
    JScrollPane jscrollpane2;
    JList<String> jlist1;
    JList<String> jlist2;
    public static void main(String[] args)
    {
        SynJList sjl=new SynJList();
    }
    public SynJList()
    {
        jpanel=new JPanel();
        jlist1=new JList<String>();
        jlist2=new JList<String>();
        jscrollpane1=new JScrollPane(jlist1);
        jscrollpane2=new JScrollPane(jlist2);
        jlist1.setListData(this.NewStringArray(100));
        jlist2.setListData(this.NewStringArray(100));
        jpanel.setLayout(new GridLayout(1, 2));
        jpanel.add(jscrollpane1);
        jpanel.add(jscrollpane2);
        this.add(jpanel);
        this.setTitle("JScrollPane中JList同步显示");
        this.setSize(300, 300);
        this.setVisible(true);
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.AddListen();
    }
    //用于创建String数组的方法
    public String[] NewStringArray(int count)
    {
        String result[]=new String[count];
        for(int i=0;i<count;i++)
        {
            result[i]=Integer.toString((int)(Math.random()*100));
        }
        return result;
    }
    //将所有要注册的监听写到一个方法中
    public void AddListen()
    {
        //两个JScrollPane中JScrollBar的事件监听注册
        this.jscrollpane1.getVerticalScrollBar().addAdjustmentListener(this);
        this.jscrollpane2.getVerticalScrollBar().addAdjustmentListener(this);
        //两个JList的事件监听注册
        this.jlist1.addListSelectionListener(this);
        this.jlist2.addListSelectionListener(this);
    }
    //两个JScrollPane中JScrollBar的事件处理
    @Override
    public void adjustmentValueChanged(AdjustmentEvent arg0)
    {
        this.jscrollpane1.getVerticalScrollBar().setValue(arg0.getValue());
        this.jscrollpane2.getVerticalScrollBar().setValue(arg0.getValue());
    }
    //两个JList的事件处理
    @Override
    public void valueChanged(ListSelectionEvent arg0)
    {
        JList templist=(JList) arg0.getSource();
        this.jlist1.setSelectedIndex(templist.getSelectedIndex());
        this.jlist2.setSelectedIndex(templist.getSelectedIndex());
    }
}

该案例可以成功的达到我们想要的目的。但是仔细分析后我们会发现以下问题:

1.如果我们拖动JScrollPane1中的JScrollBar1,将会触发AdjustmentEvent事件,在以上案例中,因为没有对事件源进行判断,所以在事件处理中会将JScrollPane1中的JScrollBar的value再设置一遍,此做法是多余的。
2.如果我们拖动JScrollPane1中的JScrollBar1,将会触发AdjustmentEvent事件,在修改JScrollPane2中JScrollBar2的value值时将触发AdjustmentEvent事件,此处的事件触发属于多余的。
3.同理,对JList也有类似的问题存在。因此对以上案例进行了该进。

6.两个JScrollPane中的JList同步显示改进案例

对在事件处理中JScrollBar的value值重复设置的问题。因为和一般使用的ActionEvent不同,不能设置ActionCommand。所以此处采用的方法是:AdjustmentEvent事件对象中用getSource()方法得到的对象与JFrame中的JScrollBar对象进行比较,从而判断事件源是哪个对象。

对在事件处理中再次触发AdjustmentEvent事件的问题。此处采用的方法是:在设置JScrollBar的value值之前,先移除JScrollBar的事件监听,在修改之后,再添加其事件监听。

对JList的问题,处理方法类似。

package com.test;
import java.awt.GridLayout;
import java.awt.event.AdjustmentEvent;
import java.awt.event.AdjustmentListener;
import javax.swing.JFrame;
import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.JScrollBar;
import javax.swing.JScrollPane;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.ListSelectionListener;
public class SynJList  extends JFrame implements AdjustmentListener,ListSelectionListener
{
    JPanel jpanel;
    JScrollPane jscrollpane1;
    JScrollPane jscrollpane2;
    JList<String> jlist1;
    JList<String> jlist2;
    public static void main(String[] args)
    {
        SynJList sjl=new SynJList();
    }
    public SynJList()
    {
        jpanel=new JPanel();
        jlist1=new JList<String>();
        jlist2=new JList<String>();
        jscrollpane1=new JScrollPane(jlist1);
        jscrollpane2=new JScrollPane(jlist2);
        jlist1.setListData(this.NewStringArray(100));
        jlist2.setListData(this.NewStringArray(100));
        jpanel.setLayout(new GridLayout(1, 2));
        jpanel.add(jscrollpane1);
        jpanel.add(jscrollpane2);
        this.add(jpanel);
        this.setTitle("JScrollPane中JList同步显示");
        this.setSize(300, 300);
        this.setVisible(true);
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.AddListen();
    }
    //用于创建String数组的方法
    public String[] NewStringArray(int count)
    {
        String result[]=new String[count];
        for(int i=0;i<count;i++)
        {
            result[i]=Integer.toString((int)(Math.random()*100));
        }
        return result;
    }
    //将所有要注册的监听写到一个方法中
    public void AddListen()
    {
        //两个JScrollPane中JScrollBar的事件监听注册
        this.jscrollpane1.getVerticalScrollBar().addAdjustmentListener(this);
        this.jscrollpane2.getVerticalScrollBar().addAdjustmentListener(this);
        //两个JList的事件监听注册
        this.jlist1.addListSelectionListener(this);
        this.jlist2.addListSelectionListener(this);
    }
    //两个JScrollPane中JScrollBar的事件处理
    @Override
    public void adjustmentValueChanged(AdjustmentEvent arg0)
    {
        if(arg0.getSource().equals(this.jscrollpane1.getVerticalScrollBar()))
        {
            this.jscrollpane2.getVerticalScrollBar().removeAdjustmentListener(this);
            this.jscrollpane2.getVerticalScrollBar().setValue(arg0.getValue());
            this.jscrollpane2.getVerticalScrollBar().addAdjustmentListener(this);
        }else if(arg0.getSource().equals(this.jscrollpane2.getVerticalScrollBar()))
        {
            this.jscrollpane1.getVerticalScrollBar().removeAdjustmentListener(this);
            this.jscrollpane1.getVerticalScrollBar().setValue(arg0.getValue());
            this.jscrollpane1.getVerticalScrollBar().addAdjustmentListener(this);
        }
    }
    //两个JList的事件处理
    @Override
    public void valueChanged(ListSelectionEvent arg0)
    {
        if(arg0.getSource().equals(this.jlist1))
        {
            System.out.println("list1changed!");
            this.jlist2.removeListSelectionListener(this);
            JList templist=(JList) arg0.getSource();
            this.jlist2.setSelectedIndex(templist.getSelectedIndex());
            this.jlist2.addListSelectionListener(this);
        }else if(arg0.getSource().equals(this.jlist2))
        {
            System.out.println("list2changed!");
            this.jlist1.removeListSelectionListener(this);
            JList templist=(JList) arg0.getSource();
            this.jlist1.setSelectedIndex(templist.getSelectedIndex());
            this.jlist1.addListSelectionListener(this);
        }
    }
}

我们会发现在方法valueChanged中有两行是测试代码

System.out.println("list1changed!");

System.out.println("list2changed!");

我们运行代码并进行JList中的选择操作(一次),控制台会输出如下的结果:

list1changed!
list1changed!

经过测试后发现,事件监听了鼠标单击中的按下和释放两个操作。

相关文章

  • Java 设置两个JScrollPane中的JList同步显示

    1.为什么需要两个JScrollPane中的JList同步显示 有些情况下,我们会遇到这种情况:有两列一一对应的数...

  • 2.安全性

    java中多线程同步包括: synchronized 显示锁 volatile 原子变量 之所以要使用同步,是因为...

  • JAVA高并发(三)

    JAVA中的线程同步机制: 在java中,按照锁的实现方式划分为两种, 内部锁:synchronized 显示...

  • JList设置样式ListCellRenderer

    效果

  • Java线程同步锁

    Java中JDK1.5之后,出了另一种线程同步机制,通过显示定义同步锁来进行同步: ReadWriteLock(读...

  • 【Code-Snippet】TextView

    1. TextView文字过长,显示省略号 【参考】 必须要同时设置XML和JAVA,而且,java中设置文字必须...

  • Windows 10不能正常同步的解决方法

    正常情况下,设置同步应该是这样的: OneDrive同步应该是这样的: 然而有时设置里显示“无法同步,请与你的设备...

  • 【转】Java 同步块

    转自 Java 同步块 Java 同步关键字(synchronized) Java 中的同步块用 synchron...

  • java同步块

    Java同步关键字(synchronized) Java中的同步块用synchronized标记。同步块在Java...

  • 深入浅出JScrollPane

    当显示区域受限时,滚动窗格,即JScrollPane,就有了用武之地,用来显示尺寸较大或者尺寸可以的动态变化的控件...

网友评论

      本文标题:Java 设置两个JScrollPane中的JList同步显示

      本文链接:https://www.haomeiwen.com/subject/tdvvuhtx.html