在JavaFX中創建一個行索引列



javafx-2 javafx-8 (1)

這是不可能的,因為你沒有顯示堆棧跟踪,標識了引發異常的行或者你的代碼已經足夠了(你的回調中沒有任何代碼可以拋出空指針異常,所以別的地方有什麼問題)。

對於實際的單元實現,您沒有顯示是否在列上設置了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", "[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]"));


        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);
    }
}

我有一個JavaFX TableView,我正在填充一個ObservableList of Tasks。 我一直在試圖創建一個列,顯示每一行的索引,作為表中的任務的ID,但我已經嘗試了這裡的方法和來自其他來源的類似方法,但很少成功。

我的代碼參考,沒有表面的錯誤(據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。





fxml