JTextField在覆盖所选内容时自动删除字符

时间:2019-03-26 17:45:29

标签: java swing user-interface windowbuilder documentfilter

我正在使用Java,并且我有一个JTextField,它只能有4位数字。我正在使用扩展DocumentFilter的类来过滤掉其他任何字符,并将字符数限制为4。

问题是,一旦我有4位数字,如果我选择了所有4位数字并尝试通过键入另一位数字来覆盖它们,它不会自动覆盖并且什么也不做,我必须明确地输入键盘上的“ Backspace”或“ Delete”删除4位数字,然后(一旦清除该字段)我可以再次键入。

如何让JTextField充当操作系统的其余部分,如果我键入一个字符,一旦选择了一些文本,它将“删除所有字符,然后写入该字符”(它替换内容)。

我有一个辅助类,JustLimitDigitFilter.java

import javax.swing.text.DocumentFilter;
import javax.swing.text.BadLocationException;

import java.awt.Toolkit;

import javax.swing.text.AttributeSet;

public class JustLimitDigitFilter extends DocumentFilter {

  int limit;

  public JustLimitDigitFilter(int limit) {
    this.limit = limit;
  }

  @Override
  public void insertString(DocumentFilter.FilterBypass fb, int offset, String text, AttributeSet attr) throws BadLocationException {

    // if (text == null) {
    //   return;
    // }
    String str = text.replaceAll("\\D", "");
    if (!str.isEmpty() && (fb.getDocument().getLength() + str.length()) <= limit) {
      super.insertString(fb, offset, str, attr);
    } else {
      Toolkit.getDefaultToolkit().beep();
    }

  }

  @Override
  public void replace(DocumentFilter.FilterBypass fb, int offset, int length, String text, AttributeSet attr)
      throws BadLocationException {

    // if (text == null) {
    //   return;
    // }
    String str = text.replaceAll("\\D", "");
    if (!str.isEmpty() && (fb.getDocument().getLength() + str.length()) <= limit) {
      super.replace(fb, offset, length, str, attr);
    } else {
      Toolkit.getDefaultToolkit().beep();
    }

  }

}

主要类别App.java

import java.awt.EventQueue;

import javax.swing.JFrame;
import javax.swing.JPanel;
import java.awt.BorderLayout;
import javax.swing.JTextField;
import javax.swing.text.AbstractDocument;
import javax.swing.text.DocumentFilter;

public class App {

  private JFrame frame;
  private JTextField textField;

  /**
   * Launch the application.
   */
  public static void main(String[] args) {
    EventQueue.invokeLater(new Runnable() {
      public void run() {
        try {
          App window = new App();
          window.frame.setVisible(true);
        } catch (Exception e) {
          e.printStackTrace();
        }
      }
    });
  }

  /**
   * Create the application.
   */
  public App() {
    initialize();
  }

  /**
   * Initialize the contents of the frame.
   */
  private void initialize() {
    frame = new JFrame();
    frame.setBounds(100, 100, 191, 96);
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.getContentPane().setLayout(null);

    JPanel panel = new JPanel();
    panel.setBounds(6, 6, 179, 62);
    frame.getContentPane().add(panel);
    panel.setLayout(null);

    textField = new JTextField();
    textField.setBounds(6, 6, 167, 26);
    panel.add(textField);
    textField.setColumns(10);

    // without this code below this, the textfield is “normal” when
    // something is selected if I write it overwrites the selection

    AbstractDocument doc = (AbstractDocument) textField.getDocument();
    doc.setDocumentFilter(new JustLimitDigitFilter(4));
  }

}

除了我明确的疑问外,任何建议都值得欢迎,因为我是新手。

2 个答案:

答案 0 :(得分:2)

replace()中的方法DocumentFilter实际上执行两个操作。首先,它从length开始删除offset个字符,然后在text插入offset。因此,当replace()包含允许的最大字符数时,JTextField方法中的以下行什么也不会发生...

if (!str.isEmpty() && (fb.getDocument().getLength() + str.length()) <= limit) {

如果JTextField已满,则其长度将为最大字符数,因此加上str的长度将始终大于{{1} }。

答案 1 :(得分:1)

扩大Abra的答案:

replace(...)方法中的“ length”参数包含将要删除的字符数。

因此您可以将if语句更改为:

//if (!str.isEmpty() && (fb.getDocument().getLength() + str.length()) <= limit)
if (!str.isEmpty() && (fb.getDocument().getLength() + str.length() - length) <= limit)

更改之后,您可以将insert(...)方法简化为:

@Override
public void insertString(FilterBypass fb, int offset, String text, AttributeSet attributes)
    throws BadLocationException
{
    replace(fb, offset, 0, text, attributes);
}