• 追加された行はこの色です。
  • 削除された行はこの色です。
TITLE:JTableのセル文字揃え
#navi(../)
RIGHT:Posted by [[terai]] at 2008-08-25
*JTableのセル文字揃え [#y99ee441]
JTableのセルに表示されている文字列の揃えを変更します。
---
category: swing
folder: CellTextAlignment
title: JTableのセル文字揃え
tags: [JTable, TableCellRenderer, Alignment]
author: aterai
pubdate: 2008-08-25T14:45:47+09:00
description: JTableのセルに表示されている文字列の揃えを変更します。
image: https://lh6.googleusercontent.com/_9Z4BYR88imo/TQTIs6qWcBI/AAAAAAAAATM/AnH_ZWdWA5o/s800/CellTextAlignment.png
---
* 概要 [#summary]
`JTable`のセルに表示されている文字列の揃えを変更します。

-&jnlp;
-&jar;
-&zip;
#download(https://lh6.googleusercontent.com/_9Z4BYR88imo/TQTIs6qWcBI/AAAAAAAAATM/AnH_ZWdWA5o/s800/CellTextAlignment.png)

#screenshot

**サンプルコード [#ocb54d34]
#code{{
* サンプルコード [#sourcecode]
#code(link){{
TableColumn col = table.getColumnModel().getColumn(1);
col.setCellRenderer(new HorizontalAlignmentTableRenderer(
                          new DefaultTableCellRenderer()));
}}
#code{{
class HorizontalAlignmentTableRenderer implements TableCellRenderer {
  private final TableCellRenderer renderer;
  public HorizontalAlignmentTableRenderer(TableCellRenderer renderer) {
    this.renderer = renderer;
  }
  public Component getTableCellRendererComponent(JTable table, Object value,
            boolean isSelected, boolean hasFocus, int row, int column) {
    Component c = renderer.getTableCellRendererComponent(table, value,
              isSelected, hasFocus, row, column);
    if(c instanceof JLabel) initLabel((JLabel)c, row);
col.setCellRenderer(new HorizontalAlignmentTableRenderer());
// ...
class HorizontalAlignmentTableRenderer extends 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);
    if (c instanceof JLabel) {
      initLabel((JLabel) c, row);
    }
    return c;
  }

  private void initLabel(JLabel l, int row) {
    if(leftRadio.isSelected()) {
      l.setHorizontalAlignment(JLabel.LEFT);
    }else if(centerRadio.isSelected()) {
      l.setHorizontalAlignment(JLabel.CENTER);
    }else if(rightRadio.isSelected()) {
      l.setHorizontalAlignment(JLabel.RIGHT);
    }else if(customRadio.isSelected()) {
      l.setHorizontalAlignment(row%3==0?JLabel.LEFT:
                               row%3==1?JLabel.CENTER:
                                        JLabel.RIGHT);
    if (leftRadio.isSelected()) {
      l.setHorizontalAlignment(SwingConstants.LEFT);
    } else if (centerRadio.isSelected()) {
      l.setHorizontalAlignment(SwingConstants.CENTER);
    } else if (rightRadio.isSelected()) {
      l.setHorizontalAlignment(SwingConstants.RIGHT);
    } else if (customRadio.isSelected()) {
      l.setHorizontalAlignment(row % 3 == 0 ? SwingConstants.LEFT:
                               row % 3 == 1 ? SwingConstants.CENTER:
                                              SwingConstants.RIGHT);
    }
  }
}
}}

