3

我需要在Ctrl-TabCtrl-Shift-Tab上切换标签。但它仅在TabPane 处于焦点时才有效。我有带有 setContent(textArea) 的标签。当textArea处于焦点时,我按Ctrl-Tab,焦点位于菜单中的某个位置:

    TextArea@3f40a15a[styleClass=text-input text-area]
    TextArea@7b7c82a4[styleClass=text-input text-area]
    TabPane[id=null, styleClass=tab-pane]
    TextArea[id=null, styleClass=text-input text-area]
    MenuButtonSkinBase$MenuLabeledImpl[id=null, styleClass=label]
    TextArea[id=null, styleClass=text-input text-area]
    MenuButtonSkinBase$MenuLabeledImpl[id=null, styleClass=label]
    TextArea[id=null, styleClass=text-input text-area]
    TabPane[id=null, styleClass=tab-pane]
    TextArea[id=null, styleClass=text-input text-area]
    MenuButtonSkinBase$MenuLabeledImpl[id=null, styleClass=label]
    TextArea[id=null, styleClass=text-input text-area]

我跟踪这样的焦点变化:

    scene.focusOwnerProperty().addListener(new ChangeListener<Node>() {

        @Override
        public void changed(ObservableValue<? extends Node> ov, Node t, Node t1) {
            System.out.println(t1);
        }
    });

我几乎在每个组件(包括菜单、工具栏、textArea 和窗格)上都将FocusTraversable设置为false

我创建了选项卡并使用了setContent(textArea)

我还想在每次用户打开/关闭选项卡并在选项卡之间切换(使用鼠标和键盘热键)时请求关注 textArea

    tabPane.getSelectionModel().selectedIndexProperty().addListener(new ChangeListener<Number>() {

        @Override
        public void changed(ObservableValue<? extends Number> ov, Number t, Number t1) {
            Platform.runLater(new Runnable() {

                @Override
                public void run() {
                    if (tabPane.getTabs().size() > 0) {
                        FileTab tab = (FileTab)tabPane.getTabs().get(tabPane.getSelectionModel().getSelectedIndex());
                        tab.getTextArea().requestFocus();
                    }
                }
            });

        }
    });

这不起作用。我试过tab.getContent().requestFocus() - 也不起作用。

关于在 TabPane 子处于焦点时更改选项卡:似乎 TabPane 获得了 KeyEvent,但因为它不在焦点中(焦点中的 textArea),所以它只是跳过了这个事件。也许我可以通过在TabPane触发KeyEvent来复制默认功能,或者甚至可以自己重新实现它。或者,也许我可以从此侦听器调用 nextTab/previousTab 默认操作:

tabPane.addEventFilter(KeyEvent.KEY_PRESSED, new EventHandler() {

        @Override
        public void handle(KeyEvent t) {
            if (t.getCode().equals(KeyCode.TAB) && t.isControlDown()) {
                  //Event.fireEvent(tabPane, event); //I'm not sure how to do it like this
                  //Maybe it's better to just call default actions on TabPane.
                  t.consume();
            }
        }
    }); 

http://s23.postimg.org/poeujzwbf/screenshot_185.png

但是我仍然不知道如何在标签切换后将焦点设置在TextArea上。也许我应该在延迟一段时间后尝试调用Platform.runLater()

4

2 回答 2

4

选项卡开关的解决方案:

    scene.addEventFilter(KeyEvent.KEY_PRESSED, new EventHandler<KeyEvent>() {

        @Override
        public void handle(KeyEvent t) {
            if (t.getCode().equals(KeyCode.TAB) && t.isControlDown()) {
                int size = tabPane.getTabs().size();

                if (size > 0) {
                    SkinBase skin = (SkinBase) tabPane.getSkin();
                    TabPaneBehavior tabPaneBehavior = (TabPaneBehavior) skin.getBehavior();

                    int selectedIndex = tabPane.getSelectionModel().getSelectedIndex();

                    if (!t.isShiftDown()) {
                        if (selectedIndex < size -1) {
                            tabPaneBehavior.selectNextTab();
                        } else {
                            tabPaneBehavior.selectTab(tabPane.getTabs().get(0));
                        }
                    } else {
                        if (selectedIndex > 0) {
                            tabPaneBehavior.selectPreviousTab();
                        } else {
                            tabPaneBehavior.selectTab(tabPane.getTabs().get(size - 1));
                        }
                    }

                    t.consume();
                }

            }
        }
    });

在选定选项卡上设置焦点的解决方案:

        tabPane.getSelectionModel().selectedIndexProperty().addListener(new ChangeListener<Number>() {

        @Override
        public void changed(ObservableValue<? extends Number> ov, Number t, final Number t1) {
            Platform.runLater(new Runnable() {

                @Override
                public void run() {
                    if (tabPane.getTabs().size() > 0) {
                        final FileTab tab = (FileTab) tabPane.getTabs().get(t1.intValue());

                        final Timer timer = new Timer();
                        timer.schedule(new TimerTask() {

                            @Override
                            public void run() {
                                Platform.runLater(new Runnable() {

                                    @Override
                                    public void run() {
                                        tab.getContent().requestFocus();
                                        timer.cancel();
                                        timer.purge();
                                    }
                                });
                            }
                        }, 25);
                    }
                }
            });

        }
    });

将焦点设置在选定选项卡上的替代解决方案(基于@Uluk Biy的https://stackoverflow.com/a/16920898 ):

        tabPane.getSelectionModel().selectedIndexProperty().addListener(new ChangeListener<Number>() {

        @Override
        public void changed(ObservableValue<? extends Number> ov, Number t, final Number t1) {
            Platform.runLater(new Runnable() {

                @Override
                public void run() {
                    if (tabPane.getTabs().size() > 0) {
                        final FileTab tab = (FileTab) tabPane.getTabs().get(t1.intValue());

                        final Timeline animation = new Timeline(
                                new KeyFrame(Duration.millis(25),
                                new EventHandler<ActionEvent>() {
                                    @Override
                                    public void handle(ActionEvent actionEvent) {
                                        Platform.runLater(new Runnable() {

                                            @Override
                                            public void run() {
                                                tab.getContent().requestFocus();
                                            }
                                        });
                                    }
                                }));
                        animation.setCycleCount(1);
                        animation.play();
                    }
                }
            });

        }
    });

这应该是比仅使用计时器更好的解决方案。

于 2013-09-27T08:47:55.247 回答
0

只需使用以下内容来关注第一个选项卡(即选项卡 0):

Platform.runLater(() -> tabPane.getTabs().get(0).getContent().requestFocus());
于 2020-04-25T11:37:08.510 回答