0

我使用基于 Swing 的简单 GUI 用 Ja​​va 编写了一个小注释器,但我遇到了一个让我感到害怕的问题。问题是:我有一个 jlist 和两个修改此类 jlist 的 jbutton,这两个按钮具有相同的侦听器,但它们的工作方式不同。它应该像这样工作:一旦您在 jlist 中选择了一行,您可以将其标记为 ON TOPIC 或 OFF TOPIC(使用两个按钮),该行更改颜色,然后选择下一行但是;即使选择在右行(下一个),它也会为关闭主题按钮突出显示下一个,为什么?

这是代码:

public class TweetsAnnotator {

static Boolean[] annotations = null;
@SuppressWarnings("rawtypes")
static JList jl;
static JButton offbutton = new JButton("OFF Topic");
static JButton onbutton = new JButton("ON Topic");
static String file = "inception_TweetList";

public TweetsAnnotator() {
}

/**
 * @param args
 * @throws IOException 
 * @throws FileNotFoundException 
 * @throws ClassNotFoundException 
 */
@SuppressWarnings({ "unchecked", "rawtypes" })
public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {

    // Read Tweets from file
    ObjectInputStream load = new ObjectInputStream(new FileInputStream(file));
    ArrayList<String> list = (ArrayList<String>) load.readObject();
    load.close();
    System.out.println(list.size() + " Tweets read from: " + file);

    // Check and read annotations
    File fileannot = new File(file + "Annotations");
    if (fileannot.exists()) {
        System.out.println("esiste, leggo");
        ObjectInputStream loadannot = new ObjectInputStream(new FileInputStream(file + "Annotations"));
        annotations = (Boolean[]) loadannot.readObject();
        loadannot.close();
    } else {
        System.out.println("non esiste, creo poi leggo");
        ObjectOutputStream save = new ObjectOutputStream(new FileOutputStream(file + "Annotations"));
        Boolean[] creatannotations = new Boolean[list.size()];
        for (int i=0; i<list.size(); i++) {
            creatannotations[i] = (Boolean) null;
        }
        save.writeObject(creatannotations);
        save.close();
        ObjectInputStream loadannot = new ObjectInputStream(new FileInputStream(file + "Annotations"));
        annotations = (Boolean[]) loadannot.readObject();
        loadannot.close();
    }
    System.out.println(annotations.length + " Annotations loaded");     

    // Buttons
    offbutton.setActionCommand("off");
    offbutton.addActionListener(new ButtonListener());
    offbutton.setEnabled(false);
    onbutton.setActionCommand("on");
    onbutton.addActionListener(new ButtonListener());
    onbutton.setEnabled(false);

    // ButtonPanel
    JPanel buttonPane = new JPanel();
    buttonPane.setLayout(new BoxLayout(buttonPane, BoxLayout.X_AXIS));
    buttonPane.add(onbutton);
    buttonPane.add(offbutton);

    // JList
    jl = new JList((Object[])list.toArray());
    jl.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
    jl.setLayoutOrientation(JList.VERTICAL);
    jl.setVisibleRowCount(-1);
    jl.setCellRenderer(new MyCellRenderer());
    ListSelectionListener listSelectionListener = new ListSelectionListener() {

        public void valueChanged(ListSelectionEvent e) {
            if (e.getValueIsAdjusting() == false) {
                if (jl.getSelectedIndex() == -1) {
                    //No selection, disable buttons.
                    onbutton.setEnabled(false);
                    offbutton.setEnabled(false);
                } else {
                    //Selection, enable buttons.
                    onbutton.setEnabled(true);
                    offbutton.setEnabled(true);
                }
            }
        }
    };
    jl.addListSelectionListener(listSelectionListener);

    // JScrollPane
    JScrollPane listScroller = new JScrollPane(jl);

    // JFrame
    JFrame frame = new JFrame(file);
    frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
    frame.setExtendedState(frame.getExtendedState()|JFrame.MAXIMIZED_BOTH);
    frame.addWindowListener(new WindowCloseHandler());

    // Add and show
    frame.getContentPane().add(listScroller, BorderLayout.CENTER);
    frame.getContentPane().add(buttonPane, BorderLayout.PAGE_END);
    frame.pack();
    frame.setVisible(true);

}

private static class MyCellRenderer extends DefaultListCellRenderer {   

    private static final long serialVersionUID = 1L;

    @SuppressWarnings("rawtypes")
    public Component getListCellRendererComponent( JList list, Object value, int index, boolean isSelected, boolean cellHasFocus ) {   
        Component c = super.getListCellRendererComponent( list, value, index, isSelected, cellHasFocus );   
        if ( annotations[index] == null ) {   
            c.setBackground( Color.white );   
        }   
        else if (annotations[index] == true) {   
            c.setBackground( Color.green );   
        } else {
            c.setBackground( Color.red);
        }
        return c;   
    } 
}

private static class ButtonListener implements ActionListener {

