java - TableView - 尝试使其表现得像 Excel

标签 java javafx tableview keyboard-events

我正在尝试使 TableView 表现得像 Excel。我无法选择正确的单元格。

在 Excel 中,如果您选择一列(例如该列中的 5 个相邻单元格),然后单击 SHIFT + LEFT,Excel 将自动选择所有相应的最左侧单元格。 例如 假设在 Excel 中我选择了 [A1][A2][A3],然后单击 SHIFT + RIGHT,知道所选单元格将是: [A1][B1][A2][B2][A3][B3]

如何使用 TableView 实现类似的行为。

我尝试过执行以下操作,但没有成功。

// Table view
TableView tv = new TableView();

// Some more code goes here

// Select right most element
tv.getSelectionModel().selectRightCell();

selectRightCell()selectLeftCell() 函数的问题在于,它仅选择一个单元格(所选单元格的左侧或右侧),而不是所有单元格所选单元格的左侧或右侧单元格。

最佳答案

下面是示例代码,您可以通过鼠标拖动或按control+箭头键进行选择。

import com.sun.javafx.scene.control.behavior.TableCellBehavior;
import javafx.application.Application;
import javafx.beans.property.SimpleStringProperty;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.event.EventHandler;
import javafx.geometry.Insets;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.control.*;
import javafx.scene.control.cell.PropertyValueFactory;
import javafx.scene.input.KeyCode;
import javafx.scene.input.KeyEvent;
import javafx.scene.input.MouseDragEvent;
import javafx.scene.input.MouseEvent;
import javafx.scene.layout.VBox;
import javafx.scene.text.Font;
import javafx.stage.Stage;
import javafx.util.Callback;

import java.util.Arrays;

public class DragSelectionTable extends Application {

    private TableView<Person> table = new TableView<Person>();

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

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

    @Override
    public void start(Stage stage) {
        Scene scene = new Scene(new Group());
        stage.setTitle("Table View Sample");
        stage.setWidth(450);
        stage.setHeight(500);

        final Label label = new Label("Address Book");
        label.setFont(new Font("Arial", 20));

        TableColumn<Person, String> firstNameCol = new TableColumn<>("First Name");
        firstNameCol.setMinWidth(100);
        firstNameCol.setCellValueFactory(new PropertyValueFactory<Person, String>("firstName"));

        TableColumn<Person, String> lastNameCol = new TableColumn<>("Last Name");
        lastNameCol.setMinWidth(100);
        lastNameCol.setCellValueFactory(new PropertyValueFactory<Person, String>("lastName"));

        TableColumn<Person, String> emailCol = new TableColumn<>("Email");
        emailCol.setMinWidth(200);
        emailCol.setCellValueFactory(new PropertyValueFactory<Person, String>("email"));

        final Callback<TableColumn<Person, String>, TableCell<Person, String>> cellFactory = new DragSelectionCellFactory();
        firstNameCol.setCellFactory(cellFactory);
        lastNameCol.setCellFactory(cellFactory);
        emailCol.setCellFactory(cellFactory);

        table.setEditable(true);
        table.setItems(data);
        table.getColumns().addAll(Arrays.asList(firstNameCol, lastNameCol, emailCol));
        table.getSelectionModel().setCellSelectionEnabled(true);
        table.getSelectionModel().setSelectionMode(SelectionMode.MULTIPLE);
        table.addEventHandler(KeyEvent.KEY_RELEASED, new ControlDownSelectionEventHandler());

        final VBox vbox = new VBox();
        vbox.setSpacing(5);
        vbox.setPadding(new Insets(10, 0, 0, 10));
        vbox.getChildren().addAll(label, table);

        ((Group) scene.getRoot()).getChildren().addAll(vbox);

        stage.setScene(scene);
        stage.show();
    }

    public class DragSelectionCellFactory implements Callback<TableColumn<Person, String>, TableCell<Person, String>> {
        @Override
        public TableCell<Person, String> call(final TableColumn<Person, String> col) {
            return new DragSelectionCell();
        }
    }

