悬停时激活JMenuBar



JMenuBar直到第一次单击它才开始显示选定的JMenuItems或显示JMenu弹出窗口。单击JMenuBar中的某个位置后,所有这些项都会对鼠标悬停做出响应。

我想绕过需要的初始点击,让它在鼠标悬停时自动激活。有办法做到这一点吗?

方法是在JMenu上添加一个MouseListener并侦听事件mouseEntered。在事件处理程序中,您只需要使用doClick单击它。例如,

jMenuFile.addMouseListener(new MouseListener(){
    public void mouseEntered(MouseEvent e) {
       jMenuFile.doClick();
    }
  ...
});

一旦以编程方式点击鼠标,它就会自动打开弹出菜单。要激活整个JMenuBar,必须在每个JMenu上添加一个侦听器。为此,最好单独创建一个侦听器对象。

我在吧台上有两个菜单项,所以我做了:

MouseListener ml = new MouseListener(){
    public void mouseClicked(MouseEvent e) {}
    public void mousePressed(MouseEvent e) {}
    public void mouseReleased(MouseEvent e) {}
    public void mouseExited(MouseEvent e) {}
    public void mouseEntered(MouseEvent e) {
        ((JMenu)e.getSource()).doClick();
    }
  };
  jMenuFile.addMouseListener(ml);
  jMenuHelp.addMouseListener(ml);

如果你在工具栏上有这么多菜单项,你可以迭代它:

for (Component c: jMenuBar1.getComponents()) {
    if (c instanceof JMenu){
        c.addMouseListener(ml);
    }
}

Roman C的初始和接受的答案不会自动关闭具有作为JMenuBar一部分的子MenuItems的菜单。

运行((JMenu)e.getSource()).doClick();在mouseEntered上模拟单击其中一个JMenu父项,但不能简单地添加到mouseExited方法,因为MouseListener需要附加到子MenuItems和JMenu子项。(它在MenuBar的正常赋值中没有这样做——只附加到父JMenu对象)。

此外,当鼠标离开整个菜单结构(即子菜单下拉菜单)时,试图让MouseExit侦听器启动"关闭"方法会出现问题。

以下是从我的实时应用程序中得到的一个完全有效的答案:

我解决鼠标关闭菜单的方法是在构造函数的顶部运行一个布尔变量"isMouseOut"来跟踪,然后以更面向对象的方式分配MouseListener,以在用户与菜单交互时跟踪多个MouseIn-MouseOut事件。它调用一个单独的menuClear方法,该方法作用于布尔值"isMouseOut"的状态。该类实现MouseListener。事情就是这样做的。

创建一个ArrayList,首先将所有菜单项添加到此数组。像这样:

    Font menuFont = new Font("Arial", Font.PLAIN, 12);
    JMenuBar menuBar = new JMenuBar();
    getContentPane().add(menuBar, BorderLayout.NORTH); 
// Array of MenuItems
    ArrayList<JMenuItem> aMenuItms = new ArrayList<JMenuItem>();
    JMenuItem mntmRefresh = new JMenuItem("Refresh");
    JMenuItem mntmNew = new JMenuItem("New");
    JMenuItem mntmNormal = new JMenuItem("Normal");
    JMenuItem mntmMax = new JMenuItem("Max");
    JMenuItem mntmStatus = new JMenuItem("Status");
    JMenuItem mntmFeedback = new JMenuItem("Send Feedback");
    JMenuItem mntmEtsyTWebsite = new JMenuItem("EtsyT website");
    JMenuItem mntmAbout = new JMenuItem("About");
    aMenuItms.add(mntmRefresh);
    aMenuItms.add(mntmNew);
    aMenuItms.add(mntmNormal);
    aMenuItms.add(mntmMax);
    aMenuItms.add(mntmStatus);
    aMenuItms.add(mntmFeedback);
    aMenuItms.add(mntmEtsyTWebsite);
    aMenuItms.add(mntmAbout);

然后在这个阶段迭代arrayList,使用for()循环添加MouseListener:

  for (Component c : aMenuItms) {
        if (c instanceof JMenuItem) {
            c.addMouseListener(ml);
        }
    }

现在为MenuBar设置JMenu父项:

// Now set JMenu parents on MenuBar
    final JMenu mnFile = new JMenu("File");
    menuBar.add(mnFile).setFont(menuFont);
    final JMenu mnView = new JMenu("View");
    menuBar.add(mnView).setFont(menuFont);
    final JMenu mnHelp = new JMenu("Help");
    menuBar.add(mnHelp).setFont(menuFont);

然后将下拉菜单Items children添加到JMenu parent:

// Now set menuItems as children of JMenu parents
    mnFile.add(mntmRefresh).setFont(menuFont);
    mnFile.add(mntmNew).setFont(menuFont);
    mnView.add(mntmNormal).setFont(menuFont);
    mnView.add(mntmMax).setFont(menuFont);
    mnHelp.add(mntmStatus).setFont(menuFont);
    mnHelp.add(mntmFeedback).setFont(menuFont);
    mnHelp.add(mntmEtsyTWebsite).setFont(menuFont);
    mnHelp.add(mntmAbout).setFont(menuFont);

将mouseListeners作为单独的步骤添加到JMenu父级:

    for (Component c : menuBar.getComponents()) {
        if (c instanceof JMenu) {
            c.addMouseListener(ml);
        }
    }

现在,子menuItem元素都有自己的侦听器,这些侦听器与父JMenu元素和MenuBar本身是分开的-在MouseListener()实例化中识别对象类型很重要,这样您就可以在鼠标悬停时自动打开菜单(在本例中是3x JMenu父项),但也可以避免子异常错误,并允许清晰地识别mouseOUT菜单结构,而不尝试监视鼠标位置。MouseListener如下所示:

MouseListener ml = new MouseListener() {
        public void mouseClicked(MouseEvent e) {
        }
        public void mousePressed(MouseEvent e) {
        }
        public void mouseReleased(MouseEvent e) {
        }
        public void mouseExited(MouseEvent e) {
            isMouseOut = true;
            timerMenuClear();
        }
        public void mouseEntered(MouseEvent e) {
            isMouseOut = false;
            Object eSource = e.getSource();
            if(eSource == mnHelp || eSource == mnView || eSource == mnFile){
                ((JMenu) eSource).doClick();
            }
        }
    }; 

上面只是模拟鼠标点击JMenu的"parents"(本例中为3x),因为它们是子菜单下拉菜单的触发器。timerMenuClear()方法调用MenuSelectionManager来清空在实际鼠标OUT:时活动的任何选定路径点

public void timerMenuClear(){
    ActionListener task = new ActionListener() {
      public void actionPerformed(ActionEvent e) {
          if(isMouseOut == true){
              System.out.println("Timer");
          MenuSelectionManager.defaultManager().clearSelectedPath();
          }
      }
  };        
    //Delay timer half a second to ensure real mouseOUT
  Timer timer = new Timer(1000, task); 
  timer.setInitialDelay(500);        
  timer.setRepeats(false);
  timer.start();
}

我花了一些时间进行测试,监控在JVM开发过程中我可以访问的值,但它确实很有效!即使使用嵌套菜单:)我希望很多人觉得这个完整的例子非常有用。

相关内容

  • 没有找到相关文章