【问题标题】:JTextArea. How can I replace text when printing new one?文本区域。打印新文本时如何替换文本?
【发布时间】:2020-10-30 22:21:39
【问题描述】:

我在它上面创建了 JFrame 和 JTextArea。 JTextArea 有默认文本“This is text for demo version”,已通过 setText() 方法设置。

目标是实现这个逻辑: 如果我开始打印文本,则应删除旧文本并出现新文本。 之后,当打印新文本时,如果我点击“Enter” - 新文本应该保存到private ArrayList<String> textList

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

我尝试在 keyReleased(KeyEvent e) {} 中添加TestTextArea.this.replaceRange(keyText,0, 30);; (30 是默认字符串 "This is text for demo version" 的最后一个索引)。但是每次我打印任何东西时,它都会导致 IllegalArgumentException 并且似乎旧文本在背景上仍然可见。

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, 200, 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,0, 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 后,我输入的所有文本都会被保存。但此刻事实证明,当你输入第一个字符时,默认文本被删除,但甚至没有一个字符被保存。

对不起,提前为问题的可能不太成功的陈述。

【问题讨论】:

  • @Abra 我需要一个输入字段,但它不应该看起来像 JTextField。可以在输入字段中输入许多句子,并且单词必须换行到下一行。此外,当输入一行或至少一个字符时,用户可以按Enter,并且他输入的数据必须保存到列表中。这就是我选择 JTextArea 的原因,因为它可以被拉伸以适应整个 JFrame。感谢您对 Text Prompt 的建议,但我还不想求助于使用第三方库。
  • 如果我理解您的要求,您需要一个操作,以便在按下 Enter 时 1) 将当前文本保存到 ArrayList 2) 设置一个标志,以便将要添加到 Document 的下一个字符首先从文档中删除之前的文本。
  • @camickr 事实上,一切都更加复杂。这只是一个演示示例。在我的实际应用程序中,这个 JFrame 是在单击特定按钮后出现的附加组件。以下是逻辑。 1) 如果我们按 Esc 键,附加的框架应该被处理掉。 2)如果我们按回车,输入的值不等于空字符串,则将其写入列表,并处理附加帧。但是在这个演示的具体示例中,我想完全实现我上面描述的逻辑,其他所有处理框架的事情都非常容易完成。
  • 其他逻辑与您提出的问题无关。我要求明确这一具体要求。您最初说:我希望在我输入文本并按 Enter 后,保存我输入的所有文本。因此,您建议只保存文本。现在你是说文本应该被“保存”和“删除”所以文本区域不包含文本?
  • 如果这应该是一个额外的应用程序 JFrame,也许 JDialog 会是一个更好的选择。

标签: java swing jtextarea keylistener


【解决方案1】:

我想我理解您的要求。

  • 当输入第一个字符时,清除JTextArea并捕获JTextArea中输入的文本,

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

此代码符合这些要求。

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

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

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, 200, 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("", 0, 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;
                }
            } 
        }
        
    }

}

【讨论】:

  • 是的,你理解的很正确,非常感谢,问题解决了!
【解决方案2】:

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

我没有检查,但我认为 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 后,您可以在 JTextArea 上使用 setDocument()

【讨论】:

  • 看起来不错,只有一个问题,在进入smth之前我的默认文本消失了:)
  • 能否请您编辑您的代码并编写删除方法?我在父 PlainDocument 类中发现只有 removeUpdate() 方法。
  • @Neo 为什么?您希望“revoveUpdate() 方法会发生什么?
  • 我看到评论 // ... remove() 的类似内容,但我不知道应该重写哪个方法。无论如何,我修复了默认文本不可见的问题。我只是在将 setText() 添加到 JTextArea 之前添加了 ``` setDocument() ```。但我面临新的问题。如果我单击并将鼠标光标放在默认文本字符之间的某个位置并开始输入,则默认文本不会被删除。它只是通过添加新输入的字符进行编辑
  • 是的,我想这不符合您的要求,抱歉。只有在编辑完成时才会调用插入/删除方法。您也许可以通过听焦点来做某事?
猜你喜欢
  • 2011-11-14
  • 1970-01-01
  • 2022-07-23
  • 2011-04-27
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多