#include "LogHighlighter.hpp" #define TO_EOL "(([\\s\\S]*)|([\\d\\D]*)|([\\w\\W]*))$" #define REGEX_IPV6_ADDR \ R"(\[\s*((([0-9A-Fa-f]{1,4}:){7}([0-9A-Fa-f]{1,4}|:))|(([0-9A-Fa-f]{1,4}:){6}(:[0-9A-Fa-f]{1,4}|((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){5}(((:[0-9A-Fa-f]{1,4}){1,2})|:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){4}(((:[0-9A-Fa-f]{1,4}){1,3})|((:[0-9A-Fa-f]{1,4})?:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){3}(((:[0-9A-Fa-f]{1,4}){1,4})|((:[0-9A-Fa-f]{1,4}){0,2}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){2}(((:[0-9A-Fa-f]{1,4}){1,5})|((:[0-9A-Fa-f]{1,4}){0,3}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){1}(((:[0-9A-Fa-f]{1,4}){1,6})|((:[0-9A-Fa-f]{1,4}){0,4}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(:(((:[0-9A-Fa-f]{1,4}){1,7})|((:[0-9A-Fa-f]{1,4}){0,5}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:)))(%.+)?\s*\])" #define REGEX_IPV4_ADDR \ R"((\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5]))" #define REGEX_PORT_NUMBER R"(([0-9]|[1-9]\d{1,3}|[1-5]\d{4}|6[0-5]{2}[0-3][0-5])*)" namespace Qv2ray::ui { SyntaxHighlighter::SyntaxHighlighter(bool darkMode, QTextDocument *parent) : QSyntaxHighlighter(parent) { HighlightingRule rule; if (darkMode) { tcpudpFormat.setForeground(QColor(0, 200, 230)); ipHostFormat.setForeground(Qt::yellow); warningFormat.setForeground(QColor(255, 160, 15)); warningFormat2.setForeground(Qt::cyan); } else { ipHostFormat.setForeground(QColor(0, 52, 130)); tcpudpFormat.setForeground(QColor(0, 52, 130)); warningFormat.setBackground(QColor(255, 160, 15)); warningFormat.setForeground(Qt::white); warningFormat2.setForeground(Qt::darkCyan); } const static QColor darkGreenColor(10, 180, 0); acceptedFormat.setForeground(darkGreenColor); acceptedFormat.setFontItalic(true); acceptedFormat.setFontWeight(QFont::Bold); rule.pattern = QRegularExpression("\\saccepted\\s"); rule.format = acceptedFormat; highlightingRules.append(rule); // rejectedFormat.setFontWeight(QFont::Bold); rejectedFormat.setBackground(Qt::red); rejectedFormat.setForeground(Qt::white); rejectedFormat.setFontWeight(QFont::Bold); rule.pattern = QRegularExpression("\\srejected\\s" TO_EOL); rule.format = rejectedFormat; highlightingRules.append(rule); // dateFormat.setForeground(darkMode ? Qt::cyan : Qt::darkCyan); rule.pattern = QRegularExpression("\\d\\d\\d\\d/\\d\\d/\\d\\d"); rule.format = dateFormat; highlightingRules.append(rule); // timeFormat.setForeground(darkMode ? Qt::cyan : Qt::darkCyan); rule.pattern = QRegularExpression("\\d\\d:\\d\\d:\\d\\d"); rule.format = timeFormat; highlightingRules.append(rule); // debugFormat.setForeground(Qt::darkGray); rule.pattern = QRegularExpression("\\[D[Ee][Bb][Uu].*?\\]"); rule.format = debugFormat; highlightingRules.append(rule); // infoFormat.setForeground(darkMode ? Qt::lightGray : Qt::darkCyan); rule.pattern = QRegularExpression("\\[I[Nn][Ff][Oo].*?\\]"); rule.format = infoFormat; highlightingRules.append(rule); // warningFormat.setFontWeight(QFont::Bold); warningFormat2.setFontWeight(QFont::Bold); rule.pattern = QRegularExpression("\\[W[Aa][Rr][Nn].*?\\]"); rule.format = warningFormat2; highlightingRules.append(rule); // rule.pattern = QRegularExpression("\\[E[Rr][Rr][Oo].*?\\]"); rule.format = rejectedFormat; highlightingRules.append(rule); // v2rayComponentFormat.setForeground(darkMode ? darkGreenColor : Qt::darkYellow); rule.pattern = QRegularExpression(R"( (\w+\/)+\w+: )"); rule.format = v2rayComponentFormat; highlightingRules.append(rule); // failedFormat.setFontWeight(QFont::Bold); failedFormat.setBackground(Qt::red); failedFormat.setForeground(Qt::white); rule.pattern = QRegularExpression("failed"); rule.format = failedFormat; highlightingRules.append(rule); // rule.pattern = QRegularExpression(">>>>+"); rule.format = warningFormat; highlightingRules.append(rule); // rule.pattern = QRegularExpression("<<<<+"); rule.format = warningFormat; highlightingRules.append(rule); { // IP IPv6 Host; rule.pattern = QRegularExpression(REGEX_IPV4_ADDR ":" REGEX_PORT_NUMBER); rule.pattern.setPatternOptions(QRegularExpression::ExtendedPatternSyntaxOption); rule.format = ipHostFormat; highlightingRules.append(rule); // rule.pattern = QRegularExpression(REGEX_IPV6_ADDR ":" REGEX_PORT_NUMBER); rule.pattern.setPatternOptions(QRegularExpression::ExtendedPatternSyntaxOption); rule.format = ipHostFormat; highlightingRules.append(rule); // rule.pattern = QRegularExpression("([a-zA-Z0-9]([a-zA-Z0-9\\-]{0,61}[a-zA-Z0-9])?\\.)+[a-zA-Z]{2,6}(/|):" REGEX_PORT_NUMBER); rule.pattern.setPatternOptions(QRegularExpression::PatternOption::ExtendedPatternSyntaxOption); rule.format = ipHostFormat; highlightingRules.append(rule); } for (const auto &pattern: {"tcp:", "udp:"}) { tcpudpFormat.setFontWeight(QFont::Bold); rule.pattern = QRegularExpression(pattern); rule.format = tcpudpFormat; highlightingRules.append(rule); } } void SyntaxHighlighter::highlightBlock(const QString &text) { for (const HighlightingRule &rule: highlightingRules) { QRegularExpressionMatchIterator matchIterator = rule.pattern.globalMatch(text); while (matchIterator.hasNext()) { QRegularExpressionMatch match = matchIterator.next(); setFormat(match.capturedStart(), match.capturedLength(), rule.format); } } setCurrentBlockState(0); } } // namespace Qv2ray::ui