Невозможно заполнить JavaFX TableView, созданный в SceneBuilder 2

Я пытаюсь воссоздать пример табличного представления, используя конструктор сцен 2, но мой TableView не может быть заполнен.

Я определил свою таблицу в JavaFx SceneBuilder следующим образом:

<?xml version="1.0" encoding="UTF-8"?>

<?import java.lang.*?>
<?import java.util.*?>
<?import javafx.scene.*?>
<?import javafx.scene.control.*?>
<?import javafx.scene.layout.*?>

<AnchorPane id="AnchorPane" prefHeight="380.0" prefWidth="462.0" xmlns:fx="http://javafx.com/fxml/1" xmlns="http://javafx.com/javafx/8" fx:controller="tableviewsample.FXMLDocumentController">
    <children>
        <Button fx:id="button" layoutX="325.0" layoutY="324.0" onAction="#handleButtonAction" text="Click Me!" />
        <Label fx:id="label" layoutX="126" layoutY="120" minHeight="16" minWidth="69" />
        <TableView fx:id="table" layoutX="26.0" layoutY="29.0" prefHeight="221.0" prefWidth="411.0">
            <columns>
                <TableColumn fx:id="firstNameCol" prefWidth="75.0" text="First Name" />
                <TableColumn fx:id="lastNameCol" prefWidth="75.0" text="Last Name" />
                <TableColumn fx:id="emailCol" prefWidth="75.0" text="Email" />
            </columns>
        </TableView>
        <TextField fx:id="addFirstName" layoutX="26.0" layoutY="284.0" prefHeight="25.0" prefWidth="81.0" promptText="First Name" />
        <TextField fx:id="addLastName" layoutX="121.0" layoutY="284.0" prefHeight="25.0" prefWidth="89.0" promptText="Last Name" />
        <TextField fx:id="addEmail" layoutX="222.0" layoutY="284.0" promptText="Email" />
    </children>
</AnchorPane>

Мой код контроллера:

package tableviewsample;

import java.net.URL;
import java.util.ResourceBundle;
import javafx.beans.property.SimpleStringProperty;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.event.ActionEvent;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.Label;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.TextField;

public class FXMLDocumentController implements Initializable {

    @FXML
    private Label label;

    @FXML
    private TableView<Person> table;// = new TableView<Person>();

    @FXML private TableColumn firstNameCol ;
    @FXML private TableColumn lastNameCol ;
    @FXML private TableColumn emailCol ;

    @FXML TextField addFirstName;
    @FXML TextField addLastName;
    @FXML TextField addEmail;

    private final ObservableList<Person> data
            = FXCollections.observableArrayList(
                    new Person("Jacob", "Smith", "[email protected]"),
                    new Person("Isabella", "Johnson", "[email protected]"),
                    new Person("Ethan", "Williams", "[email protected]"),
                    new Person("Emma", "Jones", "[email protected]"),
                    new Person("Michael", "Brown", "[email protected]")
            );

    @FXML
    private void handleButtonAction(ActionEvent event) {
        System.out.println("You clicked me!");
        data.add(new Person(
                addFirstName.getText(),
                addLastName.getText(),
                addEmail.getText()));
        addFirstName.clear();
        addLastName.clear();
        addEmail.clear();
    }

    @Override
    public void initialize(URL url, ResourceBundle rb) {
        // TODO
        firstNameCol.setMinWidth(100);
        lastNameCol.setMinWidth(100);
        emailCol.setMinWidth(200);
        table.getItems().setAll(this.data);
    }

    public static class Person {

        private final SimpleStringProperty firstName;
        private final SimpleStringProperty lastName;
        private final SimpleStringProperty email;

        private Person(String fName, String lName, String email) {
            this.firstName = new SimpleStringProperty(fName);
            this.lastName = new SimpleStringProperty(lName);
            this.email = new SimpleStringProperty(email);
        }

        public String getFirstName() {
            return firstName.get();
        }

        public void setFirstName(String fName) {
            firstName.set(fName);
        }

