在 JTextPane (Java 7) 中包装长字

2022-09-03 06:54:53

在截至6的所有Java版本中,JTextPane放在JScrollPane中的默认行为是:如果可能的话,在单词边界处换行。如果没有,那么无论如何都要包装它们。

在 JDK 7 中,默认行为似乎是:如果可能,在单词边界处换行。如果没有,只需扩展 JTextPane 的宽度(切勿换行长字)。

很容易重现这一点,这是一个SSCCE:


public class WrappingTest extends JFrame
{

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

    public WrappingTest ()
    {
        setSize(200,200);
        getContentPane().setLayout(new BorderLayout());
        JTextPane jtp = new JTextPane();
        JScrollPane jsp = new JScrollPane(jtp);
        jsp.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);
        getContentPane().add(jsp,BorderLayout.CENTER);
        setVisible(true);
    }

}

只需在JDK 6和JDK 7中运行它,写一些小单词,写一个长单词,你就会看到区别。

我的问题很简单...JDK 7中新的默认行为完全搞砸了我的一个程序(他们应该在Oracle上更小心地更改这种默认值......它们似乎不重要,但是当您使用JTextPane显示通常包含很长的字母字符串的数据时,它们并不是那么不重要 - 事实上,我将提交一个错误报告,但我希望有一个解决方法,而/如果他们不能解决它)。有什么办法可以回到以前的行为吗?

请注意,我已经检查了相关问题的答案 如何在JTextPane中实现自动换行,以及如何让它换行不带空格的字符串?但它没有回答这个问题 - 它提供了一种使JTextPane换行的方法,而根本不考虑空格,但对我来说,如果可能的话,所需的行为是在空格处拆分行, 如果不可能,则在其他地方(如以前的Java版本)。


答案 1

对我来说,修复工作(在1.7.0_09下测试)

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

public class WrapTestApp extends JFrame {

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

    public WrapTestApp () {
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setSize(200,200);
        getContentPane().setLayout(new BorderLayout());
        JTextPane jtp = new JTextPane();
        jtp.setEditorKit(new WrapEditorKit());
        JScrollPane jsp = new JScrollPane(jtp);
        jsp.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);
        getContentPane().add(jsp, BorderLayout.CENTER);
        jtp.setText("ExampleOfTheWrapLongWordWithoutSpaces");
        setVisible(true);
    }

    class WrapEditorKit extends StyledEditorKit {
        ViewFactory defaultFactory=new WrapColumnFactory();
        public ViewFactory getViewFactory() {
            return defaultFactory;
        }

    }

    class WrapColumnFactory implements ViewFactory {
        public View create(Element elem) {
            String kind = elem.getName();
            if (kind != null) {
                if (kind.equals(AbstractDocument.ContentElementName)) {
                    return new WrapLabelView(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);
        }
    }

    class WrapLabelView extends LabelView {
        public WrapLabelView(Element elem) {
            super(elem);
        }

        public float getMinimumSpan(int axis) {
            switch (axis) {
                case View.X_AXIS:
                    return 0;
                case View.Y_AXIS:
                    return super.getMinimumSpan(axis);
                default:
                    throw new IllegalArgumentException("Invalid axis: " + axis);
            }
        }

    }
}

答案 2

从@dk89很好,但唉,给定的解决方法不起作用:JDK 7显然仍然没有提供在JTextComponent上设置自定义BreakIterator的等待;甚至在GlyphView上也没有,其中BreakIterator的生成是私有的。如果我们逐个字符地插入字符串char,它仍然不起作用:我认为具有相同样式(AttributeSet)的连续文本运行被折叠在一起。

我花了两天时间尝试按照其他地方的建议做一个自定义的EditorKit,但它不能很好地工作(至少使用JDK 1.7.0_4)作为文本。

我尝试了如何存储在JTextPanes中的自动换行文本中给出的解决方案,这些文本是JList中的单元格,也是在 http://www.experts-exchange.com/Programming/Languages/Java/Q_20393892.html

但是我发现,当JTextPane小于句子中最长的单词时,不再调用breakView。因此,当只有一个(长)单词时,它根本不起作用。这就是我们的情况,因为我们在相当小的空间中显示用户提供的类似标识符的字符串,通常没有空格。

我终于找到了一个简单的解决方案,从bug报告中的建议中派生出来:确实,逐个插入字符串char,但交替样式!因此,我们有和字符一样多的段,并且字符串在字符边界处被包裹。直到下一个“错误修复”?

代码片段:

private JTextPane tp;
private SimpleAttributeSet sas = new SimpleAttributeSet();

tp= new JTextPane();
sas.addAttribute( "A", "C" ); // Arbitrary attribute names and value, not used actually

    // Set the global attributes (italics, etc.)
    tp.setParagraphAttributes(styleParagraphAttributes, true);

    Document doc = tp.getDocument();
    try
    {
        doc.remove(0, doc.getLength()); // Clear
        for (int i = 0; i < textToDisplay.length(); i++)
        {
            doc.insertString(doc.getLength(), textToDisplay.substring(i, i+1),
                    // Change attribute every other char
                    i % 2 == 0 ? null : sas);
        }
    }
    catch (BadLocationException ble)
    {
        log.warn("Cannot happen...", ble);
    }

如 bug 中所述,它们应该提供一种简单的方法(也许是一些属性,或者一些可注入的东西)来恢复到旧行为。


推荐