JEditorPane.scrollToReference() с динамически генерируемым HTML

Я не могу понять, как JEditorPane.scrollToReference() предназначен для использования с динамически генерируемой HTML-страницей. Я хотел бы открыть диалоговое окно с JEditorPane, которое прокручивается на якорь по моему выбору. Независимо от того, как я подхожу к проблеме, представление всегда прокручивается внизу страницы.

Как уродливый обходной путь, я в настоящее время анализирую весь HTML-документ на предмет привязки тегов, сохраняя их смещения в Map<String, Integer>, а потом я звоню:

editorPane.setCaretPosition(anchorMap.get("anchor-name"));

... который даже не дает привлекательного результата, потому что позиция каретки в видимом прямоугольнике выглядит непредсказуемой и редко находится в верхней части окна. Я ищу более похожее на браузер поведение, когда закрепленный текст появляется в верхней части видимой области.

Ниже приведен скриншот того, что происходит с моим неудобным обходным решением (на "заголовке 6" есть привязка), не касаясь полосы прокрутки:

http://i49.tinypic.com/jkaj9s.png

Я думаю, что что-то упустил, но не могу понять, что именно.

Источник моего текущего решения:

import javax.swing.*;
import javax.swing.text.MutableAttributeSet;
import javax.swing.text.html.*;
import javax.swing.text.html.HTMLEditorKit.*;
import javax.swing.text.html.parser.*;
import java.io.*;
import java.awt.*;
import java.util.HashMap;

public class AnchorTest
{
    public static void main(String[] args)
    {
        final String html = generateLongPage();
        final HashMap<String, Integer> anchors = anchorPositions(html);

        SwingUtilities.invokeLater(new Runnable()
        {
            public void run()
            {
                JEditorPane editor  = new JEditorPane("text/html", html);
                JScrollPane scroller= new JScrollPane(editor);

                scroller.setPreferredSize(new Dimension(400, 250));

                //editor.scrollToReference("anchor6");  // doesn't work...
                editor.setCaretPosition(anchors.get("anchor6")); //sorta works

                JOptionPane.showMessageDialog(new JPanel(), scroller, "",
                        JOptionPane.PLAIN_MESSAGE);
            }});
    }

    public static HashMap<String, Integer> anchorPositions(String html)
    {
        final HashMap<String, Integer> map = new HashMap<String, Integer>();

        Reader reader = new StringReader(html);
        HTMLEditorKit.Parser parser = new ParserDelegator();

        try
        {
            ParserCallback cb = new ParserCallback()
            {
                public void handleStartTag(HTML.Tag t,
                                           MutableAttributeSet a,
                                           int pos)
                {
                    if (t == HTML.Tag.A) {
                        String name = 
                            (String)a.getAttribute(HTML.Attribute.NAME);
                        map.put(name, pos);
                    }
                }
            };

            parser.parse(reader, cb, true);
        }
        catch (IOException ignore) {}

        return map;
    }


    public static String generateLongPage()
    {
        StringBuilder sb = new StringBuilder(
                "<html><head><title>hello</title></head><body>\n");

        for (int i = 0; i < 10; i++) {
            sb.append(String.format(
                    "<h1><a name='anchor%d'>header %d</a></h1>\n<p>", i, i));

            for (int j = 0; j < 100; j++) {
                sb.append("blah ");
            }
            sb.append("</p>\n\n");
        }

        return sb.append("</body></html>").toString();
    }
}

1 ответ

Решение

Возможно, проблема в том, что панель еще не видна или еще не выложена, когда scrollToReference выполняется, поэтому его размер не может быть определен.

Реализация scrollToReference имеет следующий блок:

Rectangle r = modelToView(iter.getStartOffset());
if (r != null) {
   ...
   scrollRectToVisible(r);
}

В опубликованном примере прямоугольник null за anchor6, поскольку вид еще не виден или его размер не является положительным.

Попробуйте это грязное исправление, чтобы отложить прокрутку:

SwingUtilities.invokeLater(new Runnable() {
    public void run() {
        editor.scrollToReference("anchor6");
    }
});
JOptionPane.showMessageDialog(new JPanel(), scroller, "",
        JOptionPane.PLAIN_MESSAGE);
Другие вопросы по тегам