1

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。虽然这种方法是私有的,但我不确定这是否是正确的方法。任何想法?

4

1 回答 1

2

我自己解决了。我默认禁用 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);
}

}

于 2013-01-29T09:44:50.307 回答