调用焦点JButton的ActionListener的默认键是Space,但是如何将其更改为另一个键?
答案 0 :(得分:3)
这不是很简单,需要:
JComponent.WHEN_FOCUSED
条件相关联示例:
import java.awt.event.ActionEvent;
import java.awt.event.KeyEvent;
import javax.swing.*;
@SuppressWarnings("serial")
public class AlterSpaceBinding extends JPanel {
private JButton myButton = new JButton("My Button -- associate with \"B\" key");
public AlterSpaceBinding() {
myButton = alterDefaultButtonAction(myButton, KeyEvent.VK_B);
myButton.addActionListener(l -> {
System.out.println("button pressed");
});
add(myButton);
add(new JButton("Second Button -- no change"));
}
public static JButton alterDefaultButtonAction(JButton button, int desiredKeyCode) {
// get the correct InputMap
int condition = JComponent.WHEN_FOCUSED;
InputMap inputMap = button.getInputMap(condition);
ActionMap actionMap = button.getActionMap();
// empty action that does nothing
Action emptyAction = new AbstractAction() {
@Override
public void actionPerformed(ActionEvent e) {
// This does NOTHING
}
};
// get both key strokes for space key, but pressed and released
KeyStroke spaceKeyPressed = KeyStroke.getKeyStroke(KeyEvent.VK_SPACE, 0, false);
KeyStroke spaceKeyReleased = KeyStroke.getKeyStroke(KeyEvent.VK_SPACE, 0, true);
// get input map key for pressed and released
String keyForSpacePressed = (String) inputMap.get(spaceKeyPressed);
String keyForSpaceReleased = (String) inputMap.get(spaceKeyReleased);
// get actions for press and release
Action actionForSpacePressed = actionMap.get(keyForSpacePressed);
Action actionForSpaceReleased = actionMap.get(keyForSpaceReleased);
// substitute empty action
actionMap.put(keyForSpacePressed, emptyAction);
actionMap.put(keyForSpaceReleased, emptyAction);
// key stroke for desired key code
KeyStroke desiredKeyPressed = KeyStroke.getKeyStroke(desiredKeyCode, 0, false);
KeyStroke desiredKeyReleased = KeyStroke.getKeyStroke(desiredKeyCode, 0, true);
// put in the original actions to where wanted
inputMap.put(desiredKeyPressed, desiredKeyPressed.toString());
actionMap.put(desiredKeyPressed.toString(), actionForSpacePressed);
inputMap.put(desiredKeyReleased, desiredKeyReleased.toString());
actionMap.put(desiredKeyReleased.toString(), actionForSpaceReleased);
return button;
}
private static void createAndShowGui() {
AlterSpaceBinding mainPanel = new AlterSpaceBinding();
JFrame frame = new JFrame("AlterSpaceBinding");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.getContentPane().add(mainPanel);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
public static void main(String[] args) {
SwingUtilities.invokeLater(() -> createAndShowGui());
}
}
答案 1 :(得分:3)
您需要:
更新组件的InputMap
,以添加新的KeyStroke
指向现有的Action
。
防止KeyStroke
中现有的InputMap
调用现有的Action
。如果您希望能够同时使用两个KeyStroke来调用默认的Action,则此步骤是可选的。
注意:
InputMap
InputMap
的更新。您可以执行以下操作修改Hovercrafts示例中的代码:
import java.awt.event.ActionEvent;
import java.awt.event.KeyEvent;
import javax.swing.*;
@SuppressWarnings("serial")
public class AlterSpaceBinding extends JPanel {
private JButton myButton = new JButton("My Button -- associate with \"B\" key");
public AlterSpaceBinding() {
myButton = alterDefaultButtonAction(myButton, KeyEvent.VK_B);
myButton.addActionListener(l -> {
System.out.println("button pressed");
});
add(myButton);
add(new JButton("Second Button -- no change"));
}
public static JButton alterDefaultButtonAction(JButton button, int desiredKeyCode) {
// get the correct InputMap
InputMap inputMap = button.getInputMap(JComponent.WHEN_FOCUSED);
// get both key strokes for space key, but pressed and released
KeyStroke spaceKeyPressed = KeyStroke.getKeyStroke(KeyEvent.VK_SPACE, 0, false);
KeyStroke spaceKeyReleased = KeyStroke.getKeyStroke(KeyEvent.VK_SPACE, 0, true);
// key stroke for desired key code
KeyStroke desiredKeyPressed = KeyStroke.getKeyStroke(desiredKeyCode, 0, false);
KeyStroke desiredKeyReleased = KeyStroke.getKeyStroke(desiredKeyCode, 0, true);
// share the Action with desired KeyStroke
inputMap.put(desiredKeyPressed, inputMap.get(spaceKeyPressed));
inputMap.put(desiredKeyReleased, inputMap.get(spaceKeyReleased));
// disable original KeyStrokes (optional)
inputMap.put(spaceKeyPressed, "none");
inputMap.put(spaceKeyReleased, "none");
return button;
}
private static void createAndShowGui() {
AlterSpaceBinding mainPanel = new AlterSpaceBinding();
JFrame frame = new JFrame("AlterSpaceBinding");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.getContentPane().add(mainPanel);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
public static void main(String[] args) {
SwingUtilities.invokeLater(() -> createAndShowGui());
}
}
上面的代码不需要:
Action
。ActionMap
。请查看Using Key Bindings,以了解操纵组件的InputMap
和ActionMap
的更多示例。
注意:
更可能的情况是,您想更新应用程序中所有按钮的绑定,以使您拥有通用的LAF。在这种情况下,您可以从UIManager更新所有JButton共享的“ focusInputMap”:
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
public class SSCCE extends JPanel
{
SSCCE()
{
add( new JButton("Button 1" ) );
add( new JButton("Button 2" ) );
add( new JButton("Button 3" ) );
}
private static void createAndShowGUI()
{
JFrame frame = new JFrame("SSCCE");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(new SSCCE());
frame.pack();
frame.setLocationByPlatform( true );
frame.setVisible( true );
// Update shared InputMap
InputMap inputMap = (InputMap)UIManager.get("Button.focusInputMap");
int desiredKeyCode = KeyEvent.VK_B; // type "b" to invoke Action on button
KeyStroke spaceKeyPressed = KeyStroke.getKeyStroke(KeyEvent.VK_SPACE, 0, false);
KeyStroke spaceKeyReleased = KeyStroke.getKeyStroke(KeyEvent.VK_SPACE, 0, true);
// key stroke for desired key code
KeyStroke desiredKeyPressed = KeyStroke.getKeyStroke(desiredKeyCode, 0, false);
KeyStroke desiredKeyReleased = KeyStroke.getKeyStroke(desiredKeyCode, 0, true);
// share the Action with desired KeyStroke
inputMap.put(desiredKeyPressed, inputMap.get(spaceKeyPressed));
inputMap.put(desiredKeyReleased, inputMap.get(spaceKeyReleased));
// optionally disable original KeyStrokes
inputMap.put(spaceKeyPressed, "none");
inputMap.put(spaceKeyReleased, "none");
}
public static void main(String[] args) throws Exception
{
java.awt.EventQueue.invokeLater( () -> createAndShowGUI() );
/*
EventQueue.invokeLater(new Runnable()
{
public void run()
{
createAndShowGUI();
}
});
*/
}
}