JavaFX TableView不会触发选择回调

时间:2018-01-01 14:30:37

标签: java events javafx tableview

我遇到了JavaFX TableView没有触发选择回调的问题。我设置了这两个回调:

tableView.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
tableView.getSelectionModel().selectedItemProperty().addListener((observable) -> {
    System.out.println(LocalDateTime.now() + " Item selection changed.");
});
tableView.getSelectionModel().selectedIndexProperty().addListener(observable -> {
    System.out.println(LocalDateTime.now() + " Item index changed.");
});

有一件事让我感到困惑。选择更改事件仅在第一次选择和开始多选时触发。每次变化似乎都会发生变化。那是为什么?

然而,我的一个大问题是,当我选择多个条目然后选择我点击的最后一个条目时,因此,撤消多选但不更改焦点的项目,不会触发任何事件。希望这会说清楚:

enter image description here

为什么会发生这种情况?我怎么知道选择何时改变了?

我的最小例子是四个文件:

Main.java:

package sample;

import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;

public class Main extends Application {

    @Override
    public void start(Stage primaryStage) throws Exception {
        Parent root = FXMLLoader.load(getClass().getResource("sample.fxml"));
        primaryStage.setTitle("Hello World");
        primaryStage.setScene(new Scene(root, 300, 275));
        primaryStage.show();
    }


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

Record.java

package sample;

import javafx.beans.property.SimpleStringProperty;

public class Record {
    private SimpleStringProperty c1 = new SimpleStringProperty();
    private SimpleStringProperty c2 = new SimpleStringProperty();

    public Record(String c1, String c2) {
        this.c1.setValue(c1);
        this.c2.setValue(c2);
    }

    public String getC1() {
        return c1.get();
    }

    public SimpleStringProperty c1Property() {
        return c1;
    }

    public void setC1(String c1) {
        this.c1.set(c1);
    }

    public String getC2() {
        return c2.get();
    }

    public SimpleStringProperty c2Property() {
        return c2;
    }

    public void setC2(String c2) {
        this.c2.set(c2);
    }

    @Override
    public String toString() {
        return String.format("Record{c1=%s, c2=%s}", c1, c2);
    }
}

Controller.java

package sample;

import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.fxml.FXML;
import javafx.scene.control.SelectionMode;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.cell.PropertyValueFactory;

import java.time.LocalDateTime;

public class Controller {
    private ObservableList<Record> records = FXCollections.observableArrayList();

    @FXML
    private TableView<Record> tableView;

    @FXML
    private TableColumn<Record, String> c1Column;

    @FXML
    private TableColumn<Record, String> c2Column;

    @FXML
    private void initialize() {
        records.add(new Record("c1 - 1", "c2 - 1"));
        records.add(new Record("c1 - 2", "c2 - 2"));
        records.add(new Record("c1 - 3", "c2 - 3"));
        records.add(new Record("c1 - 4", "c2 - 4"));
        records.add(new Record("c1 - 5", "c2 - 5"));

        tableView.setItems(records);
        c1Column.setCellValueFactory(new PropertyValueFactory<>("c1"));
        c2Column.setCellValueFactory(new PropertyValueFactory<>("c2"));

        tableView.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
        tableView.getSelectionModel().selectedItemProperty().addListener((observable) -> {
            System.out.println(LocalDateTime.now() + " Item selection changed.");
        });
        tableView.getSelectionModel().selectedIndexProperty().addListener(observable -> {
            System.out.println(LocalDateTime.now() + " Item index changed.");
        });
    }
}

和sample.fxml

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.control.TableColumn?>
<?import javafx.scene.control.TableView?>
<?import javafx.scene.layout.StackPane?>
<StackPane xmlns="http://javafx.com/javafx/8.0.141" xmlns:fx="http://javafx.com/fxml/1"
           fx:controller="sample.Controller">
    <TableView fx:id="tableView" prefHeight="200.0" prefWidth="200.0">
        <columns>
            <TableColumn fx:id="c1Column" prefWidth="75.0" text="C1"/>
            <TableColumn fx:id="c2Column" prefWidth="75.0" text="C2"/>
        </columns>
    </TableView>
</StackPane>

1 个答案:

答案 0 :(得分:2)

问题是您使用的是InvalidationListener。只要可观察值变为无效,就会触发这些侦听器:也就是说,最后计算的值可能不是最新的。

可以懒惰地评估JavaFX属性。如果它们变为无效,则通知失效侦听器,但不一定计算它们的新值。如果它们所依赖的值在重新计算之前再次更改,则它们已经无效,因此不会再次通知失效侦听器。如果在中间时间重新计算它们,那么它们将在重新计算时变为有效,并且将再次通知失效监听器。

此机制是出于性能原因而设计的:您可以想象,例如大窗格的布局取决于其中许多项目的大小和位置;但是你不想在每次改变时重新计算它:你只需要知道何时需要重新计算(当它无效时)。

从您的代码中看,它的工作方式似乎是selectedItem声明对selectedIndex的依赖。当用户选择某些内容时,会selectedIndex直接设置(使其无效并更改),这会导致selectedItem无效。但是,selectedItem在重新计算之前未明确更改。

如果您实际重新计算侦听器中的值,则可以看到所有这些:

    tableView.getSelectionModel().selectedItemProperty().addListener((observable) -> {
        System.out.println(LocalDateTime.now() + " Item selection changed: "+tableView.getSelectionModel().getSelectedItem());
    });
    tableView.getSelectionModel().selectedIndexProperty().addListener((observable) -> {
        System.out.println(LocalDateTime.now() + " Item index changed: " + tableView.getSelectionModel().getSelectedIndex());
    });

典型用法是使用这些属性注册ChangeListener,而不是InvalidationListenerChangeListener通知属性的先前值和新值,因此它们会强制重新计算值:

    tableView.getSelectionModel().selectedItemProperty().addListener((observable, oldItem, newItem) -> {
        System.out.println(LocalDateTime.now() + " Item selection changed: "+ oldItem + " -> " + newItem);
    });
    tableView.getSelectionModel().selectedIndexProperty().addListener((observable, oldIndex, newInde) -> {
        System.out.println(LocalDateTime.now() + " Item index changed: " + oldIndex + " -> " + newIndex);
    });

最后请注意,对于多项选择,selectedItemselectedIndex属性仅引用最后一个选项。可以在不更改最后选择的情况下更改所选项目的列表。在这种情况下,属性根本不会发生变化,因此不会通知更改侦听器和失效侦听器。因此,对于多项选择,在项目(或索引)列表上使用监听器比在项目或索引本身上使用监听器更自然:

    tableView.getSelectionModel().getSelectedItems().addListener((Change<? extends Record> c) -> {
            System.out.println("Selected items: "+tableView.getSelectionModel().getSelectedItems());
    });

(或类似于tableView.getSelectionModel().getSelectedIndices().addListener((Change<? extends Number> c) ...);)。

有关InvalidationListenerChangeListener之间差异的详情,请参阅Javadocs for ObservableObservableValue