JTreeのセルエディタにJComboBoxなどを配置したJPanelを使用する
Total: 4466
, Today: 1
, Yesterday: 1
Posted by aterai at
Last-modified:
概要
JTree
のセルエディタ、セルレンダラーとして、JComboBox
などを配置したJPanel
を使用します。
Screenshot
Advertisement
サンプルコード
class PluginCellEditor extends DefaultCellEditor {
private final PluginPanel panel;
private transient Node node;
public PluginCellEditor(JComboBox<String> comboBox) {
super(comboBox);
panel = new PluginPanel(comboBox);
}
@Override public Component getTreeCellEditorComponent(
JTree tree, Object value, boolean selected, boolean expanded,
boolean leaf, int row) {
Node node = panel.extractNode(value);
panel.setContents(node);
this.node = node;
return panel;
}
@Override public Object getCellEditorValue() {
Object o = super.getCellEditorValue();
if (node == null) {
return o;
}
DefaultComboBoxModel<String> m =
(DefaultComboBoxModel<String>) panel.comboBox.getModel();
Node n = new Node(panel.pluginName.getText(), node.plugins);
n.setSelectedPluginIndex(m.getIndexOf(o));
return n;
}
@Override public boolean isCellEditable(EventObject e) {
Object source = e.getSource();
if (!(source instanceof JTree) || !(e instanceof MouseEvent)) {
return false;
}
JTree tree = (JTree) source;
MouseEvent me = (MouseEvent) e;
TreePath path = tree.getPathForLocation(me.getX(), me.getY());
if (path == null) {
return false;
}
Object node = path.getLastPathComponent();
if (!(node instanceof DefaultMutableTreeNode)) {
return false;
}
Rectangle r = tree.getPathBounds(path);
if (r == null) {
return false;
}
Dimension d = panel.getPreferredSize();
r.setSize(new Dimension(d.width, r.height));
if (r.contains(me.getX(), me.getY())) {
showComboPopup(tree, me);
return true;
}
return delegate.isCellEditable(e);
}
private void showComboPopup(final JTree tree, final MouseEvent me) {
EventQueue.invokeLater(new Runnable() {
@Override public void run() {
Point pt = SwingUtilities.convertPoint(tree, me.getPoint(), panel);
Component o = SwingUtilities.getDeepestComponentAt(panel, pt.x, pt.y);
if (o instanceof JComboBox) {
panel.comboBox.showPopup();
} else if (o != null) {
Container c = SwingUtilities.getAncestorOfClass(
JComboBox.class, (Component) o);
if (c instanceof JComboBox) {
panel.comboBox.showPopup();
}
}
}
});
}
}
View in GitHub: Java, Kotlin解説
上記のサンプルでは、JLabel
とJComboBox
を配置したJPanel
を描画や編集に移譲するTreeCellRenderer
とTreeCellEditor
を作成して、それぞれ、JTree#setCellRenderer(...)
、JTree#setCellEditor(...)
で設定しています。
TreeCellEditor
にはコンストラクタでJComboBox
を設定するDefaultCellEditor
を使用しているが、このJComboBox
はJPanel
の子要素になるため一回目のクリックでノードが編集開始されたときにJComboBox
のドロップダウンリストを開くことができない- 二回目ならすでにセルエディタとして
JPanel
自体がJTree
の前面に表示されているので子コンポーネントのJComboBox
をクリックすればドロップダウンリストが開く
- 二回目ならすでにセルエディタとして
- そのため、このサンプルでは
TreeCellEditor#isCellEditable(...)
をオーバーライドし、ノード(JPanel
)のクリックされた位置に存在するコンポーネントがJComboBox
(またはJComboBox
内にあるArrowButton
)の場合は、編集が開始された後(EventQueue.invokeLater(...)
を使用してセルエディタが表示された後で実行)、JComboBox.showPopup()
メソッドを実行してドロップダウンリストを開くように設定している