    public void actionPerformed(ActionEvent e) {
        int ind = jl.getSelectedIndex() +1;
        if (e.getActionCommand().equals("on")) {
            System.out.println("ON");
            annotations[jl.getSelectedIndex()] = true;
        }
        if (e.getActionCommand().equals("off")) {
            System.out.println("OFF");
            annotations[jl.getSelectedIndex()] = false;

        }
        jl.clearSelection();
        jl.setSelectedIndex(ind);
    }
}

private static class WindowCloseHandler extends WindowAdapter {

    public void windowClosing(WindowEvent evt) {
        ObjectOutputStream save = null;
        try {
            save = new ObjectOutputStream(new FileOutputStream(file + "Annotations"));
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        try {
            save.writeObject(annotations);
        } catch (IOException e) {
            e.printStackTrace();
        }
        try {
            save.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        System.out.println("Saved.");
    }
}
}

即使 jl.requestFocus(); 工作正常,我尝试了另一件事让它工作:我交换了这两行

buttonPane.add(onbutton);
buttonPane.add(offbutton);

但为什么?对不起,如果我再问一次,但这真的很奇怪,不是吗?

4

2 回答 2

2

在你的单元格渲染中,试试这个

if (annotations[index] == null) {
    //c.setBackground(Color.white);
} else if (annotations[index] == true) {
    c.setBackground(Color.green);
} else {
    c.setBackground(Color.red);
}

对于“小”扩展的东西,您也可以尝试

private static class MyCellRenderer extends DefaultListCellRenderer {

    private static final long serialVersionUID = 1L;

    private static final Border SELECTION_BORDER = new LineBorder(UIManager.getColor("List.selectionBackground"));
    private static final Border EMPTY_BORDER = new EmptyBorder(1, 1, 1, 1);

    @SuppressWarnings("rawtypes")
    public Component getListCellRendererComponent(JList list, Object value, int index, boolean isSelected, boolean cellHasFocus) {
        Component c = super.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus);
        setBorder(EMPTY_BORDER);
        if (annotations[index] == null) {

            if (isSelected) {

                setBorder(SELECTION_BORDER);

            }

            c.setBackground(Color.white);
        } else if (annotations[index] == true) {
            c.setBackground(Color.green);
        } else {
            c.setBackground(Color.red);
        }
        return c;
    }
}

基本上,您所做的是更改选择突出显示颜色,隐藏选择。至于为什么您确实得到了一个“似乎”突出显示所选行的油漆工件,对我来说仍然是一个谜:P

更新了知道为什么

您有时得到的绘制工件是hasFocus单元格渲染器参数的结果,绘制了一个焦点矩形。

现在,如果您想保留现有的单元格渲染器,请在 ActionEvent 中尝试此操作

 //jl.clearSelection();
 jl.setSelectedIndex(ind);
 jl.requestFocus();

据我所知,重绘管理器似乎有点奇怪。如果我将现有代码添加jl.repaint()到您的现有代码中(在setSelectedIndex调用下),我可以让它永远不会绘制焦点矩形:P

于 2012-08-04T08:55:51.510 回答
2

没有解决您的问题,没有解决某些问题,永远不要运行您的代码,只需将其拆分为逻辑块,添加最低要求的规则,抱歉最多 5 分钟

import java.awt.*;
import java.awt.event.*;
import java.io.*;
import java.util.ArrayList;
import javax.swing.*;
import javax.swing.event.*;

public class TweetsAnnotator {

    private Boolean[] annotations = null;
    private JList jl;
    private JButton offbutton = new JButton("OFF Topic");
    private JButton onbutton = new JButton("ON Topic");
    private String file = "inception_TweetList";
    private ArrayList<String> list;
    private ObjectInputStream load;
    private ObjectInputStream loadannot;
    private ObjectOutputStream save;

