我正在使用Eclipse中的Swing设计器构建一个GUI,以列出存储在XML文件中的电影。在将框架和内容加载到各种JList
之后,我将应用程序设置为在选择项目时更新列表。所以,如果你选择一个类型,该类型的所有电影都会被播放,同样适用于组、集和季。我也有一个刷新列表按钮,重新加载列表。
我遇到的问题是:当我选择索引0
处的项目时,与选择索引1
或更高时相比,会调用额外的侦听器动作。然后(我的主要关注点),如果我选择刷新按钮,将为之前选择的JList
添加到listmodel
的每个项目触发一个侦听器。
例如,如果我从剧集列表中选择1
&然后刷新,只调用一个侦听器。但是,如果我从剧集列表中选择0
&然后刷新,我得到35个听众(这是列表中的剧集总数)。
对于像剧集这样的小事,这不是一个大问题,但当这种情况发生在电影专栏时,我大约有1500名听众被解雇。
package local.testarea;
import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.DefaultListModel;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JList;
import javax.swing.JScrollPane;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.ListSelectionListener;
import net.miginfocom.swing.MigLayout;
@SuppressWarnings( { "rawtypes", "unchecked" } )
public class ListDemo {
private JFrame frame;
private JList list, list_1;
private JButton btnNewButton;
private DefaultListModel<String> list2;
private DefaultListModel<Integer> list1;
/**
* Launch the application.
*/
public static void main( String[] args )
{
EventQueue.invokeLater( new Runnable() {
public void run()
{
try
{
ListDemo window = new ListDemo();
window.frame.setVisible( true );
}
catch ( Exception e )
{
e.printStackTrace();
}
}
} );
}
/**
* Create the application.
*/
public ListDemo()
{
list1 = new DefaultListModel<Integer>();
list2 = new DefaultListModel<String>();
initialize();
updateAll();
}
private void updateAll()
{
list1.clear();
list2.clear();
for ( int i = 0; i < 101; i++ )
{
list1.addElement( i );
}
list2.addElement( "Even" );
list2.addElement( "Odd" );
}
private void updateLists( int selected )
{
list1.clear();
list2.clear();
switch( selected )
{
case 0:
for ( int i = 0; i < 101; i++)
{
if ( i % 2 == 0 )
{
list1.addElement( i );
}
}
list2.addElement( "Even" );
break;
case 1:
for ( int i = 0; i < 101; i++)
{
if ( i % 2 != 0 )
{
list1.addElement( i );
}
}
list2.addElement( "Odd" );
break;
default:
int z = selected - 10;
list1.addElement( z );
list2.addElement( "Even" );
list2.addElement( "Odd" );
break;
}
}
/**
* Initialize the contents of the frame.
*/
private void initialize()
{
frame = new JFrame();
frame.setBounds( 100, 100, 450, 481 );
frame.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
frame.getContentPane().setLayout(new MigLayout("", "[grow][grow]", "[grow][]"));
JScrollPane scrollPane = new JScrollPane();
frame.getContentPane().add(scrollPane, "cell 0 0,grow");
list = new JList( list1 );
list.addListSelectionListener(new ListSelectionListener() {
public void valueChanged(ListSelectionEvent e) {
System.out.println( "List 1 trigger" );
System.out.println( "selected item: " + list_1.getSelectedIndex() );
if ( list.getSelectedIndex() >= 0 )
{
int z = 10 + list.getSelectedIndex();
updateLists( z );
}
}
});
scrollPane.setViewportView(list);
JScrollPane scrollPane_1 = new JScrollPane();
frame.getContentPane().add(scrollPane_1, "cell 1 0,grow");
list_1 = new JList( list2 );
list_1.addListSelectionListener(new ListSelectionListener() {
public void valueChanged(ListSelectionEvent e) {
System.out.println( "List 2 trigger" );
System.out.println( "selected item: " + list_1.getSelectedIndex() );
if ( list_1.getSelectedIndex() >= 0 )
{
updateLists( list_1.getSelectedIndex() );
}
}
});
scrollPane_1.setViewportView(list_1);
btnNewButton = new JButton("New button");
btnNewButton.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
System.out.println( "refresh" );
updateAll();
}
});
frame.getContentPane().add(btnNewButton, "cell 1 1");
}
}
在选择仍在进行时修改列表,这会导致触发额外的选择事件(递归地)。你还莫名其妙地修改了两个列表。
String
列表应该只是作为Integer
列表的过滤器。此外,我不明白为什么当选择单个值时,您要从Integer
列表中删除所有项目。它使它看起来像一个JList
不是你想要使用的。
无论如何,我修改了你的ListSelectionListener
:
public class ListDemo extends JFrame{
private DefaultListModel<Integer> listModel1 = new DefaultListModel<>();
private DefaultListModel<String> listModel2 = new DefaultListModel<>();
private JList<Integer> list1 = new JList<>(listModel1);
private JList<String> list2 = new JList<>(listModel2);
int size = 101;
public ListDemo() {
list1.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
add(new JScrollPane(list1));
list2.addListSelectionListener(new ListParityFilter());
list2.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
listModel2.addElement("All");
listModel2.addElement("Even");
listModel2.addElement("Odd");
list2.setSelectedIndex(0);
add(new JScrollPane(list2), BorderLayout.EAST);
setDefaultCloseOperation(EXIT_ON_CLOSE);
pack();
setLocationRelativeTo(null);
setVisible(true);
}
private class ListParityFilter implements ListSelectionListener {
public void valueChanged(ListSelectionEvent e) {
if (e.getValueIsAdjusting())
return;
System.out.println("Selected List2 item: " + list2.getSelectedValue());
listModel1.clear();
switch (list2.getSelectedIndex()) {
case 0:
for (int i = 0; i < size; i++)
listModel1.addElement(i);
break;
case 1:
for (int i = 0; i < size; i += 2)
listModel1.addElement(i);
break;
case 2:
for (int i = 1; i < size; i += 2)
listModel1.addElement(i);
break;
}
}
}
public static void main(String[] args) {
new ListDemo();
}
}
指出:
- 我为过滤器列表添加了"All"选项,而不是刷新按钮。
- 过滤器列表不应该被修改(没有原因)。
- 检查
e.getValueIsAdjusting()
,这样你就不会执行不必要的操作。 - 我将两个列表的选择模型设置为
SINGLE_SELECTION
。 - 我删除了
Integer
列表选择侦听器,因为它没有任何用途。在两种情况下,将被调用一次:- 当一个值被选中时(所选索引>= 0)。
- 当一个过滤器被应用,因为列表被清除(选定的索引= -1)。
- 在
JFrame
上使用pack()
而不是设置其大小和位置。