9

我的应用程序中有一个 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
        }
    });
4

2 回答 2

8

有两种方法

  • 如果完成,请DocumentListenersetText("...")添加前删除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
    }
});
于 2012-06-20T13:02:04.943 回答
4

请记住,所有事件侦听器都在 Swing 事件线程上执行。因此,事情可能不会按照您希望的顺序进行。在这种情况下,任何解决方案都将是 hack-ish,因为您无法完全控制摆动线程以及谁在其上发布事件。

我想说的是:假设您选择使用一些标志来让您的听众知道这是一个程序性更改。这是可能的情况(我假设您遵循通过摆动线程执行任何 UI 更新的良好规则invokeLater):

  1. 设置标志以跳过事件
  2. 设置文本
  3. 将标志设置为假

如果您在一次调用中完成所有操作,则 setText 将触发发布到事件队列末尾的更新事件,因此,当它们被执行时,该标志已经为假。

因此,您应该在一次invokeLater通话中执行第 1 步和第 2 步,甚至invokeAndWait,然后发布另一个事件invokeLater以取消设置标志。并祈祷您的用户不要那么快地在这两个调用之间进行一些更改,否则,它们也将被视为程序性更改。

于 2012-06-20T13:30:31.727 回答