ListSelectionListener 在调用 setSelected 方法时不会触发事件

ListSelectionListener does not fire events when calling setSelected methods

这里是第一个问题,希望我做对了。

下面是我的问题的一个最小示例,我快速提出了我的项目代表。我已经为包含一些对象的 JList 创建了一个自定义渲染器(在示例中,我使用字符串进行说明)。我的问题是,据我所知,如果我将 ListSelectionListener 添加到列表中,setSelected...() 方法系列不会触发触发条件 if(e.getValueIsAdjusting()).[= 的事件15=]

在下面的示例中,问题在启动程序时立即显而易见:即使在将 JLabel 的选定文本分配为 "none" 之后调用 list.setSelectedIndex(2);,它也不会改变,直到您,用户,单击列表项。事实上,它必须与当前选择的项目不同。

我想要这个功能,这样我的程序流程将是,在列表中的用户 adds/removes 项之后,一旦选定的列表项发生更改,"view" 会立即更新.

是我做错了什么,还是我的处理方式不正确?

谢谢。 :)

import java.awt.BorderLayout;
import java.beans.*;
import javax.swing.*;

public class Example implements PropertyChangeListener {
    String ITEM_SELECTED = "Item Selected";
    String DELETE_SELECTED = "Delete Selected";
    PropertyChangeSupport pcs = new PropertyChangeSupport(this);
    JLabel selected;

    JList<String> list;
    DefaultListModel<String> model;

    public static void main(String[] args) {
        new Example();
    }

    public Example() {
        JFrame frame = new JFrame();
        JPanel mainPanel = new JPanel(new BorderLayout());

        model = new DefaultListModel<>();
        model.addElement("Entry 1");
        model.addElement("Entry 2");
        model.addElement("Entry 3");
        model.addElement("Entry 4");
        model.addElement("Entry 5");

        list = new JList<>(model);

        selected = new JLabel("Currently selected: none");
        list.setSelectedIndex(2);

        list.addListSelectionListener(e -> {
                if(e.getValueIsAdjusting()) {
                    pcs.firePropertyChange(ITEM_SELECTED, null, null);
                }
            }
        );

        JButton removeItem = new JButton("Remove item");

        removeItem.addActionListener(e -> {
                pcs.firePropertyChange(DELETE_SELECTED, null, null);
            }
        );

        mainPanel.add(new JScrollPane(list), BorderLayout.WEST);
        mainPanel.add(removeItem, BorderLayout.NORTH);

        mainPanel.add(selected, BorderLayout.CENTER);       

        pcs.addPropertyChangeListener(this);

        frame.add(mainPanel);
        frame.setSize(300, 400);
        frame.setLocationRelativeTo(null);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.setVisible(true);
    }

    @Override
    public void propertyChange(PropertyChangeEvent evt) {
        String event = (String) evt.getPropertyName();

        if (event.equals(ITEM_SELECTED)) {
            selected.setText("Currently selected: " + list.getSelectedValue());
        }

        if (event.equals(DELETE_SELECTED)) {
            int selectedIndex = list.getSelectedIndex();
            model.removeElement(list.getSelectedValue());

            if (selectedIndex > 0)
                list.setSelectedIndex(selectedIndex-1);
            else if (selectedIndex == 0)
                list.setSelectedIndex(selectedIndex);
        }
    }
}

e.getValueIsAdjusting() 方法检查值是否仍在调整中。在您的情况下,list.setSelectedIndex(2) 触发的事件看起来 e.getValueIsAdjusting() return false.

您可以通过执行相反的检查来解决此问题:!e.getValueIsAdjusting().

这将检查响应的任何更改是否是该事件链的最后一个。