Qt XML查看器?(类似于Notepad++)

Qt XML Viewer? (Like Notepad++)

本文关键字:类似于 Notepad++ XML Qt      更新时间:2023-10-16

我有一个包含QTextBrowserQDialog,这个对话框的唯一功能是在QTextBrowser中显示XML文档。QDialogstd::string作为参数,然后使用QXmlStreamReaderQXmlStreamWriter将该字符串转换为"漂亮打印"格式的xml字符串,然后在UI上将该字符串设置为QTextBrowser

这一切都很好,但理想情况下,我希望能够点击打开/关闭标签,并让它进行一些巧妙的高亮显示,还能够折叠和展开标签,本质上,我想要的是在XML模式下与Notepad++相同的功能。

我想知道是否有人知道有任何Qt C++兼容库可以实现这一功能,或者Qt本身是否有任何东西可以具有这一功能(尽管我仔细查看了一下,但找不到任何东西)。

注意:这个QTextBrowser只是显示它被设置为只读的XML,所以不需要编辑它。我只需要这个功能来查看(如果有区别的话)

欢迎任何建议欢呼

您可以看看QScintilla,它应该非常强大,但它只是Qt的一个端口。

或者例如QSyntaxHighlighter

查看此页面,了解如何使用QSyntaxHighlighter显示xml的示例:http://www.yasinuludag.com/blog/?p=49

这里是QSyntaxHighlighter的重新实现,它突出显示xml,就像notepad++做一样

class XMLHighlighter : public QSyntaxHighlighter
{
public:
    XMLHighlighter(QTextDocument *parent) : QSyntaxHighlighter(parent) {
        HighlightingRule rule;
        // >text<
        QTextCharFormat xmlValueElementFormat;
        xmlValueElementFormat.setForeground(Qt::black);
        xmlValueElementFormat.setFontWeight(QFont::Bold);
        rule.pattern = QRegExp(">[^n]*<");
        rule.format = xmlValueElementFormat;
        highlightingRules.append(rule);
        // keywords
        QTextCharFormat keywordFormat;
        keywordFormat.setForeground(Qt::blue);
        keywordFormat.setFontWeight(QFont::Bold);
        QStringList keywords;
        keywords << "\b?xml\b" << "/>" << ">" << "<";
        foreach (const QString &keyword, keywords) {
            rule.pattern = QRegExp(keyword);
            rule.format = keywordFormat;
            highlightingRules.append(rule);
        }
        // <Text> </Text>
        QTextCharFormat xmlElementFormat;
        xmlElementFormat.setForeground(Qt::blue);
        rule.pattern = QRegExp("\b[A-Za-z0-9_]+(?=[/>])");
        rule.format = xmlElementFormat;
        highlightingRules.append(rule);
        // < Text= >
        QTextCharFormat xmlAttributeFormat;
        xmlAttributeFormat.setForeground(Qt::red);
        rule.pattern = QRegExp("\b[A-Za-z0-9_]+(?=\=)");
        rule.format = xmlAttributeFormat;
        highlightingRules.append(rule);
        // <!-- Text -->
        QTextCharFormat singleLineCommentFormat;
        singleLineCommentFormat.setForeground(Qt::gray);
        rule.pattern = QRegExp("<!--[^n]*-->");
        rule.format = singleLineCommentFormat;
        highlightingRules.append(rule);
        // = "Text"
        QColor valueColor(128, 0, 255);
        valueFormat.setForeground(valueColor);
        valueFormat.setFontWeight(QFont::Bold);
        valueStartExpression.setPattern(""");
        valueEndExpression.setPattern(""(?=[s></])");
    }
    virtual ~XMLHighlighter() {
    }
private:
    struct HighlightingRule
    {
        QRegExp pattern;
        QTextCharFormat format;
    };
    QVector<HighlightingRule> highlightingRules;
    QTextCharFormat valueFormat;
    QRegExp valueStartExpression;
    QRegExp valueEndExpression;
    void highlightBlock(const QString & text)
    {
        //for every pattern
        foreach (const HighlightingRule &rule, highlightingRules)
        {
            QRegExp expression(rule.pattern);
            int index = expression.indexIn(text);
            while(index >= 0) {
                int length = expression.matchedLength();
                this->setFormat(index, length, rule.format);
                index = expression.indexIn(text, index + length);
            }
        }
        this->setCurrentBlockState(0);
        int startIndex = 0;
        if(this->previousBlockState() != 1)
            startIndex = valueStartExpression.indexIn(text);
        while(startIndex >= 0)
        {
            int endIndex = valueEndExpression.indexIn(text, startIndex);
            int commentLength;
            if (endIndex == -1) {
                this->setCurrentBlockState(1);
                commentLength = text.length() - startIndex;
            }
            else {
                commentLength = endIndex - startIndex + valueEndExpression.matchedLength();
            }
            this->setFormat(startIndex, commentLength, valueFormat);
            startIndex = valueStartExpression.indexIn(text, startIndex + commentLength);
        }
    }
};