美文网首页
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同步显示

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