在JavaFX中创建行索引列

时间:2015-10-26 18:26:40

标签: java javafx javafx-2 javafx-8 fxml

我有一个JavaFX TableView,我用一个Observable Tasks of Tasks填充。我一直在尝试创建一个显示每行索引的列,该列用作表中任务的ID,但我已经尝试了方法here和其他方法的类似方法来源收效甚微。

我的参考代码,没有表面错误(就Eclipse而言):

    @FXML private TableColumn<Task, String> taskIndexCol;
    Callback<TableColumn<Task, String>, TableCell<Task, String>> cb =
            new Callback<TableColumn<Task, String>, TableCell<Task, String>>(){
                @Override
                public TableCell<Task, String> call(TableColumn<Task, String> col) {
                    TableCell<Task, String> cell = new TableCell<Task, String>() {
                        @Override
                        protected void updateItem(String item, boolean empty) {
                            super.updateItem(item, empty);
                            if (item == null) {
                                setText("");
                            } else {
                                setText(getIndex()+"");
                            }
                        }
                    };
                    return cell;
                }
    };

    taskIndexCol.setCellFactory(cb);

目前,当我尝试设置列的CellFactory时,我的代码给了我一个NullPointerException。我已经用填充的任务列表尝试了它,但这没有用。我已经被困了很长时间了 - 理论上这应该很简单,因为它只是对行进行编号?感觉就像我跳过一百万个篮球去做一些令人沮丧的简单事情。

编辑:最后一行给了我NPE。

1 个答案:

答案 0 :(得分:5)

无法告诉Null指针异常的原因,因为您没有显示堆栈跟踪,识别出引发异常的行,或者发布了足够的代码(回调中的代码都没有抛出空指针异常,所以其他地方出了问题。)

对于您的实际单元格实现,您没有显示您是否在列上设置了cellValueFactory。如果不这样做,那么item始终null,因此您永远不会在该列的单元格中看到任何文字。您可以检查empty属性(或方法参数),以检查单元格是空行还是具有实际数据。 (注意这意味着该列实际上根本不需要提供任何数据:它可以是TableColumn<Task, Void>。)

此外,依靠使用updateIndex(...)代替updateItem(...)可能更安全。保证在索引更改时调用updateIndex;如果您实现updateItem,则假设在项目之前设置了索引,这意味着您依赖于实现细节。

如果使用Java 8 lambda表达式,您的代码会更短,更容易阅读:

taskIndexCol.setCellFactory(col -> new TableCell<Task, String>() {
    @Override
    protected void updateIndex(int index) {
        super.updateIndex(index);
        if (isEmpty() || index < 0) {
            setText(null);
        } else {
            setText(Integer.toString(index));
        }
    }
});

或者

taskIndexCol.setCellFactory(col -> {
    TableCell<Task, String> cell = new TableCell<>();
    cell.textProperty().bind(Bindings.when(cell.emptyProperty())
        .then("")
        .otherwise(cell.indexProperty().asString()));
    return cell ;
});

这是一个SSCCE:

import java.util.function.Function;

import static javafx.beans.binding.Bindings.when ;

import javafx.application.Application;
import javafx.beans.binding.Bindings;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
import javafx.beans.value.ObservableValue;
import javafx.scene.Scene;
import javafx.scene.control.TableCell;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.layout.BorderPane;
import javafx.stage.Stage;

public class TableViewWithIndexColumn extends Application {

    @Override
    public void start(Stage primaryStage) {
        TableView<Person> table = new TableView<>();
        table.setEditable(true);

        table.getItems().addAll(
                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"));


        TableColumn<Person, String> firstNameCol = createColumn("First Name",
                Person::firstNameProperty, 150);
        TableColumn<Person, String> lastNameCol = createColumn("Last Name",
                Person::lastNameProperty, 150);
        TableColumn<Person, String> emailCol = createColumn("Email",
                Person::emailProperty, 150);

        // index column doesn't even need data...
        TableColumn<Person, Void> indexCol = createColumn("Index", person -> null, 50);

        // cell factory to display the index:
//        indexCol.setCellFactory(col -> {
//            
//            // just a default table cell:
//            TableCell<Person, Void> cell = new TableCell<>();
//            
//            cell.textProperty().bind(
//                when(cell.emptyProperty())
//                    .then("")
//                    .otherwise(cell.indexProperty().asString()));
//            
//            return cell ;
//        });

        indexCol.setCellFactory(col -> new TableCell<Person, Void>() {
            @Override
            public void updateIndex(int index) {
                super.updateIndex(index);
                if (isEmpty() || index < 0) {
                    setText(null);
                } else {
                    setText(Integer.toString(index));
                }
            }
        });

        table.getColumns().add(indexCol);
        table.getColumns().add(firstNameCol);
        table.getColumns().add(lastNameCol);
        table.getColumns().add(emailCol);

        primaryStage.setScene(new Scene(new BorderPane(table), 600, 400));
        primaryStage.show();
    }

    private <S, T> TableColumn<S, T> createColumn(String title,
            Function<S, ObservableValue<T>> property, double width) {
        TableColumn<S, T> col = new TableColumn<>(title);
        col.setCellValueFactory(cellData -> property.apply(cellData.getValue()));
        col.setPrefWidth(width);
        return col;
    }

    public static class Person {
        private final StringProperty firstName = new SimpleStringProperty();
        private final StringProperty lastName = new SimpleStringProperty();
        private final StringProperty email = new SimpleStringProperty();

        public Person() {
            this("", "", "");
        }

        public Person(String firstName, String lastName, String email) {
            setFirstName(firstName);
            setLastName(lastName);
            setEmail(email);
        }

        public final StringProperty firstNameProperty() {
            return this.firstName;
        }

        public final java.lang.String getFirstName() {
            return this.firstNameProperty().get();
        }

        public final void setFirstName(final java.lang.String firstName) {
            this.firstNameProperty().set(firstName);
        }

        public final StringProperty lastNameProperty() {
            return this.lastName;
        }

        public final java.lang.String getLastName() {
            return this.lastNameProperty().get();
        }

        public final void setLastName(final java.lang.String lastName) {
            this.lastNameProperty().set(lastName);
        }

        public final StringProperty emailProperty() {
            return this.email;
        }

        public final java.lang.String getEmail() {
            return this.emailProperty().get();
        }

        public final void setEmail(final java.lang.String email) {
            this.emailProperty().set(email);
        }

    }

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