mirror of
https://gitee.com/vnotex/vnote.git
synced 2025-07-05 22:09:52 +08:00

- Won't treat all the text to the end as code block once we add a new line within a code block; - Fix color column by adding the code block indentation bias.
298 lines
8.0 KiB
C++
298 lines
8.0 KiB
C++
#ifndef HGMARKDOWNHIGHLIGHTER_H
|
|
#define HGMARKDOWNHIGHLIGHTER_H
|
|
|
|
#include <QTextCharFormat>
|
|
#include <QSyntaxHighlighter>
|
|
#include <QAtomicInt>
|
|
#include <QMap>
|
|
#include <QSet>
|
|
#include <QString>
|
|
|
|
#include "vtextblockdata.h"
|
|
|
|
extern "C" {
|
|
#include <pmh_parser.h>
|
|
}
|
|
|
|
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<HighlightingStyle> &styles,
|
|
const QHash<QString, QTextCharFormat> &codeBlockStyles,
|
|
int waitInterval,
|
|
QTextDocument *parent = 0);
|
|
~HGMarkdownHighlighter();
|
|
|
|
// Request to update highlihgt (re-parse and re-highlight)
|
|
void setCodeBlockHighlights(const QVector<HLUnitPos> &p_units);
|
|
|
|
const QMap<int, bool> &getPotentialPreviewBlocks() const;
|
|
|
|
const QVector<VElementRegion> &getHeaderRegions() const;
|
|
|
|
const QSet<int> &getPossiblePreviewBlocks() const;
|
|
|
|
void clearPossiblePreviewBlocks(const QVector<int> &p_blocksToClear);
|
|
|
|
// Parse and only update the highlight results for rehighlight().
|
|
void updateHighlightFast();
|
|
|
|
signals:
|
|
void highlightCompleted();
|
|
|
|
// QVector is implicitly shared.
|
|
void codeBlocksUpdated(const QVector<VCodeBlock> &p_codeBlocks);
|
|
|
|
// Emitted when image regions have been fetched from a new parsing result.
|
|
void imageLinksUpdated(const QVector<VElementRegion> &p_imageRegions);
|
|
|
|
// Emitted when header regions have been fetched from a new parsing result.
|
|
void headersUpdated(const QVector<VElementRegion> &p_headerRegions);
|
|
|
|
protected:
|
|
void highlightBlock(const QString &text) Q_DECL_OVERRIDE;
|
|
|
|
public slots:
|
|
// Parse and rehighlight immediately.
|
|
void updateHighlight();
|
|
|
|
private slots:
|
|
void handleContentChange(int position, int charsRemoved, int charsAdded);
|
|
|
|
// @p_fast: if true, just parse and update styles.
|
|
void startParseAndHighlight(bool p_fast = false);
|
|
|
|
private:
|
|
QRegExp codeBlockStartExp;
|
|
QRegExp codeBlockEndExp;
|
|
QTextCharFormat codeBlockFormat;
|
|
QTextCharFormat m_linkFormat;
|
|
QTextCharFormat m_imageFormat;
|
|
QTextCharFormat m_colorColumnFormat;
|
|
|
|
QTextDocument *document;
|
|
QVector<HighlightingStyle> highlightingStyles;
|
|
QHash<QString, QTextCharFormat> m_codeBlockStyles;
|
|
QVector<QVector<HLUnit> > 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<QVector<HLUnitStyle> > m_codeBlockHighlights;
|
|
|
|
int m_numOfCodeBlockHighlightsToRecv;
|
|
|
|
// All HTML comment regions.
|
|
QVector<VElementRegion> m_commentRegions;
|
|
|
|
// All image link regions.
|
|
QVector<VElementRegion> m_imageRegions;
|
|
|
|
// All header regions.
|
|
// May contains illegal elements.
|
|
// Sorted by start position.
|
|
QVector<VElementRegion> m_headerRegions;
|
|
|
|
// Timer to signal highlightCompleted().
|
|
QTimer *m_completeTimer;
|
|
|
|
QAtomicInt parsing;
|
|
|
|
// Whether highlight results for blocks are ready.
|
|
bool m_blockHLResultReady;
|
|
|
|
QTimer *timer;
|
|
int waitInterval;
|
|
|
|
// Block number of those blocks which possible contains previewed image.
|
|
QSet<int> m_possiblePreviewBlocks;
|
|
|
|
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(bool p_fast = false);
|
|
|
|
void parseInternal();
|
|
|
|
// Init highlight elements for all the blocks from parse results.
|
|
void initBlockHighlightFromResult(int nrBlocks);
|
|
|
|
// Init highlight elements for blocks from one parse result.
|
|
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);
|
|
|
|
// Check if [p_pos, p_end) is a valid header.
|
|
bool isValidHeader(unsigned long p_pos, unsigned long p_end);
|
|
|
|
VTextBlockData *currentBlockData() const;
|
|
|
|
VTextBlockData *previousBlockData() const;
|
|
};
|
|
|
|
inline const QVector<VElementRegion> &HGMarkdownHighlighter::getHeaderRegions() const
|
|
{
|
|
return m_headerRegions;
|
|
}
|
|
|
|
inline const QSet<int> &HGMarkdownHighlighter::getPossiblePreviewBlocks() const
|
|
{
|
|
return m_possiblePreviewBlocks;
|
|
}
|
|
|
|
inline void HGMarkdownHighlighter::clearPossiblePreviewBlocks(const QVector<int> &p_blocksToClear)
|
|
{
|
|
for (auto i : p_blocksToClear) {
|
|
m_possiblePreviewBlocks.remove(i);
|
|
}
|
|
}
|
|
|
|
inline VTextBlockData *HGMarkdownHighlighter::currentBlockData() const
|
|
{
|
|
return static_cast<VTextBlockData *>(currentBlockUserData());
|
|
}
|
|
|
|
inline VTextBlockData *HGMarkdownHighlighter::previousBlockData() const
|
|
{
|
|
QTextBlock block = currentBlock().previous();
|
|
if (!block.isValid()) {
|
|
return NULL;
|
|
}
|
|
|
|
return static_cast<VTextBlockData *>(block.userData());
|
|
}
|
|
#endif
|