Java 为什么DocumentFilter没有给出预期的结果?

Java 为什么DocumentFilter没有给出预期的结果?,java,swing,jtextfield,documentfilter,Java,Swing,Jtextfield,Documentfilter,我想这一定是代码中的一个简单错误或我的误解,但我无法使用DocumentFilter来检测insertString事件。下面是一个简单的大写字母过滤器,但它不像insertString(..)方法似乎从未被调用那样重要 为什么没有调用DocumentFilter的insertString(..)方法 过滤器应用于顶部的JTextField。每次调用insertString(..)时,它都应该将信息附加到中心的JTextArea。目前,文本字段中没有导致文本附加到文本区域的操作 import ja

我想这一定是代码中的一个简单错误或我的误解,但我无法使用
DocumentFilter
来检测
insertString
事件。下面是一个简单的大写字母过滤器,但它不像
insertString(..)
方法似乎从未被调用那样重要

为什么没有调用
DocumentFilter
insertString(..)
方法

过滤器应用于顶部的
JTextField
。每次调用
insertString(..)
时,它都应该将信息附加到
中心的
JTextArea
。目前,文本字段中没有导致文本附加到文本区域的操作

import java.awt.*;
import javax.swing.*;
import javax.swing.border.EmptyBorder;
import javax.swing.text.*;

public class FilterUpperCaseLetters {

    private JComponent ui = null;
    private final JTextField textField = new JTextField(25);
    private final JTextArea textArea = new JTextArea(5, 20);

    FilterUpperCaseLetters() {
        initUI();
    }

    public void initUI() {
        // The document filter that seems to do nothing.
        DocumentFilter capsFilter = new DocumentFilter() {
            @Override
            public void insertString(
                    DocumentFilter.FilterBypass fb,
                    int offset,
                    String string,
                    AttributeSet attr) throws BadLocationException {
                textArea.append("insertString! " + string + "\n");
                if (!string.toUpperCase().equals(string)) {
                    textArea.append("Insert!\n");
                    super.insertString(fb, offset, string, attr);
                } else {
                    textArea.append("DON'T insert!\n");
                }
            }
        };
        AbstractDocument abstractDocument
                = (AbstractDocument) textField.getDocument();
        abstractDocument.setDocumentFilter(capsFilter);

        ui = new JPanel(new BorderLayout(4, 4));
        ui.setBorder(new EmptyBorder(4, 4, 4, 4));

        ui.add(textField, BorderLayout.PAGE_START);
        ui.add(new JScrollPane(textArea), BorderLayout.CENTER);
    }

    public JComponent getUI() {
        return ui;
    }

    public static void main(String[] args) {
        Runnable r = new Runnable() {
            @Override
            public void run() {
                FilterUpperCaseLetters o = new FilterUpperCaseLetters();

                JFrame f = new JFrame(o.getClass().getSimpleName());
                f.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
                f.setLocationByPlatform(true);

                f.setContentPane(o.getUI());
                f.pack();
                f.setMinimumSize(f.getSize());

                f.setVisible(true);
            }
        };
        SwingUtilities.invokeLater(r);
    }
}


文本组件使用
replaceSelection(…)
方法,该方法将依次调用
AbstractDocument的
replace(…)
方法,该方法将调用
DocumentFilter的
replace(…)
方法

只有在使用
Document.insertString(…)
方法直接更新
文档时,才会调用
DocumentFilter
insertString(…)
方法

因此,实际上您需要重写这两种方法,以确保完成大写转换

一个简单的示例演示了如何轻松实现这两种方法:

import java.awt.*;
import javax.swing.*;
import javax.swing.text.*;

public class UpperCaseFilter extends DocumentFilter
{
    public void insertString(FilterBypass fb, int offs, String str, AttributeSet a)
        throws BadLocationException
    {
        replace(fb, offs, 0, str, a);
    }

    public void replace(FilterBypass fb, final int offs, final int length, final String text, final AttributeSet a)
        throws BadLocationException
    {
        if (text != null)
        {
            super.replace(fb, offs, length, text.toUpperCase(), a);
        }
    }

    private static void createAndShowGUI()
    {
        JTextField textField = new JTextField(10);
        AbstractDocument doc = (AbstractDocument) textField.getDocument();
        doc.setDocumentFilter( new UpperCaseFilter() );

        JFrame frame = new JFrame("Upper Case Filter");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.setLayout( new java.awt.GridBagLayout() );
        frame.add( textField );
        frame.setSize(220, 200);
        frame.setLocationByPlatform( true );
        frame.setVisible( true );
    }

    public static void main(String[] args) throws Exception
    {
        EventQueue.invokeLater( () -> createAndShowGUI() );
    }
}

“为什么不调用
insertString
?”为什么应该调用它?您是否尝试过查看方法的调用顺序<代码>插入字符串
/
删除
/
替换
并查看何时调用,可能尝试查看是否可以确定需要在哪些条件下应用过滤器?还需要覆盖
替换
:-)