背景

在JavaFX中,当MenuBar的Menu或者其Sub-Menu包含的MenuItem合计超过一个屏幕所能显示的高度时,菜单能够滚动显示。此时有一个Bug,滚动的位置会被记录,同一个层级的所有ContextMenu都会使用同一个滚动位置,也就是说当在一个ContextMenu中向下滚动超过一些距离后,另一个ContextMenu可能就完全无法看到ContextMenu。

分析

这个Bug根本的原因就是不同的ContextMenu共用了同一个滚动位置,按理来说,不同的ContextMenu应该有各自的滚动位置记录。可以想办法看能否让各个ContextMenu独自维护滚动位置,不过我这里没有找到办法。

我现在的方案是:当菜单显示后,立刻重置滚动位置到0,并通过隐藏菜单再次显示的方式来刷新,这个方案有个缺陷就是离开菜单之后再回来滚动位置被重置(这个方案也是经过打印,调试查看类的层次,刷新处理等多个环节的调研才找到…)。

代码

//在菜单显示之后立刻校正滚动条位置
curMenu.setOnShown(new EventHandler<Event>() {
    private boolean firedBySelf = false;//用于标识是方案本身触发的刷新,还是外部操作触发的刷新

    @Override
    public void handle(Event t) {
        if (firedBySelf) {
            firedBySelf = false;
            return;
        }

     //找到最核心的显示皮肤类ContextMenuContent
        ContextMenu parentPopup = curMenu.getItems().get(0).getParentPopup();
        final Scene scene = parentPopup.getScene();
        Parent root = scene.getRoot();
        Group group = (Group) root.getChildrenUnmodifiable().get(0);
        ContextMenuContent content = (ContextMenuContent) group.getChildren().get(0);

        Class cla = ContextMenuContent.class;
        try {
       //通过反射修改滚动位置——私有变量ty
            Field field;
            field = cla.getDeclaredField("ty");
            field.setAccessible(true);
            field.set(content, 0);

       //刷新——隐藏再显示
            curMenu.hide();
            firedBySelf = true;
            curMenu.show();
        } catch (SecurityException ex) {
            log.error("", ex);
        } catch (IllegalArgumentException ex) {
            log.error("", ex);
        } catch (IllegalAccessException ex) {
            log.error("", ex);
        } catch (NoSuchFieldException ex) {
            log.error("", ex);
        }
    }
});

发表评论

邮箱地址不会被公开。 必填项已用*标注