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!
经过测试后发现,事件监听了鼠标单击中的按下和释放两个操作。
网友评论