5

在 OpenJFX 11.0.2 和 12.0.1 SDK(Windows 10、x64)中重现,在 JavaFX 8 中不可重现

右键单击表格列,然后尝试调整列的大小。在您再次手动单击列之前,不会显示调整大小的光标并且无法调整列的大小。

任何解决方法的想法?我需要使用contextMenuTableColumns,因此无法使用使标题忽略鼠标右键单击的潜在解决方法。

在此处输入图像描述

import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.stage.Stage;

public class Foo extends Application {


    @Override
    public void start(Stage stage) throws Exception {
        TableView<Object> testView = new TableView<>();
        testView.getColumns().addAll(new TableColumn<Object, Object>("C1"), new TableColumn<Object, Object>("C2"), new TableColumn<Object, Object>("C3"));

        stage.setScene(new Scene(testView));
        stage.show();
    }

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

2 回答 2

3

好的,我找到了以下(非常非常脏)的解决方法。我以前从未尝试过,因为我认为它会阻止上下文菜单显示(正如我在原始问题中指出的那样),但显然只是简单地消耗了每个TableColumnHeader作品的鼠标事件并且上下文菜单仍然正确显示(也适用于没有 TableColumns上下文菜单)。

不确定内部是否会出现任何问题,但由于默认情况下右键单击似乎没有做任何有用的事情,我希望不会。

当然lookupAll需要在渲染后调用。

注意 1:如果您已TableMenuButtonVisible设置为 true,则每次将列设置为可见时都需要执行此操作。

注2:它越来越脏。在将列设置为可见后再次调用它(见注 1)并不总是足够的(也不能通过Platform.runLater调用)。我认为那是因为此时尚未呈现列标题。你要么

  • 需要等到Set<Node>完全填满,即它的大小必须是amountOfVisibleColumns + 1。如果它等于可见列的数量,它将不适用于新显示的列。
  • 或之前调用layout()TableViewlookupAll
  • 或者如果你有一个扩展类TableView,覆盖layoutChildren并执行查找,如果可见列的数量发生了变化

注意3onMousePressed :如果按钮不是,您需要跟踪旧的并执行它SECONDARY,否则列的重新排序将不起作用。

如果您能想到任何更清洁的方法,请告诉我。

在此处输入图像描述

import java.util.Set;

import javafx.application.Application;
import javafx.scene.Node;
import javafx.scene.Scene;
import javafx.scene.control.ContextMenu;
import javafx.scene.control.MenuItem;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.skin.TableColumnHeader;
import javafx.scene.input.MouseButton;
import javafx.stage.Stage;

public class Foo extends Application {

    @Override
    public void start(Stage stage) throws Exception {
        TableView<Object> testView = new TableView<>();
        testView.getColumns().addAll(createColumn("C1"), createColumn("C2"), createColumn("C3"));

        stage.setOnShown(ev -> {
            Set<Node> headers = testView.lookupAll("TableColumnHeader");
            for (Node header : headers) {
                if (header != null) {
                    ((TableColumnHeader) header).setOnMousePressed(e -> {
                        if (e.getButton() == MouseButton.SECONDARY) {
                            e.consume();
                        }
                    });
                }
            }
        });

        stage.setScene(new Scene(testView));
        stage.show();
    }

    private TableColumn<Object, Object> createColumn(String text) {
        MenuItem item = new MenuItem("Context");
        item.setOnAction(e -> {
            System.out.println("Action");
        });

        ContextMenu contextMenu = new ContextMenu();
        contextMenu.getItems().add(item);

        TableColumn<Object, Object> column = new TableColumn<>(text);
        column.setContextMenu(contextMenu);

        return column;
    }

    public static void main(String[] args) {
        launch(args);
    }
}
于 2019-05-14T11:29:17.453 回答
2

编辑:在 Java 错误跟踪器中找到了描述的错误,并提交了带有修复程序的 PR:
https
://github.com/openjdk/jfx/pull/483 编辑 2:我的 PR 被接受并合并回来。该错误现已修复,您可以使用 17-ea+11 进行测试。:-)

我也有同样的问题。此错误是由mousePressedHandler添加的TableColumnHeader. 这个类有更多的问题,例如如果我通过点击一个列来关闭一个,排序将被触发PopupControlsetConsumeAutoHidingEvents(true)那些方法需要改变,也许addEventHandler应该使用方法而不是方便的setOn...方法。

当我要展示我的PopupControl

public class MyTableColumnHeader extends TableColumnHeader {

    public MyTableColumnHeader(TableColumnBase tc) {
        super(tc);
        addEventHandler(MouseEvent.MOUSE_PRESSED, this::onMousePressed);
    }

    private void onMousePressed(MouseEvent mouseEvent) {
        if (mouseEvent.getButton() == MouseButton.SECONDARY) {
            showPopup();
            // Consume here, so the column won't get 'stuck'.
            mouseEvent.consume();
        }
    }

    private void showPopup() {
        ...
    }
}

最终,有人应该至少打开一个错误。在不久的将来,我可能还会去看看。

于 2021-03-23T00:28:41.153 回答