将JTextField添加到JMenuBar会取消KeyListener响应! - 怎么修?

时间:2016-03-20 11:40:44

标签: java swing jtextfield keylistener jmenubar

以下是SSCCE代码;它运行正常且KeyListener正确响应,但向JTextField添加JMenuBar会导致KeyListener完全无法响应。

向下取消注释/评论添加JTextField的行,以查看差异:menuBar.add(textField);

import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;
import javax.swing.JFrame;
import javax.swing.JMenu;
import javax.swing.JMenuBar;
import javax.swing.JMenuItem;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.SwingUtilities;

public class JMenueWithKeyListenerExample extends JPanel {

    JTextField textField;
    JMenuBar menuBar;
    JFrame f;

    public JMenueWithKeyListenerExample() {

        textField = new JTextField();

        menuBar = new JMenuBar();
        JMenu fileMenue = new JMenu("File");
        JMenuItem menuItem1 = new JMenuItem("Item 1");
        JMenuItem menuItem2 = new JMenuItem("Item 2");

        fileMenue.add(menuItem1);
        fileMenue.add(menuItem2);

        menuBar.add(fileMenue);
      //menuBar.add(textField); // switch between comment & uncomment to see the difference

        f = new JFrame();
        f.setJMenuBar(menuBar);
        f.add(this);

        f.addKeyListener(new KeyAdapter() {

            @Override
            public void keyPressed(KeyEvent e) {
                if ((e.getKeyCode() == KeyEvent.VK_Z) && ((e.getModifiers() & KeyEvent.CTRL_MASK) != 0)) {
                    System.out.println("undo");
                }
            }
        });

        f.setSize(400, 400);
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.setVisible(true);
    }

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                new JMenueWithKeyListenerExample();
            }
        });
    }
}

1 个答案:

答案 0 :(得分:3)

添加快捷键的首选方法是使用input map and action map,如下所示:

f.getRootPane().getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW)
        .put(KeyStroke.getKeyStroke(KeyEvent.VK_Z, KeyEvent.CTRL_DOWN_MASK),
             "Undo");
f.getRootPane().getActionMap()
        .put("Undo", new AbstractAction() {
    @Override
    public void actionPerformed(ActionEvent e) {
        System.out.println("undo");
    }
});

当然,在实际代码中,您应该定义一个常量,而不是直接对"Undo"操作名称进行硬编码。也许可以单独定义动作,给它一个正确的名称和其他有用的属性。