我正在使用ListCellRenderer编辑JList中条目的外观。一旦选中它们(通过单击它们,此代码位于click事件中),我将调用ListCellRenderer来更改文本的颜色。如果再次选择它们,我希望文本恢复为正常颜色。我遇到的问题是,一旦我选择了第二个条目,第一个条目就会恢复其正常颜色。如何将所选条目保留为选定的颜色,直到实际上取消选择它们为止?这是我实际调用该函数的部分:

for(int i = 0; i < selectedEntries.size() - 1; i++){
    System.out.println("Inside the for loop at entry:" + i);
    if(selectedEntries.get(i).equals(selectedEntry)){
        selectedEntries.remove(i);
        removed = true;
        renderer.getListCellRendererComponent(logEntries, value, index, false, false);
        System.out.println("Duplicate Entry Removed From List");
    }
}

if(!removed){
    selectedEntries.add(selectedEntry);
    renderer.getListCellRendererComponent(logEntries, value, index, true, false);
}


为了便于解释,selectedEntries是一个ArrayList,其中包含每个选定条目的文本。

最佳答案

一旦选中它们(通过单击它们,此代码位于click事件中),我将调用ListCellRenderer更改文本的颜色


不,那不是应该的工作方式,ListCellRenderer将再次被调用(由JList调用),而isSelected参数将是true,您将以不同的方式呈现值

ListCellRenderer负责呈现整个状态(选中或未选中)。

请参阅Writing a Custom Cell Renderer了解更多详细信息

例如



import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Font;
import javax.swing.DefaultListCellRenderer;
import javax.swing.DefaultListModel;
import javax.swing.JFrame;
import javax.swing.JList;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;

public class Test {

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

    public Test() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                    ex.printStackTrace();
                }

                JFrame frame = new JFrame("Testing");
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.add(new TestPane());
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    public class TestPane extends JPanel {

        public TestPane() {
            setLayout(new BorderLayout());
            DefaultListModel<String> model = new DefaultListModel<>();
            model.addElement("Apples");
            model.addElement("Bananas");
            model.addElement("Peachs");
            model.addElement("Pears");
            JList<String> listOfStrings = new JList<>(model);
            listOfStrings.setCellRenderer(new FancyPancyListCellRenderer());
            add(new JScrollPane(listOfStrings));
        }

        @Override
        public Dimension getPreferredSize() {
            return new Dimension(200, 200);
        }

    }

    public static class FancyPancyListCellRenderer  extends DefaultListCellRenderer {

        protected static final Font SELECTED_FONT = new Font("Comic Sans MS", Font.PLAIN, 12);

        @Override
        public Component getListCellRendererComponent(JList<?> list, Object value, int index, boolean isSelected, boolean cellHasFocus) {

            super.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus);
            if (isSelected) {
                setForeground(Color.YELLOW);
                setFont(SELECTED_FONT);
            } else {
                setFont(UIManager.getFont("Label.font"));
            }
            return this;

        }

    }

}


另外,MouseListener确实不是检测选择更改的合适方法,如果用户使用键盘选择行,会发生什么情况?您应该改用ListSelectionListener(但不能解决此问题)。

请查看How to Write a List Selection ListenerHow to Use Lists以获得更多详细信息

07-25 22:13