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

2020-03-30  本文已影响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!

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

上一篇 下一篇

猜你喜欢

热点阅读