JTextArea打印新的文本时如何替换文本?

问题描述

我在其上面创建了JFrame和JTextArea。 JTextArea具有默认文本“ 这是演示版本的文本”,该文本已通过setText()方法设置。

enter image description here

目标是实施以下逻辑: 如果我开始打印文本,则应删除旧文本,然后添加新的文本。 之后,在打印新文本时,如果单击“ Enter”,则应将新文本保存到private ArrayList<String> textList

enter image description here

主要问题是在打印文本的第一个符号时如何替换旧文本?

我试图在keyReleased(KeyEvent e){}中添加TestTextArea.this.replaceRange(keyText,30);; (30是默认字符串“ 这是演示版本的文本”的最后一个索引)。但是每次我打印任何内容时,它都会导致 IllegalArgumentException ,并且似乎旧文本仍在背景上可见。

enter image description here

import javax.swing.*;
import java.awt.*;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import java.util.ArrayList;

import static javax.swing.WindowConstants.EXIT_ON_CLOSE;

public class TestTextArea extends JTextArea {

    private String text = "This is text for demo version";
    private ArrayList<String> textList = new ArrayList<>();

    TestTextArea() {
        setBackground(new Color(23,28,34,240));
        setForeground(new Color(6,200,109));
        setCaretColor(new Color(6,109));
        setCaretPosition(0);
        setFont(new Font("Helvetica Neue",Font.BOLD,16));
        setText(text);
        setLineWrap(true);
        setWrapStyleWord(true);
        setFocusable(true);
        setEnabled(true);
        setEditable(true);
        setVisible(true);

        addKeyListener(new KeyListener() {
            @Override
            public void keyTyped(KeyEvent e) {

            }

            @Override
            public void keyPressed(KeyEvent e) {

            }

            @Override
            public void keyReleased(KeyEvent e) {
                String keyText = KeyEvent.getKeyText(e.getKeyCode());
                TestTextArea.this.replaceRange(keyText,30);
                if (keyText.equals("Enter")) {
                    textList.add(TestTextArea.this.getText());
                }
            }
        });

    }
    public static void main(String []args) {
        JFrame f = new JFrame();

        TestTextArea area = new TestTextArea();
        f.add(area);

        f.setSize(400,200);
        f.setDefaultCloseOperation(EXIT_ON_CLOSE);
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }
}

还有其他解决方法吗?我希望在键入文本并按Enter后,保存所有键入的文本。但是目前,事实证明,当您输入第一个字符时,将删除默认文本,但甚至不会保存任何一个字符。

对不起,对于这个问题的陈述可能不是很成功。

解决方法

我想我理解您的要求。

  • 键入第一个字符后,清除JTextArea并在JTextArea中捕获键入的文本,

  • 按下Enter键后,将键入的文本保存在List中。

此代码符合这些要求。

我使用JTextArea。扩展Swing组件或任何Java类的唯一原因是重写一个或多个类方法。

我在JTextArea构造函数中设置JTextArea的大小。然后我pack JFrame

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Font;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import java.util.ArrayList;
import java.util.List;

import javax.swing.JFrame;
import javax.swing.JTextArea;
import javax.swing.SwingUtilities;

public class TextEntryExample implements Runnable {

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new TextEntryExample());
    }
    
    private List<String> textList;
    
    private String text;
    
    private JTextArea textArea;
    
    public TextEntryExample() {
        this.text = "This is text for demo version";
        this.textList = new ArrayList<>();
    }

    @Override
    public void run() {
        JFrame f = new JFrame();
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        this.textArea = createPrompt();
        f.add(textArea,BorderLayout.CENTER);

        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }
    
    private JTextArea createPrompt() {
        JTextArea textArea = new JTextArea(10,30);
        textArea.addKeyListener(new PromptListener());
        textArea.setBackground(new Color(23,28,34,240));
        textArea.setForeground(new Color(6,200,109));
        textArea.setCaretColor(new Color(6,109));
        textArea.setCaretPosition(0);
        textArea.setFont(new Font("Helvetica Neue",Font.BOLD,16));
        textArea.setText(text);
        textArea.setLineWrap(true);
        textArea.setWrapStyleWord(true);
        
        return textArea;
    }
    
    public class PromptListener implements KeyListener {
        
        private boolean entry;
        
        public PromptListener() {
            this.entry = false;
        }

        @Override
        public void keyTyped(KeyEvent e) {
        }

        @Override
        public void keyPressed(KeyEvent event) {
            if (!entry) {
                String oldText = textArea.getText();
                textArea.replaceRange("",oldText.length());
                entry = true;
            }
        }

        @Override
        public void keyReleased(KeyEvent event) {
            if (entry) {
                String keyText = KeyEvent.getKeyText(event.getKeyCode());
                if (keyText.equals("Enter")) {
                    textList.add(textArea.getText());
                    entry = false;
                }
            } 
        }
        
    }

}
,

这可能不是最简单的方法,但是您可以将文档替换为可清除第一次编辑时的文本的文档。

我还没有检查,但是我认为JTextArea使用DefaultStyledDocument。您可以扩展它并覆盖编辑方法以检查标志-如果已设置,请清除文本并清除标志。我已经为JTextField做过类似的事情(使用PlainDocument)-这是它的外观:

public class PromptDocument
    extends PlainDocument
{
    private bool clearOnEdit = false;

    public void insertString(int offset,String str,AttributeSet a)
        throws BadLocationException
    {
        if (clearOnEdit) {
            super.remove(offset,getLength());
            clearOnEdit = false;
        }
        super.insertString(offset,str,a);
    }
    public void remove(int offset,int len)
        throws BadLocationException
    {
        if (clearOnEdit) {
            super.remove(offset,getLength());
            clearOnEdit = false;
        } else {
            super.remove(offset,len);
        }
    }
    public void setClearOnEdit(final boolean clear) {
        clearOnEdit = clear;
    }
}

它对于JTextArea应该是相似的(我还没有测试过,可能遗漏了一些东西)。创建JTextArea后,只需在其上使用setDocument()

相关问答

错误1:Request method ‘DELETE‘ not supported 错误还原:...
错误1:启动docker镜像时报错:Error response from daemon:...
错误1:private field ‘xxx‘ is never assigned 按Alt...
报错如下,通过源不能下载,最后警告pip需升级版本 Requirem...