对 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。
我现在的问题是:
- 这些不同的重点是什么?
- 如何更改按钮文本周围显示为虚线轮廓的焦点,以便 space 栏 和 回车键会触发事件“否”按钮?
关于问题 1:
没有2种“焦点”。这两种方法都按照各自的名称进行操作:
JButton.requestFocus()
(更好的是 JButton.requestFocusInWindow()
)请求 focus 按钮,而 getRootPane().setDefaultButton(JButton)
设置 selected 按钮,LAF 单独处理。
关于问题 2:
对话框的模式是问题所在。
因此,可能的解决方案是
- 创建对话框时将模式设置为 false,例如使用
new DialogMinimal(this, false);
并通过调用 bNo.requestFocusInWindow()
而不是 getRootPane().setDefaultButton(bNo);
and/or bNo.requestFocus();
来获得焦点,但是如果对话框必须是模态的,这不是解决方案。
- 按照用户 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
[...]
}
我在 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。
我现在的问题是:
- 这些不同的重点是什么?
- 如何更改按钮文本周围显示为虚线轮廓的焦点,以便 space 栏 和 回车键会触发事件“否”按钮?
关于问题 1:
没有2种“焦点”。这两种方法都按照各自的名称进行操作:
JButton.requestFocus()
(更好的是 JButton.requestFocusInWindow()
)请求 focus 按钮,而 getRootPane().setDefaultButton(JButton)
设置 selected 按钮,LAF 单独处理。
关于问题 2:
对话框的模式是问题所在。
因此,可能的解决方案是
- 创建对话框时将模式设置为 false,例如使用
new DialogMinimal(this, false);
并通过调用bNo.requestFocusInWindow()
而不是getRootPane().setDefaultButton(bNo);
and/orbNo.requestFocus();
来获得焦点,但是如果对话框必须是模态的,这不是解决方案。 - 按照用户 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
[...]
}