Swing/MultipleButtonsInTreeNode のバックアップ(No.5)
- バックアップ一覧
- 差分 を表示
- 現在との差分 を表示
- 現在との差分 - 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?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
のみマウスでクリック可能