Подтвердить что ты не робот

Как установить кнопку Java по умолчанию для реакции на клавишу ENTER _released_?

В моем приложении используется кнопка "По умолчанию". Я хочу, чтобы он реагировал, когда клавиша ENTER отпущена. Не при нажатии клавиши ENTER.

Я удалил KeyStroke из InputMap кнопки. Но это не сработало для меня. Как мне это сделать?


import javax.swing.*;
import java.awt.*;
import java.awt.event.ActionEvent;

public class ButtonTest {
    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                buildFrame();
            }
        });
    }

    private static void buildFrame() {
        JFrame f = new JFrame("Test");
        f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);

        JButton button = new JButton(new AbstractAction("Button") {
            @Override
            public void actionPerformed(ActionEvent e) {
                System.out.println("ButtonTest::actionPerformed: CALLED");
            }
        });

        JButton button2 = new JButton("Button 2");
        InputMap im = button.getInputMap();
        im.put(KeyStroke.getKeyStroke("ENTER"), "none");
        im.put(KeyStroke.getKeyStroke("released ENTER"), "released");

        f.setLayout(new GridBagLayout());
        f.add(button);
        f.add(button2);
        f.getRootPane().setDefaultButton(button);

        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }
}

Это пример кода.

4b9b3361

Ответ 1

JRootPane rootPane = SwingUtilities.getRootPane(/* Your JButton  */); 
rootPane.setDefaultButton(/* Your JButton  */);

Ответ 2

ключи для кнопки по умолчанию (то есть кнопка, которая запускается при вводе, независимо от того, какой компонент является focusOwner) связаны в корневой компонент componentInputMap, то есть в его inputMap типа WHEN_IN_FOCUSED_WINDOW. Так технически, что место для настройки:

JComponent content = new JPanel();
content.add(new JTextField("some focusable"));
content.add(new JTextField("something else"));

JXFrame frame = wrapInFrame(content, "default button on released");
Action action = new AbstractAction("do something") {

    @Override
    public void actionPerformed(ActionEvent e) {
        LOG.info("clicked");
    }
};
JButton button = new JButton(action);
content.add(button);
frame.getRootPane().setDefaultButton(button);
// remove the binding for pressed
frame.getRootPane().getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW)
    .put(KeyStroke.getKeyStroke("ENTER"), "none");
// retarget the binding for released
frame.getRootPane().getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW)
    .put(KeyStroke.getKeyStroke("released ENTER"), "press");

Остерегайтесь: это по-прежнему отличается от нажатого/отпущенного поведения кнопки, отличной от значения по умолчанию, потому что действие rootPane просто вызывает кнопку .doClick, не пройдя шаги постановки на охрану/нажав кнопкуModel, чтобы косвенно вызвать действие.

Ответ 3

    InputMap im = button.getInputMap();
    im.put(KeyStroke.getKeyStroke("ENTER"), "pressed");
    im.put(KeyStroke.getKeyStroke("released ENTER"), "released");

Эффект от этого заключается в том, что кнопка выполняет свое действиеPerformed только один раз, когда ENTER освобождается (если я правильно понимаю, это то, что вы хотите).

EDIT: текущий код показывает, что actionPerformed выполняется только при выпуске ENTER (хотя визуально кнопка появляется нажатой уже при нажатии ENTER)

import javax.swing.*;
import java.awt.*;
import java.awt.event.ActionEvent;

public class ButtonTest {
    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                buildFrame();
            }
        });
    }

    private static void buildFrame() {
        JFrame f = new JFrame("Test");
        f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);

        JButton button = new JButton(new AbstractAction("Button") {
            @Override
            public void actionPerformed(ActionEvent e) {
                System.out.println("ButtonTest::actionPerformed: CALLED");
            }
        });

        InputMap im = button.getInputMap();
        im.put(KeyStroke.getKeyStroke("ENTER"), "pressed");
        im.put(KeyStroke.getKeyStroke("released ENTER"), "released");

        f.add(button);
        f.getRootPane().setDefaultButton(button);

        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }
}

Ответ 4

Вы можете использовать версию getKeyStroke(), которая позволяет установить onKeyRelease в true, как в этом ответ

Изменить: вы можете остановить повторы, как в этом .