TableView - 尝试使其行为与Excel类似

时间:2015-02-13 11:48:12

标签: 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()函数的问题在于它只选择单个单元格(所选单元格的左侧或右侧)而不是所选单元格的所有左侧或右侧单元格。

1 个答案:

答案 0 :(得分:0)

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

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