6

当尝试单击子菜单中的某个项目时,很自然地会在其下方的菜单项上快速绘制鼠标。Windows 和 Mac 都通过在打开菜单之前放置​​一个小的延迟来处理这个问题。Swing JMenus 不处理此问题,并且鼠标短暂悬停的菜单将在鼠标到达预期的菜单项之前打开。

例如,在下图中,如果我尝试选择Item 3,但在此过程中我的鼠标短暂滑过Menu 2Menu 1子菜单会在我到达之前消失。

有没有人有解决这个问题的任何提示或建议?我的想法是定义一个自定义 MenuUI,将计时器添加到其鼠标处理程序中。

一块屏幕

这是一些简单的示例代码,可以说明我的问题:

public class Thing extends JFrame {
    public Thing()
    {
        super();
        this.setSize(new Dimension(500, 500));
        final JPopupMenu pMenu = new JPopupMenu();
        for (int i = 0; i < 5; i++)
        {
            JMenu menu = new JMenu("Menu " + i);
            pMenu.add(menu);
            for (int j = 0; j < 10; j++)
            {
                menu.add(new JMenuItem("Item " + j));
            }
        }

        this.addMouseListener(new MouseAdapter() {

            @Override
            public void mouseReleased(MouseEvent e) {
                pMenu.show(Thing.this, e.getX(), e.getY());
            }
        });
    }

    public static void main(String[] args)
    {
        Thing t = new Thing();
        t.setVisible(true);
    }
}
4

3 回答 3

2

调用setDelay(delay)您的menu变量,其中delay参数是等待菜单显示的毫秒数,作为 int。

以下代码行将延迟设置为 1 秒,因此用户必须将鼠标悬停在菜单项“Menu n”上 1 秒,然后才会显示子菜单:menu.setDelay(1000);

这是编辑后的代码片段:

for (int i = 0; i < 5; i++)
{
    JMenu menu = new JMenu("Menu " + i);
    pMenu.add(menu);
    for (int j = 0; j < 10; j++)
    {
        menu.add(new JMenuItem("Item " + j));
    }
    menu.setDelay(1000);
}
于 2013-09-24T20:56:22.913 回答
0

我想出了一个非常hacky的解决方案。

我制作了一个扩展 BasicMenuUI 的 UI 类。我重写该createMouseInputListener方法以返回自定义MouseInputListener而不是内部的私有handler对象BasicMenuUI

然后,我从 GrepCode[1] 中获取了MouseInputListener实现代码handler,并将其复制到我的自定义侦听器中。我做了一个改变,在 mouseEntered. 我的最终代码mouseEntered如下所示:

public void mouseEntered(MouseEvent e) {
        timer.schedule(new TimerTask() {

            @Override
            public void run() {
                if (menuItem.isShowing())
                {
                    Point mouseLoc = MouseInfo.getPointerInfo().getLocation();
                    Point menuLoc = menuItem.getLocationOnScreen();
                    if (mouseLoc.x >= menuLoc.x && mouseLoc.x <= menuLoc.x + menuItem.getWidth() &&
                            mouseLoc.y >= menuLoc.y && mouseLoc.y <= menuLoc.y + menuItem.getHeight())
                    {
                        originalMouseEnteredStuff();
                    }
                }
            }
        }, 100);
    }

在调用 中的原始代码之前mouseEntered,我检查以确保鼠标仍在此菜单区域内。我不希望鼠标刷过的所有菜单在 100 毫秒后弹出。

请让我知道是否有人为此找到了更好的解决方案。

[1] http://www.grepcode.com/file_/repository.grepcode.com/java/root/jdk/openjdk/7-b147/javax/swing/plaf/basic/BasicMenuUI.java/?v=source

于 2013-09-25T16:07:36.387 回答
0

非常感谢,你拯救了我的一天!该解决方案按预期工作,但我建议使用 Swing 计时器来确保代码由 EDT 执行。

此外,您应该在调用原始内容之前将菜单延迟临时设置为零。否则用户必须等待两倍的延迟时间。

@Override
public void mouseEntered(MouseEvent e) {
    if (menu.isTopLevelMenu() || menu.getDelay() == 0) {
        originalMouseEnteredStuff(e);
    } else {
        final javax.swing.Timer timer = new javax.swing.Timer(menu.getDelay(), new DelayedMouseEnteredAction(e));
        timer.setRepeats(false);
        timer.start();
    }
}
class DelayedMouseEnteredAction implements ActionListener
{
    private final MouseEvent mouseEnteredEvent;

    private DelayedMouseEnteredAction(MouseEvent mouseEnteredEvent) {
        this.mouseEnteredEvent = mouseEnteredEvent;
    }

    @Override
    public void actionPerformed(ActionEvent actionEvent) {
        if (menu.isShowing()) {
            final Point mouseLocationOnScreen = MouseInfo.getPointerInfo().getLocation();
            final Rectangle menuBoundsOnScreen = new Rectangle(menu.getLocationOnScreen(), menu.getSize());
            if (menuBoundsOnScreen.contains(mouseLocationOnScreen)) {
                /*
                 * forward the mouse event only if the mouse cursor is yet
                 * located in the menus area.
                 */
                int menuDelay = menu.getDelay();
                try {
                    /*
                     * Temporary remove the delay. Otherwise the delegate would wait the
                     * delay a second time e.g. before highlighting the menu item.
                     */
                    menu.setDelay(0);
                    originalMouseEnteredStuff(mouseEnteredEvent);
                } finally {
                    // reset the delay
                    menu.setDelay(menuDelay);
                }
            }
        }
    }
}
于 2019-04-11T07:05:36.487 回答