JList 行を動的に変更しようとしています。n 番目の行の色を変更して強調表示する必要があります (n はコンパイル時に不明です)。カスタム ListCellRenderer を使用した多くの例を見ましたが、すべて「静的」でした。
つまり、x 行の JList があります。実行時に、「ビジネス ロジック」が n 番目の行が重要であることを検出します。背景を緑にして、1 秒待ってから、再び白にします。もう1つ、行の選択を変更しないでください。
そうするための最良の方法は何ですか?
JList 行を動的に変更しようとしています。n 番目の行の色を変更して強調表示する必要があります (n はコンパイル時に不明です)。カスタム ListCellRenderer を使用した多くの例を見ましたが、すべて「静的」でした。
つまり、x 行の JList があります。実行時に、「ビジネス ロジック」が n 番目の行が重要であることを検出します。背景を緑にして、1 秒待ってから、再び白にします。もう1つ、行の選択を変更しないでください。
そうするための最良の方法は何ですか?
シンプルに、次を使用してカスタム ListCellRenderer を JList に設定します。
list.setCellRenderer(myListCellrenderer);
オーバーライドされたメソッド getListCellRendererComponent() 内で、次のようにします。
public Component getListCellRendererComponent(.....) {
Component c = super.getListCellRendererComponent();
c.setBackGround(Color.blue)
return c;
}
上記の例では、カスタム レンダラーが DefaultListCellRenderer をオーバーライドすることを想定しています。
SUN の ListDemo サンプルに基づいています。
リストにないテキストをテキストフィールドに入力してハイライトすると、それが追加されます。
テキストがリスト内にあるときにハイライトを押すと、リスト内のエントリが一時的に青くハイライトされます。
ここでの一致フィールドを使用したソリューションはデモ用であることに注意してください。より正確な実装のために、提案された他のアイデアを検討し、使用を検討してくださいjavax.swing.Timer
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.event.*;
public class ListDemo extends JPanel {
private JList list;
private DefaultListModel listModel;
public String match = null;
private static final String hireString = "Highlight";
private JTextField employeeName;
public ListDemo() {
super(new BorderLayout());
listModel = new DefaultListModel();
listModel.addElement("Test1");
listModel.addElement("Test2");
listModel.addElement("Test3");
list = new JList(listModel);
list.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
list.setSelectedIndex(0);
list.setVisibleRowCount(5);
list.setCellRenderer(new MyListCellRenderer());
JScrollPane listScrollPane = new JScrollPane(list);
JButton hireButton = new JButton(hireString);
HireListener hireListener = new HireListener(hireButton);
hireButton.setActionCommand(hireString);
hireButton.addActionListener(hireListener);
hireButton.setEnabled(false);
employeeName = new JTextField(10);
employeeName.addActionListener(hireListener);
employeeName.getDocument().addDocumentListener(hireListener);
listModel.getElementAt(list.getSelectedIndex()).toString();
JPanel buttonPane = new JPanel();
buttonPane.setLayout(new BoxLayout(buttonPane,
BoxLayout.LINE_AXIS));
buttonPane.add(Box.createHorizontalStrut(5));
buttonPane.add(employeeName);
buttonPane.add(hireButton);
buttonPane.setBorder(BorderFactory.createEmptyBorder(5,5,5,5));
add(listScrollPane, BorderLayout.CENTER);
add(buttonPane, BorderLayout.PAGE_END);
}
class MyListCellRenderer extends JLabel implements ListCellRenderer {
public MyListCellRenderer() {
setOpaque(true);
}
public Component getListCellRendererComponent(JList paramlist, Object value, int index, boolean isSelected, boolean cellHasFocus) {
setText(value.toString());
if (value.toString().equals(match)) {
setBackground(Color.BLUE);
SwingWorker worker = new SwingWorker() {
@Override
public Object doInBackground() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) { /*Who cares*/ }
return null;
}
@Override
public void done() {
match = null;
list.repaint();
}
};
worker.execute();
} else
setBackground(Color.RED);
return this;
}
}
class HireListener implements ActionListener, DocumentListener {
private boolean alreadyEnabled = false;
private JButton button;
public HireListener(JButton button) {
this.button = button;
}
public void actionPerformed(ActionEvent e) {
String name = employeeName.getText();
if (listModel.contains(name)) {
match = name;
list.repaint();
employeeName.requestFocusInWindow();
employeeName.selectAll();
return;
}
if (name.equals("")) {
Toolkit.getDefaultToolkit().beep();
employeeName.requestFocusInWindow();
employeeName.selectAll();
return;
}
int index = list.getSelectedIndex();
if (index == -1)
index = 0;
else
index++;
listModel.insertElementAt(employeeName.getText(), index);
employeeName.requestFocusInWindow();
employeeName.setText("");
list.setSelectedIndex(index);
list.ensureIndexIsVisible(index);
}
public void insertUpdate(DocumentEvent e) {
enableButton();
}
public void removeUpdate(DocumentEvent e) {
handleEmptyTextField(e);
}
public void changedUpdate(DocumentEvent e) {
if (!handleEmptyTextField(e))
enableButton();
}
private void enableButton() {
if (!alreadyEnabled)
button.setEnabled(true);
}
private boolean handleEmptyTextField(DocumentEvent e) {
if (e.getDocument().getLength() <= 0) {
button.setEnabled(false);
alreadyEnabled = false;
return true;
}
return false;
}
}
private static void createAndShowGUI() {
JFrame frame = new JFrame("ListDemo");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
JComponent newContentPane = new ListDemo();
newContentPane.setOpaque(true);
frame.setContentPane(newContentPane);
frame.pack();
frame.setVisible(true);
}
public static void main(String[] args) {
javax.swing.SwingUtilities.invokeLater(new Runnable() {
public void run() { createAndShowGUI(); }
});
}
}
メソッドを実装するカスタム ListCellRenderer は、と、それが再読み込みしている値のgetListCellRendererComponent
両方にアクセスできます。これにより、 n番目の行を緑JList
にペイントするタイミングを決定する方法について、いくつかのオプションが提供されます。
JList
化して、bg に使用する色をレンダラーに要求させることができます。サブクラスは、ビジネス ロジックがn番目の行が緑色になるJList
時期であると判断したときに再描画をトリガーし、Swing を開始して再描画をトリガーし、bg を通常の状態に戻すことができます。Timer
getListCellRendererComponent
、状態が正しい場合は bg を緑色に設定するオプションもあります。ここでも、SwingTimer
を設定してバッキング オブジェクトの状態を元に戻すオプションがあります。