JTextPane - 在没有JScrollPane的情况下滚动可见文本

时间:2012-11-14 14:49:40

标签: java swing jtextpane

所以在我的应用程序中,我使用了一个自定义组件,它是JTextPane的扩展。我需要为样式扩展它。我已经设法关闭自动换行,因为这个组件可以充当JTextField,即不是多行的。

然而,我面临的问题是,如果文本跨越可见区域,则在JScrollpane内部不使用时,JTextPane似乎不能很好地处理非单词包装文本。使用JTextField时,无法移动可见区域外的文本。

我不想使用JScrollpane,因为我有许多假设JTextComponent的遗留代码。

所以我的问题是,有没有办法让可见区域与插入符号位置保持内联,这样当键入到达可见边缘或尝试选择跨越可见区域的文本时,该文本将移动到视图中,如JTextField确实

我的问题的一个工作示例如下。它在JTextPane上方显示了一个JTextField。如果您在文本字段中键入内容,您将看到可见区域随文本移动。如果您在文本窗格中键入,则不会发生这种情况。

另外,我在java 7中这样做。

提前致谢。

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

public class NoAutoScroll {
    public NoAutoScroll() {
        JFrame frame = new JFrame();
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        JTextField textField = new JTextField();
        textField.setText("This text field can show the text outside the visible area.");
        textField.setPreferredSize(new Dimension(150, 30));

        JTextPane textPane = new JTextPane();
        textPane.setEditorKit(new MyEditorKit());
        textPane.setText("This text pane cannot show the text outside the visible area.");
        textPane.setPreferredSize(new Dimension(150, 30));
        textPane.setBorder(textField.getBorder());

        JPanel mainPanel = new JPanel(new GridBagLayout());
        GridBagConstraints constraints = new GridBagConstraints();
        constraints.gridx = 0;
        constraints.gridy = 0;
        mainPanel.add(new JLabel("JTextField"), constraints);
        constraints.gridx = 1;
        constraints.gridy = 0;
        mainPanel.add(textField, constraints);
        constraints.gridx = 0;
        constraints.gridy = 1;
        mainPanel.add(new JLabel("JTextPane"), constraints);
        constraints.gridx = 1;
        constraints.gridy = 1;
        mainPanel.add(textPane, constraints);

        frame.add(mainPanel);
        frame.pack();
        frame.setVisible(true);
    }

    public class MyViewFactory implements ViewFactory {

        private final class NonBreakingLabelView extends LabelView {
            private NonBreakingLabelView(Element elem) {
                super(elem);
            }

            @Override
            public int getBreakWeight(int axis, float pos, float len){
                return BadBreakWeight;
            }
        }

        @Override
        public View create(final Element elem) {
            String kind = elem.getName();
            if (kind != null) {
                if (kind.equals(AbstractDocument.ContentElementName)) {
                return new NonBreakingLabelView(elem);
                } else if (kind.equals(AbstractDocument.ParagraphElementName)) {
                    return new ParagraphView(elem);
                } else if (kind.equals(AbstractDocument.SectionElementName)) {
                return new BoxView(elem, View.Y_AXIS);
                } else if (kind.equals(StyleConstants.ComponentElementName)) {
                return new ComponentView(elem);
                } else if (kind.equals(StyleConstants.IconElementName)) {
                return new IconView(elem);
                }
            }

            // default to text display
            return new LabelView(elem);
        }
    }

    @SuppressWarnings("serial")
    public class MyEditorKit extends StyledEditorKit {

        @Override
        public ViewFactory getViewFactory() {
            return new MyViewFactory();
        }
    }

    public static void main(String[] args) {
        new NoAutoScroll();
    }
}

1 个答案:

答案 0 :(得分:2)

正确的方式

    如果没有JScrollPane,则
  • 无效,您可以隐藏JScrollBars

脏黑客

  • 持有两个Document s,第二个且可见只会ScrollIncrememt从添加到JTextPane的鼠标事件生成,但是没有理由使用JTextPane,请使用{ {1}}而不是