1

很明显,我可以使用 EM_GETCHARFORMAT 一次完成一个字符,但它非常慢。

一种想法是以某种方式使用 ITextDocument/ITextFont 接口,另一种是使用 EM_STREAMOUT 消息并手动解析 RTF。但我无法决定哪种方法更好,并且对实现细节非常模糊。将不胜感激任何帮助,谢谢!

4

1 回答 1

2

我找到了一个让我满意并认为会与您分享的解决方案:

ITextRange界面包含一个非常有用的方法Expand,可用于查找连续运行的常量字符 ( tomCharFormat) 和段落 ( tomParaFormat) 格式。

这是一些示例代码(警告:代码是概念验证意大利面条,没有任何错误处理,根据需要应用重构):

    // Get necessary interfaces
    IRichEditOle* ole;
    SendMessage(hwndRichEdit, EM_GETOLEINTERFACE, 0, (LPARAM)&ole);

    ITextDocument* doc;
    ole->QueryInterface(__uuidof(ITextDocument), (void**)&doc);

    long start = 0;

    // Get total length:        
    ITextRange* range;
    doc->Range(start, start, &range);
    range->Expand(tomStory, NULL);
    long eof;
    range->GetEnd(&eof);

    // Extract formatting:

    struct TextCharFormat { long start, length; DWORD effects; }
    std::vector<TextCharFormat> fs;

    while(start < eof - 1)
    {
        doc->Range(start, start, &range);

        long n;
        range->Expand(tomCharFormat, &n); // <-- Magic happens here

        ITextFont* font;
        range->GetFont(&font);

        DWORD effects = 0;
        long flag;

        font->GetBold(&flag);
        if (flag == tomTrue) effects |= CFE_BOLD;

        font->GetItalic(&flag);
        if (flag == tomTrue) effects |= CFE_ITALIC;

        font->GetUnderline(&flag);
        if (flag == tomSingle) effects |= CFE_UNDERLINE;

        font->GetStrikeThrough(&flag);
        if (flag == tomTrue) effects |= CFE_STRIKEOUT;

        if (effects)
        {
            TextCharFormat f = { start, n, effects };
            fs.push_back(f);
        }
        start += n;
    }
于 2011-05-03T06:37:04.470 回答