Сочетание клавиш (не Ctrl или Alt) в Java - PullRequest
1 голос
/ 11 декабря 2011

Я делаю проект, который должен определить, была ли нажата определенная комбинация клавиш. Любые клавиши, такие как Ctrl , Alt , A - Z и 0 - 9 может использоваться как комбинация клавиш.

Я искал некоторые коды, используя KeyStroke.getKeyStroke, но похоже, что он не допускает комбинаций без Ctrl , Shift , Alt и т. Д.

Мое приложение должно обнаруживать комбинации, даже БЕЗ Ctrl или Alt , например. просто комбинация простого A + B + C . Это тоже можно сделать с KeyStroke.getKeyStroke? Ответы с благодарностью.

Ответы [ 3 ]

1 голос
/ 18 ноября 2012

AFAIK, нет способа получить статус ключа, который не имеет ничего общего с:

KeyStroke keyStroke = KeyStroke.getKeyStroke("A");
// NOT supported
if (keyStroke.isPressed()) {
  // do something
}

Единственное, что поддерживается, это уведомление о изменении статуса отдельной клавиши, fi на самом низком уровне в KeyListener

public void keyPressed(KeyEvent e) {
    if (KeyEvent.VK_A == e.getKeyCode()) {
        ....
    }
}

Для обработки комбинаций клавиш (кроме специальных клавиш-модификаторов) невозможно отследить, какие клавиши были нажаты /выпущен, чтобы узнать, какие являются в нажатом состоянии.

Ниже приведен пример того, как это делается на уровне привязок клавиш.Основные ингредиенты

  • Триггер: сущность, содержащая логику накопительного «постановки на охрану» конечного действия для выполнения
  • Действие, которое отключает / выключает триггер
  • Привязки клавиш для отпущенных / нажатых отдельных нажатий клавиш, привязанных к соответствующему действию постановки на охрану

Некоторый код:

// the logic container
public static interface Trigger {
    public void addTrigger(String trigger);
    public void arm(ActionEvent e);
    public void disarm(ActionEvent e);
}

// a particular implementation which uses the actionCommand 
// as identifiers 
public static class ActionTrigger implements Trigger {
    private Action triggered;
    private List<String> triggers = new ArrayList<>();
    private List<String> armed = new ArrayList<>();

    public ActionTrigger(Action triggered) {
        this.triggered = triggered;
    }

    @Override
    public void arm(ActionEvent e) {
        String command = e.getActionCommand();
        if (!triggers.remove(command)) return;
        armed.add(command);
        if (triggers.isEmpty()) {
            triggered.actionPerformed(e);
        }
    }

    @Override
    public void disarm(ActionEvent e) {
        String command = e.getActionCommand();
        if (!armed.remove(command)) return;
        triggers.add(command);
    }

    @Override
    public void addTrigger(String trigger) {
        triggers.add(trigger);
    }
}

// an Action notifying the trigger of dis/arms
public static class ArmingAction extends AbstractAction {
    private Trigger trigger;
    private boolean arm;

    /**
     * @param trigger
     */
    public ArmingAction(Trigger trigger, String command, boolean arm) {
        this.trigger = trigger;
        this.arm = arm;
        putValue(ACTION_COMMAND_KEY, command);
    }

    @Override
    public void actionPerformed(ActionEvent e) {
        if (arm) {
            trigger.arm(e);
        } else {
            trigger.disarm(e);
        }
    }
}

// usage
// the action to trigger with multiple keys
Action action = new AbstractAction("real") {

    @Override
    public void actionPerformed(ActionEvent e) {
        LOG.info("******triggered: " + e);
    }

};

JComponent comp = new JPanel();
ActionMap actionMap = comp.getActionMap();
InputMap inputMap = comp.getInputMap(JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT);
Trigger trigger = new ActionTrigger(action);
// the key combination
char[] chars = {'A', 'S', 'D'};
for (int i = 0; i < chars.length; i++) {
    // the identifier
    String command = "step" + chars[i];
    trigger.addTrigger(command);
    // binding for pressed
    String pressedID = "pressed" + chars[i];
    actionMap.put(pressedID, new ArmingAction(trigger, command, true));
    inputMap.put(KeyStroke.getKeyStroke("pressed " + chars[i]), pressedID);
    // binding for released
    String releasedID = "released" + chars[i];
    actionMap.put(releasedID, new ArmingAction(trigger, command, false));
    inputMap.put(KeyStroke.getKeyStroke("released " + chars[i]), releasedID);
}


comp.add(new JButton("multibindings ... a s d"));
1 голос
/ 11 декабря 2011

Зависит от того, является ли тип приложения приложением с графическим интерфейсом или нет, для связанного с ним интерфейса необходимо искать

DocumentListener и KeyBindings , и я предлагаю отказаться от реализации KeyListener для более сложного кода и обходного пути Focus

или добавление addAWTEventListener (прослушиватель AWTEventListener, long eventMask) и есть возможность прослушивания KeyEvents и MouseEvents тоже

0 голосов
/ 18 ноября 2013

Вы можете воспроизводить keyPressed и keyReleased метод KeyListener. Если вы печатаете с клавиатуры, событие keyPressed сначала происходит перед keyReleased. Вы можете получить событие нажатия клавиши и поймать эту комбинацию клавиш, выпущенную с помощью бизнес-логики.

public void keyPressed (KeyEvent evt) {

      keyHit = KeyEvent.getKeyText(evt.getKeyCode());            
      System.out.println("Key Pressed is "+keyHit);

}

public void keyReleased (KeyEvent evt) {

      stringBuffer.append(keyHit)      
      System.out.println("Key Released is "+keyHit);

}

...