Javafx 2 JavaFX2 TableView如何在对象更改时更新单元格

Javafx 2 JavaFX2 TableView如何在对象更改时更新单元格,javafx-2,Javafx 2,我正在创建一个TableView来显示有关自定义对象EntityEvents列表的信息 表视图必须有2列。 显示相应EntityEvent名称的第一列。 第二列将显示一个按钮。按钮文本依赖于EntityEvent的属性。如果该属性为零,则该属性将为“创建”,否则为“编辑” 我做得很好,只是在相应的EntityEvent对象更改时找不到更新TableView行的方法 非常重要:我不能将EntityEvent类更改为使用JavaFX属性,因为它们不在我的控制之下。此类使用PropertyChange

我正在创建一个TableView来显示有关自定义对象EntityEvents列表的信息

表视图必须有2列。 显示相应EntityEvent名称的第一列。 第二列将显示一个按钮。按钮文本依赖于EntityEvent的属性。如果该属性为零,则该属性将为“创建”,否则为“编辑”

我做得很好,只是在相应的EntityEvent对象更改时找不到更新TableView行的方法

非常重要:我不能将EntityEvent类更改为使用JavaFX属性,因为它们不在我的控制之下。此类使用PropertyChangeSupport在监视的属性更改时通知侦听器

注: 我意识到向列表中添加新元素可能会导致TableView自身重新绘制,但这不是我需要的。我这么说可能是因为我读过一些影响这种行为的bug。 我试着用这种方法来强制重画,因为我无法让它工作

有人知道怎么做吗

非常感谢

下面是一个简化的代码示例,演示了该场景:

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类中。我正在做一些测试,很快就会回来。非常感谢。我没法让它工作。该表根本没有更新。你确定那会迫使桌子重新油漆吗?真奇怪!我的代码不起作用,你的代码起作用了。这里可能有点小错误。这完美地解决了我的问题。再次非常感谢。