【问题标题】:Can't populate JavaFX TableView created in SceneBuilder 2无法填充在 SceneBuilder 2 中创建的 JavaFX TableView
【发布时间】:2015-03-04 13:48:00
【问题描述】:

我正在尝试使用场景构建器 2 重新创建 table view sample,但无法填充我的 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", "jacob.smith@example.com"),
                    new Person("Isabella", "Johnson", "isabella.johnson@example.com"),
                    new Person("Ethan", "Williams", "ethan.williams@example.com"),
                    new Person("Emma", "Jones", "emma.jones@example.com"),
                    new Person("Michael", "Brown", "michael.brown@example.com")
            );

    @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 个文本字段中键入一些值并单击按钮,这些值仍然没有插入到表中。 你能找出代码有什么问题吗?另外,在不相关的注释中,如何使表格可滚动?

【问题讨论】:

    标签: javafx javafx-2 scenebuilder


    【解决方案1】:

    在句柄鼠标按钮你可以写: /setItems(); 按照上面绿色勾号的代码

    【讨论】:

      【解决方案2】:

      你实际上犯了两个错误。

      首先,您尝试使用getItems().setAll() 将数据模型分配给TableView。看看the documentation of setAll()。它将给定列表的所有值复制到 ObservableList 中。这不是您想要的,您希望 TableView 开始观察 data 并响应更改。这样做的方法是直接分配一个引用,通过setItems(data)

      查看TableView文档中的代码示例。您忘记将 Person 中的各种 StringProperties 连接到列。他们就是这样做的:

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

      最后,我有一点旁注。请注意,这不是您的特定问题的原因。

      我看到你正在使用raw types。这通常被认为是一件坏事,您可能会从编译器那里收到一些警告。您可以通过说明对象使用的泛型类型来解决该问题。请注意,在上面的代码示例中,我为 PropertyValueFactories 声明 TableView.items 中包含的类的类型是 Person,而行中显示的对象类型是 String

      至于您也要求的滚动,一旦 TableView 中有足够的行,它就会自动执行。

      【讨论】:

      • @A.感谢您为我发现了这个错误!!!!,我按照您的建议更改为 table.setItems(this.data),我的按钮可以向表中添加行!。詹姆斯为我发现了另一个错误。你们俩都给了我很大的帮助,我坚持了将近 1 天。
      【解决方案3】:

      您尚未在您的TableColumns 上设置任何cellValueFactorys(tutorial you linked 中的示例 12-5)。

      您可以在 FXML 中使用

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

      或在控制器中使用

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

      (对于其他列也是如此)

      【讨论】:

      • @Janes。谢谢。我在 initilize 方法中为表格列设置了 cellValueFactorys,当我运行它时表格被填充。现在,该按钮仍然无法向表中添加新行。任何想法?再次感谢!
      • 要么查看 A Boschman 的答案,要么在按钮处理程序中执行 table.getItems().add(...),而不是 data.add(...)
      猜你喜欢
      • 2013-06-28
      • 1970-01-01
      • 2016-11-20
      • 1970-01-01
      • 2017-11-13
      • 2014-10-08
      • 2018-10-14
      • 2018-11-18
      • 1970-01-01
      相关资源
      最近更新 更多