Search code examples
javaswingjformattedtextfieldnumber-formatting

Make a JFormattedTextField behave like ATM input


I would like to know if there is anyway to make a JformattedTextField or jtextField behave like an atm money input. With that I mean you enter from the right to left, say you enter 10 you need to press 2 more 0's so that it will be 10.00 . The program enters the decimal point automatically as he types from right to left? If the 2 0's are not entered it will just be .10 . Is this possible? How would that be returned to me if I want to use that string to do calculations on then? I tried the abstract formatter but this doesn't work so nicely. I want to use this for input for the amount of money received by a customer. But make it idiot proof.


Solution

  • This forces the user to always enter text from the right no matter where the caret is positioned. All previous characters are shifted left as a new character is inserted. Formatting will be applied based on your formatter:

    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();
                }
            });
        }
    }
    

    How would that be returned to me if I want to use that string to do calculations on then?

    You would need to create a method, maybe getValue() that would use the format.parse(...) method to return an actual number.