Swing/MultipleButtonsInTreeNode のバックアップ(No.2)
- バックアップ一覧
- 差分 を表示
- 現在との差分 を表示
- 現在との差分 - Visual を表示
- ソース を表示
- Swing/MultipleButtonsInTreeNode へ行く。
- 1 (2018-02-13 (火) 15:56:13)
- 2 (2018-02-13 (火) 18:33:02)
- 3 (2018-02-15 (木) 14:23:42)
- 4 (2020-02-13 (木) 14:44:55)
- 5 (2021-07-31 (土) 08:07:58)
- category: swing folder: MultipleButtonsInTreeNode title: JTreeのノードにクリック可能なJButtonを複数配置する tags: [JTree, TreeCellEditor, TreeCellRenderer, JButton, JPanel] author: aterai pubdate: 2018-02-12T19:19:08+09:00 description: JTreeのセルエディタとしてクリック可能なJButtonを複数したJPanelを設定します。 image: https://drive.google.com/uc?export=view&id=1av1tLvTqv_249jfJdwdsuhEyB5i3zikC7Q
概要
JTree
のセルエディタとしてクリック可能なJButton
を複数したJPanel
を設定します。
Screenshot
Advertisement
サンプルコード
class ButtonCellEditor extends AbstractCellEditor implements TreeCellEditor {
private final ButtonPanel panel = new ButtonPanel();
protected ButtonCellEditor() {
super();
panel.b1.addActionListener(e -> {
System.out.println("b1: " + panel.renderer.getText());
stopCellEditing();
});
panel.b2.addActionListener(e -> {
System.out.println("b2: " + panel.renderer.getText());
stopCellEditing();
});
panel.b3.addActionListener(e -> {
System.out.println("b3: " + panel.renderer.getText());
stopCellEditing();
});
}
@Override public Component getTreeCellEditorComponent(
JTree tree, Object value, boolean isSelected, boolean expanded,
boolean leaf, int row) {
Component c = panel.renderer.getTreeCellRendererComponent(
tree, value, true, expanded, leaf, row, true);
return panel.remakePanel(c);
}
@Override public Object getCellEditorValue() {
return panel.renderer.getText();
}
@Override public boolean isCellEditable(EventObject e) {
Object source = e.getSource();
if (!(source instanceof JTree) || !(e instanceof MouseEvent)) {
return false;
}
JTree tree = (JTree) source;
Point p = ((MouseEvent) e).getPoint();
TreePath path = tree.getPathForLocation(p.x, p.y);
if (Objects.isNull(path)) {
return false;
}
Rectangle r = tree.getPathBounds(path);
if (Objects.isNull(r)) {
return false;
}
// r.width = panel.getButtonAreaWidth();
// return r.contains(p);
if (r.contains(p)) {
TreeNode node = (TreeNode) path.getLastPathComponent();
int row = tree.getRowForLocation(p.x, p.y);
Component c = tree.getCellRenderer().getTreeCellRendererComponent(tree, " ", true, true, node.isLeaf(), row, true);
c.setBounds(r);
c.setLocation(0, 0);
// tree.doLayout();
tree.revalidate();
p.translate(-r.x, -r.y);
Component o = SwingUtilities.getDeepestComponentAt(c, p.x, p.y);
if (o instanceof JButton) {
return true;
}
}
return false;
}
}
View in GitHub: Java, Kotlin解説
上記のサンプルでは、3
つのJButton
とJLabel
を継承するDefaultTreeCellRenderer
をFlowLayout
で配置するJPanel
を2
つ作成し、それぞれをJTree
のセルエディタとセルレンダラーとして設定しています。
- セルレンダラーの
JButton
は表示にのみ使用するめ、ActionListener
などは設定しない - セルエディタの
JButton
には、JTree
のノードが編集可能になって自身がクリックされたときに実行するActionListener
を設定- この
ActionListener
が実行されたあと、AbstractCellEditor#stopCellEditing()
を実行してJTree
のノード編集を終了する JTree
のノード編集を終了しないと、編集中のノードの各ボタンにフォーカスが描画されたり、別ノードのボタンを2
回クリックしないとそのボタンのイベントが実行できない
- この
TreeCellEditor#isCellEditable(...)
をオーバーライドし、マウスで各JButton
の表示領域がクリックされた場合のみ編集可能に設定JButton
の表示領域がクリックされたかどうかの判断は、編集開始前なのでセルエディタではなくセルレンダラー内をSwingUtilities.getDeepestComponentAt(...)
で検索してJButton
が返されるかを調査している
- 編集中のノードの
JButton
のみマウスでクリック可能