DocumentListener 运行时如何删除 JTextField 的内容?
Posted
技术标签:
【中文标题】DocumentListener 运行时如何删除 JTextField 的内容?【英文标题】:How to remove content of JTextField while DocumentListener is running? 【发布时间】:2020-06-03 15:30:58 【问题描述】:对于以下代码,我收到 IllegalStateException(尝试在通知中变异):
private class DocumentHandler implements DocumentListener
public void changedUpdate(DocumentEvent ev)
// unused
public void insertUpdate(DocumentEvent ev)
if(textInput.getText().equals("..."))
JOptionPane.showMessageDialog(null, "...");
textInput.setText("");
为什么我不能在 DocumentListener 处于活动状态时更改 TextField?
我试图在 TextField 设置为“”时删除 DocumentListener,但这根本没有帮助。 我知道以前有人问过一个非常相似的问题,但我没有得到那个答案......
谢谢
【问题讨论】:
【参考方案1】:一般来说,您不会在使用 DocumentListener 的同时收听 Document 的状态。我知道的两种可能的解决方案:
从您的侦听器中,将进行您希望进行的更改的代码放入 Runnable 中,并通过调用SwingUtilities.invokeLater(yourRunnable)
将 Runnable 排队到 Swing 事件线程中。这是一个无耻的混蛋
更好:不要使用 DocumentListener 而是使用 DocumentFilter,因为这种类型的侦听器适用于在文本在组件中可视化之前对 Document 进行更改。
不相关的附带问题:您的代码显示出令人担忧的耦合程度,因为您尝试从侦听器中更改特定文本组件中的文本。 DocumentListeners 应该完全不知道他们收听的文档的文本组件,实际上可以添加到多个文档中。
一个 DocumentFilter 有 3 个方法需要被覆盖并做你期望它们做的事情:你期望它们做的事情:
insertString
:在文档中插入一个字符串
remove
:从文档中删除文本
replace
:替换文档中的文字
更重要的是,这些方法在文本组件呈现文档更改之前执行它们的操作。
所以在我的方法覆盖中,我提取了当前文档的文本,并使用参数来创建新文本的外观,例如我所做的替换方法:
@Override
public void replace(FilterBypass fb, int offset, int length, String text, AttributeSet attrs)
throws BadLocationException
String currentText = fb.getDocument().getText(0, fb.getDocument().getLength());
StringBuilder sb = new StringBuilder(currentText);
String newText = sb.replace(offset, offset + length, text).toString();
然后我对 newText 做一个布尔测试,看它是否“好”,如果是,则调用 super 的方法,这里是 replace(...)
,传入所有参数。如果没有,如果 newText 没有通过测试,那么我会从文档中删除所有文本并显示一个 JOptionPane。
所以在这个例子中,我用这个作为我的测试方法:
private boolean isTextOk(String text)
return !BAD_TEXTS.contains(text);
它测试文本是否是任何不允许的字符串,这里是"...", " ", "oops", "OOPS"
,但它可以是您想要的任何字符串。同样,如果文本通过文本,则调用 super 的方法,否则删除文本:
if (isTextOk(newText))
super.replace(fb, offset, length, text, attrs);
else
badText(fb);
badText(fb)
在哪里:
private void badText(FilterBypass fb) throws BadLocationException
remove(fb, 0, fb.getDocument().getLength());
JOptionPane.showMessageDialog(null, "Don't do this!", "Bad Text Entered",
JOptionPane.WARNING_MESSAGE);
整个例子是:
import java.util.Arrays;
import java.util.List;
import javax.swing.*;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.DocumentFilter;
import javax.swing.text.PlainDocument;
@SuppressWarnings("serial")
public class ClearThreeDots extends JPanel
private JTextField textField = new JTextField(40);
public ClearThreeDots()
((PlainDocument) textField.getDocument()).setDocumentFilter(new MyDocFilter());
add(textField);
private static void createAndShowGui()
ClearThreeDots mainPanel = new ClearThreeDots();
JFrame frame = new JFrame("Clear Three Dots");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.getContentPane().add(mainPanel);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
public static void main(String[] args)
SwingUtilities.invokeLater(() -> createAndShowGui());
class MyDocFilter extends DocumentFilter
private static final List<String> BAD_TEXTS = Arrays.asList("...", " ", "oops", "OOPS");
@Override
public void insertString(FilterBypass fb, int offset, String string, AttributeSet attr)
throws BadLocationException
String currentText = fb.getDocument().getText(0, fb.getDocument().getLength());
StringBuilder sb = new StringBuilder(currentText);
String newText = sb.insert(offset, string).toString();
if (isTextOk(newText))
super.insertString(fb, offset, string, attr);
else
badText(fb);
@Override
public void remove(FilterBypass fb, int offset, int length) throws BadLocationException
String currentText = fb.getDocument().getText(0, fb.getDocument().getLength());
StringBuilder sb = new StringBuilder(currentText);
String newText = sb.replace(offset, offset + length, "").toString();
if (isTextOk(newText))
super.remove(fb, offset, length);
else
badText(fb);
@Override
public void replace(FilterBypass fb, int offset, int length, String text, AttributeSet attrs)
throws BadLocationException
String currentText = fb.getDocument().getText(0, fb.getDocument().getLength());
StringBuilder sb = new StringBuilder(currentText);
String newText = sb.replace(offset, offset + length, text).toString();
if (isTextOk(newText))
super.replace(fb, offset, length, text, attrs);
else
badText(fb);
private boolean isTextOk(String text)
return !BAD_TEXTS.contains(text);
private void badText(FilterBypass fb) throws BadLocationException
remove(fb, 0, fb.getDocument().getLength());
JOptionPane.showMessageDialog(null, "Don't do this!", "Bad Text Entered",
JOptionPane.WARNING_MESSAGE);
【讨论】:
感谢您的回答!因此,我研究了整个 Internet,但找不到有关如何使用 DocumentFilter 的有用教程。你能这么好心向我解释一下吗? :))) @MoritzSteinbacher:我在这方面写了很多,你可以随时查看我的previous answers on this topic。还有this tutorial got me started in learning how to use the tool. @MoritzSteinbacher:请查看编辑以回答更多信息以上是关于DocumentListener 运行时如何删除 JTextField 的内容?的主要内容,如果未能解决你的问题,请参考以下文章
如何知道啥 JTextField 进入 DocumentListener
如何从 Document Listener 更新 JComboBox 的列表?
使用 DocumentListener 实现检查 JTextField