我的应用程序中有一个textField,当用户单击JList中的项目时,它将以编程方式启动(textField.setText())。稍后用户将手动更改此值。我坚持使用文档侦听器来检测此文本字段中的更改。当以编程方式发生更改时,它必须不执行任何操作,但如果手动发生,它应该将背景更改为红色。
如何检测textField是手动填写还是由textField.setText()填写?
txtMode.getDocument().addDocumentListener(new DocumentListener() {
public void insertUpdate(DocumentEvent e) {
if (!mode.equals(e.getDocument()))
txtMode.setBackground(Color.red);
}
public void removeUpdate(DocumentEvent e) {
if (mode.equals(e.getDocument()))
txtMode.setBackground(Color.white);
}
public void changedUpdate(DocumentEvent e) {
//To change body of implemented methods
}
});
方法
- 删除
DocumentListener
setText("...")
如果完成后重新添加DocumentListener
法典
public void attachDocumentListener(JComponent compo){
compo.addDocumentListener(someDocumentListener);
}
//similair void for remove....
- 使用
boolean
值禁用"如果需要",但您必须更改DocumentListener
例如
txtMode.getDocument().addDocumentListener(new DocumentListener() {
public void insertUpdate(DocumentEvent e) {
if (!mode.equals(e.getDocument()))
if (!updateFromModel){
txtMode.setBackground(Color.red);
}
}
public void removeUpdate(DocumentEvent e) {
if (mode.equals(e.getDocument()))
if (!updateFromModel){
txtMode.setBackground(Color.white);
}
}
public void changedUpdate(DocumentEvent e) {
//To change body of implemented methods
}
});
请记住,所有事件侦听器都在 Swing 事件线程上执行。因此,事情可能不会按照您希望的确切顺序进行。在这种情况下,任何解决方案都将是黑客式的,因为您无法完全控制摆动线程以及谁在其上发布事件。
我想说的是:假设你选择使用某个标志来让你的听众知道这是一个程序化的变化。这是可能的情况(我假设您遵循通过 invokeLater
从 swing 线程进行任何 UI 更新的好规则):
- 设置标志以跳过事件
- 设置文本
- 将标志设置为假
如果您在一次调用中完成所有这些操作,setText 将触发发布到事件队列末尾的更新事件,因此,在执行它们时,该标志已经为 false。
因此,您应该在一个invokeLater
调用中执行步骤1和2,甚至invokeAndWait
,然后发布另一个带有invokeLater
的事件以取消设置标志。并祈祷您的用户不要太快,以至于在这两个调用之间进行一些更改,否则,它们也会被视为编程更改。