Сочетание клавиш широкого применения - Java Swing
Я хотел бы создать сочетание клавиш для приложения Java Swing. Цикл по всем компонентам и добавление ярлыка для каждого, имеет побочные эффекты, связанные с фокусом, и кажется грубым решением проблемы.
У кого-нибудь есть более чистое решение?
6 ответов
Установите пользовательский KeyEventDispatcher. Класс KeyboardFocusManager также является хорошим местом для этой функциональности.
Для каждого окна используйте JComponent.registerKeyboardAction
с условием WHEN_IN_FOCUSED_WINDOW
, В качестве альтернативы используйте:
JComponent.getInputMap(WHEN_IN_FOCUSED_WINDOW).put(keyStroke, command);
JComponent.getActionMap().put(command,action);
как описано в документации API registerKeyboardAction.
Для людей, интересующихся (как и я), как использовать KeyEventDispatcher, вот пример, который я собрал. Он использует HashMap для хранения всех глобальных действий, потому что я не люблю большие if (key == ..) then .. else if (key == ..) then .. else if (key ==..) ..
строит.
/** map containing all global actions */
private HashMap<KeyStroke, Action> actionMap = new HashMap<KeyStroke, Action>();
/** call this somewhere in your GUI construction */
private void setup() {
KeyStroke key1 = KeyStroke.getKeyStroke(KeyEvent.VK_A, KeyEvent.CTRL_DOWN_MASK);
actionMap.put(key1, new AbstractAction("action1") {
@Override
public void actionPerformed(ActionEvent e) {
System.out.println("Ctrl-A pressed: " + e);
}
});
// add more actions..
KeyboardFocusManager kfm = KeyboardFocusManager.getCurrentKeyboardFocusManager();
kfm.addKeyEventDispatcher( new KeyEventDispatcher() {
@Override
public boolean dispatchKeyEvent(KeyEvent e) {
KeyStroke keyStroke = KeyStroke.getKeyStrokeForEvent(e);
if ( actionMap.containsKey(keyStroke) ) {
final Action a = actionMap.get(keyStroke);
final ActionEvent ae = new ActionEvent(e.getSource(), e.getID(), null );
SwingUtilities.invokeLater( new Runnable() {
@Override
public void run() {
a.actionPerformed(ae);
}
} );
return true;
}
return false;
}
});
}
Использование SwingUtils.invokeLater(), возможно, не является необходимым, но, вероятно, будет хорошей идеей не блокировать глобальный цикл обработки событий.
Когда у вас есть меню, вы можете добавить глобальные сочетания клавиш в пункты меню:
JMenuItem item = new JMenuItem(action);
KeyStroke key = KeyStroke.getKeyStroke(
KeyEvent.VK_R, KeyEvent.CTRL_DOWN_MASK);
item.setAccelerator(key);
menu.add(item);
Немного упрощенный пример:
KeyboardFocusManager keyManager;
keyManager=KeyboardFocusManager.getCurrentKeyboardFocusManager();
keyManager.addKeyEventDispatcher(new KeyEventDispatcher() {
@Override
public boolean dispatchKeyEvent(KeyEvent e) {
if(e.getID()==KeyEvent.KEY_PRESSED && e.getKeyCode()==27){
System.out.println("Esc");
return true;
}
return false;
}
});
Используйте следующий кусок кода
ActionListener a=new ActionListener(){
public void actionPerformed(ActionEvent ae)
{
// your code
}
};
getRootPane().registerKeyboardAction(a,KeyStroke.getKeyStroke("ctrl D"),JComponent.WHEN_IN_FOCUSED_WINDOW);
Замените "Ctrl D" на ярлык, который вы хотите.