如何区分textField.setText()并在java中手动将文本添加到textField?

时间:2012-06-20 12:53:24

标签: java swing jtextfield documentlistener

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

2 个答案:

答案 0 :(得分:8)

有两种方式

  • 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
    }
});

答案 1 :(得分:4)

请记住,所有事件侦听器都在Swing Event Thread上执行。所以事情可能不会按照你想要的顺序排列。在这种情况下,任何解决方案都将是hack-ish,因为你无法完全控制swing线程以及谁在其上发布事件。

我想说的是:假设你选择使用一些旗帜让听众知道这是一个程序化的改变。这是可能的场景(我假设您遵循通过invokeLater从swing线程执行任何UI更新的好规则):

  1. 将标志设置为跳过事件
  2. 的setText
  3. 将旗帜设为假
  4. 如果你在一次调用中完成所有操作,setText将触发发布到事件队列末尾的更新事件,因此,当它们被执行时,标志将已经为假。

    因此,您应该在一次invokeLater调用中执行第1步和第2步,甚至是invokeAndWait,然后使用invokeLater发布另一个事件以取消设置该标记。并且祈祷你的用户没有那么快在这两次调用之间做出一些改变,否则,他们也被认为是程序化的改变。