使 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仍然需要字符[重复]

JFormattedTextField 仅返回默认值

JFormattedTextField 接受不同的位数

具有任意小数位数的 JFormattedTextfield

如何从 JformattedTextfield 检索货币格式值

为 JFormattedTextField 分配一个浮点数