        public String getLastName() {
            return lastName.get();
        }

        public void setLastName(String fName) {
            lastName.set(fName);
        }

        public String getEmail() {
            return email.get();
        }

        public void setEmail(String fName) {
            email.set(fName);
        }
    }
}

Основной код:

package tableviewsample;

import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;

public class TableViewSample extends Application {

    @Override
    public void start(Stage stage) throws Exception {
        Parent root = FXMLLoader.load(getClass().getResource("FXMLDocument.fxml"));
        Scene scene = new Scene(root);
        stage.setScene(scene);
        stage.show();
    }

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

Снимок экрана:

скриншот

Когда я запускаю его, в таблице есть пустые строки. Если я ввожу некоторые значения в 3 текстовых поля и нажимаю кнопку, значения все равно не вставляются в таблицу. Не могли бы вы найти, что не так с кодом? Кроме того, в несвязанной заметке, как я могу сделать таблицу прокручиваемой?


person Gary    schedule 06.01.2015    source источник


Ответы (3)


Вы не установили cellValueFactory для TableColumn (пример 12-5 в учебник, на который вы ссылаетесь).

Вы можете сделать это в FXML с помощью

<TableColumn fx:id="firstNameCol" prefWidth="75.0" text="First Name" >
    <cellValueFactory><PropertyValueFactory property="firstName"/></cellValueFactory>
</TableColumn>

или в контроллере с

firstNameCol.setCellValueFactory(new PropertyValueFactory("firstName"));

(и аналогично для других столбцов)

person James_D    schedule 06.01.2015
comment
@Джейнс. Спасибо. Я устанавливаю cellValueFactorys для столбцов таблицы внутри метода инициализации, таблица заполняется при ее запуске. теперь кнопка по-прежнему не может добавлять новые строки в таблицу. Есть идеи? еще раз спасибо! - person Gary; 06.01.2015
comment
Либо посмотрите ответ Бошмана, либо сделайте table.getItems().add(...) в обработчике кнопки вместо data.add(...) - person James_D; 06.01.2015

Вы на самом деле сделали две ошибки.

Прежде всего, вы пытаетесь присвоить модель данных TableView, используя getItems().setAll(). Взгляните на документация setAll(). Он копирует все значения данного списка в ObservableList. Это не то, что вы хотите, вы хотите, чтобы TableView начал наблюдать data и реагировать на изменения. Это можно сделать, назначив ссылку напрямую через setItems(data).

Взгляните на примеры кода в документации Таблица. Вы забыли подключить различные StringProperties в Person к столбцам. Вот как они это делают:

firstNameCol.setCellValueFactory(new PropertyValueFactory<Person, String>("firstName"));
lastNameCol.setCellValueFactory(new PropertyValueFactory<Person, String>("lastName"));

Напоследок у меня есть небольшая заметка. Обратите внимание, что это НЕ является причиной вашей конкретной проблемы.

Я вижу, вы используете необработанные типы. Обычно это считается плохой вещью, и вы, вероятно, получаете предупреждения об этом от компилятора. Вы можете решить эту проблему, указав общие типы, используемые вашими объектами. Обратите внимание, что в приведенном выше примере кода я указываю для PropertyValueFactories, что тип класса, содержащегося в TableView.items, равен Person, а тип объекта, отображаемого в строках, — String.

Что касается прокрутки, о которой вы также просили, она выполняется автоматически, когда в TableView достаточно строк.

person Reinstate Monica    schedule 06.01.2015
comment
@А. Спасибо, что нашли для меня эту ошибку!!!!, я изменил на table.setItems(this.data), как вы предложили, моя кнопка может добавлять строки в таблицу!. Джеймс нашел для меня другую ошибку. Вы оба мне очень помогли, я застрял на этом почти на 1 день. - person Gary; 06.01.2015

В дескрипторе кнопки мыши вы можете написать: /setItems(); следуйте коду зеленой галочки выше

person Shubham Jain    schedule 11.09.2018