Swing/StripeTable のバックアップの現在との差分(No.2)
TITLE:TableCellRendererでセルの背景色を変更
TableCellRendererでセルの背景色を変更
編集者:Terai Atsuhiro~
作成日:2004-01-19
更新日:2022-02-01 (火) 22:40:22
概要
TableCellRenderer
を継承するレンダラーを作ってテーブルのセルを修飾します。
Screenshot
Advertisement
概要
TableCellRendererを継承するレンダラーを作ってテーブルのセルを修飾します。サンプルコード
#spanend
#spanadd
class StripeTableRenderer extends DefaultTableCellRenderer {
#spanend
private static final Color evenColor = new Color(240, 240, 255);
@Override public Component getTableCellRendererComponent(JTable table, Object value,
boolean isSelected, boolean hasFocus,
int row, int column) {
super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
if (isSelected) {
setForeground(table.getSelectionForeground());
setBackground(table.getSelectionBackground());
} else {
setForeground(table.getForeground());
setBackground((row % 2 == 0) ? evenColor : table.getBackground());
}
setHorizontalAlignment((value instanceof Number) ? RIGHT : LEFT);
return this;
}
#spanadd
}
#spanend
#spanadd
View in GitHub: Java, Kotlin#screenshot
解説
上記のサンプルでは、以下のようなセルレンダラーを作成し、Object
を継承するクラスのデフォルトレンダラーとして設定しています。
サンプルコード
public class TestRenderer extends DefaultTableCellRenderer { private static final Color ec = new Color(240, 240, 255); public TestRenderer() { super(); setOpaque(true); setBorder(BorderFactory.createEmptyBorder(0,5,0,5)); } public Component getTableCellRendererComponent( JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) { super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column); if(isSelected) { setForeground(table.getSelectionForeground()); setBackground(table.getSelectionBackground()); }else{ setForeground(table.getForeground()); setBackground((row%2==0)?ec:table.getBackground()); } setHorizontalAlignment((value instanceof Number)?RIGHT:LEFT); return this; } }
- 奇数偶数で行の背景色を変更してテーブルをストライプ模様にする
- 第
0
列のカラムのセルを右寄せ- ここでは
TableColumn#setCellRenderer(TableCellRenderer)
を使用せずにObject.classのDefaultRenderer
ひとつにまとめているため、TableModel#getColumnClass(0)
がInteger.class
を返すよう設定 -
JTable#setDefaultRenderer(Class, TableCellRenderer)
でクラスに関連付けるより、各カラムに関連付けした方が優先順位が高い
- ここでは
- &jnlp;
- &jar;
- &zip;
#spanend #spanadd table.setDefaultRenderer(Object.class, new StripeTableRenderer()); #spanend #spanadd
解説
上記のサンプルでは、第0列のカラムのセルを右寄せにし、奇数偶数で行の背景色を変更してテーブルをストライプ模様にするレンダラーを作成し、これをObjectを継承するクラスのデフォルトレンダラーとして設定しています。- -
あるセルが描画されるとき、設定された
TableCellRenderer
のgetTableCellRendererComponent
メソッドが呼び出されます。レンダラーは引数などから得られる情報(選択されているか何行何列目かなど)を使ってコンポーネントを修飾してからreturn
します。
table.setDefaultRenderer(Object.class, new TestRenderer());
サンプルのTestRenderer
ではJLabel
を継承するDefaultTableCellRenderer
を継承しているので自分自身(this
)をsetForeground
、setHorizontalAlignment
などのメソッドで修飾し直し、さらに自分自身(this
)を戻り値としています。このようにコンポーネントを使い回しているためセルの数が膨大になってもオブジェクトを大量に生成しなくて済むようになっています。
TableCellRendererがあるセルが描画するとき、getTableCellRendererComponentメソッドが呼び出されます。引数から得られる、選択されているか何行何列目かなどの情報を使って、コンポーネント(ここではJLabel)を修飾し、そのコンポーネントを戻します。これが上記のセルの描画に利用されます*1。
また返されたコンポーネントはセルの描画のためだけに利用されマウスイベントなどは無視されます。
セルレンダラーを作成する代わりに、以下のようにJTable#prepareRendererメソッドをオーバーライドする方法もあります。
JTable table = new JTable(sorter) { public Component prepareRenderer(TableCellRenderer renderer, int row, int column) { Component c = super.prepareRenderer(renderer, row, column); if(row%2==0 && !isRowSelected(row)) { c.setForeground(getForeground()); c.setBackground(evenColor); } return c; } };
- -
セルレンダラーで色を変更する代わりに以下のように
JTable#prepareRenderer
メソッドをオーバーライドする方法もあります。この場合使用するセルレンダラーに関係なくテーブル全体での前処理が可能です。このためNumber
クラス用デフォルトレンダラーのJTable$NumberRenderer
がそのまま使用され、TableModel#getColumnClass(int)
がNumber.class
を返すようにしておけば自動的に右寄せになります。
参考リンク
- Sorting and Otherwise Manipulating Data - How to Use Tables (The Java™ Tutorials > Creating a GUI with JFC/Swing > Using Swing Components)
- SwingのJTableコンポーネントでセルを描く
#spanend #spanadd JTable table = new JTable(model) { #spanend private final Color evenColor = new Color(240, 240, 255); @Override public Component prepareRenderer(TableCellRenderer tcr, int row, int column) { Component c = super.prepareRenderer(tcr, row, column); if (isRowSelected(row)) { c.setForeground(getSelectionForeground()); c.setBackground(getSelectionBackground()); } else { c.setForeground(getForeground()); c.setBackground((row % 2 == 0) ? evenColor : getBackground()); } return c; } #spanadd }; #spanend #spanadd
コメント
- -
JDK 1.6.0_10
以降では、以下のように設定する方法もあります。
#spanend
#spanadd
UIManager.put("Table.alternateRowColor", Color.PINK);
#spanend
#spanadd
- 注: デフォルトの
Boolean
用レンダラーの背景色は変更されない
#spanend
#spanadd
import java.awt.*;
#spanend
#spanadd
import javax.swing.*;
#spanend
#spanadd
import javax.swing.table.*;
#spanend
#spanadd
public class TableAlternateRowColorTest {
#spanend
public static void main(String... args) {
EventQueue.invokeLater(() -> {
UIManager.put("Table.alternateRowColor", Color.ORANGE);
String[] columnNames = {"String", "Integer", "Boolean"};
Object[][] data = {
{"A", 1, true}, {"B", 2, false}, {"C", 0, true}
};
TableModel model = new DefaultTableModel(data, columnNames) {
@Override public Class<?> getColumnClass(int column) {
return getValueAt(0, column).getClass();
}
};
JFrame f = new JFrame();
f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
f.getContentPane().add(new JScrollPane(new JTable(model)));
f.setSize(320, 240);
f.setLocationRelativeTo(null);
f.setVisible(true);
});
}
#spanadd
}
#spanend
#spanadd
参考リンク
- Sorting and Otherwise Manipulating Data - How to Use Tables (The Java™ Tutorials > Creating a GUI with JFC/Swing > Using Swing Components)
- SwingのJTableコンポーネントでセルを描く