问题

ObservableList中删除项目时,会触发change event,其中getFrom()给出了删除位置,而 getRemoved() 给出了已删除项目的列表。 The documentation说:



并不是这样说的,但是我认为这意味着项目列表是原始列表中的连续子列表。我已经用这种假设编写了很多代码,但是现在 TreeTableView 's selection model遇到了困难,这种行为不是那样的。



例如,一个简单的树表具有三个“节点”行。如果我选择那三行...

java - JavaFX的ListChangeListener.Change.getRemoved()是否可以返回不连续的项目?-LMLPHP

...然后单击并仅选择中间一行...

java - JavaFX的ListChangeListener.Change.getRemoved()是否可以返回不连续的项目?-LMLPHP

...在treeTableView.getSelectionModel().getSelectedItems()上触发的change事件如下所示:

{ [TreeItem [ value: Node 1 ], TreeItem [ value: Node 3 ]] removed at 0,  }

在单个更改事件中,它报告从selectedItems列表的索引0中删除了“节点1”和“节点3”。

我本来希望Change对象具有两个单独的删除事件,这些事件由next()调用分隔。第一次调用next()会告诉我在索引0处删除了“节点1”,第二次调用next()会告诉我在索引1处删除了“节点3”。但是不,我得到了一个包含两行的事件立即列出。


getRemoved()真的可以返回不连续的项目吗?这是我对列表更改事件的工作方式的误解,还是TreeTableView中的错误?

通常,我不愿意责怪标准库,但这并不是我在JavaFX中发现的第一个错误,因此这并非不可想象。

更新资料

如果我添加一个对setShowRoot(false)的调用,则行为会改变。我得到了我期望的结果,清除过程分为两部分:
{ [TreeItem [ value: Node 1 ]] removed at 0, [TreeItem [ value: Node 3 ]] removed at 1,  }

另外,这是我的MCVE:
import java.util.*;

import javafx.application.*;
import javafx.beans.property.*;
import javafx.collections.*;
import javafx.scene.*;
import javafx.scene.control.*;
import javafx.stage.*;

public class TreeTableSelectionEvents extends Application {
    public void start(Stage stage) {
        // Root node.
        TreeItem<String> root = new TreeItem<>("Root");

        root.setExpanded(true);

        root.getChildren().setAll(Arrays.asList(
            new TreeItem<>("Node 1"),
            new TreeItem<>("Node 2"),
            new TreeItem<>("Node 3")
        ));

        // Single column.
        TreeTableColumn<String, String> column = new TreeTableColumn<>("Column");

        column.setPrefWidth(150);
        column.setCellValueFactory((TreeTableColumn.CellDataFeatures<String, String> p) -> {
            return new ReadOnlyStringWrapper(p.getValue().getValue());
        });

        // Tree table.
        TreeTableView<String> table = new TreeTableView<>(root);

        table.getColumns().add(column);
        table.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
        // table.setShowRoot(false);

        table.getSelectionModel().getSelectedItems().addListener(
            (ListChangeListener.Change<? extends TreeItem<String>> change) -> {
                System.out.printf("item change = %s, list is now %s%n", change, change.getList());
            }
        );

        table.getSelectionModel().getSelectedIndices().addListener(
            (ListChangeListener.Change<? extends Integer> change) -> {
                System.out.printf("index change = %s, list is now %s%n", change, change.getList());
            }
        );

        // Stage.
        stage.setScene(new Scene(table));
        stage.show();
    }
}

最佳答案

没错,该事件应包含两个单独的“删除”更改。从1.8.0_74开始,TreeTableView的选择模型似乎无可救药了。它甚至与TreeView的选择模型不一致,后者也是很棘手的(但事实并非如此)。失败模式太多,现有的错误和回归错误太多,很难判断Oracle是否意识到问题。我建议提出另一个错误。下面的代码提供了一个不错的沙盒,用于使用功能。

