使 JFormattedTextField 的行为类似于 ATM 输入
Posted
技术标签:
【中文标题】使 JFormattedTextField 的行为类似于 ATM 输入【英文标题】:Make a JFormattedTextField behave like ATM input 【发布时间】:2013-06-12 20:29:57 【问题描述】:我想知道是否有办法让 JformattedTextField 或 jtextField 表现得像 atm 货币输入。我的意思是你从右到左输入,说你输入 10 你需要再按 2 个 0 ,这样它就会是 10.00 。程序在他从右到左键入时自动输入小数点?如果没有输入 2 0,它只会是 .10 。这可能吗?如果我想使用该字符串进行计算,那将如何返回给我?我尝试了抽象格式化程序,但这并不能很好地工作。我想用它来输入客户收到的金额。但是让它证明是白痴。
【问题讨论】:
我认为您可以将侦听器附加到文本字段并执行您想做的任何功能。 【参考方案1】:这会强制用户始终从右侧输入文本,无论插入符号位于何处。插入新字符时,所有先前的字符都向左移动。将根据您的格式化程序应用格式化:
import java.awt.*;
import java.text.*;
import javax.swing.*;
import javax.swing.text.*;
public class ABMTextField extends JTextField
private DecimalFormat format;
private String decimal;
public ABMTextField(DecimalFormat format)
this.format = format;
decimal = Character.toString( format.getDecimalFormatSymbols().getDecimalSeparator() );
setColumns( format.toPattern().length() );
setHorizontalAlignment(JFormattedTextField.TRAILING);
setText( format.format(0.0) );
AbstractDocument doc = (AbstractDocument)getDocument();
doc.setDocumentFilter( new ABMFilter() );
@Override
public void setText(String text)
Number number = format.parse(text, new ParsePosition(0));
if (number != null)
super.setText( text );
public class ABMFilter 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, int offs, int length, String str, AttributeSet a)
throws BadLocationException
if ("0123456789".contains(str))
Document doc = fb.getDocument();
StringBuilder sb = new StringBuilder( doc.getText(0, doc.getLength()) );
int decimalOffset = sb.indexOf( decimal );
if (decimalOffset != -1)
sb.deleteCharAt(decimalOffset);
sb.insert(decimalOffset + 1, decimal);
sb.append(str);
try
String text = format.format( format.parse( sb.toString() ) );
super.replace(fb, 0, doc.getLength(), text, a);
catch(ParseException e)
else
Toolkit.getDefaultToolkit().beep();
public void remove(DocumentFilter.FilterBypass fb, int offset, int length)
throws BadLocationException
Document doc = fb.getDocument();
StringBuilder sb = new StringBuilder( doc.getText(0, doc.getLength()) );
int decimalOffset = sb.indexOf( decimal );
if (decimalOffset != -1)
sb.deleteCharAt(decimalOffset);
sb.insert(decimalOffset - 1, decimal);
sb.deleteCharAt( sb.length() - 1) ;
try
String text = format.format( format.parse( sb.toString() ) );
super.replace(fb, 0, doc.getLength(), text, null);
catch(ParseException e)
private static void createAndShowUI()
DecimalFormat format = new DecimalFormat("###,##0.00");
ABMTextField abm = new ABMTextField( format );
JPanel panel = new JPanel();
panel.add( abm );
JFrame frame = new JFrame("ABMTextField");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add( panel );
frame.setSize(200, 200);
frame.setLocationByPlatform( true );
frame.setVisible( true );
public static void main(String[] args)
EventQueue.invokeLater(new Runnable()
public void run()
createAndShowUI();
);
如果我想使用该字符串进行计算,那将如何返回给我?
您需要创建一个方法,也许是 getValue(),它会使用 format.parse(...) 方法返回一个实际数字。
【讨论】:
非常感谢,这正是我想要的。【参考方案2】:看看How to use Formatted Text Fields,尤其是Using MaskFormatter。
类似...
MaskFormatter formatter = new MaskFormatter("##.##");
JFormattedTextField field = JFormattedTextField(formatter);
例如可能会有所帮助。
简单示例
import java.awt.BorderLayout;
import java.awt.EventQueue;
import java.awt.GridBagLayout;
import java.awt.HeadlessException;
import java.text.ParseException;
import javax.swing.JFormattedTextField;
import javax.swing.JFrame;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.text.DefaultFormatterFactory;
import javax.swing.text.MaskFormatter;
public class TestFormattedTextField
public static void main(String[] args)
new TestFormattedTextField();
public TestFormattedTextField()
EventQueue.invokeLater(new Runnable()
@Override
public void run()
try
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex)
try
JFormattedTextField field = new JFormattedTextField();
MaskFormatter formatter = new MaskFormatter("##.##");
formatter.setPlaceholderCharacter('0');
field.setFormatterFactory(new DefaultFormatterFactory(formatter));
JFrame frame = new JFrame("Testing");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new GridBagLayout());
frame.add(field);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
catch (ParseException exp)
exp.printStackTrace();
);
其他示例
现在,我意识到前面的示例不能满足您的确切需求(正如您所描述的那样),这是一个简单的解决方案,我还添加了一个 DocumentFilter
示例...
哪个会输出...
Value = 0.1
Value = $0.10
哪个会输出
Value = 10.0
Value = $10.00
代码...
import java.awt.EventQueue;
import java.awt.GridBagLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.text.NumberFormat;
import java.text.ParseException;
import javax.swing.JFormattedTextField;
import javax.swing.JFrame;
import javax.swing.JTextField;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.text.AbstractDocument;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.DocumentFilter;
import javax.swing.text.MaskFormatter;
public class TestFormattedTextField
public static void main(String[] args)
new TestFormattedTextField();
public TestFormattedTextField()
EventQueue.invokeLater(new Runnable()
@Override
public void run()
try
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex)
MoneyField field = new MoneyField();
field.addActionListener(new ActionListener()
@Override
@SuppressWarnings("empty-statement")
public void actionPerformed(ActionEvent e)
MoneyField field = (MoneyField) e.getSource();
double value = field.getValue();
System.out.println("Value = " + value);
System.out.println("Value = " + NumberFormat.getCurrencyInstance().format(value));
);
JFrame frame = new JFrame("Testing");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new GridBagLayout());
frame.add(field);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
);
public class MoneyField extends JTextField
public MoneyField()
setColumns(5);
setHorizontalAlignment(RIGHT);
((AbstractDocument) getDocument()).setDocumentFilter(new Filter());
public double getValue()
String text = getText();
if (!text.contains("."))
text = "0." + text;
return Double.parseDouble(text);
protected class Filter extends DocumentFilter
protected String getNumbers(String text)
StringBuilder sb = new StringBuilder(text.length());
for (char c : text.toCharArray())
if (Character.isDigit(c))
sb.append(c);
return sb.toString();
@Override
public void replace(FilterBypass fb, int offset, int length, String text, AttributeSet attrs) throws BadLocationException
if (length > 0)
fb.remove(offset, length);
insertString(fb, offset, text, attrs);
@Override
public void insertString(FilterBypass fb, int offset, String text, AttributeSet attr) throws BadLocationException
text = getNumbers(text);
if (text.length() > 0)
int docLength = fb.getDocument().getLength();
if (docLength == 2)
text = "." + text;
if (docLength + text.length() < 6)
super.insertString(fb, offset, text, attr);
@Override
public void remove(FilterBypass fb, int offset, int length) throws BadLocationException
if (offset == 3)
offset = 2;
length = 2;
super.remove(fb, offset, length);
查看DocumentFilter examples了解更多详情
【讨论】:
+1,我看到我们都在研究 DocumentFilter 方法,我认为这是要走的路。 @camickr 我仍然认为JFormattedTextField
是更简单的方法,但DocumentFilter
将更接近于OP 要求...【参考方案3】:
对 JTextField 使用 DocumentFilter
并覆盖适当的方法来处理数字格式。如果您可以发布您尝试过但“不起作用”的内容,那就太好了。
【讨论】:
DocumentListener
不是“过滤”文档输入的合适选择。在调用 DocumentListener
时已经对文档进行了更改,并且侦听器中的任何修改都将导致抛出异常
@MadProgrammer - 你是对的。我指的是 DocumentFilter 并输入了 DocumentListener :-)
你不喜欢你的手指做一件事而大脑做另一件事吗:P
我试图为它创建一个正则表达式。我已经尝试制作一个输入掩码,输入掩码工作直到数字变得大于 99.99 。使用匹配器的正则表达式,我发现当它必须检查按下的键时会出现问题。以上是关于使 JFormattedTextField 的行为类似于 ATM 输入的主要内容,如果未能解决你的问题,请参考以下文章
Double的JFormattedTextField仍然需要字符[重复]