TITLE:TableCellRendererでセルの背景色を変更
#navi(../)
*TableCellRendererでセルの背景色を変更 [#r8e14fa9]
>編集者:[[Terai Atsuhiro>terai]]~
作成日:2004-01-19~
更新日:&lastmod;

#contents

**概要 [#tcf81150]
TableCellRendererを継承するレンダラーを作ってテーブルのセルを修飾します。

#screenshot

**サンプルコード [#p4a666c9]
#code{{
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;
  }
}
}}
-&jnlp;
-&jar;
-&zip;

**解説 [#j15250af]
上記のサンプルでは、以下のようなセルレンダラーを作成し、Objectを継承するクラスのデフォルトレンダラーとして設定しています。
-奇数偶数で行の背景色を変更してテーブルをストライプ模様にする
-第0列のカラムのセルを右寄せ
--ここでは、TableColumn#setCellRenderer(TableCellRenderer)((JTable#setDefaultRenderer(Class, TableCellRenderer)でクラスに関連付けるより、各カラムに関連付けした方が優先順位が高い))を使わず、Object.classのDefaultRendererひとつにまとめているため、TableModel#getColumnClass(0)がInteger.classを返すようにしている

#code{{
table.setDefaultRenderer(Object.class, new TestRenderer());
}}

あるセルが描画されるとき、設定されたTableCellRendererのgetTableCellRendererComponentメソッドが呼び出されます。レンダラーは、引数などから得られる情報(選択されているか、何行何列目かなど)を使って、コンポーネントを修飾してからreturnします。

サンプルのTestRendererでは、JLabelを継承するDefaultTableCellRendererを継承しているので、自分自身(this)をsetForeground、setHorizontalAlignmentなどのメソッドで修飾し直し、さらに自分自身(this)を戻り値としています。このようにコンポーネントを使い回しているため、セルの数が膨大になっても、オブジェクトを大量に生成しなくて済むようになっています。

また、返されたコンポーネントは、セルの描画のみに利用されて、コンポーネントとしては機能しません。

セルレンダラーで色を変更する代わりに、以下のようにJTable#prepareRendererメソッドをオーバーライドする方法もあります。使用するセルレンダラーに関係なく、テーブル全体で前処理することができます。このため、Numberクラス用のDefaultTableCellRendererがそのまま使われるので、TableModel#getColumnClass(int)が、Number.classを返すようにしておけば、勝手に右寄せしてくれます。
#code{{
JTable table = new JTable(model) {
  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;
  }
};
}}

**参考リンク [#sdd74788]
-[[Sorting and Otherwise Manipulating Data - How to Use Tables (The Java™ Tutorials > Creating a GUI with JFC/Swing > Using Swing Components)>http://java.sun.com/docs/books/tutorial/uiswing/components/table.html#sorting]]
-[[SwingのJTableコンポーネントでセルを描く>http://www-06.ibm.com/jp/developerworks/java/010518/j_j-jtable.shtml]]

**コメント [#sfc984fe]
- JTable#prepareRendererメソッドを使用する場合のサンプルを修正。 -- [[terai]] &new{2007-04-04 (水) 19:41:37};

#comment