TITLE:JTableのセルに複数のJButtonを配置する

Posted by aterai at 2009-10-05

JTableのセルに複数のJButtonを配置する

JTableのセル内にクリック可能な複数のJButtonを配置します。

  • &jnlp;
  • &jar;
  • &zip;
MultipleButtonsInTableCell.png

サンプルコード

class ButtonsEditorRenderer extends AbstractCellEditor
                            implements TableCellRenderer,TableCellEditor{
  private final JPanel renderer = new JPanel();
  private final JPanel editor   = new JPanel();
  private final JTable table;
  private final MouseListener ml = new MouseAdapter() {
    public void mousePressed(MouseEvent e) {
      ButtonModel m = ((JButton)e.getSource()).getModel();
      if(m.isPressed() && table.isRowSelected(table.getEditingRow())
         && !e.isShiftDown()) {
        editor.setBackground(table.getBackground());
      }
    }
  };
  public ButtonsEditorRenderer(JTable t) {
    super();
    this.table = t;
    JButton viewButton2 = new JButton(new AbstractAction("view") {;
      @Override public void actionPerformed(ActionEvent e) {
        fireEditingStopped();
        JOptionPane.showMessageDialog(table, "Viewing");
      }
    });
    JButton editButton2 = new JButton(new AbstractAction("edit") {;
      @Override public void actionPerformed(ActionEvent e) {
        //if isControlDown bug:
        //Object o = table.getModel().getValueAt(table.getSelectedRow(), 0);
        int row = table.convertRowIndexToModel(table.getEditingRow());
        Object o = table.getModel().getValueAt(row, 0);
        fireEditingStopped();
        JOptionPane.showMessageDialog(table, "Editing: "+o);
      }
    });
    viewButton2.addMouseListener(ml);
    viewButton2.setFocusable(false);
    viewButton2.setRolloverEnabled(false);

    editButton2.addMouseListener(ml);
    editButton2.setFocusable(false);
    editButton2.setRolloverEnabled(false);

    renderer.setOpaque(true);
    renderer.add(new JButton("view"));
    renderer.add(new JButton("edit"));
    editor.setOpaque(true);
    editor.add(viewButton2);
    editor.add(editButton2);
    editor.addMouseListener(new MouseAdapter() {
      @Override public void mousePressed(MouseEvent e) {
        fireEditingStopped();
      }
    });
  }
  @Override public Component getTableCellRendererComponent(
      JTable table, Object value, boolean isSelected, boolean hasFocus,
    int row, int column) {
    renderer.setBackground(isSelected?table.getSelectionBackground()
                     :table.getBackground());
    return renderer;
  }
  @Override public Component getTableCellEditorComponent(
      JTable table, Object value, boolean isSelected, int row, int column) {
    editor.setBackground(table.getSelectionBackground());
    return editor;
  }
  @Override public Object getCellEditorValue() {
    return "";
  }
}

解説

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

参考リンク

コメント

  • 第0列目が編集状態でボタンをクリックした場合、パネルが二度表示されるバグを修正。 -- aterai
  • Table Button Column « Java Tips Weblogを参考にして、JTable#editCellAtではなく、逆にTableCellEditor#stopCellEditing()を使用するように変更しました。 -- aterai
  • Ctrlキーを押しながら、editボタンをクリックすると異なる行(table.getSelectedRow())の内容が表示されるバグを修正。 -- aterai