高亮显示上下文菜单中的第一个选项,而无需将鼠标悬停在Javafx 9上

Vaz*_*zid 7 javafx java-9

当我们右键单击上下文菜单时,列表中的第一个选项将突出显示,而无需将鼠标悬停。仅在打开应用程序后第一次右键单击此操作。从javafx-9观察到此行为。直到javafx-8正常工作。

尝试使用示例代码:

import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.control.ContextMenu;
import javafx.scene.control.Label;
import javafx.scene.control.MenuItem;
import javafx.scene.layout.TilePane;
import javafx.stage.Stage;

public class SampleContextMenu extends Application {
    // labels
    Label l;

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

    // launch the application
    public void start(Stage stage) {
        // set title for the stage
        stage.setTitle("creating contextMenu ");

        // create a label
        Label label1 = new Label("This is a ContextMenu example ");

        // create a menu
        ContextMenu contextMenu = new ContextMenu();

        // create menuitems
        MenuItem menuItem1 = new MenuItem("menu item 1");
        MenuItem menuItem2 = new MenuItem("menu item 2");
        MenuItem menuItem3 = new MenuItem("menu item 3");

        // add menu items to menu
        contextMenu.getItems().add(menuItem1);
        contextMenu.getItems().add(menuItem2);
        contextMenu.getItems().add(menuItem3);

        // create a tilepane
        TilePane tilePane = new TilePane(label1);

        // setContextMenu to label
        label1.setContextMenu(contextMenu);

        // create a scene
        Scene sc = new Scene(tilePane, 200, 200);

        // set the scene
        stage.setScene(sc);

        stage.show();
    }
}
Run Code Online (Sandbox Code Playgroud)

kle*_*tra 5

经过一番挖掘后,发现罪魁祸首(可以这么说)是场景最初显示时的默认焦点遍历-将焦点放在第一个可聚焦节点上,在contextMenu的情况下,这是第一项。

首先尝试破解:将焦点对准项目后,将焦点要求重新回到场景的根源。步骤:

  • 在contextMenu上注册一个onShown处理程序
  • 在处理程序中,获取包含contextMenu的场景:此时,其focusOwner仍然为null,因此我们需要在其focusOwner属性上注册一个changeListener。
  • 在侦听器中,在第一次更改focusOwner(旧值为null)时,请求将焦点放在根上并清理侦听器

当心:这还不够好,事实证明这只是一个表面修饰,评论中提到了一些小故障

  • 请求焦点到场景根禁用键盘导航
  • 第一项仍处于活动状态:按Enter激活其动作

下一步尝试(现在真的很脏,需要访问非公共类的隐藏实现细节!):将第一次尝试的最后一步替换为

  • 抓取包含的ContextMenuContent(com.sun.xx中的内部类),它是焦点项的祖父母
  • 请求重点关注该内容以使突出显示消失
  • 更新该内容,以注意不关注项目(反射访问私有字段)

在代码中:

contextMenu.setOnShown(e -> {
    Scene scene = contextMenu.getScene();
    scene.focusOwnerProperty().addListener((src, ov, nv) -> {
        // focusOwner set after first showing
        if (ov == null) {
            // transfer focus to root
            // old hack (see the beware section) on why it doesn't work
            //  scene.getRoot().requestFocus();

            // next try: 
            // grab the containing ContextMenuContainer and force the internal
            // book-keeping into no-item-focused state
            Parent parent = nv.getParent().getParent();
            parent.requestFocus();
            // reflective setting of private field, this is my utility method, use your own ;)
            invokeSetFieldValue(ContextMenuContent.class, parent, "currentFocusedIndex", -1);
            // cleanup
            contextMenu.setOnShown(null);

         }
    });
});
Run Code Online (Sandbox Code Playgroud)

为了方便起见,这是用于内部字段反射访问的实用方法(没有火箭场景,只是纯Java;)

public static void invokeSetFieldValue(Class<?> declaringClass, Object target, String name, Object value) {
    try {
        Field field = declaringClass.getDeclaredField(name);
        field.setAccessible(true);
        field.set(target, value);
    } catch (NoSuchFieldException | SecurityException | IllegalArgumentException | IllegalAccessException e) {
        Thread.currentThread().getUncaughtExceptionHandler().uncaughtException(Thread.currentThread(), e);
    }
}
Run Code Online (Sandbox Code Playgroud)