JTreeのノードにクリック可能なJButtonを複数配置する
Total: 2418
, Today: 1
, Yesterday: 1
Posted by aterai at
Last-modified:
概要
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
のみマウスでクリック可能