0

我有JTextPane一个扩展的DefaultStyledDocument. 更新文档大约需要 2 分钟。目前,我将其分解为多个步骤,并InvokeAndWait针对每个步骤更新摆动线程。我有一个进度条和一个取消按钮。取消按钮仅在步骤之间有中断时触发。每个步骤大约需要 10 秒,因此我需要等待长达 10 秒才能停止处理文档。反正有没有让这更具响应性?JFrame完成后,我正在显示JTextPanea JScrollPane。当它最终呈现时,滚动非常敏感,我可以查看整个文档。我不想显示JFrame直到文档更新但我想继续显示更新的进度。关于如何更新文档并使 Swing 和/或取消按钮更具响应性的任何想法?

====== 编辑以响应评论 ====== 在样式化文档中使用 append(text) 方法 - 在附加之前为每一行设置样式。

public void append(String text)
{
    append(text, textStyle, paragraphStyle);
}

public void append(String text, Style ts, Style ps)
{
    try
    {
        int start = this.getLength();
        int end = this.getLength()+text.length();
        this.insertString(start, text, ts);
        this.setParagraphAttributes(start, end, ps, true);
    }

    catch (BadLocationException e)
    {
        LOG.log(Level.SEVERE, "Bad location in append", e);
    }
}

======== 编辑 ======== 这就是我的文档更新方法的样子。

 public void writeTextOutsideOfSwing(StatusDialog status, float statusPercentage)
  {
    final StringBuilder letter = new StringBuilder();
    final ArrayList<IndexTextEntry>list = new ArrayList<>();
    LOG.log(Level.INFO, "Updating document: {0}", entries.length);
    setText("");
    int step = (int)((status.getMaximum() * statusPercentage) / (double)entries.length);
    for(int j = 0; j < entries.length; j++)
    {
        if(status.cancel) break;
        final int index = j;

        list.add(entries[j]);
        if(list.size() == 100 || index == entries.length -1)
        {
            int first = index - list.size() + 2;
            int last = index + 1;
            status.setStatusBarText("Writing Concordance: Processing " + first + " - " + last + " of " + entries.length);
            try 
            { 
                SwingUtilities.invokeAndWait(()-> 
                {
                    for(int k = 0; k < list.size(); k++)
                    {
                        int i = index-list.size() + k;

                        if(!letter.toString().equals(list.get(k).getSortLetter()))
                        {
                            letter.setLength(0);
                            letter.append(list.get(k).getSortLetter());
                            String title = list.get(k).getSortLetterTitle(letter.toString());
                            appendLetter(title, i == 0);
                        }
                        else if(i > 0)
                        {
                            if(cf.getLineBetweenEntries()) 
                            {
                                clearTextAreaStyles();
                                setFontSize(cf.getLineBetweenEntriesFontSize());
                                append(" \n");
                            }
                        }
                        float indent = appendEntry(0, list.get(k));
                        appendSubEntries(indent, list.get(k));
                    }
                });
            } 
            catch(InterruptedException | InvocationTargetException ex) 
            { LOG.log(Level.SEVERE, "Writing Concorder Interrupted", ex); }
            list.clear();
        }
        status.increment(step);
    }
    LOG.info("Done updating docuemnt");
}

这些方法与 write 方法一起使用:

 private void appendSubEntries(float indent, IndexTextEntry entry)
 {
    for (IndexTextEntry subEntry : entry.getSubEntries()) 
    {
        float ind = appendEntry(indent, subEntry);
        appendSubEntries(ind, subEntry);
    }
 }

 private float appendEntry(float indent, IndexTextEntry entry)
 {

    setFontFamily(cf.getWordFormat().getFontFamily());
    setFontSize(cf.getWordFormat().getFontSize());
    setFontBold(cf.getWordFormat().getBold());
    setFontItalic(cf.getWordFormat().getItalic());
    switch (cf.getWordFormat().getAlignment()) 
    {
        case TextFormat.ALIGN_CENTER:
            setFontAlignment(EnhancedTextArea.ALIGN_CENTER);
            break;
        case TextFormat.ALIGN_RIGHT:
            setFontAlignment(EnhancedTextArea.ALIGN_RIGHT);
            break;
        default:
            setFontAlignment(EnhancedTextArea.ALIGN_LEFT);
            break;
    }
    float wi = indent + cf.getWordFormat().getIndentJAVA();
    setLeftIndent(wi);
    setFontColor(cf.getWordFormat().getColor());
    append(entry.getConcordance().getTitle());
    append("\n");

    float li = 0;
    for(ConcordanceLine line : entry.getConcordance().getLines())
        li = appendLine(wi, line);

    return li;
}

private float appendLine(float indent, ConcordanceLine line)
{
    setFontFamily(cf.getLineFormat().getFontFamily());
    setFontSize(cf.getLineFormat().getFontSize());
    switch (cf.getLineFormat().getAlignment()) 
    {
        case TextFormat.ALIGN_CENTER:
            setFontAlignment(EnhancedTextArea.ALIGN_CENTER);
            break;
        case TextFormat.ALIGN_RIGHT:
            setFontAlignment(EnhancedTextArea.ALIGN_RIGHT);
            break;
        default:
            setFontAlignment(EnhancedTextArea.ALIGN_LEFT);
            break;
    }
    float li = indent + cf.getLineFormat().getIndentJAVA();
    setLeftIndent(li);
    setFontColor(cf.getLineFormat().getColor());

    setFontBold(cf.getPageBold());
    setFontItalic(cf.getPageItalic());
    append(line.page + " ");
    setFontBold(cf.getLineBold());
    setFontItalic(cf.getLineItalic());
    append(line.lead);
    setFontBold(cf.getWordBold());
    setFontItalic(cf.getWordItalic());
    append(line.word);
    setFontBold(cf.getLineBold());
    setFontItalic(cf.getLineItalic());
    append(line.trail);
    append("\n");

    return li;
}

在获取方法之前解决所有变量

4

0 回答 0