TITLE:JTableのセルに複数のJButtonを配置する
#navi(../)
#tags(JTable, TableCellEditor, TableCellRenderer, JButton, JPanel, ActionListener)
RIGHT:Posted by &author(aterai); at 2009-10-05
*JTableのセルに複数のJButtonを配置する [#wd1c2d32]
``JTable``のセル内にクリック可能な複数の``JButton``を配置します。

-&jnlp;
-&jar;
-&zip;

//#screenshot
#ref(http://lh4.ggpht.com/_9Z4BYR88imo/TQTQRygoYeI/AAAAAAAAAfU/-Sr9o7PsQkM/s800/MultipleButtonsInTableCell.png)

**サンプルコード [#ybe99159]
#code(link){{
class ButtonsPanel extends JPanel {
  public final List<JButton> buttons =
    Arrays.asList(new JButton("view"), new JButton("edit"));
  public ButtonsPanel() {
    super();
    setOpaque(true);
    for(JButton b: buttons) {
      b.setFocusable(false);
      b.setRolloverEnabled(false);
      add(b);
    }
  }
}
}}
#code{{
class ButtonsRenderer extends ButtonsPanel
                      implements TableCellRenderer {
  public ButtonsRenderer() {
    super();
    setName("Table.cellRenderer");
  }
  @Override public Component getTableCellRendererComponent(JTable table,
        Object value, boolean isSelected, boolean hasFocus, int row, int column) {
    setBackground(isSelected?table.getSelectionBackground():table.getBackground());
    return this;
  }
}
}}
#code{{
class ButtonsEditor extends ButtonsPanel
                    implements TableCellEditor {
  public ButtonsEditor(final JTable table) {
    super();
    //---->
    //DEBUG: view button click -> control key down + edit button(same cell) press
    //       -> remain selection color
    MouseListener ml = new MouseAdapter() {
      @Override public void mousePressed(MouseEvent e) {
        ButtonModel m = ((JButton)e.getSource()).getModel();
        if(m.isPressed() && table.isRowSelected(table.getEditingRow())
                         && e.isControlDown()) {
          setBackground(table.getBackground());
        }
      }
    };
    buttons.get(0).addMouseListener(ml);
    buttons.get(1).addMouseListener(ml);
    //<----

    buttons.get(0).addActionListener(new ActionListener() {
      @Override public void actionPerformed(ActionEvent e) {
        fireEditingStopped();
        JOptionPane.showMessageDialog(table, "Viewing");
      }
    });

    buttons.get(1).addActionListener(new ActionListener() {
      @Override public void actionPerformed(ActionEvent e) {
        int row = table.convertRowIndexToModel(table.getEditingRow());
        Object o = table.getModel().getValueAt(row, 0);
        fireEditingStopped();
        JOptionPane.showMessageDialog(table, "Editing: "+o);
      }
    });

    addMouseListener(new MouseAdapter() {
      @Override public void mousePressed(MouseEvent e) {
        fireEditingStopped();
      }
    });
  }
  @Override public Component getTableCellEditorComponent(JTable table,
        Object value, boolean isSelected, int row, int column) {
    this.setBackground(table.getSelectionBackground());
    return this;
  }
  @Override public Object getCellEditorValue() {
    return "";
  }
  //Copid from AbstractCellEditor
  //protected EventListenerList listenerList = new EventListenerList();
  transient protected ChangeEvent changeEvent = null;
  @Override public boolean isCellEditable(java.util.EventObject e) {
    return true;
  }
//......
}}

**解説 [#e6806cbc]
上記のサンプルでは、``CellRenderer``用と``CellEditor``用に、``JButton``を2つ配置した``JPanel``をそれぞれ作成しています。アクションイベントを設定するのは、``CellEditor``用の``JButton``で、``CellRenderer``用の``JButton``は表示のためのダミーです。

// %%行の選択状態が切り替わるとき(例えば、一行目を選択していて、二行目のボタンをクリック)には、``CellEditor``用の``JButton``がうまくクリックできないので、``JTable``自体に以下のようなマウスリスナーを設定しています。%%
//
//#code{{
//class CellButtonsMouseListener extends MouseAdapter{
//  @Override public void mouseReleased(MouseEvent e) {
//    JTable t = (JTable)e.getComponent();
//    Point pt = e.getPoint();
//    int row  = t.rowAtPoint(pt);
//    int col  = t.columnAtPoint(pt);
//    if(t.convertRowIndexToModel(row)>=0 && t.convertColumnIndexToModel(col)==1) {
//      TableCellEditor ce = t.getCellEditor(row, col);
//      ce.stopCellEditing();
//      Component c = ce.getTableCellEditorComponent(t, null, true, row, col);
//      Point p = SwingUtilities.convertPoint(t, pt, c);
//      Component b = SwingUtilities.getDeepestComponentAt(c, p.x, p.y);
//      if(b instanceof JButton) ((JButton)b).doClick();
//    }
//  }
//}
//}}

**参考リンク [#b4c9d73f]
-[[JTableのセルにJButtonを追加して行削除>Swing/DeleteButtonInCell]]
-[[JTableのセルにHyperlinkを表示>Swing/HyperlinkInTableCell]]
-[http://tips4java.wordpress.com/2009/07/12/table-button-column/ Table Button Column « Java Tips Weblog]
-[[JTableのセル中にJRadioButtonを配置>Swing/RadioButtonsInTableCell]]
-[[JTableのCellにJCheckBoxを複数配置する>Swing/CheckBoxesInTableCell]]

**コメント [#le0df76f]
- 第``0``列目が編集状態でボタンをクリックした場合、パネルが二度表示されるバグを修正。 -- [[aterai]] &new{2009-10-06 (火) 11:56:21};
- [http://tips4java.wordpress.com/2009/07/12/table-button-column/ Table Button Column « Java Tips Weblog]を参考にして、``JTable#editCellAt``ではなく、逆に``TableCellEditor#stopCellEditing()``を使用してクリック直後に編集終了するように変更。 -- [[aterai]] &new{2009-11-03 (火) 04:36:55};
- ``Ctrl``キーを押しながら、``edit``ボタンをクリックすると異なる行(``table.getSelectedRow()``)の内容が表示されるバグを修正。 -- [[aterai]] &new{2011-03-10 (木) 02:35:35};
- すごいと思いました! -- [[いわく]] &new{2013-05-24 (金) 11:57:26};
-- こんばんは。たしかに``JTable``の``TableCellRenderer``、``TableCellEditor``の仕組みは、すごい良くできているといつも感心してしまいます :) -- [[aterai]] &new{2013-05-24 (金) 23:59:16};

#comment