#ifndef HGMARKDOWNHIGHLIGHTER_H #define HGMARKDOWNHIGHLIGHTER_H #include #include #include #include #include #include extern "C" { #include } QT_BEGIN_NAMESPACE class QTextDocument; QT_END_NAMESPACE struct HighlightingStyle { pmh_element_type type; QTextCharFormat format; }; // One continuous region for a certain markdown highlight style // within a QTextBlock. // Pay attention to the change of HighlightingStyles[] struct HLUnit { // Highlight offset @start and @length with style HighlightingStyles[styleIndex] // within a QTextBlock unsigned long start; unsigned long length; unsigned int styleIndex; }; struct HLUnitStyle { unsigned long start; unsigned long length; QString style; }; // Fenced code block only. struct VCodeBlock { int m_startPos; int m_startBlock; int m_endBlock; QString m_lang; QString m_text; }; // Highlight unit with global position and string style name. struct HLUnitPos { HLUnitPos() : m_position(-1), m_length(-1) { } HLUnitPos(int p_position, int p_length, const QString &p_style) : m_position(p_position), m_length(p_length), m_style(p_style) { } int m_position; int m_length; QString m_style; }; // Denote the region of a certain Markdown element. struct VElementRegion { VElementRegion() : m_startPos(0), m_endPos(0) {} VElementRegion(int p_start, int p_end) : m_startPos(p_start), m_endPos(p_end) {} // The start position of the region in document. int m_startPos; // The end position of the region in document. int m_endPos; // Whether this region contains @p_pos. bool contains(int p_pos) const { return m_startPos <= p_pos && m_endPos >= p_pos; } bool operator==(const VElementRegion &p_other) const { return (m_startPos == p_other.m_startPos && m_endPos == p_other.m_endPos); } bool operator<(const VElementRegion &p_other) const { if (m_startPos < p_other.m_startPos) { return true; } else if (m_startPos == p_other.m_startPos) { return m_endPos <= p_other.m_endPos; } else { return false; } } }; class HGMarkdownHighlighter : public QSyntaxHighlighter { Q_OBJECT public: HGMarkdownHighlighter(const QVector &styles, const QHash &codeBlockStyles, int waitInterval, QTextDocument *parent = 0); ~HGMarkdownHighlighter(); // Request to update highlihgt (re-parse and re-highlight) void setCodeBlockHighlights(const QVector &p_units); const QMap &getPotentialPreviewBlocks() const; signals: void highlightCompleted(); // QVector is implicitly shared. void codeBlocksUpdated(const QVector &p_codeBlocks); // Emitted when image regions have been fetched from a new parsing result. void imageLinksUpdated(const QVector &p_imageRegions); // Emitted when header regions have been fetched from a new parsing result. void headersUpdated(const QVector &p_headerRegions); protected: void highlightBlock(const QString &text) Q_DECL_OVERRIDE; public slots: void updateHighlight(); private slots: void handleContentChange(int position, int charsRemoved, int charsAdded); void timerTimeout(); private: QRegExp codeBlockStartExp; QRegExp codeBlockEndExp; QTextCharFormat codeBlockFormat; QTextCharFormat m_linkFormat; QTextCharFormat m_imageFormat; QTextCharFormat m_colorColumnFormat; QTextDocument *document; QVector highlightingStyles; QHash m_codeBlockStyles; QVector > blockHighlights; // Use another member to store the codeblocks highlights, because the highlight // sequence is blockHighlights, regular-expression-based highlihgts, and then // codeBlockHighlights. // Support fenced code block only. QVector > m_codeBlockHighlights; int m_numOfCodeBlockHighlightsToRecv; // If the ith block contains preview, then the set contains i. // If the set contains i, the ith block may contain preview. // We just use the key. QMap m_potentialPreviewBlocks; // All HTML comment regions. QVector m_commentRegions; // All image link regions. QVector m_imageRegions; // All header regions. // May contains illegal elements. // Sorted by start position. QVector m_headerRegions; // Timer to signal highlightCompleted(). QTimer *m_completeTimer; QAtomicInt parsing; QTimer *timer; int waitInterval; char *content; int capacity; pmh_element **result; static const int initCapacity; void resizeBuffer(int newCap); void highlightCodeBlock(const QString &text); // Highlight links using regular expression. // PEG Markdown Highlight treat URLs with spaces illegal. This function is // intended to complement this. void highlightLinkWithSpacesInURL(const QString &p_text); void parse(); void parseInternal(); void initBlockHighlightFromResult(int nrBlocks); void initBlockHighlihgtOne(unsigned long pos, unsigned long end, int styleIndex); // Return true if there are fenced code blocks and it will call rehighlight() later. // Return false if there is none. bool updateCodeBlocks(); // Fetch all the HTML comment regions from parsing result. void initHtmlCommentRegionsFromResult(); // Fetch all the image link regions from parsing result. void initImageRegionsFromResult(); // Fetch all the header regions from parsing result. void initHeaderRegionsFromResult(); // Whether @p_block is totally inside a HTML comment. bool isBlockInsideCommentRegion(const QTextBlock &p_block) const; // Highlights have been changed. Try to signal highlightCompleted(). void highlightChanged(); // Set the user data of currentBlock(). void updateBlockUserData(int p_blockNum, const QString &p_text); // Highlight color column in code block. void highlightCodeBlockColorColumn(const QString &p_text); }; inline const QMap &HGMarkdownHighlighter::getPotentialPreviewBlocks() const { return m_potentialPreviewBlocks; } #endif