为什么 DocumentFilter 没有给出预期的结果?
Why does the DocumentFilter not give the intended result?
我认为这一定是代码中的一个简单错误或我的误解,但我无法获得 DocumentFilter
来检测 insertString
事件。下面是一个简单的大写字母过滤器,但这并不重要,因为 insertString(..)
方法似乎从未被调用过!
为什么 DocumentFilter
的 insertString(..)
方法没有被调用?
过滤器应用于顶部的 JTextField
。每次调用 insertString(..)
时,它应该将信息附加到 CENTER
中的 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(...)
方法,该方法将调用 [=15] 的 replace(...)
方法=].
DocumentFilter
的insertString(...)
方法只在使用Document.insertString(...)
方法直接更新Document
时调用。
所以实际上你需要重写这两个方法以确保完成大写转换。
一个展示如何轻松实现这两种方法的简单示例:
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() );
}
}
我认为这一定是代码中的一个简单错误或我的误解,但我无法获得 DocumentFilter
来检测 insertString
事件。下面是一个简单的大写字母过滤器,但这并不重要,因为 insertString(..)
方法似乎从未被调用过!
为什么 DocumentFilter
的 insertString(..)
方法没有被调用?
过滤器应用于顶部的 JTextField
。每次调用 insertString(..)
时,它应该将信息附加到 CENTER
中的 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(...)
方法,该方法将调用 [=15] 的 replace(...)
方法=].
DocumentFilter
的insertString(...)
方法只在使用Document.insertString(...)
方法直接更新Document
时调用。
所以实际上你需要重写这两个方法以确保完成大写转换。
一个展示如何轻松实现这两种方法的简单示例:
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() );
}
}