JavaFX 2 TreeView - 如何更改进入编辑模式的默认行为?

时间:2013-01-28 16:47:33

标签: java javafx-2

http://docs.oracle.com/javafx/2/ui_controls/tree-view.htm上的JavaFX教程的启发我想知道如何更改在编辑模式下进入单元格的行为。我想要的行为是

    左边的鼠标点击
  • :只需选择单元格
  • 两次鼠标左键单击:选择单元格并调用一些操作
  • 右键单击
  • :在编辑模式下输入单元格

我尝试在TreeView / TreeCell上安装鼠标事件处理程序,但似乎该事件已被TreeCellBehavior使用。

在类TreeCellBehvior中有以下方法:

private void simpleSelect(MouseEvent e) {
    TreeView tv = getControl().getTreeView();
    TreeItem treeItem = getControl().getTreeItem();
    int index = getControl().getIndex();
    MultipleSelectionModel sm = tv.getSelectionModel();
    boolean isAlreadySelected = sm.isSelected(index);

    tv.getSelectionModel().clearAndSelect(index);

    // handle editing, which only occurs with the primary mouse button
    if (e.getButton() == MouseButton.PRIMARY) {
        if (e.getClickCount() == 1 && isAlreadySelected) {
            tv.edit(treeItem);
        } else if (e.getClickCount() == 1) {
            // cancel editing
            tv.edit(null);
        } else if (e.getClickCount() == 2/* && ! getControl().isEditable()*/) {
            if (treeItem.isLeaf()) {
                // attempt to edit
                tv.edit(treeItem);
            } else {
                // try to expand/collapse branch tree item
                treeItem.setExpanded(! treeItem.isExpanded());
            }
        }
    }
}

我不确定是否可以用我自己的实现替换TreeCellBehavior。虽然这种方法是私有的,但我不确定这是不是正确的方法。有什么想法吗?

1 个答案:

答案 0 :(得分:1)

我自己解决了这个问题。我默认禁用TreeView的可编辑。对于每个TreeItem,都有一个允许更改项名称的上下文菜单。如果调用上下文菜单操作,TreeView将设置为可编辑,并调用具有当前TreeItem的TreeView.edit()。现在在幕后调用startEdit()并且编辑模式处于活动状态。

然而,在按下enter并调用commitEdit()后,我有一些奇怪的行为。此方法检查单元格是否仍处于编辑模式(它是,因此返回true)导致内部调用cancelEdit()?!?!作为一种解决方法,我引入了一个commitModeProperty并检查了cancelEdit()是否设置了..否则将永远不会设置新的文本值。

这是我的代码:

public class FolderTreeCell extends TreeCell<FolderCellType> {

// workaround for a strange behaviour in commitEdit.. see initTextFieldListener() 
private BooleanProperty commitModeProperty = new SimpleBooleanProperty(false);

public FolderTreeCell() {
    assert Platform.isFxApplicationThread();
}

private ContextMenu createContextMenu() {
    MenuItem menuItem = new MenuItem("Change folder name");

    menuItem.setOnAction(new EventHandler<ActionEvent>() {
        @Override
        public void handle(ActionEvent evt) {
            getTreeView().setEditable(true);
            getTreeView().edit(getTreeItem());
        }

    });

    return new ContextMenu(menuItem);
}

private void initTextFieldListener() {
    getItem().textFieldProperty().get().setOnKeyReleased(new EventHandler<KeyEvent>() {
        @Override
        public void handle(KeyEvent evt) {
            if (evt.getCode() == KeyCode.ENTER) {
                commitEdit(getItem()); // TODO calls updateItem() when isEditing() is true causing invocation of cancelEdit() ?!?!
            } 
        }

    });
}

@Override
public void commitEdit(FolderCellType newFolderCellType) {
    commitModeProperty.set(true);
    super.commitEdit(newFolderCellType);
    commitModeProperty.set(false);
}

@Override
public void startEdit() {
    super.startEdit();

    setGraphic(getItem().getEditBox());

    if (getItem().textFieldProperty().get().getOnKeyReleased() == null) {
        initTextFieldListener();
    }

    getItem().textFieldProperty().get().selectAll();
    getItem().textFieldProperty().get().requestFocus();
}

@Override
public void cancelEdit() {
    super.cancelEdit();

    getTreeView().setEditable(false);
    if (!commitModeProperty.getValue()) {
        getItem().resetCurrentEntry();
    }

    setGraphic(getItem().getViewBox());
}

@Override
public void updateItem(FolderCellType item, boolean empty) {
    super.updateItem(item, empty);

    if (empty || item == null) {
        setText(null);
        setGraphic(null);
    } else {
        if (isEditing()) {
            setGraphic(item.getEditBox());
        } else {
            setGraphic(item.getViewBox());

            if (getContextMenu() == null) {
                setContextMenu(createContextMenu());
            }
        }
    }

    getTreeView().setEditable(false);
}

}