JavaFX:TableView 使用来自 ObservableList 的特定数据填充列

JavaFX: TableView fill column with specific data from ObservableList

我正在为大学做一个项目。我刚刚学习了 Java 和 JavaFX。

我有一个问题,也许有人可以帮助我。

使用这个小软件,您可以计划一周的任务。在表格视图中,您有从星期一到星期五的列。

我有一个数据库,我可以从中获取“Auftragsverteilung”对象(这意味着计划在特定日期执行的任务)。这些对象有一个序列号 (= seriennr) 和一个日期 (= bearbeitungsdatum)。

为了在表视图中填写一周,我从数据库中获取了一周计划的所有任务。本周的任务保存在 ObservableList rechnerWochenansichtTabelle 中。 您可以从组合框中 select 一周。这就是为什么我在组合框上有一个监听器。

现在我浏览 ObservableList 并检查日期是否是星期一、星期四等。 这样就可以了。但是如果我写 col_RW_Montag.setCellValueFactory(new PropertyValueFactory<>("seriennr")) 该列会获取保存在 observableList 中的所有序列号,但我需要那个特定日期的特定序列号。

如何从日期中获取特定序列号并用它填充列?

我希望你明白一切,因为该项目是德语的。

如果您需要更多信息或更多代码,请告诉我。

谢谢。

"Auftragsverteilung"class(型号)

public class Auftragsverteilung {

    private Date bearbeitungsdatum;
    private Integer seriennr;

    public Auftragsverteilung(Integer seriennr, Date bearbeitungsdatum) {
        super();
        this.seriennr = seriennr;
        this.bearbeitungsdatum = bearbeitungsdatum;
    }


    public Date getBearbeitungsdatum() {
        return bearbeitungsdatum;
    }

    public void setBearbeitungsdatum(Date bearbeitungsdatum) {
        this.bearbeitungsdatum = bearbeitungsdatum;
    }


    public Integer getSeriennr() {
        return seriennr;
    }

    public void setSeriennr(Integer seriennr) {
        this.seriennr = seriennr;
    }

}

"AuftragsansichtController" class(控制器)

public class RechneransichtController implements Initializable {

    @FXML
    private ComboBox<String> comboBox_RW_Wochenansicht;
    @FXML
    private TableView<Auftragsverteilung> tableRechnerWoche;
    @FXML
    private TableColumn<Auftragsverteilung, Integer> col_RW_Montag;
    @FXML
    private TableColumn<Auftragsverteilung, Integer> col_RW_Dienstag;
    @FXML
    private TableColumn<Auftragsverteilung, Integer> col_RW_Mittwoch;
    @FXML
    private TableColumn<Auftragsverteilung, Integer> col_RW_Donnerstag;
    @FXML
    private TableColumn<Auftragsverteilung, Integer> col_RW_Freitag;

    private Datenbank db = new Datenbank();

    ObservableList<Auftragsverteilung> rechnerWochenansichtTabelle = FXCollections.observableArrayList();

