使用自定义单元工厂的ListView在删除项目后不会更新

时间:2015-02-07 16:54:58

标签: java listview javafx

我正在学习JavaFX并且我想创建一个正常工作的单元工厂,直到我想从我的ListView中删除一行:

plateList.setCellFactory(new Callback<ListView<Car>, ListCell<Car>>() {

        @Override
        public ListCell<Car> call(ListView<Car> param) {
            ListCell<Car> cell = new ListCell<Car>() {

                @Override
                protected void updateItem(Car item, boolean empty) {
                    super.updateItem(item, empty);
                    if (item != null) {
                       setText(item.getPlate());
                    }
                }
            };
            return cell;
        }
    });

我使用一些示例数据填充ListView

ObservableList<Car> sample = FXCollections.observableArrayList();
    sample.add(new Car("123-abc", "opel", "corsa", 5.5));
    sample.add(new Car("123-cba", "vw", "passat", 7.5));
    plateList.setItems(sample);

现在我将看到我期望的ListView将是以下内容:

  • 123-ABC
  • 123-CBA

如果删除行ex:第一行(123-abc),ListView将如下所示:

  • 123-CBA
  • 123-CBA

这是删除部分:

@FXML
private void deleteBtnAction() {
   plateList.getItems().remove(plateList.getSelectionModel().getSelectedItem());
    ObservableList<Car> t = plateList.getItems();
    plateList.setItems(t);

}

如果我删除了单元工厂,程序将按预期工作。 非常感谢任何帮助。

2 个答案:

答案 0 :(得分:14)

尝试更改为以下内容,这是必需的,因为JavaFX重用了列表单元格,因此当传递null时,updateItem()也需要清空未使用的

super.updateItem(item, empty);
if (item != null) {
   setText(item.getPlate());
} else {
   setText("");   // <== clear the now empty cell.
}

完整SSCCE

import javafx.application.Application;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.ListCell;
import javafx.scene.control.ListView;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;
import javafx.util.Callback;

public class JavaFxListView extends Application {

    private static class Car {
        private String plate;

        public Car(String plate, String string2, String string3, double d) {
            this.plate = plate;
        }

        public String getPlate() {
            return plate;
        }

    }

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

    @Override
    public void start(Stage arg0) throws Exception {
        ListView<Car> plateList = new ListView<Car>();
        plateList.setCellFactory(new Callback<ListView<Car>, ListCell<Car>>() {

            @Override
            public ListCell<Car> call(ListView<Car> param) {
                ListCell<Car> cell = new ListCell<Car>() {

                    @Override
                    protected void updateItem(Car item, boolean empty) {
                        super.updateItem(item, empty);
                        if (item != null) {
                            setText(item.getPlate());
                        } else {
                            setText("");
                        }
                    }
                };
                return cell;
            }
        });
        Button delete = new Button("Delete");
        ObservableList<Car> sample = FXCollections.observableArrayList();
        sample.add(new Car("123-abc", "opel", "corsa", 5.5));
        sample.add(new Car("123-cba", "vw", "passat", 7.5));

        delete.setOnAction((e) -> {
            plateList.getItems().remove(plateList.getSelectionModel().getSelectedItem());
            ObservableList<Car> t = plateList.getItems();
            plateList.setItems(t);
        });

        plateList.setItems(sample);
        arg0.setScene(new Scene(new VBox(plateList, delete)));
        arg0.show();
    }
}

答案 1 :(得分:4)

根据Cell updateItem方法的Java文档,推荐的用法与接受的用法略有不同:

 protected void updateItem(T item, boolean empty) {
     super.updateItem(item, empty);

     if (empty || item == null) {
         setText(null);
         setGraphic(null);
     } else {
         setText(item.toString());
     }
 }

区别在于参数empty的使用。但是@Adam的溶剂也应该在主要情况下正常工作。