public class Test extends Application {
    public void start(Stage pStage) {
        pStage.setTitle("Test");

        final TabPane tabPane = new TabPane();

        tabPane.getTabs().addAll(
            Stream.of(true, false).map(
                pIsTreeTable -> {
                    final Tab result = new Tab(pIsTreeTable ? "TreeTableView" : "TreeView");

                    // create tree model
                    final TreeItem<String> root = new TreeItem<>("Root Node");
                    root.setExpanded(true);
                    final Collection<TreeItem<String>> children = IntStream.rangeClosed(
                        1, 5
                    ).mapToObj(pIdx -> new TreeItem<>("Child Node " + pIdx)).collect(
                        Collectors.toList()
                    );

                    // create TreeView or TreeTableView
                    final Control tree;
                    final MultipleSelectionModel<TreeItem<String>> selectionModel;
                    if (pIsTreeTable) {
                        final TreeTableView<String> treeTableView = new TreeTableView<>(
                            root
                        );
                        final TreeTableColumn<String,String> column = new TreeTableColumn<>(
                            "Column"
                        );
                        column.setCellValueFactory(
                            pTreeItem -> new ReadOnlyStringWrapper(
                                pTreeItem.getValue().getValue()
                            )
                        );
                        treeTableView.getColumns().add(column);

                        tree = treeTableView;
                        selectionModel = treeTableView.getSelectionModel();
                    } else {
                        final TreeView<String> treeView = new TreeView<>(root);

                        tree = treeView;
                        selectionModel = treeView.getSelectionModel();
                    }
                    selectionModel.setSelectionMode(SelectionMode.MULTIPLE);

                    // add buttons
                    final ToggleButton childrenBtn = new ToggleButton("Children");
                    childrenBtn.selectedProperty().addListener(
                        (pObservable, pOldVal, pNewVal) -> {
                            if (pNewVal) {
                                root.getChildren().addAll(children);
                            } else {
                                root.getChildren().clear();
                            }
                        }
                    );
                    childrenBtn.setSelected(true);
                    final ToggleButton showRootBtn = new ToggleButton("Show Root");
                    showRootBtn.setSelected(true);
                    (
                        pIsTreeTable ?
                        ((TreeTableView<?>) tree).showRootProperty() :
                        ((TreeView<?>) tree).showRootProperty()
                    ).bind(showRootBtn.selectedProperty());

                    // 'getSelectedItems()' tab
                    final Tab selectedItemsTab = new Tab("getSelectedItems()");
                    final TextArea selectedItemsTextArea = new TextArea();
                    selectionModel.getSelectedItems().addListener(
                        (ListChangeListener<TreeItem<String>>) pChange -> {
                            while (pChange.next()) {
                                if (pChange.getRemovedSize() > 0) {
                                    selectedItemsTextArea.appendText(
                                        "Removed " + pChange.getRemoved() + '\n'
                                    );
                                }
                                if (pChange.getAddedSize() > 0) {
                                    selectedItemsTextArea.appendText(
                                        "Added " + pChange.getAddedSubList() + '\n'
                                    );
                                }
                            }
                            selectedItemsTextArea.appendText(
                                "Selection: " + pChange.getList() + "\n\n"
                            );
                        }
                    );
                    selectedItemsTab.setContent(selectedItemsTextArea);

                    // 'getSelectedItem()' tab
                    final Tab selectedItemTab = new Tab("getSelectedItem()");
                    final TextArea selectedItemTextArea = new TextArea();
                    selectionModel.selectedItemProperty().addListener(
                        (pObservable, pOldVal, pNewVal) -> {
                            selectedItemTextArea.appendText("Selected " + pNewVal + '\n');
                        }
                    );
                    selectedItemTab.setContent(selectedItemTextArea);


                    // display selection data in text area
                    final TabPane selectionTabPane = new TabPane();
                    selectionTabPane.getTabs().addAll(selectedItemsTab, selectedItemTab);

                    final SplitPane splitPane = new SplitPane(
                        tree, new HBox(showRootBtn, childrenBtn), selectionTabPane
                    );
                    splitPane.setOrientation(Orientation.VERTICAL);

                    result.setContent(splitPane);

                    return result;
                }
            ).collect(Collectors.toList())
        );

        pStage.setScene(new Scene(tabPane, 300, 450));
        pStage.show();
    }

    public static void main(String[] pArgs) {launch(pArgs);}
}

相关问题:
  • 按下“Ctrl”按钮
  • 选择“子节点2”
  • 选择“子节点3”
  • 选择“子节点1”

  • ListChangeListener.Change事件表示已选择“子节点2”。
  • 选择“根节点”
  • 取消选择“显示根”按钮

  • 选择了“子节点1”,但不广播选择事件。
  • 选择“子节点2”
  • 取消选择“ child ”按钮

  • “选择”列表包括一个空值。
  • 选择“子节点1”
  • 折叠“根节点”

  • ListChangeListener.Change事件不包含任何删除。
  • 取消选择“ child ”按钮
  • 选择“根节点”
  • 取消选择“显示根”按钮

  • TreeView和TreeTableView都不广播事件。从那里,如果按下“显示根”按钮,TreeView会广播事件,但TreeTableView不会。

    08-05 23:10