    public void wochenansichtFuellen() {

        // ComboBox Listener
        comboBox_RW_Wochenansicht.getSelectionModel().selectedItemProperty().addListener((options) -> {
        tableRechnerWoche.getItems().clear();

        // the week selected from the comboBox
        String wochenAuswahl = comboBox_RW_Wochenansicht.getSelectionModel().selectedItemProperty().getValue(); // DD.MM.YYYY-DD.MM.YYYY

        // String split, first date = monday, last date = friday
        String startdatum = wochenAuswahl.substring(0, 10);
        String enddatum = wochenAuswahl.substring(11, 21);


        // db = Database object with method getTasksFromWeek
        try {
            rechnerWochenansichtTabelle.addAll(db.getRechnerAusAuftragsverteilungWoche(startdatum, enddatum));
        } catch (SQLException e) {
            e.printStackTrace();
        }

        for (int i = 0; i < rechnerWochenansichtTabelle.size(); i++) {

            Date d = rechnerWochenansichtTabelle.get(i).getBearbeitungsdatum();
            Integer seriennr = rechnerWochenansichtTabelle.get(i).getSeriennr(); // SERIENNUMER DIE REIN MUSS


            switch(simpleDateformat.format(d)) {

            case "Montag":
            col_RW_Montag.setCellValueFactory(new PropertyValueFactory<>("seriennr"));
                break;
            case "Dienstag":
            col_RW_Dienstag.setCellValueFactory(new PropertyValueFactory<>("seriennr"));
                break;
            case "Mittwoch":                                                                                                                                    
            col_RW_Mittwoch.setCellValueFactory(new PropertyValueFactory<>("seriennr"));
                break;
            case "Donnerstag":
            col_RW_Donnerstag.setCellValueFactory(new PropertyValueFactory<>("seriennr"));
                break;
            case "Freitag":
            col_RW_Freitag.setCellValueFactory(new PropertyValueFactory<>("seriennr"));
                break;
            default:
            break;          
            }

        }
        tableRechnerWoche.setItems(rechnerWochenansichtTabelle);
    });

}

您没有按应有的方式使用 ObservableList。检查 ObservableList 的文档。 ObservableList 使用观察者模式。有观察者和可观察对象。你有一个 Observables 列表,你的控制器可以观察它们。 https://en.wikipedia.org/wiki/Observer_pattern

不看你的代码,你需要类似这样的东西(下面的片段从 https://dzone.com/articles/javafx-collections-observablelist-and-observablema 复制):

observableList.addListener(new ListChangeListener() {
    @Override
    public void onChanged(ListChangeListener.Change change) {    
        System.out.println("Detected a change! ");
        while (change.next()) {
            System.out.println("Was added? " + change.wasAdded());
            System.out.println("Was removed? " + change.wasRemoved());
        }
    }
}

基本上,listener会观察你的observables,并会通过这个重写的方法告诉你发生了什么变化以及它是如何变化的。您需要在那里采取行动,即更改 table.

中的值

希望对您有所帮助。

您想将多个 Auftragsverteilung 对象放在一行中。由于 items 列表中的每个元素对应于 table 的一行,您的方法是错误的。您需要将多个 Auftragsverteilung 对象存储在单个项目中才能正确访问数据或使用自定义 cellValueFactorys 来检索数据并使用项目来标识周。

此外,我不建议依赖使用德语工作日的程序,除非您专门设置 simpleDateformat 以使用德语作为语言环境。否则更改默认语言环境可能会破坏您的程序,我不会指望您 professor/tutor 在他们用来测试您的程序的机器上使用德语语言环境。

以下代码假设您没有在 fxml 文件中指定列,而是以编程方式创建它们。在下面的代码中,我使用 Auftragsverteilung[] 存储星期一、星期二、...、星期五的对象。此外,它使用 java.time API:

@Override
public void initialize​(URL location, ResourceBundle resources) {
    final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("EEEE", Locale.GERMAN);

    for (int i = 1; i <= lastDay; i++) {
        final int dayIndex = i-1;
        DayOfWeek day = DayOfWeek.of(i);
        TableColumn<Auftragsverteilung[], Integer> column = new TableColumn<>(formatter.format(day));
        column.setCellValueFactory(cd -> {
            Auftragsverteilung auftrag = cd.getValue()[dayIndex];
            return new SimpleObjectProperty<>(auftrag == null ? null : auftrag.getSeriennr());
        });
        tableRechnerWoche.getColumns().add(column);

    }

    tableRechnerWoche.setItems(rechnerWochenansichtTabelle);
    ...
}
@FXML
private TableView<Auftragsverteilung[]> tableRechnerWoche;
@FXML
private TableColumn<Auftragsverteilung[], Integer> col_RW_Montag;
@FXML
private TableColumn<Auftragsverteilung[], Integer> col_RW_Dienstag;
@FXML
private TableColumn<Auftragsverteilung[], Integer> col_RW_Mittwoch;
@FXML
private TableColumn<Auftragsverteilung[], Integer> col_RW_Donnerstag;
@FXML
private TableColumn<Auftragsverteilung[], Integer> col_RW_Freitag;

...

public void wochenansichtFuellen() {

    // ComboBox Listener (ChangeListener)
    comboBox_RW_Wochenansicht.valueProperty().addListener((o, oldValue, newValue) -> {
        rechnerWochenansichtTabelle.clear();

        // String split, first date = monday, last date = friday
        String startdatum = newValue.substring(0, 10);
        String enddatum = newValue.substring(11, 21);

        // db = Database object with method getTasksFromWeek
        try {
            // depending on the return type and order of the elements
            // rechnerWochenansichtTabelle.add(db.getRechnerAusAuftragsverteilungWoche(startdatum, enddatum));
            // may be sufficient instead of the following code

            // store each Auftragsverteilung in array with index corresponding to the day of week of the bearbeitungsdatum
            Auftragsverteilung[] row = new Auftragsverteilung[DayOfWeek.FRIDAY.getValue()];
            for (Auftragsverteilung auftrag : db.getRechnerAusAuftragsverteilungWoche(startdatum, enddatum)) {
                LocalDate date = auftrag.getBearbeitungsdatum().toLocalDate();
                row[date.getDayOfWeek().getValue() - 1] = auftrag;
            }

            rechnerWochenansichtTabelle.add(row);
        } catch (SQLException e) {
            e.printStackTrace();
        }
    });

}