使JTextArea滚动,因此当用户键入更多文本时,始终显示插入位置

时间:2013-07-10 21:24:37

标签: java swing scroll jscrollpane jtextarea

如何使JTextArea滚动,所以当用户输入更多文字时,始终显示插入位置?

这对我来说似乎是一个愚蠢的问题,我觉得以前一定是问过,但我已经搜索过,找不到答案。

我有一个嵌入JScrollPane的多行文字区域。如果用户一直打字直到他填充文本区域,最终插入符号变得不可见(在显示的区域下方)并且用户无法看到他正在键入的内容。这似乎很奇怪,这将是默认行为。我需要做些什么来确保文本区域始终滚动到插入符号所在的行。

我应该提一下,在文本区域中启用换行。

1 个答案:

答案 0 :(得分:1)

在我看来,setXxXSize()方法将使用有限数量的组件,并牢记后面使用的Layout,具体取决于哪个Layout Manager尊重提供的大小提示程序员。

因此,对于像JTextArea这样的RowsColumns已足够的组件,提供Layout关注点以计算其大小的提示,不应该是与硬编码大小相关联,尽可能使用一些setXxXSize()方法。

这里我提供了相同的示例代码:

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.beans.EventHandler;
import javax.swing.*;

public class DialogWithScroller
{
    private JFrame frame;
    private JButton showButton;
    private MyDialog myDialog;

    private void displayGUI()
    {
        frame = new JFrame("Dialog ScrollPane Example");
        frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);

        JPanel contentPane = new JPanel(new BorderLayout(5, 5));
        showButton = new JButton("Show Dialog");
        showButton.addActionListener((ActionListener)
                EventHandler.create(ActionListener.class,
                        DialogWithScroller.this, "buttonActions", ""));     
        contentPane.add(showButton);

        frame.setContentPane(contentPane);
        frame.pack();
        frame.setLocationByPlatform(true);
        frame.setVisible(true);
    }   

    public void buttonActions(ActionEvent ae)
    {
        myDialog = new MyDialog(frame
                , "TextArea with ScrollPane", true);
    }

    public static void main(String[] args)
    {
        Runnable runnable = new Runnable()
        {
            @Override
            public void run()
            {
                new DialogWithScroller().displayGUI();
            }
        };
        EventQueue.invokeLater(runnable);
    }
}

class MyDialog extends JDialog
{
    private JTextArea tArea;
    private JButton hideButton;

    private ActionListener buttonActions =
                            new ActionListener()
    {
        @Override
        public void actionPerformed(ActionEvent ae)
        {
            MyDialog.this.dispose();
        }
    };

    public MyDialog()
    {}

    public MyDialog(Frame owner, String title, boolean modal)
    {
        super(owner, title, modal);
        displayGUI();
    }

    private void displayGUI()
    {
        JPanel contentPane = new JPanel(
                        new BorderLayout(5, 5));
        contentPane.setBorder(
                BorderFactory.createTitledBorder(
                            "My Personal Text Area"));
        /*
         * Here one can simply initialize the
         * JTextArea like this too, using the
         * constructor itself for specifying
         * the Rows and Columns, which will
         * help the layout concern to determine
         * its size
         */
        tArea = new JTextArea(20, 20);
        tArea.setLineWrap(true);
        tArea.setWrapStyleWord(true);
        JScrollPane textScroller = new JScrollPane(tArea);
        //textScroller.setViewportView(tArea);

        hideButton = new JButton("Hide Dialog");
        hideButton.addActionListener(buttonActions);

        contentPane.add(textScroller, BorderLayout.CENTER);
        contentPane.add(hideButton, BorderLayout.PAGE_END);
        setContentPane(contentPane);
        pack();
        setLocationByPlatform(true);
        setVisible(true);
    }   
}

在容器上调用Window.pack()始终被认为是明智的做法,会使此窗口的大小适合其子组件的首选大小和布局。虽然这样的调用必须在程序员完成后将所有组件添加到容器中并且在将容器设置为可见状态之前