使用Swing在Java中为JButton创建热键

6
我使用以下代码在Java Swing表单中创建热键。如果我按下ALT + N,ALT + R,ALT + 1,ALT + 2,光标将移动到正确的文本字段,并在相应的文本字段中输入值。它可以正常运行。我的问题是,我在此表单中有保存和退出JButtons。如果我按CTRL + S,就会同时选择保存按钮。如果我按CTRL + X,则会选择退出按钮。如何为JButton创建助记符?如何使用以下代码进行CTRL + S,CTRL + X操作?
提前致谢。
package hotkeys;
import java.awt.event.*;
import javax.swing.*;
import java.net.*;
public class hotkey extends JFrame {
    public static void main(String arg[]) {
        JLabel Name = new JLabel("Name");
        JTextField tf1 = new JTextField(20);
        Name.setLabelFor(tf1);
        Name.setDisplayedMnemonic('N');


        JLabel Regno = new JLabel("RegNO");
        JTextField tf2 = new JTextField(20);
        Regno.setLabelFor(tf2);
        Regno.setDisplayedMnemonic('R');

        JLabel Mark1 = new JLabel("Mark1");
        JTextField tf3 = new JTextField(20);
        Mark1.setLabelFor(tf3);
        Mark1.setDisplayedMnemonic('1');

        JLabel Mark2 = new JLabel("Mark2");
        JTextField tf4 = new JTextField(20);
        Mark2.setLabelFor(tf4);
        Mark2.setDisplayedMnemonic('2');


        JButton b1 = new JButton("Save");
        JButton b2 = new JButton("eXit");


        JFrame f = new JFrame();
        JPanel p = new JPanel();

        p.add(Name);
        p.add(tf1);
        p.add(Regno);
        p.add(tf2);
        p.add(Mark1);
        p.add(tf3);
        p.add(Mark2);
        p.add(tf4);
        p.add(b1);
        p.add(b2);

        f.add(p);
        f.setVisible(true);
        f.pack();
    }
}

2
请问您需要什么其他帮助吗?这些链接是关于在Java Swing中创建热键和使用JTextField的回车键的问题。 - mKorbel
4个回答

21

您需要在按钮的组件InputMap中注册一个keyBinding。在代码中(重复您在以前的问题中被告知要做的微妙变体:-)

// create an Action doing what you want
Action action = new AbstractAction("doSomething") {

    @Override
    public void actionPerformed(ActionEvent e) {
        System.out.println("triggered the action");
    }

};
// configure the Action with the accelerator (aka: short cut)
action.putValue(Action.ACCELERATOR_KEY, KeyStroke.getKeyStroke("control S"));

// create a button, configured with the Action
JButton toolBarButton = new JButton(action);
// manually register the accelerator in the button's component input map
toolBarButton.getActionMap().put("myAction", action);
toolBarButton.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW).put(
        (KeyStroke) action.getValue(Action.ACCELERATOR_KEY), "myAction");

10

Sun提供了关于整个键绑定问题的非常好的描述。您可以在此处找到:JavaSE键绑定教程

//编辑

编辑了我的示例代码,所以您现在可以将其复制+粘贴并使其正常工作。感谢反馈,已包含缺失的部分。

KeyStroke keySave = KeyStroke.getKeyStroke(KeyEvent.VK_S, Event.CTRL_MASK); 
Action performSave = new AbstractAction("Save") {  
    public void actionPerformed(ActionEvent e) {     
         //do your save
         System.out.println("save");
    }
}; 
JButton b1 = new JButton(performSave); 
b1.getActionMap().put("performSave", performSave);
b1.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW).put(keySave, "performSave"); 

KeyStroke keyExit = KeyStroke.getKeyStroke(KeyEvent.VK_Y, Event.CTRL_MASK); 
Action performExit = new AbstractAction("Exit") {  
    public void actionPerformed(ActionEvent e) {     
        //exit
        System.out.println("exit");
    }
}; 
JButton b2 = new JButton(performExit); 
b2.getActionMap().put("performExit", performExit);
b2.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW).put(keyExit, "performExit"); 

好的方向 - 只是有点不完整:此外,您需要a)在按钮的actionMap中注册操作b)如果您想要一个窗口范围的快捷方式,请在“组件inputMap”中注册keyStroke,即JComponent.WHEN_IN_FOCUSED_WINDOW类型的inputMap,而不是默认值(即WHEN_FOCUSED)。 - kleopatra
刚刚注意到 c) Action 应该设置其名称属性(而不是使用文本实例化按钮)- 所以现在点了个踩,一旦您纠正了这些小问题,我会撤销的 :-) - kleopatra
感谢你的反馈,kleopatra。:) 我已经编辑了代码示例,所以在复制粘贴后它将能够工作。祝你有美好的一天。 - Matschie
好的,没问题。c)现在也正确了吗?因为我以前从未使用过这些名称。哇,学到了新东西,太棒了 :) - Matschie
完美 :-) 作为最后一点糖衣涂层,您可以考虑使用 getStroke(String) 方法,这是抓取一个的推荐方式 - 尽管在一天结束时是个人口味问题。 - kleopatra