    public class DragSelectionCell extends TableCell<Person, String> {
        public DragSelectionCell() {
            setOnDragDetected(new DragDetectedEventHandler(this));
            setOnMouseDragEntered(new DragEnteredEventHandler(this));
        }

        @Override
        public void updateItem(String item, boolean empty) {
            super.updateItem(item, empty);
            if (empty) {
                setText(null);
            } else {
                setText(item);
            }
        }

    }

    public class DragDetectedEventHandler implements EventHandler<MouseEvent> {
        TableCell<Person, String> tableCell;

        public DragDetectedEventHandler(TableCell<Person, String> tableCell) {
            this.tableCell = tableCell;
        }

        @Override
        public void handle(final MouseEvent event) {
            tableCell.startFullDrag();
        }
    }

    public class DragEnteredEventHandler implements EventHandler<MouseDragEvent> {
        TableCell<Person, String> tableCell;

        public DragEnteredEventHandler(TableCell<Person, String> tableCell) {
            this.tableCell = tableCell;
        }

        @Override
        public void handle(final MouseDragEvent event) {
            performSelection(tableCell.getTableView(), tableCell.getTableColumn(), tableCell.getIndex());
        }
    }

    public class ControlDownSelectionEventHandler implements EventHandler<KeyEvent> {
        @Override
        public void handle(final KeyEvent event) {
            KeyCode code = event.getCode();
            if (event.isShiftDown() && (KeyCode.UP.equals(code) || KeyCode.DOWN.equals(code) || KeyCode.LEFT.equals(code) || KeyCode.RIGHT.equals(code))) {
                int index = table.getFocusModel().getFocusedCell().getRow();
                TableColumn column = table.getFocusModel().getFocusedCell().getTableColumn();
                performSelection(table, column, index);
            }
        }
    }

    protected void performSelection(TableView<Person> table, TableColumn<Person, String> column, int index) {
        final TablePositionBase anchor = TableCellBehavior.getAnchor(table, table.getFocusModel().getFocusedCell());
        int columnIndex = table.getVisibleLeafIndex(column);

        int minRowIndex = Math.min(anchor.getRow(), index);
        int maxRowIndex = Math.max(anchor.getRow(), index);
        TableColumnBase minColumn = anchor.getColumn() < columnIndex ? anchor.getTableColumn() : column;
        TableColumnBase maxColumn = anchor.getColumn() >= columnIndex ? anchor.getTableColumn() : column;

        table.getSelectionModel().clearSelection();
        final int minColumnIndex = table.getVisibleLeafIndex((TableColumn) minColumn);
        final int maxColumnIndex = table.getVisibleLeafIndex((TableColumn) maxColumn);
        for (int _row = minRowIndex; _row <= maxRowIndex; _row++) {
            for (int _col = minColumnIndex; _col <= maxColumnIndex; _col++) {
                table.getSelectionModel().select(_row, table.getVisibleLeafColumn(_col));
            }
        }
        table.getFocusModel().focus(index, column);
    }

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

关于java - TableView - 尝试使其表现得像 Excel,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28498921/

相关文章:

安卓删除按钮

ios - 如何定位我的 footerView 使其位于 UITableView 的底部?

ios - 在获取 indexPath 之后但在删除该 indexPath 处的单元格之前调用 reloadData() 是否不安全?

java - 如何使用 HtmlUnit 显示所有 AJAX 请求

java - 安卓/Java : onProgressUpdate() not being called?

java - 不知道为什么我无法让 addMouseListener(this())、addMouseMotionListener(this()) 工作

java - 无法调用 Openweathermap API

java - 可以从 Java 或 JavaFx 使用 mongorestore 和 mongodump 命令吗?

JavaFX GrowableDataBuffer、Canvas 性能问题

css - 如何在 Javafx 中将 CSS 用于图表栏颜色