简体   繁体   中英

How to make part of a JTextField uneditable

I wanted to develop a console-like interface, similar to IDLE. That involved determining how to prevent a certain part of the text in a JTextField from being edited. For example:

>>> help

Where the ">>> " is uneditable. The caret must never move behind a certain position, and the text behind that position cannot be edited in any way.

I looked at NavigationFilter, but it doesn't seem to prevent keyboard driven manipulation of the caret.

This shows how to do it with a NavigationFilter :

import java.awt.event.*;
import javax.swing.*;
import javax.swing.text.*;

public class NavigationFilterPrefixWithBackspace extends NavigationFilter
{
    private int prefixLength;
    private Action deletePrevious;

    public NavigationFilterPrefixWithBackspace(int prefixLength, JTextComponent component)
    {
        this.prefixLength = prefixLength;
        deletePrevious = component.getActionMap().get("delete-previous");
        component.getActionMap().put("delete-previous", new BackspaceAction());
        component.setCaretPosition(prefixLength);
    }

    @Override
    public void setDot(NavigationFilter.FilterBypass fb, int dot, Position.Bias bias)
    {
        fb.setDot(Math.max(dot, prefixLength), bias);
    }

    @Override
    public void moveDot(NavigationFilter.FilterBypass fb, int dot, Position.Bias bias)
    {
        fb.moveDot(Math.max(dot, prefixLength), bias);
    }

    class BackspaceAction extends AbstractAction
    {
        @Override
        public void actionPerformed(ActionEvent e)
        {
            JTextComponent component = (JTextComponent)e.getSource();

            if (component.getCaretPosition() > prefixLength)
            {
                deletePrevious.actionPerformed( null );
            }
        }
    }

    private static void createAndShowUI()
    {
        JTextField textField = new JTextField("Prefix_", 20);
        textField.setNavigationFilter( new NavigationFilterPrefixWithBackspace(7, textField) );

        JFrame frame = new JFrame("Navigation Filter Example");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.getContentPane().add(textField);
        frame.pack();
        frame.setLocationRelativeTo( null );
        frame.setVisible(true);
    }

    public static void main(String[] args)
    {
        SwingUtilities.invokeLater(new Runnable()
        {
            public void run()
            {
                createAndShowUI();
            }
        });
    }
}

Spent a little while figuring this out, so I thought I would share my solution for anyone else with the same dilemma. I don't know if it's optimal, but it does seem to work.

It prevents the user from using backspace behind the postion n . It also moves the caret back to n for any other events, such as (illegally) changing the caret position with the arrow keys or mouse. Finally, it resets the text and caret position after a entry is processed.

EDIT: While I'm leaving this answer here for posterity, see the accepted answer for the best way to solve this problem.

    JTextField in = new JTextField();
    final String protectMe = ">>> "; //protect this text
    final int n = protectMe.length();
    in.setText(protectMe);
    in.setCaretPosition(n);

    in.addCaretListener(new CaretListener()
    {
        @Override
        public void caretUpdate(CaretEvent e)
        {
            if (e.getDot() < n)
            {
                if (!(in.getText().length() < n))
                    in.getCaret().setDot(n);
            }
        }
    });

    in.addKeyListener(new KeyListener()
    {

        @Override
        public void keyPressed(KeyEvent arg0)
        {
            if (in.getCaret().getDot() <= n)
            {
                in.setText(protectMe + in.getText().substring(n));
                arg0.consume();
            }
        }

        @Override
        public void keyReleased(KeyEvent arg0){}

        @Override
        public void keyTyped(KeyEvent arg0){}
    });
    in.addActionListener(new ActionListener()
    {
        public void actionPerformed(ActionEvent evt)
        {
            String input = in.getText().substring(n).trim(); 

            //do something

            in.setText(protectMe);
            in.setCaretPosition(n);
        }
    });

As usual, let me know if there's anything I missed!

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM