7

这是我的代码:

import javax.swing.*;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import javax.swing.text.Document;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

public class Frame extends JFrame {

    private JTextField txt1 = new JTextField(10);
    private JTextField txt2 = new JTextField(10);
    private JButton btn = new JButton("Set Text");

    public Frame() {
        super("Latihan");
        setLayout(new FlowLayout());
        btn.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent e) {
                txt1.setText("TEST"); txt2.setText("TEST2");
            }
        });

        txt1.getDocument().addDocumentListener(new TheDocumentListener("txt1"));
        txt2.getDocument().addDocumentListener(new TheDocumentListener("txt2"));

        add(txt1);
        add(txt2);
        add(btn);
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        pack();
        setVisible(true);
    }

    public static void main (String[] args) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                new Frame();
            }
        });
    }
}

class TheDocumentListener implements DocumentListener {

    private String source;

    public TheDocumentListener(String source) {
        this.source = source;
    }
    @Override
    public void insertUpdate(DocumentEvent e) {
        System.out.println("insertUpdate from " + source);
    }

    @Override
    public void removeUpdate(DocumentEvent e) {
        System.out.println("removeUpdate from " + source);
    }

    @Override
    public void changedUpdate(DocumentEvent e) {
        System.out.println("changedUpdate from " + source);
    }
}

当我第一次单击 JButton 时,只会insertUpdate()调用:

insertUpdate from txt1
insertUpdate from txt2

但是如果我再次单击该按钮,removeUpdate()将在之前调用insertUpdate()

removeUpdate from txt1
insertUpdate from txt1
removeUpdate from txt2
insertUpdate from txt2

这是预期的行为还是我的代码有问题?

我可以制作insertUpdate执行时调用的唯一方法JTextField.setText吗?我想确保removeUpdate仅在用户删除文本字段中的文本时才被调用。怎么做?

4

1 回答 1

5

这是字符串替换的预期行为。实际上所做的是setText()删除整个字符串并设置一个新字符串。这里是实现JTextField.setText()

public void setText(String t) {
    try {
        Document doc = getDocument();
        if (doc instanceof AbstractDocument) {
            ((AbstractDocument)doc).replace(0, doc.getLength(), t,null);
        }
        else {
            doc.remove(0, doc.getLength());
            doc.insertString(0, t, null);
        }
    } catch (BadLocationException e) {
    UIManager.getLookAndFeel().provideErrorFeedback(JTextComponent.this);
    }
}

如您所见,AbstractDocument.replace()AbstractDocument文档执行。否则,remove()insert()被执行。

AbstractDocument.replace()文档:

删除从偏移到偏移+长度的文本区域,并将其替换为文本。这取决于实现如何实现,一些实现可能将其视为两个不同的操作:删除后跟插入,其他可能将替换视为一个原子操作。

所以这取决于文档的实现。PlainDocument例如继承AbstractDocument. APlainDocument是文本字段的默认文档。

如果需要,您可以随时创建自己的文档实现,或者安装文档过滤器。有关详细信息,请参阅使用文本组件教程。但不确定,您试图实现的这种行为改变背后的原因是什么。

于 2013-03-05T00:41:17.023 回答