У меня есть что-то вроде следующих двух упрощенных классов, назовем их Person и Order. Заказ состоит из одного человека:
class Person {
final SimpleStringProperty name;
Person(String name) {
this.name = new SimpleStringProperty(name);
}
@Override
public String toString() {
return name.get();
}
}
class Order {
final SimpleObjectProperty
person;
Order(Person person) {
this.person = new SimpleObjectProperty(person);
}
}
Я отображаю заказы в TableView, а один столбец — это столбец с лицом заказа (имя Человека):
TableView table = new TableView();
TableColumn col = new TableColumn("Person");
col.setCellValueFactory(data -> data.getValue().person);
Когда я меняю имя человека, я хочу, чтобы столбец «Человек» отражал это изменение. Наконец-то я заставил это работать через привязку в CellFactory столбца (как описано в https://stackoverflow.com/a/67979303/1016514). Однако столбец также можно редактировать с помощью ComboBoxTableCell, поэтому двойной щелчок позволяет выбрать другого человека (заметьте, а не другое имя):
class UpdatingCell extends ComboBoxTableCell {
public UpdatingCell(ObservableList
people) {
super(people);
}
@Override
public void startEdit() {
textProperty().unbind();
super.startEdit();
}
@Override
public void updateItem(Person item, boolean empty) {
textProperty().unbind();
super.updateItem(item, empty);
if (empty || item == null) {
setText("");
} else {
textProperty().bind(item.name);
}
}
}
Без привязки() выбор в поле со списком работает (но не обновляет имя, отображаемое в столбце); но с помощью методаbind() я получаю исключение при двойном щелчке, хотя я явно отвязал textProperty в startEdit:
Exception in thread "JavaFX Application Thread" java.lang.RuntimeException: UpdatingCell.text : A bound value cannot be set.
at javafx.beans.property.StringPropertyBase.set(StringPropertyBase.java:141)
at javafx.beans.property.StringPropertyBase.set(StringPropertyBase.java:50)
at javafx.beans.property.StringProperty.setValue(StringProperty.java:71)
at javafx.scene.control.Labeled.setText(Labeled.java:147)
at javafx.scene.control.cell.ComboBoxTableCell.startEdit(ComboBoxTableCell.java:354)
at org.people.UpdatingCell.startEdit(OrderTable.java:50)
at javafx.scene.control.TableCell.updateEditing(TableCell.java:569)
at javafx.scene.control.TableCell.lambda$new$3(TableCell.java:141)
at javafx.beans.WeakInvalidationListener.invalidated(WeakInvalidationListener.java:83)
Отладка показывает, что textProperty ячейки имеет свой Observable, переустановленный по неясным причинам где-то глубоко в работе javafx. Я бы подумал, что это обычный вариант использования с более простым решением, поэтому начинаю задаваться вопросом, не иду ли я в неправильном направлении. Как я могу сделать так, чтобы столбец одновременно отражал изменения имени человека и позволял изменять его?
Вот полный пример приложения, в котором кнопка меняет имя человека в заказ. Двойной щелчок по человеку, чтобы изменить его, вызывает исключение. Закомментировав привязку в UpdatingCell, вы можете выбрать другого человека, но имя не обновляется при нажатии кнопки.
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
import javafx.application.Application;
import javafx.beans.property.SimpleObjectProperty;
import javafx.beans.property.SimpleStringProperty;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.cell.ComboBoxTableCell;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;
class Person {
final SimpleStringProperty name;
Person(String name) {
this.name = new SimpleStringProperty(name);
}
@Override
public String toString() {
return name.get();
}
}
class Order {
final SimpleObjectProperty person;
Order(Person person) {
this.person = new SimpleObjectProperty(person);
}
}
class UpdatingCell extends ComboBoxTableCell {
public UpdatingCell(ObservableList people) {
super(people);
}
@Override
public void startEdit() {
textProperty().unbind();
super.startEdit();
}
@Override
public void updateItem(Person item, boolean empty) {
textProperty().unbind();
super.updateItem(item, empty);
if (empty || item == null) {
setText("");
} else {
// comment out to be able to select person:
textProperty().bind(item.name);
}
}
}
public class OrderTable extends Application {
private static final List names = Arrays.asList("Anna", "Bob", "Charly");
private static final AtomicInteger nameIndex = new AtomicInteger();
private static final ObservableList people = FXCollections
.observableArrayList(Arrays.asList(new Person("Zoe"), new Person("Yvonne"), new Person("Xavier")));
@Override
public void start(Stage stage) throws Exception {
Order order = new Order(people.get(0));
ObservableList orders = FXCollections.observableArrayList();
orders.add(order);
TableView table = new TableView();
TableColumn col = new TableColumn("Person");
col.setCellValueFactory(data -> data.getValue().person);
col.setCellFactory(tc -> new UpdatingCell(people));
col.setEditable(true);
table.getColumns().setAll(Collections.singleton(col));
table.getItems().setAll(orders);
table.setEditable(true);
Button nextName = new Button("Next Name");
nextName.setOnAction(event -> {
order.person.get().name.set(names.get(nameIndex.getAndIncrement() % names.size()));
System.out.println("people are %s".formatted(people));
System.out.println("person is %s".formatted(order.person.get()));
});
stage.setScene(new Scene(new VBox(table, nextName), 200, 140));
stage.show();
}
public static void main(String[] args) {
launch(args);
}
}
Подробнее здесь: https://stackoverflow.com/questions/769 ... m-property
Сохраняйте редактируемое значение TableColumn, обновляемое из свойства. ⇐ JAVA
-
- Похожие темы
- Ответы
- Просмотры
- Последнее сообщение