**解説 [#b5da87a2]
上記のサンプルでは、上部のラジオボタンで、第一列のセル文字列の揃えを変更することができます。
-left: 左揃え
-center: 中央揃え
-rihgt: 右揃え
-custom: 行ごとに左、中央、右揃えを変更
* 解説 [#explanation]
上記のサンプルでは、`JTable`の第`1`列目のセルに表示されている文字列の揃えを変更するテストを行っています。

----
JTableは、Object、Number、Booleanクラスのデフォルトセルレンダラーを持っているため、モデル((DefaultTableModelはすべての列のクラスとして、Object.classを返す))が各列のクラスを正しく返すように、TableModel#getColumnClass(int)をオーバーライドしてやると、そのクラスのデフォルトセルレンダラーが使用され((列にセルレンダラーが割り当てられていない場合))、表示は数字右揃え、文字列左揃えになります。
- `left`: 左揃え
- `center`: 中央揃え
- `right`: 右揃え
- `custom`: 行ごとに左、中央、右揃えを変更

-Object: JLabel.LEFT
-Number: JLabel.RIGHT
-Boolean: JCheckBox, CENTER

----
- `JTable`は、`Object`、`Number`、`Boolean`クラスのデフォルトセルレンダラーを持っているため、モデルが各列のクラスを正しく返すように`TableModel#getColumnClass(int)`をオーバーライドすることでそのクラスのデフォルトセルレンダラーが使用される
-- `Object`: `SwingConstants.LEFT`(文字列などは左揃え)
-- `Number`: `SwingConstants.RIGHT`(数字は右揃え)
-- `Boolean`: `JCheckBox`, `CENTER`(チェックボックスは中央揃え)
- `DefaultTableModel#TableModel#getColumnClass(int)`のデフォルトは、すべての列のクラスとして`Object.class`を返す
- 各クラスのデフォルトセルレンダラーが使用されるのは、列にセルレンダラーが割り当てられていない場合に限られる
#code{{
//JTableの手抜きサンプルなら、以下のようにオーバーライドするのが簡単?
// JTableの手抜きサンプルなら、以下のようにオーバーライドするのが簡単?(モデルが空になる場合、例外が発生する可能性がある)
String[] columnNames = {"String", "Integer", "Boolean"};
Object[][] data = {
  {"AAA", 1, true},
  {"BBB", 2, false},
};
DefaultTableModel model = new DefaultTableModel(data, columnNames) {
  @Override
  public Class<?> getColumnClass(int column) {
  @Override public Class<?> getColumnClass(int column) {
    return getValueAt(0, column).getClass();
  }
};
JTable table = new JTable(model);
}}

クラスのデフォルトセルレンダラーではなく、任意の列にセルレンダラーを割り当てて、例えば中央揃えにしたい場合は、以下のように設定します。
// -- [[terai]] &new{2007-06-14 (木) 13:19:36};

- クラスのデフォルトセルレンダラーではなく、任意の列にセルレンダラーを割り当てて文字揃えを変更する方法もある
#code{{
DefaultTableCellRenderer r = new DefaultTableCellRenderer();
r.setHorizontalAlignment(JLabel.CENTER);
r.setHorizontalAlignment(SwingConstants.CENTER);
table.getColumnModel().getColumn(2).setCellRenderer(r);
}}

----
-テスト中
--ヘッダの字揃えを以下のように行った場合、フォントの指定は第二列のみ、字揃えは全体に作用する?
--Microsoft Windows XP [Version 5.1.2600]
--Java(TM) SE Runtime Environment (build 1.6.0_07-b06)
* 参考リンク [#reference]
- [[JTableHeaderの字揃え>Swing/HorizontalAlignmentHeaderRenderer]]
-- ヘッダの文字揃えに関するテストは、上記の場所に移動
- [[JTableのセル内文字列を両端揃えにする>Swing/InterIdeographJustify]]

#code{{
col = table.getColumnModel().getColumn(2);
TableCellRenderer hr = table.getTableHeader().getDefaultRenderer();
col.setHeaderRenderer(new HeaderRenderer(hr));
}}
#code{{
static class HeaderRenderer implements TableCellRenderer {
  private final TableCellRenderer renderer;
  private final Font font = new Font("Sans-serif", Font.BOLD, 14);
  public HeaderRenderer(TableCellRenderer renderer) {
    this.renderer  = renderer;
  }
  public Component getTableCellRendererComponent(JTable table, Object value,
            boolean isSelected, boolean hasFocus, int row, int column) {
    JLabel l = (JLabel)renderer.getTableCellRendererComponent(table, value,
              isSelected, hasFocus, row, column);
    l.setHorizontalAlignment(JLabel.CENTER);
    l.setFont(font);
    return l;
  }
}
}}

//**参考リンク
**コメント [#x640db8a]
* コメント [#comment]
#comment
#comment