3

这个问题是这个问题的后续。

JMenuBar的行为类似于 Firefox 和 iTunes 中的菜单栏。也就是说,菜单栏最初是隐藏的,但是当您按下 时Alt,菜单栏就会出现。

另一个问题的答案解决了实现该功能的问题,但它带来了另一个问题:当不可见JMenuItem时,加速器不起作用。JMenuBar也就是说,你必须按下Alt之前CTRL+F(安装的加速器)才会起作用。

但是,这不应该是这种情况,因为该setAccelerator()方法声明了以下内容:

公共无效 setAccelerator(KeyStroke keyStroke)

设置在不导航菜单层次结构的情况下调用菜单项的操作侦听器的组合键。UI 负责安装正确的操作。请注意,当键入键盘加速器时,无论当前是否显示菜单,它都会起作用。

所以,我想知道这是否是另一个 Java 错误?

SSCCE(要显示菜单,请按Alt,安装的加速器CTRL+F用于“查找”,它会显示一个虚拟JOptionPane输入):

import javax.swing.JFrame;
import javax.swing.JMenu;
import javax.swing.JMenuBar;
import javax.swing.JMenuItem;
import javax.swing.JOptionPane;
import javax.swing.KeyStroke;
import javax.swing.MenuElement;
import javax.swing.MenuSelectionManager;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.WindowConstants;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;

public class MenuBarTest extends JFrame {

    public MenuBarTest() {
        JMenu jMenu1 = new JMenu();
        JMenu jMenu2 = new JMenu();
        final JMenuBar jMenuBar1 = new JMenuBar();
        JMenuItem jMenuItem1 = new JMenuItem();
        JMenuItem jMenuItem2 = new JMenuItem();

        setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);

        jMenu1.setText("File");
        jMenuItem1.setText("jMenuItem1");
        jMenu1.add(jMenuItem1);
        jMenuBar1.add(jMenu1);
        jMenu2.setText("Edit");
        jMenuItem2.setText("Find");
        jMenu2.add(jMenuItem2);
        jMenuBar1.add(jMenu2);
        setJMenuBar(jMenuBar1);

        jMenuBar1.setVisible(false);
        ChangeListener listener = new ChangeListener() {
            @Override
            public void stateChanged(ChangeEvent e) {
                MenuElement[] elements = MenuSelectionManager.defaultManager().getSelectedPath();
                jMenuBar1.setVisible(elements.length > 0 && elements[0] == jMenuBar1);
            }
        };
        MenuSelectionManager.defaultManager().addChangeListener(listener);

        jMenuItem2.setAccelerator(KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_F, java.awt.event.InputEvent.CTRL_MASK));
        jMenuItem2.setText("Find");
        jMenuItem2.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                String what = JOptionPane.showInputDialog(MenuBarTest.this, "Search for what?");
                System.out.println(what);
            }
        });

        pack();
        setSize(500,500);
        setLocationRelativeTo(null);
        setVisible(true);
    }

    public static void main(String args[]) {
        try {
            UIManager.setLookAndFeel("com.sun.java.swing.plaf.windows.WindowsLookAndFeel");
        } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
            ex.printStackTrace();
        }
        java.awt.EventQueue.invokeLater(new Runnable() {
            public void run() {
                new MenuBarTest();
            }
        });
    }
}
4

1 回答 1

4

仔细阅读你的重点

请注意,当键入键盘加速器时,无论当前是否显示菜单,它都会起作用。

这谈论的是菜单而不是其父级。这意味着菜单当前可能未显示。然而,真正的(可能没有充分记录的)需求属性是它必须显示。已经更新了我对您上一个问题的回答。

于 2013-08-09T21:02:47.680 回答