TITLE:JTableHeaderにJCheckBoxを追加してセルの値を切り替える
Posted by aterai at 2009-02-16

JTableHeaderにJCheckBoxを追加してセルの値を切り替える

JTableHeaderにJCheckBoxを追加して、同じ列のJCheckBoxで表示している値をすべて切り替えます。
  • category: swing folder: TableHeaderCheckBox title: JTableHeaderにJCheckBoxを追加してセルの値を切り替える tags: [JTable, JTableHeader, JCheckBox, TableCellRenderer, MouseListener, Icon, JLabel] author: aterai pubdate: 2009-02-16T20:29:54+09:00 description: JTableHeaderにJCheckBoxを追加して、同じ列のJCheckBoxで表示している値をすべて切り替えます。 image: https://lh6.googleusercontent.com/_9Z4BYR88imo/TQTUf8Li6CI/AAAAAAAAAmI/mj7-1IwK86o/s800/TableHeaderCheckBox.png hreflang:
       href: https://java-swing-tips.blogspot.com/2009/02/jtableheader-checkbox.html
       lang: en

概要

JTableHeaderJCheckBoxを追加して、同じ列のJCheckBoxで表示している値をすべて切り替えます。
TableHeaderCheckBox.png

サンプルコード

#spanend
#spanadd
* サンプルコード [#sourcecode]
#spanend
#spanadd
#code(link){{
#spanend
enum Status { SELECTED, DESELECTED, INDETERMINATE }
class HeaderRenderer extends JCheckBox implements TableCellRenderer {
  public HeaderRenderer(JTableHeader header, final int targetColumnIndex) {
    super();
  private final JLabel label = new JLabel("Check All");
  private final int targetColumnIndex;
#spanadd

#spanend
  public HeaderRenderer(JTableHeader header, int index) {
    super((String) null);
    this.targetColumnIndex = index;
    setOpaque(false);
    setFont(header.getFont());
    header.addMouseListener(new MouseAdapter() {
      @Override public void mouseClicked(MouseEvent e) {
        JTableHeader header = (JTableHeader)e.getSource();
        JTableHeader header = (JTableHeader) e.getComponent();
        JTable table = header.getTable();
        TableColumnModel columnModel = table.getColumnModel();
        int vci = columnModel.getColumnIndexAtX(e.getX());
        int mci = table.convertColumnIndexToModel(vci);
        if(mci == targetColumnIndex) {
        if (mci == targetColumnIndex) {
          TableColumn column = columnModel.getColumn(vci);
          Object v = column.getHeaderValue();
          boolean b = Status.DESELECTED.equals(v)?true:false;
          boolean b = column.getHeaderValue() == Status.DESELECTED;
          TableModel m = table.getModel();
          for(int i=0; i<m.getRowCount(); i++) m.setValueAt(b, i, mci);
          column.setHeaderValue(b?Status.SELECTED:Status.DESELECTED);
          header.repaint();
          for (int i = 0; i < m.getRowCount(); i++) {
            m.setValueAt(b, i, mci);
          }
          column.setHeaderValue(b ? Status.SELECTED : Status.DESELECTED);
          // header.repaint();
        }
      }
    });
  }
#spanadd

#spanend
  @Override public Component getTableCellRendererComponent(
      JTable tbl, Object val, boolean isS, boolean hasF, int row, int col) {
        JTable tbl, Object val, boolean isS, boolean hasF, int row, int col) {
    TableCellRenderer r = tbl.getTableHeader().getDefaultRenderer();
    JLabel l =(JLabel)r.getTableCellRendererComponent(tbl,"Check All",isS,hasF,row,col);
    if(val instanceof Status) {
      switch((Status)val) {
        case SELECTED:    setSelected(true);  setEnabled(true);  break;
        case DESELECTED:  setSelected(false); setEnabled(true);  break;
        case INDETERMINATE: setSelected(true);  setEnabled(false); break;
    JLabel l = (JLabel) r.getTableCellRendererComponent(tbl, val, isS, hasF, row, col);
    if (targetColumnIndex == tbl.convertColumnIndexToModel(col)) {
      if (val instanceof Status) {
        switch ((Status) val) {
        case SELECTED:
          setSelected(true);
          setEnabled(true);
          break;
        case DESELECTED:
          setSelected(false);
          setEnabled(true);
          break;
        case INDETERMINATE:
          setSelected(true);
          setEnabled(false);
          break;
        default:
          throw new AssertionError("Unknown Status");
        }
      } else {
        setSelected(true);
        setEnabled(false);
      }
    }else{
      setSelected(true); setEnabled(false);
      label.setIcon(new ComponentIcon(this));
      l.setIcon(new ComponentIcon(label));
      l.setText(null);
    }
    l.setIcon(new CheckBoxIcon(this));
    if(l.getPreferredSize().height>1000) { //XXX: Nimbus
      System.out.println(l.getPreferredSize().height);
      l.setPreferredSize(new Dimension(0, 28));
    }
    return l;
  }
}
#spandel
class CheckBoxIcon implements Icon{
#spanend
  private final JCheckBox check;
  public CheckBoxIcon(JCheckBox check) {
    this.check = check;
#spanadd

#spanend
#spanadd
class ComponentIcon implements Icon {
#spanend
  private final JComponent cmp;
#spanadd

#spanend
  public ComponentIcon(JComponent cmp) {
    this.cmp = cmp;
  }
#spanadd

#spanend
  @Override public int getIconWidth() {
    return check.getPreferredSize().width;
    return cmp.getPreferredSize().width;
  }
#spanadd

#spanend
  @Override public int getIconHeight() {
    return check.getPreferredSize().height;
    return cmp.getPreferredSize().height;
  }
#spanadd

#spanend
  @Override public void paintIcon(Component c, Graphics g, int x, int y) {
    SwingUtilities.paintComponent(
        g, check, (Container)c, x, y, getIconWidth(), getIconHeight());
        g, cmp, c.getParent(), x, y, getIconWidth(), getIconHeight());
  }
}

解説

上記のサンプルでは、JCheckBoxのアイコンを作成して、これをTableCellRenderer(=JLabel)にsetIconすることで描画しています。このため、ヘッダに追加したMouseListenerで切り替えを行っています。

解説

上記のサンプルでは、JCheckBoxのアイコンを作成し、これをTableCellRenderer(=JLabel)にsetIcon(...)メソッドで設定しています。
  • JCheckBoxをアイコン化しているので、これをクリックしてもイベントは発生しない
    • 代わりにヘッダ自体にMouseListenerを追加し、マウスクリックイベントが発生するとアイコンの入れ替えることでチェック状態の表示を更新する
  • LookAndFeelによってはTableCellRendererのソートアイコンと競合する場合がある

  • JTableのセル中にあるJCheckBoxが全てチェックされた場合、ヘッダのJCheckBoxもチェックされる
  • JTableのセル中にあるJCheckBoxのチェックが全てクリアされた場合、ヘッダのJCheckBoxのチェックもクリアされる
  • JTableのセル中にあるJCheckBoxでチェックの有無が混在している場合、ヘッダのJCheckBoxは薄くチェックされた状態(setEnabled(false)でsetSelected(true))になる
  • JTableのモデルの更新
    • JTableのセル中にあるJCheckBoxが全てチェックされた場合、ヘッダのJCheckBoxもチェックされる
    • JTableのセル中にあるJCheckBoxのチェックが全てクリアされた場合、ヘッダのJCheckBoxのチェックもクリアされる
    • JTableのセル中にあるJCheckBoxでチェックの有無が混在している場合、ヘッダのJCheckBoxは薄くチェックされた状態(setEnabled(false)setSelected(true))になる
#spanend
#spandel
model.addTableModelListener(new TableModelListener() {
#spanend
  @Override public void tableChanged(TableModelEvent e) {
    if(e.getType()==TableModelEvent.UPDATE && e.getColumn()==0) {
      int mci = 0;
      int vci = table.convertColumnIndexToView(mci);
      TableColumn column = table.getColumnModel().getColumn(vci);
      Object title = column.getHeaderValue();
      if(!Status.INDETERMINATE.equals(title)) {
        column.setHeaderValue(Status.INDETERMINATE);
      }else{
        int selected = 0, deselected = 0;
        TableModel m = table.getModel();
        for(int i=0; i<m.getRowCount(); i++) {
          if(Boolean.TRUE.equals(m.getValueAt(i, mci))) {
            selected++;
          }else{
            deselected++;
          }
        }
        if(selected==0) {
          column.setHeaderValue(Status.DESELECTED);
        }else if(deselected==0) {
          column.setHeaderValue(Status.SELECTED);
        }else{
          return;
        }
      }
      table.getTableHeader().repaint();
    }
  }
#spandel
});
#spanend
#spandel
  • -

参考リンク

参考リンク

コメント

  • LookAndFeel を Nimbus に変更したとき、TableHeader の高さがおかしい? -- aterai
  • Java Swing Tips: JTableHeader CheckBoxで、ヘッダクリックで全選択した後、テーブル中のチェックを外すと、ヘッダのチェックボックスもチェック外した方がよくないか?との指摘を頂いたので、(Gmailなどのチェックボックス風の)不定状態?を導入しました。 -- aterai

コメント