Swing/ComboBoxCellEditor のバックアップ(No.4)
- バックアップ一覧
- 差分 を表示
- 現在との差分 を表示
- 現在との差分 - Visual を表示
- ソース を表示
- Swing/ComboBoxCellEditor へ行く。
- category: swing folder: ComboBoxCellEditor title: JTreeのセルエディタにJComboBoxなどを配置したJPanelを使用する tags: [JTree, TreeCellEditor, TreeCellRenderer, JComboBox, JPanel] author: aterai pubdate: 2014-06-09T01:29:59+09:00 description: JTreeのセルエディタ、セルレンダラーとして、JComboBoxなどを配置したJPanelを使用します。 image:
概要
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()
メソッドでドロップダウンリストを開くように設定しています。