我遇到了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.");
});
有一件事让我感到困惑。选择更改事件仅在第一次选择和开始多选时触发。每次变化似乎都会发生变化。那是为什么?
然而,我的一个大问题是,当我选择多个条目然后选择我点击的最后一个条目时,因此,撤消多选但不更改焦点的项目,不会触发任何事件。希望这会说清楚:
为什么会发生这种情况?我怎么知道选择何时改变了?
我的最小例子是四个文件:
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>
答案 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
,而不是InvalidationListener
。 ChangeListener
通知属性的先前值和新值,因此它们会强制重新计算值:
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);
});
最后请注意,对于多项选择,selectedItem
和selectedIndex
属性仅引用最后一个选项。可以在不更改最后选择的情况下更改所选项目的列表。在这种情况下,属性根本不会发生变化,因此不会通知更改侦听器和失效侦听器。因此,对于多项选择,在项目(或索引)列表上使用监听器比在项目或索引本身上使用监听器更自然:
tableView.getSelectionModel().getSelectedItems().addListener((Change<? extends Record> c) -> {
System.out.println("Selected items: "+tableView.getSelectionModel().getSelectedItems());
});
(或类似于tableView.getSelectionModel().getSelectedIndices().addListener((Change<? extends Number> c) ...);
)。
有关InvalidationListener
和ChangeListener
之间差异的详情,请参阅Javadocs for Observable
和ObservableValue
。