Сочетание клавиш широкого применения - Java Swing - PullRequest
29 голосов
/ 19 сентября 2008

Я хотел бы создать комбинацию клавиш для приложения Java Swing. Цикл по всем компонентам и добавление ярлыка на каждый, имеет побочные эффекты, связанные с фокусом, и выглядит как решение грубой силы

У кого-нибудь есть более чистое решение?

Ответы [ 6 ]

39 голосов
/ 19 сентября 2008

Для каждого окна используйте JComponent.registerKeyboardAction с условием WHEN_IN_FOCUSED_WINDOW. Альтернативное использование:

JComponent.getInputMap(WHEN_IN_FOCUSED_WINDOW).put(keyStroke, command);
JComponent.getActionMap().put(command,action);

, как описано в документации к API * register6eyboardAction .

19 голосов
/ 19 сентября 2008

Установите пользовательский KeyEventDispatcher. Класс KeyboardFocusManager также является хорошим местом для этой функции.

KeyEventDispatcher

14 голосов
/ 13 декабря 2011

Для людей, которым интересно (как и я), как использовать 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 (), возможно, не является необходимым, но, вероятно, будет хорошей идеей не блокировать цикл глобальных событий.

6 голосов
/ 13 декабря 2011

Если у вас есть меню, вы можете добавить глобальные сочетания клавиш для пунктов меню:

    JMenuItem item = new JMenuItem(action);
    KeyStroke key = KeyStroke.getKeyStroke(
        KeyEvent.VK_R, KeyEvent.CTRL_DOWN_MASK);
    item.setAccelerator(key);
    menu.add(item);
1 голос
/ 21 апреля 2015

Немного упрощенный пример:

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;
  }

});
0 голосов
/ 05 июля 2013

Используйте следующий кусок кода

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" на нужный вам ярлык.

...