    public TweetsAnnotator() {// Read Tweets from file
        loadDatas();
        offbutton.setActionCommand("off"); // Buttons
        offbutton.addActionListener(new ButtonListener());
        offbutton.setEnabled(false);
        onbutton.setActionCommand("on");
        onbutton.addActionListener(new ButtonListener());
        onbutton.setEnabled(false);
        JPanel buttonPane = new JPanel(); // ButtonPanel
        buttonPane.setLayout(new BoxLayout(buttonPane, BoxLayout.X_AXIS));
        buttonPane.add(onbutton);
        buttonPane.add(offbutton);
        jl = new JList(list.toArray()); // JList
        jl.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
        jl.setLayoutOrientation(JList.VERTICAL);
        jl.setVisibleRowCount(-1);
        jl.setCellRenderer(new MyCellRenderer());
        ListSelectionListener listSelectionListener = new ListSelectionListener() {

            @Override
            public void valueChanged(ListSelectionEvent e) {
                if (e.getValueIsAdjusting() == false) {
                    if (jl.getSelectedIndex() == -1) {
                        onbutton.setEnabled(false); //No selection, disable buttons.
                        offbutton.setEnabled(false);
                    } else {
                        onbutton.setEnabled(true); //Selection, enable buttons.
                        offbutton.setEnabled(true);
                    }
                }
            }
        };
        jl.addListSelectionListener(listSelectionListener);
        JScrollPane listScroller = new JScrollPane(jl); // JScrollPane
        JFrame frame = new JFrame(file); // JFrame
        frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
        frame.setExtendedState(frame.getExtendedState() | JFrame.MAXIMIZED_BOTH);
        frame.addWindowListener(new WindowCloseHandler());
        frame.add(listScroller, BorderLayout.CENTER); // Add and show
        frame.add(buttonPane, BorderLayout.PAGE_END);
        frame.pack();
        frame.setVisible(true);
    }

    private void loadDatas() {
        try {
            load = new ObjectInputStream(new FileInputStream(file));
            list = (ArrayList<String>) load.readObject();
            System.out.println(list.size() + " Tweets read from: " + file);
            File fileannot = new File(file + "Annotations"); // Check and read annotations
            if (fileannot.exists()) {
                System.out.println("esiste, leggo");
                loadannot = new ObjectInputStream(new FileInputStream(file + "Annotations"));
                annotations = (Boolean[]) loadannot.readObject();
                loadannot.close();
            } else {
                System.out.println("non esiste, creo poi leggo");
                save = new ObjectOutputStream(new FileOutputStream(file + "Annotations"));
                Boolean[] creatannotations = new Boolean[list.size()];
                for (int i = 0; i < list.size(); i++) {
                    creatannotations[i] = (Boolean) null;
                }
                save.writeObject(creatannotations);
                loadannot = new ObjectInputStream(new FileInputStream(file + "Annotations"));
                annotations = (Boolean[]) loadannot.readObject();
            }
            System.out.println(annotations.length + " Annotations loaded");
        } catch (IOException ex) {
            ex.printStackTrace();
            System.out.println(" Something wrong happened");
        } catch (ClassNotFoundException ex) {
            ex.printStackTrace();
            System.out.println(" Something wrong happened");
        } finally {
            if (load != null) {
                try {
                    load.close();
                } catch (IOException ex) {
                }
            }
            if (save != null) {
                try {
                    save.close();
                } catch (IOException ex) {
                }
            }
            if (loadannot != null) {
                try {
                    loadannot.close();
                } catch (IOException ex) {
                }
            }
        }
        System.out.println("Load block ended");
    }

    private class MyCellRenderer extends DefaultListCellRenderer {

        private static final long serialVersionUID = 1L;

        @Override
        public Component getListCellRendererComponent(JList list, Object value, int index, boolean isSelected, boolean cellHasFocus) {
            Component c = super.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus);
            if (annotations[index] == null) {
                c.setBackground(Color.white);
            } else if (annotations[index] == true) {
                c.setBackground(Color.green);
            } else {
                c.setBackground(Color.red);
            }
            return c;
        }
    }

    private class ButtonListener implements ActionListener {

        @Override
        public void actionPerformed(ActionEvent e) {
            int ind = jl.getSelectedIndex() + 1;
            if (e.getActionCommand().equals("on")) {
                System.out.println("ON");
                annotations[jl.getSelectedIndex()] = true;
            }
            if (e.getActionCommand().equals("off")) {
                System.out.println("OFF");
                annotations[jl.getSelectedIndex()] = false;
            }
            jl.clearSelection();
            jl.setSelectedIndex(ind);
        }
    }

    private class WindowCloseHandler extends WindowAdapter {

        @Override
        public void windowClosing(WindowEvent evt) {
            ObjectOutputStream save = null;
            try {
                save = new ObjectOutputStream(new FileOutputStream(file + "Annotations"));
            } catch (FileNotFoundException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }
            try {
                save.writeObject(annotations);
            } catch (IOException e) {
                e.printStackTrace();
            }
            try {
                save.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
            System.out.println("Saved.");
        }
    }

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {

            @Override
            public void run() {
                TweetsAnnotator tweetsAnnotator = new TweetsAnnotator();
            }
        });
    }
}
于 2012-08-04T09:24:10.447 回答