1
我提供这篇文章,不仅是为了自己的学习体验,也希望对其他人有所帮助。过去我一直难以应用代码片段来绑定按键,希望我的解释和代码能够清晰明了。感谢@kleopatra提供的代码片段,我的代码基于她的代码。
(我在需要匹配的地方使用大写字母,以便更清楚地显示必须匹配的内容。)
该代码通过在getInputMap和getActionMap中匹配字符串,将Ctrl-Shift-U按键与MYACTION中的actionPerformed代码链接起来。
下面的三个MYACTION实例必须匹配,四个MYACTIONBUTTON实例和两个MAKE_THESE_MATCH字符串实例也必须匹配。可以随意命名,只要保持它们匹配即可。
按钮MYACTIONBUTTON必须将MYACTION作为定义它的JButton的参数,并且必须将getInputMap和getActionMap应用于它。
private static JButton MYACTIONBUTTON;
private static JFrame frame;
private static JPanel panel;

...

Action MYACTION = new AbstractAction()
{

  @Override
  public void actionPerformed(ActionEvent e)
  {
      // put action code here
  }
};

MYACTIONBUTTON = new JButton(MYACTION);

MYACTIONBUTTON.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW)

                           .put(getKeyStroke(VK_U, CTRL_DOWN_MASK | SHIFT_DOWN_MASK),

                                  "MAKE_THESE_MATCH"); 

MYACTIONBUTTON.getActionMap().put("MAKE_THESE_MATCH",        MYACTION);

panel.add(MYACTIONBUTTON);

frame.add(panel);

1

刚刚修改了你的代码。(在 //** 中插入了代码)
只有一个评论... Ctrl-X 是编辑命令“剪切”的快捷键(与 Ctrl-C 和 Ctrl-V 一起)。你的框架中有可编辑字段。我使用了 Ctrl-Q(退出)。

import java.awt.event.*;
import java.beans.PropertyChangeListener;

import javax.swing.*;
import javax.swing.plaf.ActionMapUIResource;

import java.net.*;

public class HotKeys extends JFrame {
public static void main(String arg[]) {
    JLabel Name = new JLabel("Name");
    JTextField tf1 = new JTextField(20);
    Name.setLabelFor(tf1);
    Name.setDisplayedMnemonic('N');

    JLabel Regno = new JLabel("RegNO");
    JTextField tf2 = new JTextField(20);
    Regno.setLabelFor(tf2);
    Regno.setDisplayedMnemonic('R');

    JLabel Mark1 = new JLabel("Mark1");
    JTextField tf3 = new JTextField(20);
    Mark1.setLabelFor(tf3);
    Mark1.setDisplayedMnemonic('1');

    JLabel Mark2 = new JLabel("Mark2");
    JTextField tf4 = new JTextField(20);
    Mark2.setLabelFor(tf4);
    Mark2.setDisplayedMnemonic('2');

    JButton b1 = new JButton("Save");
    JButton b2 = new JButton("eXit");

    JFrame f = new JFrame();
    JPanel p = new JPanel();

    p.add(Name);
    p.add(tf1);
    p.add(Regno);
    p.add(tf2);
    p.add(Mark1);
    p.add(tf3);
    p.add(Mark2);
    p.add(tf4);
    p.add(b1);
    p.add(b2);

    // *****************************************************
    ActionMap actionMap = new ActionMapUIResource();
    actionMap.put("action_save", new AbstractAction() {
        @Override
        public void actionPerformed(ActionEvent e) {
            System.out.println("Save action performed.");
        }
    });
    actionMap.put("action_exit", new AbstractAction() {
        @Override
        public void actionPerformed(ActionEvent e) {
            System.out.println("Exit action performed.");
        }
    });

    InputMap keyMap = new ComponentInputMap(p);
    keyMap.put(KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_S,
            java.awt.Event.CTRL_MASK), "action_save");
    keyMap.put(KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_Q,
            java.awt.Event.CTRL_MASK), "action_exit");
    SwingUtilities.replaceUIActionMap(p, actionMap);
    SwingUtilities.replaceUIInputMap(p, JComponent.WHEN_IN_FOCUSED_WINDOW,
            keyMap);
    // *****************************************************

    f.add(p);
    f.setVisible(true);
    f.pack();
}
}

不要使用replaceXX方法(除非你真的打算这样做)。a)你需要为所有JPanel实例安装这些新地图,至少是ActionMap。b)它无法在LAF切换后生存。 - kleopatra
稍微更正一下我的上一个评论:b) 是不正确的,因为 XXPanelUI 根本没有安装任何地图。 - kleopatra
我尝试了Ctrl-X (java.awt.event.KeyEvent.VK_X) - 但GUI忽略了它。之后我改成了java.awt.event.KeyEvent.VK_Q。也许我在尝试解释原因时错了,但事实是java.awt.event.KeyEvent.VK_X没有起作用。 - andrey

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接