table.repaint();
を強制的に再描画せずに、外部からの値に基づいて背景を正しく更新することができます。
kleopatraとHovercraft Full Of Eelsによって作成された優れたコードに基づいて使用され、テストされています
aPIに変更がなかったため、Java6/7で有効
私のSSCCEは正しく機能し、JTable.repaint()
によって再描画されます
import Java.awt.BorderLayout;
import Java.awt.Color;
import Java.awt.Component;
import Java.awt.GridLayout;
import javax.swing.DefaultComboBoxModel;
import javax.swing.JComboBox;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.border.EmptyBorder;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.DefaultTableModel;
public class MyTableAndRenderer {
private JFrame frame = new JFrame();
private JPanel panel = new JPanel();
private String[] items = {"Item 1", "Item 2", "Item 3", "Item 4"};
private DefaultComboBoxModel comboBoxModel = new DefaultComboBoxModel(items);
private JComboBox combo = new JComboBox(comboBoxModel);
private JPanel panel1 = new JPanel();
private String[] columnNames = {"First Name", "Last Name", "Sport",
"# of Years", "Vegetarian"};
private Object[][] data = {
{"Kathy", "Smith", "Item 1", new Integer(5), (false)},
{"John", "Doe", "Item 1", new Integer(3), (true)},
{"Sue", "Black", "Item 3", new Integer(2), (false)},
{"Jane", "White", "Item 3", new Integer(20), (true)},
{"Joe", "Brown", "Item 3", new Integer(10), (false)}
};
private DefaultTableModel model = new DefaultTableModel(data, columnNames) {
private static final long serialVersionUID = 1L;
@Override
public Class getColumnClass(int column) {
return getValueAt(0, column).getClass();
}
};
private JTable table = new JTable(model);
public MyTableAndRenderer() {
panel.setBorder(new EmptyBorder(10, 0, 2, 0));
panel.add(combo);
//@HFOE
/*table.setDefaultRenderer(Object.class, new DefaultTableCellRenderer() {
@Override
public Component getTableCellRendererComponent(JTable table,
Object value, boolean isSelected, boolean hasFocus,
int row, int column) {
super.getTableCellRendererComponent(table, value, isSelected,
hasFocus, row, column);
String str = combo.getSelectedItem().toString();
if (value.toString().equalsIgnoreCase(str)) {
setBackground(Color.RED);
} else {
setBackground(null);
}
return this;
}
});*/
//@kleopatra
/*table.setDefaultRenderer(Object.class, new DefaultTableCellRenderer() {
@Override
public Component getTableCellRendererComponent(JTable table,
Object value, boolean isSelected, boolean hasFocus,
int row, int column) {
String str = combo.getSelectedItem().toString();
if (value.toString().equalsIgnoreCase(str)) {
setBackground(Color.RED);
} else {
setBackground(null);
}
super.getTableCellRendererComponent(table, value, isSelected,
hasFocus, row, column);
return this;
}
});*/
table.setDefaultRenderer(Object.class, new DefaultTableCellRenderer() {
@Override
public Component getTableCellRendererComponent(JTable table,
Object value, boolean isSelected, boolean hasFocus,
int row, int column) {
super.getTableCellRendererComponent(table, value, isSelected,
hasFocus, row, column);
String str = combo.getSelectedItem().toString();
if (value.toString().equalsIgnoreCase(str)) {
setBackground(Color.RED);
table.repaint();
} else {
setBackground(null);
table.repaint();
}
return this;
}
});
table.getTableHeader().setReorderingAllowed(false);
table.setAutoCreateRowSorter(true);
table.setPreferredScrollableViewportSize(table.getPreferredSize());
panel1.setLayout(new GridLayout(1, 1, 10, 10));
panel1.add(new JScrollPane(table));
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(panel, BorderLayout.NORTH);
frame.add(panel1);
frame.pack();
frame.setVisible(true);
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
MyTableAndRenderer fs = new MyTableAndRenderer();
}
});
}
}
編集
@Devolusが書いたものをテストしましたか?私は自分の作業コードからこのスニペットを取得しましたが、答えに関係がないため、間にあるものを削除しました。私はここでJava 6を使用していますが、これは私にとってはうまくいきます。
public Component getTableCellRendererComponent(JTable table,
Object value,
boolean isSelected,
boolean hasFocus,
int row, int column)
{
Component cell = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
... determine the color value ...
cell.setBackground(back);
cell.setForeground(fore);
}
コードから(SSCCEを投稿する理由)
import Java.awt.BorderLayout;
import Java.awt.Color;
import Java.awt.Component;
import Java.awt.GridLayout;
import javax.swing.DefaultComboBoxModel;
import javax.swing.JComboBox;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.border.EmptyBorder;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.DefaultTableModel;
public class MyTableAndRenderer {
private JFrame frame = new JFrame();
private JPanel panel = new JPanel();
private String[] items = {"Item 1", "Item 2", "Item 3", "Item 4"};
private DefaultComboBoxModel comboBoxModel = new DefaultComboBoxModel(items);
private JComboBox combo = new JComboBox(comboBoxModel);
private JPanel panel1 = new JPanel();
private String[] columnNames = {"First Name", "Last Name", "Sport",
"# of Years", "Vegetarian"};
private Object[][] data = {
{"Kathy", "Smith", "Item 1", new Integer(5), (false)},
{"John", "Doe", "Item 1", new Integer(3), (true)},
{"Sue", "Black", "Item 3", new Integer(2), (false)},
{"Jane", "White", "Item 3", new Integer(20), (true)},
{"Joe", "Brown", "Item 3", new Integer(10), (false)}
};
private DefaultTableModel model = new DefaultTableModel(data, columnNames) {
private static final long serialVersionUID = 1L;
@Override
public Class getColumnClass(int column) {
return getValueAt(0, column).getClass();
}
};
private JTable table = new JTable(model);
public MyTableAndRenderer() {
panel.setBorder(new EmptyBorder(10, 0, 2, 0));
panel.add(combo);
table.setDefaultRenderer(Object.class, new DefaultTableCellRenderer() {
@Override
public Component getTableCellRendererComponent(JTable table,
Object value, boolean isSelected, boolean hasFocus,
int row, int column) {
Component c = super.getTableCellRendererComponent(table, value, isSelected,
hasFocus, row, column);
String str = combo.getSelectedItem().toString();
if (value.toString().equalsIgnoreCase(str)) {
c.setBackground(Color.RED);
} else {
c.setBackground(null);
}
return this;
}
});
table.getTableHeader().setReorderingAllowed(false);
table.setAutoCreateRowSorter(true);
table.setPreferredScrollableViewportSize(table.getPreferredSize());
panel1.setLayout(new GridLayout(1, 1, 10, 10));
panel1.add(new JScrollPane(table));
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(panel, BorderLayout.NORTH);
frame.add(panel1);
frame.pack();
frame.setVisible(true);
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
MyTableAndRenderer fs = new MyTableAndRenderer();
}
});
}
}
EDIT2
EDIT3:
可能な限り単純化されたコードであり、ここでの質問の前にテストされた後、レンダリングコンポーネントをJComponent/JLabelにキャストしても機能しないことに注意してください(JLabel.repaint()/ setOpaque()を使用)
選択したアイテムを変更すると、問題が発生します。コンボボックスとテーブルの間に暗黙の相互作用があります(コンボボックスの選択されたアイテムは、テーブルのペイント方法に影響します)。
コンボボックスポップアップが非表示になると、ホバーされた領域の再描画が自動的にトリガーされます(RepaintManagerは、ホバーされた領域のみを再描画し、テーブル全体は再描画しません)。ただし、その間に、テーブルのセルの描画方法を変更しました(最初のセルは、選択範囲と一致しなくなったため、赤で描画されなくなりました)。ただし、再描画マネージャーは、赤血球を完全にカバーしていないテーブルの小さな領域のみを再描画するように強制するため、これらの視覚的な不具合が発生します。
これが私が思いつくことができる2つの解決策です:
ActionListener
を追加し、table.repaint()
を呼び出します(簡単に実行できます)fireTableCellUpdated(row, column)
を呼び出します。2番目のソリューションのSSCCE:
import Java.awt.BorderLayout;
import Java.awt.Color;
import Java.awt.Component;
import Java.awt.GridLayout;
import Java.awt.event.ActionEvent;
import Java.awt.event.ActionListener;
import Java.util.ArrayList;
import Java.util.List;
import javax.swing.DefaultComboBoxModel;
import javax.swing.JComboBox;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.SwingUtilities;
import javax.swing.border.EmptyBorder;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.DefaultTableModel;
public class MyTableAndRenderer {
private final class DefaultTableModelExtension extends DefaultTableModel {
private static final long serialVersionUID = 1L;
private String selected;
private DefaultTableModelExtension(Object[][] data, Object[] columnNames) {
super(data, columnNames);
}
@Override
public Class getColumnClass(int column) {
return getValueAt(0, column).getClass();
}
public String getSelected() {
return selected;
}
public void setSelected(String selected) {
if (this.selected == null && selected == null || this.selected != null && this.selected.equalsIgnoreCase(selected)) {
return;
}
class Cell {
public final int row;
public final int column;
public Cell(int row, int column) {
super();
this.row = row;
this.column = column;
}
}
List<Cell> updatedCells = new ArrayList<Cell>();
if (this.selected != null) {
for (int i = 0; i < data.length; i++) {
Object[] o = data[i];
for (int j = 0; j < o.length; j++) {
Object object = o[j];
if (this.selected.toString().equalsIgnoreCase(object.toString())) {
updatedCells.add(new Cell(i, j));
}
}
}
}
this.selected = selected;
if (this.selected != null) {
for (int i = 0; i < data.length; i++) {
Object[] o = data[i];
for (int j = 0; j < o.length; j++) {
Object object = o[j];
if (this.selected.toString().equalsIgnoreCase(object.toString())) {
updatedCells.add(new Cell(i, j));
}
}
}
}
for (Cell pair : updatedCells) {
fireTableCellUpdated(pair.row, pair.column);
}
}
}
private JFrame frame = new JFrame();
private JPanel panel = new JPanel();
private String[] items = { "Item 1", "Item 2", "Item 3", "Item 4" };
private DefaultComboBoxModel comboBoxModel = new DefaultComboBoxModel(items);
private JComboBox combo = new JComboBox(comboBoxModel);
private JPanel panel1 = new JPanel();
private String[] columnNames = { "First Name", "Last Name", "Sport", "# of Years", "Vegetarian" };
private Object[][] data = { { "Kathy", "Smith", "Item 1", new Integer(5), false }, { "John", "Doe", "Item 1", new Integer(3), true },
{ "Sue", "Black", "Item 3", new Integer(2), false }, { "Jane", "White", "Item 3", new Integer(20), true },
{ "Joe", "Brown", "Item 3", new Integer(10), false } };
private DefaultTableModelExtension model = new DefaultTableModelExtension(data, columnNames);
private JTable table = new JTable(model);
public MyTableAndRenderer() {
panel.setBorder(new EmptyBorder(10, 0, 2, 0));
panel.add(combo);
combo.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
updateSelected();
}
});
// Need first synch
updateSelected();
table.setDefaultRenderer(Object.class, new DefaultTableCellRenderer() {
@Override
public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row,
int column) {
Component c = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
String str = combo.getSelectedItem().toString();
if (value.toString().equalsIgnoreCase(str)) {
c.setBackground(Color.RED);
} else {
c.setBackground(null);
}
return this;
}
});
table.getTableHeader().setReorderingAllowed(false);
table.setAutoCreateRowSorter(true);
table.setPreferredScrollableViewportSize(table.getPreferredSize());
panel1.setLayout(new GridLayout(1, 1, 10, 10));
panel1.add(new JScrollPane(table));
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(panel, BorderLayout.NORTH);
frame.add(panel1);
frame.pack();
frame.setVisible(true);
}
private void updateSelected() {
model.setSelected((String) combo.getSelectedItem());
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
MyTableAndRenderer fs = new MyTableAndRenderer();
}
});
}
}
特定の問題を修正するには、次のようにtable.repaint()を追加します。
ただし、テーブルを適切に再描画するには、外部からテーブルを更新する必要があります。この例では、コンボボックスにイベントリスナーを追加し、そこから更新する必要があります。
public Component getTableCellRendererComponent(JTable table,
Object value,
boolean isSelected,
boolean hasFocus,
int row, int column)
{
Component cell = super.getTableCellRendererComponent(table, value, isSelected,
hasFocus, row, column);
String str = combo.getSelectedItem().toString();
if (value.toString().equalsIgnoreCase(str))
{
cell.setBackground(Color.RED);
}
else
{
cell.setBackground(null);
}
return cell;
}