Отображение HTML в JEditorPane с использованием jEditor.setText() заставляет его мерцать

Я пытаюсь использовать JEditorPane для отображения HTML-текста. По какой-то причине мне приходится использовать метод setText(). Однако это заставляет JEditorPane мерцать. Причина в том, что каждый раз после обновления editorPane.setText(msgBuffer); мне приходится подводить курсор к концу документа editorPane.setCaretPosition((editorPane.getDocument()).getLength ()-1), так как я хочу, чтобы самая последняя текстовая строка отображалась внизу документа. У меня есть для вас, ребята, весь код, чтобы вы могли увидеть его сами. Я видел много рекомендаций по использованию document.insertString, но в этом отношении я должен использовать отдельные атрибуты, которые меня не интересуют. Есть ли способ заставить этот код работать без мерцания?

import javax.swing.JEditorPane;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.text.Document;
import javax.swing.text.html.HTMLEditorKit;
import javax.swing.text.html.StyleSheet;

public class CMessageWindow {

    private static final String ERROR   = "ERROR"; 
    private static final String MESSAGE = "msg";
    private  JScrollPane scrollPane;
    public  JEditorPane  editorPane;
    private  HTMLEditorKit kit;
    private  String msgBuffer=new String("");
    private static CMessageWindow window=null;
    private static JFrame frameContainer=null;

    private CMessageWindow()
    {
        editorPane  = new JEditorPane ();
        editorPane.setEditable(false);
        editorPane.setContentType("text/html");
        kit = new HTMLEditorKit();
        editorPane.setEditorKit(kit);

        StyleSheet styleSheet = kit.getStyleSheet();
        styleSheet.addRule("."+MESSAGE+" {font: 10px monaco; color: black; }");
        styleSheet.addRule("."+ERROR+" {font: 10px monaco; color: #ff2222; background-color : #cccccc; }");

        Document doc = kit.createDefaultDocument();
        editorPane.setDocument(doc);
        scrollPane = new JScrollPane(editorPane);
    }
    public static CMessageWindow getInstance(){
        if (null==window)
        {window=new CMessageWindow();}
        return window;
    }
/**
 * The core
 * @param sMessage
 * @param sType
 */
    private void updateMessages(final String sMessage, final String sType)

    {
        String sMessageHTML=""; 
        String sTypeText="";
        if (!sMessage.equals("\r\n")){ 
            sTypeText = sType+": ";
        }

        sMessageHTML = sMessage.replaceAll("(\r\n|\n)", "<br/>");
        if (!sMessageHTML.equals("<br/>")) 
        {
            sMessageHTML =   "<SPAN CLASS="+sType+">"+ sTypeText+sMessageHTML + "</SPAN>";
        }

        msgBuffer=msgBuffer.concat( sMessageHTML);
        editorPane.setText(msgBuffer);
        if ((editorPane.getDocument()).getLength()>1){
            editorPane.setCaretPosition((editorPane.getDocument()).getLength()-1);
        }  
    }

    public void setContainerFrame(JFrame jFrame){
        frameContainer = jFrame;
        javax.swing.GroupLayout layout = new javax.swing.GroupLayout(frameContainer.getContentPane());
        frameContainer.getContentPane().setLayout(layout);
        layout.setHorizontalGroup(
                layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
                .addComponent(scrollPane)
                );
        layout.setVerticalGroup(
                layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
                .addGroup(layout.createSequentialGroup()
                        .addComponent(scrollPane, javax.swing.GroupLayout.DEFAULT_SIZE, 217, Short.MAX_VALUE))
                );
    }

    public void setVisible(boolean bVisible){
        editorPane.setVisible(bVisible);
        scrollPane.setVisible(bVisible);
    }

    public void printMsg(String sMessage){
        String sType = MESSAGE;
        updateMessages(sMessage,sType);
    }

    public void printlnMsg(String sMessage){
        sMessage=sMessage.concat("\r\n");
        printMsg(sMessage);
    }

    public void printErr(String sMessage){
        String sType = ERROR;
        updateMessages(sMessage,sType);
    }

    public void printlnErr(String sMessage){
        sMessage=sMessage.concat("\r\n");
        printErr(sMessage);
    }

    public static void main(String args[]){
        CMessageWindow m_LogMgr;
        JFrame frame = new JFrame();
        m_LogMgr=CMessageWindow.getInstance();
        m_LogMgr.setContainerFrame(frame);
        frame.setVisible(true);
        frame.setSize(500, 500);

        for(int i=0;i<20;++i){
            m_LogMgr.printlnErr("MMMMMMMMMMMMMMMMMMMMMMMMMMMMMMM");
        }

        for(int i=0;i<150;++i){
            try {
                Thread.sleep(20);
            } catch (Exception e) {
            }
            m_LogMgr.printlnMsg("-----------------------");
        }

    }


}

person C graphics    schedule 03.10.2012    source источник


Ответы (2)


Вы не должны изменять компоненты пользовательского интерфейса вне EDT.

Если вы добавите вызов, скажем, в свои updateMessages, чтобы обновление происходило в EDT, мерцание исчезнет.

Например:

private void updateMessages(final String sMessage, final String sType)

{
    SwingUtilities.invokeLater( new Runnable() {
        public void run() {
            String sMessageHTML="";
            String sTypeText="";
            if (!sMessage.equals("\r\n")){
                sTypeText = sType+": ";
            }
            sMessageHTML = sMessage.replaceAll("(\r\n|\n)", "<br/>");
            if (!sMessageHTML.equals("<br/>"))
            {
                sMessageHTML =   "<SPAN CLASS="+sType+">"+ sTypeText+sMessageHTML + "</SPAN>";
            }

            msgBuffer=msgBuffer.concat( sMessageHTML);
            editorPane.setText(msgBuffer);
            if ((editorPane.getDocument()).getLength()>1){
                editorPane.setCaretPosition((editorPane.getDocument()).getLength()-1);
            }
        }
    });
}

Обратите внимание, что вам не следует выполнять длительные операции в EDT, потому что в противном случае вы «заблокируете» свой пользовательский интерфейс.

person TacticalCoder    schedule 03.10.2012
comment
или без каких-либо изменений в видимом графическом интерфейсе - person mKorbel; 04.10.2012

Если у вас есть длительный ввод-вывод и последующее обновление графического интерфейса, вы можете использовать вместо него SwingWorker (см. http://en.wikipedia.org/wiki/Event_dispatching_thread)

SwingWorker<Document, Void> worker = new SwingWorker<Document, Void>() {
    public Document doInBackground() throws IOException {
        return loadXML(); // heavy task
    }

    public void done() {
        try {
            Document doc = get();
            display(doc);
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
};
worker.execute();
person ahus1    schedule 23.11.2014