4

我正在尝试获得一个带有单个按钮的简单 JFrame,以在发生任何这些事件时触发事件:

  • 按下 Enter 键并且 JButton 具有焦点
  • 按下空格键并且 JButton 具有焦点
  • 单击 JButton。

似乎 Enter 和 Spacebar 是“免费”的,以及使用 JButton 上的 addActionListener 的默认鼠标点击;麻烦的是,我读过键绑定取决于使用的外观。

我试图通过在 JButton 的动作映射中添加 Enter 和空格键来获得跨 LaF 的通用行为,甚至添加了一个随机键(“m”)以确保 ActionMap 正在完成工作(它是),但现在鼠标点击丢失。我似乎能够获得所有键和鼠标点击的唯一方法是同时使用动作映射和 addActionListener。

有没有办法让这些键和鼠标绑定在所有 LaF 中一致地工作,而无需尝试检测可能出现的所有可能的 LaF?我可以注册一个会同时触发键和鼠标事件的动作监听器吗?

我最喜欢的解决方案是将鼠标单击添加到 JButton 动作映射并检测在动作内部发生的键或鼠标单击。

我仍然在这里学习绳索,所以这可能不是最好或最有效的做事方式;我敢肯定它是过度设计的。这是一种训练练习,我正在尝试所有我能得到的东西。欢迎任何和所有编码风格的评论!

import java.awt.EventQueue;
import java.awt.event.ActionEvent;
import java.awt.event.KeyEvent;

import javax.swing.AbstractAction;
import javax.swing.Action;
import javax.swing.InputMap;
import javax.swing.JButton;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.KeyStroke;
import javax.swing.UIManager;

public class Example extends JFrame {

// ============================
private class BtnListener extends AbstractAction {
    private static final long serialVersionUID = 1L;

    public void actionPerformed(ActionEvent ae) {
        System.out.println("\nclick button listener triggered");
        System.out.println(ae.getSource().getClass().toString());
    }
} // class BtnListener

private static final int NO_MODIFIER = 0;
private static final boolean ON_KEY_PRESS = false;
private static final KeyStroke ENTER_PRESSED = KeyStroke.getKeyStroke(
        KeyEvent.VK_ENTER, NO_MODIFIER, ON_KEY_PRESS);
private static final KeyStroke M_PRESSED = KeyStroke.getKeyStroke(
        KeyEvent.VK_M, NO_MODIFIER, ON_KEY_PRESS);
private static final KeyStroke SPACEBAR_PRESSED = KeyStroke.getKeyStroke(
        KeyEvent.VK_SPACE, NO_MODIFIER, ON_KEY_PRESS);
private JButton btnButton;
private final AbstractAction btnListener = new BtnListener();
private JPanel buttonPanel;
private JFrame frmMain;

public static void main(String[] args) {
    Example ex = new Example();
    ex.displayFrame();
}

Action btnActionListener = new AbstractAction() {
    private static final long serialVersionUID = 1L;

    public void actionPerformed(ActionEvent e) {
        System.out.println("\nkey button action triggerred");
        System.out.println(e.getSource().getClass().toString());
        if (e.getSource() instanceof JButton) {
            System.out.println("button");
        } else {
            System.out.println("Something else");
        }
    }
};

public Example() {
    initialize();
}

public void displayFrame() {
    try {
        UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
    } catch (Throwable e) {
        e.printStackTrace();
    }
    EventQueue.invokeLater(new Runnable() {
        public void run() {
            try {
                frmMain.setVisible(true);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    });
}

private void initialize() {

    frmMain = new JFrame();
    btnButton = new JButton("Abutton");

    // Comment this out, you lose the mouse click
    btnButton.addActionListener(btnListener);

    // Comment out ActionMaps, but keep addActionListner (above), and
            // only lose M_PRESSED
    InputMap buttonFocusedMap = btnButton
            .getInputMap(JComponent.WHEN_FOCUSED);

    buttonFocusedMap.put(ENTER_PRESSED, "blah");
    btnButton.getActionMap().put("blah", btnActionListener);

    buttonFocusedMap.put(SPACEBAR_PRESSED, "blort");
    btnButton.getActionMap().put("blort", btnActionListener);

    buttonFocusedMap.put(M_PRESSED, "gaaak");
    btnButton.getActionMap().put("gaaak", btnActionListener);

    // Is there a way to add a mouse click to the ActionMap?

    buttonPanel = new JPanel();
    buttonPanel.add(btnButton);

    frmMain.getContentPane().add(buttonPanel);
    frmMain.setBounds(100, 100, 500, 432);
    frmMain.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
}

}
4

2 回答 2

9

BasicButtonListener,由 使用BasicButtonUI,确保所有按钮(检查、单选、切换)Space在聚焦时都绑定到。这适用于跨平台,即使单个外观可能会唯一地呈现各种按钮模型状态。按压Space唤起被按压 UIAction,释放Space唤起被释放 UIAction。当在按钮范围内按下和释放鼠标时也会发生同样的情况;按住时在按钮外部拖动以查看武装状态变化。

在任何一种情况下,按下释放的组合都会调用按钮的actionPerformed()方法。

Enter无论焦点如何,绑定到 的一种便捷方法Action是通过根窗格的setDefaultButton()方法。此示例说明了单击按钮的所有三种方式。

于 2012-05-23T01:08:38.233 回答
0

您可以(并且可能应该)做的是创建一个 AbstractAction。您可以将 AbstractAction 用于您的 ActionMap 以及单击(执行 JButton.setAction(...))

于 2012-05-22T22:00:35.970 回答