我有一个GUI应用程序,它使用InputVerifier在产生焦点之前检查文本字段的内容。这一切都很正常。然而,昨天发现了一个问题 - 它似乎是一个错误,但我无法在任何地方找到它。在我将此报告为错误之前,我想我会问:我在这里遗漏了一些明显的东西吗?
情况:
如果我尝试使用鼠标或使用tab键离开此控件,我不能。 FocusLost事件不会触发,控件也会正确保留焦点。
但是,如果我尝试以反向标签顺序离开控件,使用Shift-Tab ,有时会触发FocusLost事件。如果发生这种情况,单独的线程报告没有控件具有焦点,即getFocusOwner()返回null。
<小时/> 编辑:下面是一个显示问题的小示例程序。这个问题与额外的线程无关 - 线程只是为了让问题更加明显。如果有竞争条件,它就在Swing的某个地方。
要查看问题,请转到第二个文本框并清空它。控件应该保留焦点,并且除非按下shift-tab键离开它。与完整的应用程序不同,错误似乎在100%的时间发生在这里。在OpenJDK 6和Oracle Java 7下都是如此。
这几乎是显而易见的错误,而且它发生在多个Java环境中。因此,我怀疑我错过了一些明显的东西。任何人吗?
public class FocusBugDemo extends JFrame {
static JTextField txtOne = new JTextField("Ignore this control");
static JTextField txtTwo = new JTextField("Delete this text, then press shift-tab");
static JLabel lblFocus = new JLabel("");
static KeyboardFocusManager kfm = new DefaultKeyboardFocusManager();
public static void main(String[] args) {
new FocusBugDemo();
Thread t = new Thread() {
@Override
public void run() {
while(true) {
Component c = kfm.getFocusOwner();
String focusInfo = "elsewhere";
if (c == null) { focusInfo = "null";
} else if (c == txtOne) { focusInfo = "txtOne";
} else if (c == txtTwo) { focusInfo = "txtTwo";
}
lblFocus.setText(System.currentTimeMillis() + " - Focus owner " + focusInfo);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
}
}
}
};
t.start();
}
private FocusBugDemo() {
super("Focus bug demo");
setDefaultCloseOperation(EXIT_ON_CLOSE);
setPreferredSize(new Dimension(300,100));
setLayout(new GridLayout(3,1));
NotEmpty validator = new NotEmpty();
txtOne.setInputVerifier(validator);
txtTwo.setInputVerifier(validator);
add(txtOne);
add(txtTwo);
add(lblFocus);
pack();
setVisible(true);
}
private class NotEmpty extends InputVerifier {
@Override
public boolean verify(JComponent input) {
JTextField txtField = (JTextField) input;
return (txtField.getText().length() > 0);
}
}
}
答案 0 :(得分:2)
现在向Oracle报告了错误7167871。
答案 1 :(得分:1)
使用你的sscce,我无法再现你在Mac OS X,Java 6上描述的效果,它支持@ CatalinaIsland的观察。特别是,焦点永远不会使用 tab 或 shift-tab 留下空文本字段;只有在停用框架时,焦点才会变为null
。
我看到两个线程访问多个字段而根本没有同步。至少,您应该使用EventQueue.invokeLater()
中的t
来更新GUI,如Concurrency in Swing中所述,并在下面显示。
更广泛的问题是:您尝试使用t
解决哪些焦点问题?
import java.awt.Component;
import java.awt.DefaultKeyboardFocusManager;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.GridLayout;
import java.awt.KeyboardFocusManager;
import javax.swing.InputVerifier;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JTextField;
public class FocusDemo {
private static final JTextField txtOne =
new JTextField("Ignore this control");
private static final JTextField txtTwo =
new JTextField("Delete this text, then press shift-tab");
private static final JLabel lblFocus = new JLabel("");
public static void main(String[] args) {
new FocusDemo();
Thread t = new Thread() {
@Override
public void run() {
while (true) {
EventQueue.invokeLater(new Runnable() {
KeyboardFocusManager kfm =
new DefaultKeyboardFocusManager();
@Override
public void run() {
Component c = kfm.getFocusOwner();
String focusInfo = "elsewhere";
if (c == null) {
focusInfo = "null";
} else if (c == txtOne) {
focusInfo = "txtOne";
} else if (c == txtTwo) {
focusInfo = "txtTwo";
}
lblFocus.setText(System.currentTimeMillis()
+ " - Focus owner " + focusInfo);
}
});
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace(System.err);
}
}
}
};
t.start();
}
private FocusDemo() {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
JFrame f = new JFrame("Focus bug demo");
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.setPreferredSize(new Dimension(300, 100));
f.setLayout(new GridLayout(3, 1));
NotEmpty validator = new NotEmpty();
txtOne.setInputVerifier(validator);
txtTwo.setInputVerifier(validator);
f.add(txtOne);
f.add(txtTwo);
f.add(lblFocus);
f.pack();
f.setVisible(true);
}
});
}
private class NotEmpty extends InputVerifier {
@Override
public boolean verify(JComponent input) {
JTextField txtField = (JTextField) input;
return (txtField.getText().length() > 0);
}
}
}