对 JButtons 的关注在 Java 中是如何工作的?

How does focus on JButtons actually work in Java?

我在 Java Swing 中发现了一个奇怪的异常现象。 按时间顺序添加到 UI 的第一个 JButton 总是在用户按下 space 栏时触发,假设他在此之前没有单击另一个按钮。如果调用 getRootPane().setDefaultButton(JButton)JButton.requestFocus(),甚至会发生此行为。 当请求关注 JButton 时,似乎至少有两种不同的“焦点”。 “焦点”或突出显示之一是按钮上文本周围的虚线矩形,而另一个是指定按钮周围较粗的轮廓。

只要按下 space 栏,带有虚线轮廓文本的按钮就会触发。 每当按下回车键时,带有粗边框的按钮就会触发。

我准备了一个可编译的最小示例来说明这种行为。根本没有涉及关键mapping/binding。

import java.awt.Container;
import java.awt.Dimension;
import java.awt.Insets;
import java.awt.Toolkit;
import java.awt.event.ActionEvent;
import javax.swing.JButton;
import javax.swing.JDialog;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.WindowConstants;

public class ButtonFocusAnomalyExample extends JFrame {
    public ButtonFocusAnomalyExample() {
        super();
        setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
        int frameWidth = 300;
        int frameHeight = 300;
        setSize(frameWidth, frameHeight);
        Dimension d = Toolkit.getDefaultToolkit().getScreenSize();
        int x = (d.width - getSize().width) / 2;
        int y = (d.height - getSize().height) / 2;
        setLocation(x, y);
        setTitle("Any Frame");
        setResizable(false);
        Container cp = getContentPane();
        cp.setLayout(null);
        setVisible(true);
        new DialogMinimal(this, true); // Runs the Dialog
    }

    public static void main(String[] args) {
        new ButtonFocusAnomalyExample();
    }

    static class DialogMinimal extends JDialog {
        private final JTextField output = new JTextField();

        public DialogMinimal(final JFrame owner, final boolean modal) {
            super(owner, modal);
            setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
            int frameWidth = 252;
            int frameHeight = 126;
            setSize(frameWidth, frameHeight);
            Dimension d = Toolkit.getDefaultToolkit().getScreenSize();
            int x = (d.width - getSize().width) / 2;
            int y = (d.height - getSize().height) / 2;
            setLocation(x, y);
            setTitle("Minimal Button Focus Example");
            Container cp = getContentPane();
            cp.setLayout(null);
            JButton bYes = new JButton();
            bYes.setBounds(0, 0, 100, 33);
            bYes.setText("Yes (Space)");
            bYes.addActionListener(this::bYes_ActionPerformed);
            JPanel buttonPanel = new JPanel(null, true);
            buttonPanel.add(bYes);
            JButton bNo = new JButton();
            bNo.setBounds(108, 0, 120, 33);
            bNo.setText("No (Enter/Return)");
            getRootPane().setDefaultButton(bNo); // Set "No" as default button
            bNo.requestFocus(); // Get focus on "No" button
            bNo.addActionListener(this::bNo_ActionPerformed);
            buttonPanel.add(bNo);
            buttonPanel.setBounds(8, 8, 400, 92);
            buttonPanel.setOpaque(false);
            cp.add(buttonPanel);
            output.setBounds(8, 50, 220, 32);
            cp.add(output);
            setResizable(false);
            setVisible(true);
        }

        public void bYes_ActionPerformed(final ActionEvent evt) {
            output.setText("Yes"); // Still fires on every space bar press
        }

        public void bNo_ActionPerformed(final ActionEvent evt) {
            output.setText("No"); // Only fires on every return/enter press
        }
    }
}

这是它的样子:

还可以找到可执行代码here

我现在的问题是:

  1. 这些不同的重点是什么?
  2. 如何更改按钮文本周围显示为虚线轮廓的焦点,以便 space 栏 回车键会触发事件“否”按钮?

关于问题 1:

没有2种“焦点”。这两种方法都按照各自的名称进行操作:

JButton.requestFocus()(更好的是 JButton.requestFocusInWindow())请求 focus 按钮,而 getRootPane().setDefaultButton(JButton) 设置 selected 按钮,LAF 单独处理。


关于问题 2:

对话框的模式是问题所在。

因此,可能的解决方案是

  1. 创建对话框时将模式设置为 false,例如使用 new DialogMinimal(this, false); 并通过调用 bNo.requestFocusInWindow() 而不是 getRootPane().setDefaultButton(bNo); and/or bNo.requestFocus(); 来获得焦点,但是如果对话框必须是模态的,这不是解决方案。
  2. 按照用户 camickr 的建议实施 Dialog Focus 中的 RequestFocusListener
public DialogMinimal(final JFrame owner, final boolean modal) {
    Button bNo = new JButton();
    [...]
    // bNo.requestFocusInWindow(); // obsolete now
    getRootPane().setDefaultButton(bNo); // To fire on enter key
    bNo.addAncestorListener(new RequestFocusListener()); // To fire on space bar
    [...]
}