JavaFX 2 обновить ячейку таблицы, чтобы изменить стиль

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

Класс ExampleRow является прокси для реальных данных, которые поступают из электронной таблицы, его другая функция - хранить информацию выделения. Поскольку я не могу знать, сколько будет столбцов, я просто держу список столбцов, которые должны быть выделены (перестановка столбцов не будет поддерживаться). Класс ExampleTableCell просто устанавливает текст для ячейки и применяет выделение, если необходимо.

Если я выделю подсветку до того, как таблица будет нарисована [ячейка (2,2)], то при запуске приложения ячейка будет отображаться красным текстом. Проблема состоит в том, что нажатие кнопки приводит к выделению ячейки (1,1), но таблица не меняется. Если я изменяю размер окна приложения до нуля, то снова открываю его, подсветка ячейки (1,1) рисуется правильно - вероятно, потому что этот процесс вызывает полную перерисовку.

Я хотел бы знать, как я могу инициировать перерисовку вновь выделенных ячеек (или всех видимых ячеек), чтобы стиль был правильным?

ТИА

package example;

import java.util.HashSet;
import java.util.Set;
import javafx.application.Application;
import javafx.beans.property.SimpleIntegerProperty;
import javafx.beans.property.SimpleObjectProperty;
import javafx.beans.value.ObservableValue;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.TableCell;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.layout.BorderPane;
import javafx.scene.paint.Color;
import javafx.stage.Stage;
import javafx.util.Callback;

public class CellHighlightExample extends Application {

    private final int columnCount = 4;
    private final int rowCount = 5;
    private TableView<ExampleRow> table = new TableView<>();

    @Override
    public void start(Stage stage) {
        BorderPane root = new BorderPane();
        Scene scene = new Scene(root);

        Callback<TableColumn.CellDataFeatures<ExampleRow, String>, ObservableValue<String>> cellValueFactory = new Callback<TableColumn.CellDataFeatures<ExampleRow, String>, ObservableValue<String>>() {
            @Override
            public ObservableValue<String> call(TableColumn.CellDataFeatures<ExampleRow, String> p) {
                int row = p.getValue().getRow();
                int col = p.getTableView().getColumns().indexOf(p.getTableColumn());
                return new SimpleObjectProperty<>("(" + row + ", " + col + ")");
            }
        };

        Callback<TableColumn<ExampleRow, String>, TableCell<ExampleRow, String>> cellFactory = new Callback<TableColumn<ExampleRow, String>, TableCell<ExampleRow, String>>() {
            @Override
            public TableCell<ExampleRow, String> call(TableColumn<ExampleRow, String> p) {
                return new ExampleTableCell<>();
            }
        };

        for (int i = 0, n = columnCount; i < n; i++) {
            TableColumn<ExampleRow, String> column = new TableColumn<>();
            column.setCellValueFactory(cellValueFactory);
            column.setCellFactory(cellFactory);
            table.getColumns().add(column);
        }

        ObservableList<ExampleRow> rows = FXCollections.observableArrayList();
        for (int i = 0, n = rowCount; i < n; i++) {
            ExampleRow row = new ExampleRow(i);
            //Force a cell to be highlighted to show that highlighting works.
            if (i == 2) { row.addHighlightedColumn(2); }
            rows.add(row);
        }
        table.setItems(rows);

        Button b = new Button("Click to Highlight");
        b.setOnAction(new EventHandler<ActionEvent>() {
            @Override
            public void handle(ActionEvent t) {
                ExampleRow row = table.getItems().get(1);
                row.addHighlightedColumn(1);
                //How to trigger a redraw of the table or cell to reflect the new highlighting?
            }
        });

        root.setTop(b);
        root.setCenter(table);
        stage.setScene(scene);
        stage.show();
    }

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

    private class ExampleTableCell<S extends ExampleRow, T extends String> extends TableCell<S, T> {

        @Override
        public void updateItem(T item, boolean empty) {
            super.updateItem(item, empty);

            if (item == null) {
                setText(null);
                setGraphic(null);
            } else {
                setText(item);
                int colIndex = getTableView().getColumns().indexOf(getTableColumn());
                ExampleRow row = getTableView().getItems().get(getIndex());
                if (row.isHighlighted(colIndex)) {
                    setTextFill(Color.RED);
                }
            }
        }
    }

    private class ExampleRow {

        private SimpleIntegerProperty row;
        private Set<Integer> highlightedColumns = new HashSet<>();

        public ExampleRow(int row) {
            this.row = new SimpleIntegerProperty(row);
        }

        public int getRow() { return row.get(); }
        public void setRow(int row) { this.row.set(row); }
        public SimpleIntegerProperty rowProperty() { return row; }

        public boolean isHighlighted(int col) {
            if (highlightedColumns.contains(col)) {
                return true;
            }
            return false;
        }

        public void addHighlightedColumn(int col) {
            highlightedColumns.add(col);
        }
    }
}

1 ответ

Решение

Существует много дискуссий по этой проблеме, а именно обновление таблицы после изменения элемента (ов).
Увидеть
Элементы обновления JavaFX 2.1 TableView
вопросы
http://javafx-jira.kenai.com/browse/RT-21822
http://javafx-jira.kenai.com/browse/RT-22463
http://javafx-jira.kenai.com/browse/RT-22599

Решение состоит в том, чтобы вызвать внутренний метод обновления табличного представления. Некоторые предлагают удалить элементы таблицы и добавить их снова против..., но кажется, что самый простой обходной путь для вашего случая:

b.setOnAction(new EventHandler<ActionEvent>() {
    @Override
    public void handle(ActionEvent t) {
        ExampleRow row = table.getItems().get(1);
        row.addHighlightedColumn(1);
        //How to trigger a redraw of the table or cell to reflect the new highlighting?
        // Workaround
        table.getColumns().get(0).setVisible(false);
        table.getColumns().get(0).setVisible(true);
    }
});

которые найдены в комментариях к проблеме, связанных выше. Это действительно обходной путь или иллюзия? Вы должны копать глубже себя.

Другие вопросы по тегам