Как назначить клавишу Enter в качестве триггерной клавиши для всех кнопок JButton в моем приложении Java?

Я пишу приложение Java Swing, используя внешний вид Metal. Каждый раз, когда в моем приложении есть JButton, пользователь использует клавишу Tab, чтобы переместить фокус на кнопку, а затем нажимает клавишу Enter. Ничего не произошло! Если он нажмет клавишу пробела, сработают события кнопки. Как назначить клавишу «Ввод» для запуска тех же событий, что и клавиша «Пробел»? Спасибо за помощь.


person Alfred B. Thordarson    schedule 13.01.2009    source источник


Ответы (6)


Я нашел следующее:

http://tips4java.wordpress.com/2008/10/25/enter-key-and-button/

Где Роб Камик пишет, что при использовании JDK5 и более поздних версий вы просто добавляете...

UIManager.put("Button.defaultButtonFollowsFocus", Boolean.TRUE);

...к приложению для решения проблемы. Это помогло мне! И я не могу представить ничего проще. Однако при использовании более старых версий Java вам придется сделать что-то вроде того, что Ричард и Питер описывают в своих ответах на этот вопрос.

person Alfred B. Thordarson    schedule 13.01.2009
comment
Ааа, очень мило ;-) Хорошая статья. Я просмотрел класс MetalLookAndFeel во время работы над своим ответом, и это всего на несколько строк выше Button.focusInputMap, который я использовал. - person Peter Štibraný; 13.01.2009

Вот полный пример. Ричард был близок, но вам же нужно, чтобы карта нажала ENTER для действия, а не просто отпустила. Чтобы заставить его работать для ВСЕХ кнопок, я поместил это сопоставление в карту ввода по умолчанию для кнопок. Добавьте импорт, и он должен работать.


public class Main implements Runnable {
    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Main());
    }

    @Override
    public void run() {
        setupEnterActionForAllButtons();

        JFrame frame = new JFrame("Button test");
        frame.getContentPane().add(createButton(), BorderLayout.NORTH);
        frame.getContentPane().add(createButton(), BorderLayout.SOUTH);
        frame.pack();
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        frame.setVisible(true);
    }

    private void setupEnterActionForAllButtons() {
        InputMap im = (InputMap) UIManager.getDefaults().get("Button.focusInputMap");
        Object pressedAction = im.get(KeyStroke.getKeyStroke("pressed SPACE"));
        Object releasedAction = im.get(KeyStroke.getKeyStroke("released SPACE"));

        im.put(KeyStroke.getKeyStroke("pressed ENTER"), pressedAction);
        im.put(KeyStroke.getKeyStroke("released ENTER"), releasedAction);
    }

    private JButton createButton() {
        JButton b = new JButton("press enter");
        b.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                System.out.println("Pressed");
            }
        });

        return b;
    }
}
person Peter Štibraný    schedule 13.01.2009
comment
Большое спасибо, Питер. Я думаю, что лучшие ответы - это ответы на основе кода. Пальцы вверх! - person Alfred B. Thordarson; 13.01.2009

На самом деле, это вопрос внешнего вида. Это (и должно быть) зависит от внешнего вида и ощущения того, какая клавиша запускает сфокусированную кнопку.

Обходной путь «кнопки по умолчанию» работает, поскольку используемый вами L&F использует ввод для кнопки по умолчанию.

Обходной путь Питера явно изменяет клавишу «фокус действий» L&F по умолчанию, что несколько более убедительно, если вы спросите меня.

Я бы добавил, что я не думаю, что многие пользователи захотят перейти к кнопке, а затем нажать Enter (большинство даже не заметит индикатор фокуса) — они хотят, чтобы действие по умолчанию было «правильным» и работало везде, где они нажимают. войти. Это можно сделать только с входными картами, как предлагает Ричард.

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

person Draemon    schedule 14.01.2009

Вы делаете это, назначая карту ввода/действия для клавиши Enter. Что-то вроде следующего:

// save the command mapping for space
Object spaceMap = button.getInputMap.get(KeyStroke.getKeyStroke(KeyEvent.VK_SPACE, 0, true));
// add a mapping from enter to the same command.
button.getInputMap().put(KeyStroke.getKeyStroke(KeyEvent.VK_ENTER, 0, true),spaceMap); 
person Richard Campbell    schedule 13.01.2009
comment
Я не хочу делать это со всеми кнопками в моем приложении, слишком много хлопот. - person Alfred B. Thordarson; 13.01.2009
comment
Вы можете создать подкласс JButton и добавить его в конструктор. - person Richard Campbell; 13.01.2009

Вы также можете установить «кнопку по умолчанию» на кнопку, которая была в последний раз сфокусирована.

Я сделал это в приложении, и все методы сделать это - кошмар для обслуживания и отладки. Дело в том, что это явно не то, к чему стремились разработчики Swing.

person davetron5000    schedule 13.01.2009
comment
Согласен, это был бы кошмар. - person Alfred B. Thordarson; 13.01.2009

Расширение к приведенным выше ответам, чтобы сделать то же самое с радио, флажками. Вызывается перед созданием компонентов

void setupEnterAction(String componentName){
    String keyName = componentName + ".focusInputMap";
    InputMap im = (InputMap) UIManager.getDefaults().get(keyName);
    Object pressedAction = im.get(KeyStroke.getKeyStroke("pressed SPACE"));
    Object releasedAction = im.get(KeyStroke.getKeyStroke("released SPACE"));
    im.put(KeyStroke.getKeyStroke("pressed ENTER"), pressedAction);
    im.put(KeyStroke.getKeyStroke("released ENTER"), releasedAction);
}

public void setEnterEvent(){
    setupEnterAction("Button");
    setupEnterAction("RadioButton");
    setupEnterAction("CheckBox");
}
person rinjan    schedule 15.09.2015