JTextField在覆盖选中时自动删除字符

问题描述 投票:0回答:2

我正在使用Java,我有一个只能有4位数字的JTextField。我正在使用扩展DocumentFilter的类来过滤掉任何其他字符,并将字符数限制为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));
  }

}

除了我明确怀疑之外,任何建议都是受欢迎的,因为我是新手。

java swing user-interface windowbuilder documentfilter
2个回答
2
投票

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

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

如果JTextField已满,则其长度将是最大字符数,因此添加str的长度将始终大于limit


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);
}
© www.soinside.com 2019 - 2024. All rights reserved.