Javafx 2 JavaFX2 TableView如何在对象更改时更新单元格
我正在创建一个TableView来显示有关自定义对象EntityEvents列表的信息 表视图必须有2列。 显示相应EntityEvent名称的第一列。 第二列将显示一个按钮。按钮文本依赖于EntityEvent的属性。如果该属性为零,则该属性将为“创建”,否则为“编辑” 我做得很好,只是在相应的EntityEvent对象更改时找不到更新TableView行的方法 非常重要:我不能将EntityEvent类更改为使用JavaFX属性,因为它们不在我的控制之下。此类使用PropertyChangeSupport在监视的属性更改时通知侦听器 注: 我意识到向列表中添加新元素可能会导致TableView自身重新绘制,但这不是我需要的。我这么说可能是因为我读过一些影响这种行为的bug。 我试着用这种方法来强制重画,因为我无法让它工作 有人知道怎么做吗 非常感谢 下面是一个简化的代码示例,演示了该场景:Javafx 2 JavaFX2 TableView如何在对象更改时更新单元格,javafx-2,Javafx 2,我正在创建一个TableView来显示有关自定义对象EntityEvents列表的信息 表视图必须有2列。 显示相应EntityEvent名称的第一列。 第二列将显示一个按钮。按钮文本依赖于EntityEvent的属性。如果该属性为零,则该属性将为“创建”,否则为“编辑” 我做得很好,只是在相应的EntityEvent对象更改时找不到更新TableView行的方法 非常重要:我不能将EntityEvent类更改为使用JavaFX属性,因为它们不在我的控制之下。此类使用PropertyChange
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import javafx.application.Application;
import javafx.beans.property.ReadOnlyStringWrapper;
import javafx.beans.value.ObservableValue;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.geometry.Insets;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.ContentDisplay;
import javafx.scene.control.TableCell;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableColumn.CellDataFeatures;
import javafx.scene.control.TableView;
import javafx.scene.layout.StackPane;
import javafx.scene.layout.VBox;
import javafx.scene.paint.Color;
import javafx.stage.Stage;
import javafx.util.Callback;
public class Main extends Application {
//=============================================================================================
public class EntityEvent {
private String m_Name;
private PropertyChangeSupport m_NamePCS = new PropertyChangeSupport(this);
private int m_ActionCounter;
private PropertyChangeSupport m_ActionCounterPCS = new PropertyChangeSupport(this);
public EntityEvent(String name, int actionCounter) {
m_Name = name;
m_ActionCounter = actionCounter;
}
public String getName() {
return m_Name;
}
public void setName(String name) {
String lastName = m_Name;
m_Name = name;
System.out.println("Name changed: " + lastName + " -> " + m_Name);
m_NamePCS.firePropertyChange("Name", lastName, m_Name);
}
public void addNameChangeListener(PropertyChangeListener listener) {
m_NamePCS.addPropertyChangeListener(listener);
}
public int getActionCounter() {
return m_ActionCounter;
}
public void setActionCounter(int actionCounter) {
int lastActionCounter = m_ActionCounter;
m_ActionCounter = actionCounter;
System.out.println(m_Name + ": ActionCounter changed: " + lastActionCounter + " -> " + m_ActionCounter);
m_ActionCounterPCS.firePropertyChange("ActionCounter", lastActionCounter, m_ActionCounter);
}
public void addActionCounterChangeListener(PropertyChangeListener listener) {
m_ActionCounterPCS.addPropertyChangeListener(listener);
}
}
//=============================================================================================
private class AddPersonCell extends TableCell<EntityEvent, String> {
Button m_Button = new Button("Undefined");
StackPane m_Padded = new StackPane();
AddPersonCell(final TableView<EntityEvent> table) {
m_Padded.setPadding(new Insets(3));
m_Padded.getChildren().add(m_Button);
m_Button.setOnAction(new EventHandler<ActionEvent>() {
@Override public void handle(ActionEvent actionEvent) {
// Do something
}
});
}
@Override protected void updateItem(String item, boolean empty) {
super.updateItem(item, empty);
if (!empty) {
setContentDisplay(ContentDisplay.GRAPHIC_ONLY);
setGraphic(m_Padded);
m_Button.setText(item);
}
}
}
//=============================================================================================
private ObservableList<EntityEvent> m_EventList;
//=============================================================================================
@SuppressWarnings("unchecked")
@Override
public void start(Stage primaryStage) {
primaryStage.setTitle("Table View test.");
VBox container = new VBox();
m_EventList = FXCollections.observableArrayList(
new EntityEvent("Event 1", -1),
new EntityEvent("Event 2", 0),
new EntityEvent("Event 3", 1)
);
final TableView<EntityEvent> table = new TableView<EntityEvent>();
table.setItems(m_EventList);
TableColumn<EntityEvent, String> eventsColumn = new TableColumn<>("Events");
TableColumn<EntityEvent, String> actionCol = new TableColumn<>("Actions");
actionCol.setSortable(false);
eventsColumn.setCellValueFactory(new Callback<CellDataFeatures<EntityEvent, String>, ObservableValue<String>>() {
public ObservableValue<String> call(CellDataFeatures<EntityEvent, String> p) {
EntityEvent event = p.getValue();
event.addActionCounterChangeListener(new PropertyChangeListener() {
@Override
public void propertyChange(PropertyChangeEvent event) {
// TODO: I'd like to update the table cell information.
}
});
return new ReadOnlyStringWrapper(event.getName());
}
});
actionCol.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<EntityEvent, String>, ObservableValue<String>>() {
@Override
public ObservableValue<String> call(TableColumn.CellDataFeatures<EntityEvent, String> ev) {
String text = "NONE";
if(ev.getValue() != null) {
text = (ev.getValue().getActionCounter() != 0) ? "Edit" : "Create";
}
return new ReadOnlyStringWrapper(text);
}
});
// create a cell value factory with an add button for each row in the table.
actionCol.setCellFactory(new Callback<TableColumn<EntityEvent, String>, TableCell<EntityEvent, String>>() {
@Override
public TableCell<EntityEvent, String> call(TableColumn<EntityEvent, String> personBooleanTableColumn) {
return new AddPersonCell(table);
}
});
table.getColumns().setAll(eventsColumn, actionCol);
table.setColumnResizePolicy(TableView.CONSTRAINED_RESIZE_POLICY);
// Add Resources Button
Button btnInc = new Button("+");
btnInc.setOnAction(new EventHandler<ActionEvent>() {
@Override
public void handle(ActionEvent ev) {
System.out.println("+ clicked.");
EntityEvent entityEvent = table.getSelectionModel().getSelectedItem();
if (entityEvent == null) {
System.out.println("No Event selected.");
return;
}
entityEvent.setActionCounter(entityEvent.getActionCounter() + 1);
// TODO: I expected the TableView to be updated since I modified the object.
}
});
// Add Resources Button
Button btnDec = new Button("-");
btnDec.setOnAction(new EventHandler<ActionEvent>() {
@Override
public void handle(ActionEvent ev) {
System.out.println("- clicked.");
EntityEvent entityEvent = table.getSelectionModel().getSelectedItem();
if (entityEvent == null) {
System.out.println("No Event selected.");
return;
}
entityEvent.setActionCounter(entityEvent.getActionCounter() - 1);
// TODO: I expected the TableView to be updated since I modified the object.
}
});
container.getChildren().add(table);
container.getChildren().add(btnInc);
container.getChildren().add(btnDec);
Scene scene = new Scene(container, 300, 600, Color.WHITE);
primaryStage.setScene(scene);
primaryStage.show();
}
//=============================================================================================
public Main() {
}
//=============================================================================================
public static void main(String[] args) {
launch(Main.class, args);
}
}
尝试这些类,尤其是JavaBeanStringProperty和JavaBeanTygerProperty。我没用过这些,但我想你可以这样做
TableColumn<EntityEvent, Integer> actionCol = new TableColumn<>("Actions");
actionCol.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<EntityEvent, Integer> ev) {
return new JavaBeanIntegerPropertyBuilder().bean(ev.getValue()).name("actionCounter").build();
});
// ...
public class AddPersonCell extends TableCell<EntityEvent, Integer>() {
final Button button = new Button();
public AddPersonCell() {
setPadding(new Insets(3));
setContentDisplay(ContentDisplay.GRAPHIC_ONLY);
button.setOnAction(...);
}
@Override
public void updateItem(Integer actionCounter, boolean empty) {
if (empty) {
setGraphic(null);
} else {
if (actionCounter.intValue()==0) {
button.setText("Create");
} else {
button.setText("Add");
}
setGraphic(button);
}
}
}
它将与上面的适配器类一起工作。显然,如果您有调用addActionChangeListener和addNameChangeListener方法的现有代码,您可能希望保留这些现有方法和现有属性更改侦听器,但我认为没有理由不能同时拥有它们。尝试这些类,特别是JavaBeanStringProperty和JavaBeanIntergerProperty。我没用过这些,但我想你可以这样做
TableColumn<EntityEvent, Integer> actionCol = new TableColumn<>("Actions");
actionCol.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<EntityEvent, Integer> ev) {
return new JavaBeanIntegerPropertyBuilder().bean(ev.getValue()).name("actionCounter").build();
});
// ...
public class AddPersonCell extends TableCell<EntityEvent, Integer>() {
final Button button = new Button();
public AddPersonCell() {
setPadding(new Insets(3));
setContentDisplay(ContentDisplay.GRAPHIC_ONLY);
button.setOnAction(...);
}
@Override
public void updateItem(Integer actionCounter, boolean empty) {
if (empty) {
setGraphic(null);
} else {
if (actionCounter.intValue()==0) {
button.setText("Create");
} else {
button.setText("Add");
}
setGraphic(button);
}
}
}
它将与上面的适配器类一起工作。显然,如果您有调用addActionChangeListener和addNameChangeListener方法的现有代码,您可能希望保留这些现有方法和现有的属性更改侦听器,但我认为没有理由不能同时使用这两种方法。实际上,EntityEvent可以更改。但是,我不能添加JavaFX属性。它必须是UI域独立的。我要做一些测试。如果我得到了什么,我会告诉你的。因为JavaFX现在是核心JDK的一部分,JavaFX属性不依赖于任何本地UI资源,即它们在无头模式下运行,所以你可以非常有说服力地说引入JavaFX属性是一种与UI域无关的更改。我同意你的观点,但是这次我不能将JavaFX属性添加到EntityEvent类中。我正在做一些测试,很快就会回来。非常感谢。我没法让它工作。该表根本没有更新。你确定那会迫使桌子重新油漆吗?真奇怪!我的代码不起作用,你的代码起作用了。这里可能有点小错误。这完美地解决了我的问题。再次非常感谢。事实上,实体事件是可以更改的。但是,我不能添加JavaFX属性。它必须是UI域独立的。我要做一些测试。如果我得到了什么,我会告诉你的。因为JavaFX现在是核心JDK的一部分,JavaFX属性不依赖于任何本地UI资源,即它们在无头模式下运行,所以你可以非常有说服力地说引入JavaFX属性是一种与UI域无关的更改。我同意你的观点,但是这次我不能将JavaFX属性添加到EntityEvent类中。我正在做一些测试,很快就会回来。非常感谢。我没法让它工作。该表根本没有更新。你确定那会迫使桌子重新油漆吗?真奇怪!我的代码不起作用,你的代码起作用了。这里可能有点小错误。这完美地解决了我的问题。再次非常感谢。