2085 lines
61 KiB
C++
2085 lines
61 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#include "ui/text/text_entity.h"
|
|
|
|
#include "auth_session.h"
|
|
#include "lang/lang_tag.h"
|
|
#include "base/qthelp_url.h"
|
|
#include "ui/emoji_config.h"
|
|
#include "data/data_user.h"
|
|
#include "data/data_session.h"
|
|
|
|
namespace TextUtilities {
|
|
namespace {
|
|
|
|
QString ExpressionMailNameAtEnd() {
|
|
// Matches email first part (before '@') at the end of the string.
|
|
// First we find a domain without protocol (like "gmail.com"), then
|
|
// we find '@' before it and then we look for the name before '@'.
|
|
return qsl("[a-zA-Z\\-_\\.0-9]{1,256}$");
|
|
}
|
|
|
|
QString Quotes() {
|
|
// UTF8 quotes and ellipsis
|
|
return QString::fromUtf8("\xC2\xAB\xC2\xBB\xE2\x80\x9C\xE2\x80\x9D\xE2\x80\x98\xE2\x80\x99\xE2\x80\xA6");
|
|
}
|
|
|
|
QString ExpressionSeparators(const QString &additional) {
|
|
static const auto quotes = Quotes();
|
|
return qsl("\\s\\.,:;<>|'\"\\[\\]\\{\\}\\~\\!\\?\\%\\^\\(\\)\\-\\+=\\x10") + quotes + additional;
|
|
}
|
|
|
|
QString Separators(const QString &additional) {
|
|
static const auto quotes = Quotes();
|
|
return qsl(" \x10\n\r\t.,:;<>|'\"[]{}~!?%^()-+=")
|
|
+ QChar(0xfdd0) // QTextBeginningOfFrame
|
|
+ QChar(0xfdd1) // QTextEndOfFrame
|
|
+ QChar(QChar::ParagraphSeparator)
|
|
+ QChar(QChar::LineSeparator)
|
|
+ quotes
|
|
+ additional;
|
|
}
|
|
|
|
QString SeparatorsBold() {
|
|
return Separators(qsl("`/"));
|
|
}
|
|
|
|
QString SeparatorsItalic() {
|
|
return Separators(qsl("`*/"));
|
|
}
|
|
|
|
QString SeparatorsMono() {
|
|
return Separators(qsl("*/"));
|
|
}
|
|
|
|
QString ExpressionHashtag() {
|
|
return qsl("(^|[") + ExpressionSeparators(qsl("`\\*/")) + qsl("])#[\\w]{2,64}([\\W]|$)");
|
|
}
|
|
|
|
QString ExpressionHashtagExclude() {
|
|
return qsl("^#?\\d+$");
|
|
}
|
|
|
|
QString ExpressionMention() {
|
|
return qsl("(^|[") + ExpressionSeparators(qsl("`\\*/")) + qsl("])@[A-Za-z_0-9]{1,32}([\\W]|$)");
|
|
}
|
|
|
|
QString ExpressionBotCommand() {
|
|
return qsl("(^|[") + ExpressionSeparators(qsl("`\\*")) + qsl("])/[A-Za-z_0-9]{1,64}(@[A-Za-z_0-9]{5,32})?([\\W]|$)");
|
|
}
|
|
|
|
QRegularExpression CreateRegExp(const QString &expression) {
|
|
auto result = QRegularExpression(
|
|
expression,
|
|
QRegularExpression::UseUnicodePropertiesOption);
|
|
#ifndef OS_MAC_OLD
|
|
result.optimize();
|
|
#endif // OS_MAC_OLD
|
|
return result;
|
|
}
|
|
|
|
QSet<int32> CreateValidProtocols() {
|
|
auto result = QSet<int32>();
|
|
auto addOne = [&result](const QString &string) {
|
|
result.insert(hashCrc32(string.constData(), string.size() * sizeof(QChar)));
|
|
};
|
|
addOne(qsl("itmss")); // itunes
|
|
addOne(qsl("http"));
|
|
addOne(qsl("https"));
|
|
addOne(qsl("ftp"));
|
|
addOne(qsl("tg")); // local urls
|
|
return result;
|
|
}
|
|
|
|
QSet<int32> CreateValidTopDomains() {
|
|
auto result = QSet<int32>();
|
|
auto addOne = [&result](const QString &string) {
|
|
result.insert(hashCrc32(string.constData(), string.size() * sizeof(QChar)));
|
|
};
|
|
addOne(qsl("ac"));
|
|
addOne(qsl("ad"));
|
|
addOne(qsl("ae"));
|
|
addOne(qsl("af"));
|
|
addOne(qsl("ag"));
|
|
addOne(qsl("ai"));
|
|
addOne(qsl("al"));
|
|
addOne(qsl("am"));
|
|
addOne(qsl("an"));
|
|
addOne(qsl("ao"));
|
|
addOne(qsl("aq"));
|
|
addOne(qsl("ar"));
|
|
addOne(qsl("as"));
|
|
addOne(qsl("at"));
|
|
addOne(qsl("au"));
|
|
addOne(qsl("aw"));
|
|
addOne(qsl("ax"));
|
|
addOne(qsl("az"));
|
|
addOne(qsl("ba"));
|
|
addOne(qsl("bb"));
|
|
addOne(qsl("bd"));
|
|
addOne(qsl("be"));
|
|
addOne(qsl("bf"));
|
|
addOne(qsl("bg"));
|
|
addOne(qsl("bh"));
|
|
addOne(qsl("bi"));
|
|
addOne(qsl("bj"));
|
|
addOne(qsl("bm"));
|
|
addOne(qsl("bn"));
|
|
addOne(qsl("bo"));
|
|
addOne(qsl("br"));
|
|
addOne(qsl("bs"));
|
|
addOne(qsl("bt"));
|
|
addOne(qsl("bv"));
|
|
addOne(qsl("bw"));
|
|
addOne(qsl("by"));
|
|
addOne(qsl("bz"));
|
|
addOne(qsl("ca"));
|
|
addOne(qsl("cc"));
|
|
addOne(qsl("cd"));
|
|
addOne(qsl("cf"));
|
|
addOne(qsl("cg"));
|
|
addOne(qsl("ch"));
|
|
addOne(qsl("ci"));
|
|
addOne(qsl("ck"));
|
|
addOne(qsl("cl"));
|
|
addOne(qsl("cm"));
|
|
addOne(qsl("cn"));
|
|
addOne(qsl("co"));
|
|
addOne(qsl("cr"));
|
|
addOne(qsl("cu"));
|
|
addOne(qsl("cv"));
|
|
addOne(qsl("cx"));
|
|
addOne(qsl("cy"));
|
|
addOne(qsl("cz"));
|
|
addOne(qsl("de"));
|
|
addOne(qsl("dj"));
|
|
addOne(qsl("dk"));
|
|
addOne(qsl("dm"));
|
|
addOne(qsl("do"));
|
|
addOne(qsl("dz"));
|
|
addOne(qsl("ec"));
|
|
addOne(qsl("ee"));
|
|
addOne(qsl("eg"));
|
|
addOne(qsl("eh"));
|
|
addOne(qsl("er"));
|
|
addOne(qsl("es"));
|
|
addOne(qsl("et"));
|
|
addOne(qsl("eu"));
|
|
addOne(qsl("fi"));
|
|
addOne(qsl("fj"));
|
|
addOne(qsl("fk"));
|
|
addOne(qsl("fm"));
|
|
addOne(qsl("fo"));
|
|
addOne(qsl("fr"));
|
|
addOne(qsl("ga"));
|
|
addOne(qsl("gd"));
|
|
addOne(qsl("ge"));
|
|
addOne(qsl("gf"));
|
|
addOne(qsl("gg"));
|
|
addOne(qsl("gh"));
|
|
addOne(qsl("gi"));
|
|
addOne(qsl("gl"));
|
|
addOne(qsl("gm"));
|
|
addOne(qsl("gn"));
|
|
addOne(qsl("gp"));
|
|
addOne(qsl("gq"));
|
|
addOne(qsl("gr"));
|
|
addOne(qsl("gs"));
|
|
addOne(qsl("gt"));
|
|
addOne(qsl("gu"));
|
|
addOne(qsl("gw"));
|
|
addOne(qsl("gy"));
|
|
addOne(qsl("hk"));
|
|
addOne(qsl("hm"));
|
|
addOne(qsl("hn"));
|
|
addOne(qsl("hr"));
|
|
addOne(qsl("ht"));
|
|
addOne(qsl("hu"));
|
|
addOne(qsl("id"));
|
|
addOne(qsl("ie"));
|
|
addOne(qsl("il"));
|
|
addOne(qsl("im"));
|
|
addOne(qsl("in"));
|
|
addOne(qsl("io"));
|
|
addOne(qsl("iq"));
|
|
addOne(qsl("ir"));
|
|
addOne(qsl("is"));
|
|
addOne(qsl("it"));
|
|
addOne(qsl("je"));
|
|
addOne(qsl("jm"));
|
|
addOne(qsl("jo"));
|
|
addOne(qsl("jp"));
|
|
addOne(qsl("ke"));
|
|
addOne(qsl("kg"));
|
|
addOne(qsl("kh"));
|
|
addOne(qsl("ki"));
|
|
addOne(qsl("km"));
|
|
addOne(qsl("kn"));
|
|
addOne(qsl("kp"));
|
|
addOne(qsl("kr"));
|
|
addOne(qsl("kw"));
|
|
addOne(qsl("ky"));
|
|
addOne(qsl("kz"));
|
|
addOne(qsl("la"));
|
|
addOne(qsl("lb"));
|
|
addOne(qsl("lc"));
|
|
addOne(qsl("li"));
|
|
addOne(qsl("lk"));
|
|
addOne(qsl("lr"));
|
|
addOne(qsl("ls"));
|
|
addOne(qsl("lt"));
|
|
addOne(qsl("lu"));
|
|
addOne(qsl("lv"));
|
|
addOne(qsl("ly"));
|
|
addOne(qsl("ma"));
|
|
addOne(qsl("mc"));
|
|
addOne(qsl("md"));
|
|
addOne(qsl("me"));
|
|
addOne(qsl("mg"));
|
|
addOne(qsl("mh"));
|
|
addOne(qsl("mk"));
|
|
addOne(qsl("ml"));
|
|
addOne(qsl("mm"));
|
|
addOne(qsl("mn"));
|
|
addOne(qsl("mo"));
|
|
addOne(qsl("mp"));
|
|
addOne(qsl("mq"));
|
|
addOne(qsl("mr"));
|
|
addOne(qsl("ms"));
|
|
addOne(qsl("mt"));
|
|
addOne(qsl("mu"));
|
|
addOne(qsl("mv"));
|
|
addOne(qsl("mw"));
|
|
addOne(qsl("mx"));
|
|
addOne(qsl("my"));
|
|
addOne(qsl("mz"));
|
|
addOne(qsl("na"));
|
|
addOne(qsl("nc"));
|
|
addOne(qsl("ne"));
|
|
addOne(qsl("nf"));
|
|
addOne(qsl("ng"));
|
|
addOne(qsl("ni"));
|
|
addOne(qsl("nl"));
|
|
addOne(qsl("no"));
|
|
addOne(qsl("np"));
|
|
addOne(qsl("nr"));
|
|
addOne(qsl("nu"));
|
|
addOne(qsl("nz"));
|
|
addOne(qsl("om"));
|
|
addOne(qsl("pa"));
|
|
addOne(qsl("pe"));
|
|
addOne(qsl("pf"));
|
|
addOne(qsl("pg"));
|
|
addOne(qsl("ph"));
|
|
addOne(qsl("pk"));
|
|
addOne(qsl("pl"));
|
|
addOne(qsl("pm"));
|
|
addOne(qsl("pn"));
|
|
addOne(qsl("pr"));
|
|
addOne(qsl("ps"));
|
|
addOne(qsl("pt"));
|
|
addOne(qsl("pw"));
|
|
addOne(qsl("py"));
|
|
addOne(qsl("qa"));
|
|
addOne(qsl("re"));
|
|
addOne(qsl("ro"));
|
|
addOne(qsl("ru"));
|
|
addOne(qsl("rs"));
|
|
addOne(qsl("rw"));
|
|
addOne(qsl("sa"));
|
|
addOne(qsl("sb"));
|
|
addOne(qsl("sc"));
|
|
addOne(qsl("sd"));
|
|
addOne(qsl("se"));
|
|
addOne(qsl("sg"));
|
|
addOne(qsl("sh"));
|
|
addOne(qsl("si"));
|
|
addOne(qsl("sj"));
|
|
addOne(qsl("sk"));
|
|
addOne(qsl("sl"));
|
|
addOne(qsl("sm"));
|
|
addOne(qsl("sn"));
|
|
addOne(qsl("so"));
|
|
addOne(qsl("sr"));
|
|
addOne(qsl("ss"));
|
|
addOne(qsl("st"));
|
|
addOne(qsl("su"));
|
|
addOne(qsl("sv"));
|
|
addOne(qsl("sx"));
|
|
addOne(qsl("sy"));
|
|
addOne(qsl("sz"));
|
|
addOne(qsl("tc"));
|
|
addOne(qsl("td"));
|
|
addOne(qsl("tf"));
|
|
addOne(qsl("tg"));
|
|
addOne(qsl("th"));
|
|
addOne(qsl("tj"));
|
|
addOne(qsl("tk"));
|
|
addOne(qsl("tl"));
|
|
addOne(qsl("tm"));
|
|
addOne(qsl("tn"));
|
|
addOne(qsl("to"));
|
|
addOne(qsl("tp"));
|
|
addOne(qsl("tr"));
|
|
addOne(qsl("tt"));
|
|
addOne(qsl("tv"));
|
|
addOne(qsl("tw"));
|
|
addOne(qsl("tz"));
|
|
addOne(qsl("ua"));
|
|
addOne(qsl("ug"));
|
|
addOne(qsl("uk"));
|
|
addOne(qsl("um"));
|
|
addOne(qsl("us"));
|
|
addOne(qsl("uy"));
|
|
addOne(qsl("uz"));
|
|
addOne(qsl("va"));
|
|
addOne(qsl("vc"));
|
|
addOne(qsl("ve"));
|
|
addOne(qsl("vg"));
|
|
addOne(qsl("vi"));
|
|
addOne(qsl("vn"));
|
|
addOne(qsl("vu"));
|
|
addOne(qsl("wf"));
|
|
addOne(qsl("ws"));
|
|
addOne(qsl("ye"));
|
|
addOne(qsl("yt"));
|
|
addOne(qsl("yu"));
|
|
addOne(qsl("za"));
|
|
addOne(qsl("zm"));
|
|
addOne(qsl("zw"));
|
|
addOne(qsl("arpa"));
|
|
addOne(qsl("aero"));
|
|
addOne(qsl("asia"));
|
|
addOne(qsl("biz"));
|
|
addOne(qsl("cat"));
|
|
addOne(qsl("com"));
|
|
addOne(qsl("coop"));
|
|
addOne(qsl("info"));
|
|
addOne(qsl("int"));
|
|
addOne(qsl("jobs"));
|
|
addOne(qsl("mobi"));
|
|
addOne(qsl("museum"));
|
|
addOne(qsl("name"));
|
|
addOne(qsl("net"));
|
|
addOne(qsl("org"));
|
|
addOne(qsl("post"));
|
|
addOne(qsl("pro"));
|
|
addOne(qsl("tel"));
|
|
addOne(qsl("travel"));
|
|
addOne(qsl("xxx"));
|
|
addOne(qsl("edu"));
|
|
addOne(qsl("gov"));
|
|
addOne(qsl("mil"));
|
|
addOne(qsl("local"));
|
|
addOne(qsl("xn--lgbbat1ad8j"));
|
|
addOne(qsl("xn--54b7fta0cc"));
|
|
addOne(qsl("xn--fiqs8s"));
|
|
addOne(qsl("xn--fiqz9s"));
|
|
addOne(qsl("xn--wgbh1c"));
|
|
addOne(qsl("xn--node"));
|
|
addOne(qsl("xn--j6w193g"));
|
|
addOne(qsl("xn--h2brj9c"));
|
|
addOne(qsl("xn--mgbbh1a71e"));
|
|
addOne(qsl("xn--fpcrj9c3d"));
|
|
addOne(qsl("xn--gecrj9c"));
|
|
addOne(qsl("xn--s9brj9c"));
|
|
addOne(qsl("xn--xkc2dl3a5ee0h"));
|
|
addOne(qsl("xn--45brj9c"));
|
|
addOne(qsl("xn--mgba3a4f16a"));
|
|
addOne(qsl("xn--mgbayh7gpa"));
|
|
addOne(qsl("xn--80ao21a"));
|
|
addOne(qsl("xn--mgbx4cd0ab"));
|
|
addOne(qsl("xn--l1acc"));
|
|
addOne(qsl("xn--mgbc0a9azcg"));
|
|
addOne(qsl("xn--mgb9awbf"));
|
|
addOne(qsl("xn--mgbai9azgqp6j"));
|
|
addOne(qsl("xn--ygbi2ammx"));
|
|
addOne(qsl("xn--wgbl6a"));
|
|
addOne(qsl("xn--p1ai"));
|
|
addOne(qsl("xn--mgberp4a5d4ar"));
|
|
addOne(qsl("xn--90a3ac"));
|
|
addOne(qsl("xn--yfro4i67o"));
|
|
addOne(qsl("xn--clchc0ea0b2g2a9gcd"));
|
|
addOne(qsl("xn--3e0b707e"));
|
|
addOne(qsl("xn--fzc2c9e2c"));
|
|
addOne(qsl("xn--xkc2al3hye2a"));
|
|
addOne(qsl("xn--mgbtf8fl"));
|
|
addOne(qsl("xn--kprw13d"));
|
|
addOne(qsl("xn--kpry57d"));
|
|
addOne(qsl("xn--o3cw4h"));
|
|
addOne(qsl("xn--pgbs0dh"));
|
|
addOne(qsl("xn--j1amh"));
|
|
addOne(qsl("xn--mgbaam7a8h"));
|
|
addOne(qsl("xn--mgb2ddes"));
|
|
addOne(qsl("xn--ogbpf8fl"));
|
|
addOne(QString::fromUtf8("\xd1\x80\xd1\x84"));
|
|
return result;
|
|
}
|
|
|
|
// accent char list taken from https://github.com/aristus/accent-folding
|
|
inline QChar RemoveOneAccent(uint32 code) {
|
|
switch (code) {
|
|
case 7834: return QChar(97);
|
|
case 193: return QChar(97);
|
|
case 225: return QChar(97);
|
|
case 192: return QChar(97);
|
|
case 224: return QChar(97);
|
|
case 258: return QChar(97);
|
|
case 259: return QChar(97);
|
|
case 7854: return QChar(97);
|
|
case 7855: return QChar(97);
|
|
case 7856: return QChar(97);
|
|
case 7857: return QChar(97);
|
|
case 7860: return QChar(97);
|
|
case 7861: return QChar(97);
|
|
case 7858: return QChar(97);
|
|
case 7859: return QChar(97);
|
|
case 194: return QChar(97);
|
|
case 226: return QChar(97);
|
|
case 7844: return QChar(97);
|
|
case 7845: return QChar(97);
|
|
case 7846: return QChar(97);
|
|
case 7847: return QChar(97);
|
|
case 7850: return QChar(97);
|
|
case 7851: return QChar(97);
|
|
case 7848: return QChar(97);
|
|
case 7849: return QChar(97);
|
|
case 461: return QChar(97);
|
|
case 462: return QChar(97);
|
|
case 197: return QChar(97);
|
|
case 229: return QChar(97);
|
|
case 506: return QChar(97);
|
|
case 507: return QChar(97);
|
|
case 196: return QChar(97);
|
|
case 228: return QChar(97);
|
|
case 478: return QChar(97);
|
|
case 479: return QChar(97);
|
|
case 195: return QChar(97);
|
|
case 227: return QChar(97);
|
|
case 550: return QChar(97);
|
|
case 551: return QChar(97);
|
|
case 480: return QChar(97);
|
|
case 481: return QChar(97);
|
|
case 260: return QChar(97);
|
|
case 261: return QChar(97);
|
|
case 256: return QChar(97);
|
|
case 257: return QChar(97);
|
|
case 7842: return QChar(97);
|
|
case 7843: return QChar(97);
|
|
case 512: return QChar(97);
|
|
case 513: return QChar(97);
|
|
case 514: return QChar(97);
|
|
case 515: return QChar(97);
|
|
case 7840: return QChar(97);
|
|
case 7841: return QChar(97);
|
|
case 7862: return QChar(97);
|
|
case 7863: return QChar(97);
|
|
case 7852: return QChar(97);
|
|
case 7853: return QChar(97);
|
|
case 7680: return QChar(97);
|
|
case 7681: return QChar(97);
|
|
case 570: return QChar(97);
|
|
case 11365: return QChar(97);
|
|
case 508: return QChar(97);
|
|
case 509: return QChar(97);
|
|
case 482: return QChar(97);
|
|
case 483: return QChar(97);
|
|
case 7682: return QChar(98);
|
|
case 7683: return QChar(98);
|
|
case 7684: return QChar(98);
|
|
case 7685: return QChar(98);
|
|
case 7686: return QChar(98);
|
|
case 7687: return QChar(98);
|
|
case 579: return QChar(98);
|
|
case 384: return QChar(98);
|
|
case 7532: return QChar(98);
|
|
case 385: return QChar(98);
|
|
case 595: return QChar(98);
|
|
case 386: return QChar(98);
|
|
case 387: return QChar(98);
|
|
case 262: return QChar(99);
|
|
case 263: return QChar(99);
|
|
case 264: return QChar(99);
|
|
case 265: return QChar(99);
|
|
case 268: return QChar(99);
|
|
case 269: return QChar(99);
|
|
case 266: return QChar(99);
|
|
case 267: return QChar(99);
|
|
case 199: return QChar(99);
|
|
case 231: return QChar(99);
|
|
case 7688: return QChar(99);
|
|
case 7689: return QChar(99);
|
|
case 571: return QChar(99);
|
|
case 572: return QChar(99);
|
|
case 391: return QChar(99);
|
|
case 392: return QChar(99);
|
|
case 597: return QChar(99);
|
|
case 270: return QChar(100);
|
|
case 271: return QChar(100);
|
|
case 7690: return QChar(100);
|
|
case 7691: return QChar(100);
|
|
case 7696: return QChar(100);
|
|
case 7697: return QChar(100);
|
|
case 7692: return QChar(100);
|
|
case 7693: return QChar(100);
|
|
case 7698: return QChar(100);
|
|
case 7699: return QChar(100);
|
|
case 7694: return QChar(100);
|
|
case 7695: return QChar(100);
|
|
case 272: return QChar(100);
|
|
case 273: return QChar(100);
|
|
case 7533: return QChar(100);
|
|
case 393: return QChar(100);
|
|
case 598: return QChar(100);
|
|
case 394: return QChar(100);
|
|
case 599: return QChar(100);
|
|
case 395: return QChar(100);
|
|
case 396: return QChar(100);
|
|
case 545: return QChar(100);
|
|
case 240: return QChar(100);
|
|
case 201: return QChar(101);
|
|
case 399: return QChar(101);
|
|
case 398: return QChar(101);
|
|
case 477: return QChar(101);
|
|
case 233: return QChar(101);
|
|
case 200: return QChar(101);
|
|
case 232: return QChar(101);
|
|
case 276: return QChar(101);
|
|
case 277: return QChar(101);
|
|
case 202: return QChar(101);
|
|
case 234: return QChar(101);
|
|
case 7870: return QChar(101);
|
|
case 7871: return QChar(101);
|
|
case 7872: return QChar(101);
|
|
case 7873: return QChar(101);
|
|
case 7876: return QChar(101);
|
|
case 7877: return QChar(101);
|
|
case 7874: return QChar(101);
|
|
case 7875: return QChar(101);
|
|
case 282: return QChar(101);
|
|
case 283: return QChar(101);
|
|
case 203: return QChar(101);
|
|
case 235: return QChar(101);
|
|
case 7868: return QChar(101);
|
|
case 7869: return QChar(101);
|
|
case 278: return QChar(101);
|
|
case 279: return QChar(101);
|
|
case 552: return QChar(101);
|
|
case 553: return QChar(101);
|
|
case 7708: return QChar(101);
|
|
case 7709: return QChar(101);
|
|
case 280: return QChar(101);
|
|
case 281: return QChar(101);
|
|
case 274: return QChar(101);
|
|
case 275: return QChar(101);
|
|
case 7702: return QChar(101);
|
|
case 7703: return QChar(101);
|
|
case 7700: return QChar(101);
|
|
case 7701: return QChar(101);
|
|
case 7866: return QChar(101);
|
|
case 7867: return QChar(101);
|
|
case 516: return QChar(101);
|
|
case 517: return QChar(101);
|
|
case 518: return QChar(101);
|
|
case 519: return QChar(101);
|
|
case 7864: return QChar(101);
|
|
case 7865: return QChar(101);
|
|
case 7878: return QChar(101);
|
|
case 7879: return QChar(101);
|
|
case 7704: return QChar(101);
|
|
case 7705: return QChar(101);
|
|
case 7706: return QChar(101);
|
|
case 7707: return QChar(101);
|
|
case 582: return QChar(101);
|
|
case 583: return QChar(101);
|
|
case 602: return QChar(101);
|
|
case 605: return QChar(101);
|
|
case 7710: return QChar(102);
|
|
case 7711: return QChar(102);
|
|
case 7534: return QChar(102);
|
|
case 401: return QChar(102);
|
|
case 402: return QChar(102);
|
|
case 500: return QChar(103);
|
|
case 501: return QChar(103);
|
|
case 286: return QChar(103);
|
|
case 287: return QChar(103);
|
|
case 284: return QChar(103);
|
|
case 285: return QChar(103);
|
|
case 486: return QChar(103);
|
|
case 487: return QChar(103);
|
|
case 288: return QChar(103);
|
|
case 289: return QChar(103);
|
|
case 290: return QChar(103);
|
|
case 291: return QChar(103);
|
|
case 7712: return QChar(103);
|
|
case 7713: return QChar(103);
|
|
case 484: return QChar(103);
|
|
case 485: return QChar(103);
|
|
case 403: return QChar(103);
|
|
case 608: return QChar(103);
|
|
case 292: return QChar(104);
|
|
case 293: return QChar(104);
|
|
case 542: return QChar(104);
|
|
case 543: return QChar(104);
|
|
case 7718: return QChar(104);
|
|
case 7719: return QChar(104);
|
|
case 7714: return QChar(104);
|
|
case 7715: return QChar(104);
|
|
case 7720: return QChar(104);
|
|
case 7721: return QChar(104);
|
|
case 7716: return QChar(104);
|
|
case 7717: return QChar(104);
|
|
case 7722: return QChar(104);
|
|
case 7723: return QChar(104);
|
|
case 817: return QChar(104);
|
|
case 7830: return QChar(104);
|
|
case 294: return QChar(104);
|
|
case 295: return QChar(104);
|
|
case 11367: return QChar(104);
|
|
case 11368: return QChar(104);
|
|
case 205: return QChar(105);
|
|
case 237: return QChar(105);
|
|
case 204: return QChar(105);
|
|
case 236: return QChar(105);
|
|
case 300: return QChar(105);
|
|
case 301: return QChar(105);
|
|
case 206: return QChar(105);
|
|
case 238: return QChar(105);
|
|
case 463: return QChar(105);
|
|
case 464: return QChar(105);
|
|
case 207: return QChar(105);
|
|
case 239: return QChar(105);
|
|
case 7726: return QChar(105);
|
|
case 7727: return QChar(105);
|
|
case 296: return QChar(105);
|
|
case 297: return QChar(105);
|
|
case 304: return QChar(105);
|
|
case 302: return QChar(105);
|
|
case 303: return QChar(105);
|
|
case 298: return QChar(105);
|
|
case 299: return QChar(105);
|
|
case 7880: return QChar(105);
|
|
case 7881: return QChar(105);
|
|
case 520: return QChar(105);
|
|
case 521: return QChar(105);
|
|
case 522: return QChar(105);
|
|
case 523: return QChar(105);
|
|
case 7882: return QChar(105);
|
|
case 7883: return QChar(105);
|
|
case 7724: return QChar(105);
|
|
case 7725: return QChar(105);
|
|
case 305: return QChar(105);
|
|
case 407: return QChar(105);
|
|
case 616: return QChar(105);
|
|
case 308: return QChar(106);
|
|
case 309: return QChar(106);
|
|
case 780: return QChar(106);
|
|
case 496: return QChar(106);
|
|
case 567: return QChar(106);
|
|
case 584: return QChar(106);
|
|
case 585: return QChar(106);
|
|
case 669: return QChar(106);
|
|
case 607: return QChar(106);
|
|
case 644: return QChar(106);
|
|
case 7728: return QChar(107);
|
|
case 7729: return QChar(107);
|
|
case 488: return QChar(107);
|
|
case 489: return QChar(107);
|
|
case 310: return QChar(107);
|
|
case 311: return QChar(107);
|
|
case 7730: return QChar(107);
|
|
case 7731: return QChar(107);
|
|
case 7732: return QChar(107);
|
|
case 7733: return QChar(107);
|
|
case 408: return QChar(107);
|
|
case 409: return QChar(107);
|
|
case 11369: return QChar(107);
|
|
case 11370: return QChar(107);
|
|
case 313: return QChar(97);
|
|
case 314: return QChar(108);
|
|
case 317: return QChar(108);
|
|
case 318: return QChar(108);
|
|
case 315: return QChar(108);
|
|
case 316: return QChar(108);
|
|
case 7734: return QChar(108);
|
|
case 7735: return QChar(108);
|
|
case 7736: return QChar(108);
|
|
case 7737: return QChar(108);
|
|
case 7740: return QChar(108);
|
|
case 7741: return QChar(108);
|
|
case 7738: return QChar(108);
|
|
case 7739: return QChar(108);
|
|
case 321: return QChar(108);
|
|
case 322: return QChar(108);
|
|
case 803: return QChar(108);
|
|
case 319: return QChar(108);
|
|
case 320: return QChar(108);
|
|
case 573: return QChar(108);
|
|
case 410: return QChar(108);
|
|
case 11360: return QChar(108);
|
|
case 11361: return QChar(108);
|
|
case 11362: return QChar(108);
|
|
case 619: return QChar(108);
|
|
case 620: return QChar(108);
|
|
case 621: return QChar(108);
|
|
case 564: return QChar(108);
|
|
case 7742: return QChar(109);
|
|
case 7743: return QChar(109);
|
|
case 7744: return QChar(109);
|
|
case 7745: return QChar(109);
|
|
case 7746: return QChar(109);
|
|
case 7747: return QChar(109);
|
|
case 625: return QChar(109);
|
|
case 323: return QChar(110);
|
|
case 324: return QChar(110);
|
|
case 504: return QChar(110);
|
|
case 505: return QChar(110);
|
|
case 327: return QChar(110);
|
|
case 328: return QChar(110);
|
|
case 209: return QChar(110);
|
|
case 241: return QChar(110);
|
|
case 7748: return QChar(110);
|
|
case 7749: return QChar(110);
|
|
case 325: return QChar(110);
|
|
case 326: return QChar(110);
|
|
case 7750: return QChar(110);
|
|
case 7751: return QChar(110);
|
|
case 7754: return QChar(110);
|
|
case 7755: return QChar(110);
|
|
case 7752: return QChar(110);
|
|
case 7753: return QChar(110);
|
|
case 413: return QChar(110);
|
|
case 626: return QChar(110);
|
|
case 544: return QChar(110);
|
|
case 414: return QChar(110);
|
|
case 627: return QChar(110);
|
|
case 565: return QChar(110);
|
|
case 776: return QChar(116);
|
|
case 211: return QChar(111);
|
|
case 243: return QChar(111);
|
|
case 210: return QChar(111);
|
|
case 242: return QChar(111);
|
|
case 334: return QChar(111);
|
|
case 335: return QChar(111);
|
|
case 212: return QChar(111);
|
|
case 244: return QChar(111);
|
|
case 7888: return QChar(111);
|
|
case 7889: return QChar(111);
|
|
case 7890: return QChar(111);
|
|
case 7891: return QChar(111);
|
|
case 7894: return QChar(111);
|
|
case 7895: return QChar(111);
|
|
case 7892: return QChar(111);
|
|
case 7893: return QChar(111);
|
|
case 465: return QChar(111);
|
|
case 466: return QChar(111);
|
|
case 214: return QChar(111);
|
|
case 246: return QChar(111);
|
|
case 554: return QChar(111);
|
|
case 555: return QChar(111);
|
|
case 336: return QChar(111);
|
|
case 337: return QChar(111);
|
|
case 213: return QChar(111);
|
|
case 245: return QChar(111);
|
|
case 7756: return QChar(111);
|
|
case 7757: return QChar(111);
|
|
case 7758: return QChar(111);
|
|
case 7759: return QChar(111);
|
|
case 556: return QChar(111);
|
|
case 557: return QChar(111);
|
|
case 558: return QChar(111);
|
|
case 559: return QChar(111);
|
|
case 560: return QChar(111);
|
|
case 561: return QChar(111);
|
|
case 216: return QChar(111);
|
|
case 248: return QChar(111);
|
|
case 510: return QChar(111);
|
|
case 511: return QChar(111);
|
|
case 490: return QChar(111);
|
|
case 491: return QChar(111);
|
|
case 492: return QChar(111);
|
|
case 493: return QChar(111);
|
|
case 332: return QChar(111);
|
|
case 333: return QChar(111);
|
|
case 7762: return QChar(111);
|
|
case 7763: return QChar(111);
|
|
case 7760: return QChar(111);
|
|
case 7761: return QChar(111);
|
|
case 7886: return QChar(111);
|
|
case 7887: return QChar(111);
|
|
case 524: return QChar(111);
|
|
case 525: return QChar(111);
|
|
case 526: return QChar(111);
|
|
case 527: return QChar(111);
|
|
case 416: return QChar(111);
|
|
case 417: return QChar(111);
|
|
case 7898: return QChar(111);
|
|
case 7899: return QChar(111);
|
|
case 7900: return QChar(111);
|
|
case 7901: return QChar(111);
|
|
case 7904: return QChar(111);
|
|
case 7905: return QChar(111);
|
|
case 7902: return QChar(111);
|
|
case 7903: return QChar(111);
|
|
case 7906: return QChar(111);
|
|
case 7907: return QChar(111);
|
|
case 7884: return QChar(111);
|
|
case 7885: return QChar(111);
|
|
case 7896: return QChar(111);
|
|
case 7897: return QChar(111);
|
|
case 415: return QChar(111);
|
|
case 629: return QChar(111);
|
|
case 7764: return QChar(112);
|
|
case 7765: return QChar(112);
|
|
case 7766: return QChar(112);
|
|
case 7767: return QChar(112);
|
|
case 11363: return QChar(112);
|
|
case 420: return QChar(112);
|
|
case 421: return QChar(112);
|
|
case 771: return QChar(112);
|
|
case 672: return QChar(113);
|
|
case 586: return QChar(113);
|
|
case 587: return QChar(113);
|
|
case 340: return QChar(114);
|
|
case 341: return QChar(114);
|
|
case 344: return QChar(114);
|
|
case 345: return QChar(114);
|
|
case 7768: return QChar(114);
|
|
case 7769: return QChar(114);
|
|
case 342: return QChar(114);
|
|
case 343: return QChar(114);
|
|
case 528: return QChar(114);
|
|
case 529: return QChar(114);
|
|
case 530: return QChar(114);
|
|
case 531: return QChar(114);
|
|
case 7770: return QChar(114);
|
|
case 7771: return QChar(114);
|
|
case 7772: return QChar(114);
|
|
case 7773: return QChar(114);
|
|
case 7774: return QChar(114);
|
|
case 7775: return QChar(114);
|
|
case 588: return QChar(114);
|
|
case 589: return QChar(114);
|
|
case 7538: return QChar(114);
|
|
case 636: return QChar(114);
|
|
case 11364: return QChar(114);
|
|
case 637: return QChar(114);
|
|
case 638: return QChar(114);
|
|
case 7539: return QChar(114);
|
|
case 223: return QChar(115);
|
|
case 346: return QChar(115);
|
|
case 347: return QChar(115);
|
|
case 7780: return QChar(115);
|
|
case 7781: return QChar(115);
|
|
case 348: return QChar(115);
|
|
case 349: return QChar(115);
|
|
case 352: return QChar(115);
|
|
case 353: return QChar(115);
|
|
case 7782: return QChar(115);
|
|
case 7783: return QChar(115);
|
|
case 7776: return QChar(115);
|
|
case 7777: return QChar(115);
|
|
case 7835: return QChar(115);
|
|
case 350: return QChar(115);
|
|
case 351: return QChar(115);
|
|
case 7778: return QChar(115);
|
|
case 7779: return QChar(115);
|
|
case 7784: return QChar(115);
|
|
case 7785: return QChar(115);
|
|
case 536: return QChar(115);
|
|
case 537: return QChar(115);
|
|
case 642: return QChar(115);
|
|
case 809: return QChar(115);
|
|
case 222: return QChar(116);
|
|
case 254: return QChar(116);
|
|
case 356: return QChar(116);
|
|
case 357: return QChar(116);
|
|
case 7831: return QChar(116);
|
|
case 7786: return QChar(116);
|
|
case 7787: return QChar(116);
|
|
case 354: return QChar(116);
|
|
case 355: return QChar(116);
|
|
case 7788: return QChar(116);
|
|
case 7789: return QChar(116);
|
|
case 538: return QChar(116);
|
|
case 539: return QChar(116);
|
|
case 7792: return QChar(116);
|
|
case 7793: return QChar(116);
|
|
case 7790: return QChar(116);
|
|
case 7791: return QChar(116);
|
|
case 358: return QChar(116);
|
|
case 359: return QChar(116);
|
|
case 574: return QChar(116);
|
|
case 11366: return QChar(116);
|
|
case 7541: return QChar(116);
|
|
case 427: return QChar(116);
|
|
case 428: return QChar(116);
|
|
case 429: return QChar(116);
|
|
case 430: return QChar(116);
|
|
case 648: return QChar(116);
|
|
case 566: return QChar(116);
|
|
case 218: return QChar(117);
|
|
case 250: return QChar(117);
|
|
case 217: return QChar(117);
|
|
case 249: return QChar(117);
|
|
case 364: return QChar(117);
|
|
case 365: return QChar(117);
|
|
case 219: return QChar(117);
|
|
case 251: return QChar(117);
|
|
case 467: return QChar(117);
|
|
case 468: return QChar(117);
|
|
case 366: return QChar(117);
|
|
case 367: return QChar(117);
|
|
case 220: return QChar(117);
|
|
case 252: return QChar(117);
|
|
case 471: return QChar(117);
|
|
case 472: return QChar(117);
|
|
case 475: return QChar(117);
|
|
case 476: return QChar(117);
|
|
case 473: return QChar(117);
|
|
case 474: return QChar(117);
|
|
case 469: return QChar(117);
|
|
case 470: return QChar(117);
|
|
case 368: return QChar(117);
|
|
case 369: return QChar(117);
|
|
case 360: return QChar(117);
|
|
case 361: return QChar(117);
|
|
case 7800: return QChar(117);
|
|
case 7801: return QChar(117);
|
|
case 370: return QChar(117);
|
|
case 371: return QChar(117);
|
|
case 362: return QChar(117);
|
|
case 363: return QChar(117);
|
|
case 7802: return QChar(117);
|
|
case 7803: return QChar(117);
|
|
case 7910: return QChar(117);
|
|
case 7911: return QChar(117);
|
|
case 532: return QChar(117);
|
|
case 533: return QChar(117);
|
|
case 534: return QChar(117);
|
|
case 535: return QChar(117);
|
|
case 431: return QChar(117);
|
|
case 432: return QChar(117);
|
|
case 7912: return QChar(117);
|
|
case 7913: return QChar(117);
|
|
case 7914: return QChar(117);
|
|
case 7915: return QChar(117);
|
|
case 7918: return QChar(117);
|
|
case 7919: return QChar(117);
|
|
case 7916: return QChar(117);
|
|
case 7917: return QChar(117);
|
|
case 7920: return QChar(117);
|
|
case 7921: return QChar(117);
|
|
case 7908: return QChar(117);
|
|
case 7909: return QChar(117);
|
|
case 7794: return QChar(117);
|
|
case 7795: return QChar(117);
|
|
case 7798: return QChar(117);
|
|
case 7799: return QChar(117);
|
|
case 7796: return QChar(117);
|
|
case 7797: return QChar(117);
|
|
case 580: return QChar(117);
|
|
case 649: return QChar(117);
|
|
case 7804: return QChar(118);
|
|
case 7805: return QChar(118);
|
|
case 7806: return QChar(118);
|
|
case 7807: return QChar(118);
|
|
case 434: return QChar(118);
|
|
case 651: return QChar(118);
|
|
case 7810: return QChar(119);
|
|
case 7811: return QChar(119);
|
|
case 7808: return QChar(119);
|
|
case 7809: return QChar(119);
|
|
case 372: return QChar(119);
|
|
case 373: return QChar(119);
|
|
case 778: return QChar(121);
|
|
case 7832: return QChar(119);
|
|
case 7812: return QChar(119);
|
|
case 7813: return QChar(119);
|
|
case 7814: return QChar(119);
|
|
case 7815: return QChar(119);
|
|
case 7816: return QChar(119);
|
|
case 7817: return QChar(119);
|
|
case 7820: return QChar(120);
|
|
case 7821: return QChar(120);
|
|
case 7818: return QChar(120);
|
|
case 7819: return QChar(120);
|
|
case 221: return QChar(121);
|
|
case 253: return QChar(121);
|
|
case 7922: return QChar(121);
|
|
case 7923: return QChar(121);
|
|
case 374: return QChar(121);
|
|
case 375: return QChar(121);
|
|
case 7833: return QChar(121);
|
|
case 376: return QChar(121);
|
|
case 255: return QChar(121);
|
|
case 7928: return QChar(121);
|
|
case 7929: return QChar(121);
|
|
case 7822: return QChar(121);
|
|
case 7823: return QChar(121);
|
|
case 562: return QChar(121);
|
|
case 563: return QChar(121);
|
|
case 7926: return QChar(121);
|
|
case 7927: return QChar(121);
|
|
case 7924: return QChar(121);
|
|
case 7925: return QChar(121);
|
|
case 655: return QChar(121);
|
|
case 590: return QChar(121);
|
|
case 591: return QChar(121);
|
|
case 435: return QChar(121);
|
|
case 436: return QChar(121);
|
|
case 377: return QChar(122);
|
|
case 378: return QChar(122);
|
|
case 7824: return QChar(122);
|
|
case 7825: return QChar(122);
|
|
case 381: return QChar(122);
|
|
case 382: return QChar(122);
|
|
case 379: return QChar(122);
|
|
case 380: return QChar(122);
|
|
case 7826: return QChar(122);
|
|
case 7827: return QChar(122);
|
|
case 7828: return QChar(122);
|
|
case 7829: return QChar(122);
|
|
case 437: return QChar(122);
|
|
case 438: return QChar(122);
|
|
case 548: return QChar(122);
|
|
case 549: return QChar(122);
|
|
case 656: return QChar(122);
|
|
case 657: return QChar(122);
|
|
case 11371: return QChar(122);
|
|
case 11372: return QChar(122);
|
|
case 494: return QChar(122);
|
|
case 495: return QChar(122);
|
|
case 442: return QChar(122);
|
|
case 65298: return QChar(50);
|
|
case 65302: return QChar(54);
|
|
case 65314: return QChar(66);
|
|
case 65318: return QChar(70);
|
|
case 65322: return QChar(74);
|
|
case 65326: return QChar(78);
|
|
case 65330: return QChar(82);
|
|
case 65334: return QChar(86);
|
|
case 65338: return QChar(90);
|
|
case 65346: return QChar(98);
|
|
case 65350: return QChar(102);
|
|
case 65354: return QChar(106);
|
|
case 65358: return QChar(110);
|
|
case 65362: return QChar(114);
|
|
case 65366: return QChar(118);
|
|
case 65370: return QChar(122);
|
|
case 65297: return QChar(49);
|
|
case 65301: return QChar(53);
|
|
case 65305: return QChar(57);
|
|
case 65313: return QChar(65);
|
|
case 65317: return QChar(69);
|
|
case 65321: return QChar(73);
|
|
case 65325: return QChar(77);
|
|
case 65329: return QChar(81);
|
|
case 65333: return QChar(85);
|
|
case 65337: return QChar(89);
|
|
case 65345: return QChar(97);
|
|
case 65349: return QChar(101);
|
|
case 65353: return QChar(105);
|
|
case 65357: return QChar(109);
|
|
case 65361: return QChar(113);
|
|
case 65365: return QChar(117);
|
|
case 65369: return QChar(121);
|
|
case 65296: return QChar(48);
|
|
case 65300: return QChar(52);
|
|
case 65304: return QChar(56);
|
|
case 65316: return QChar(68);
|
|
case 65320: return QChar(72);
|
|
case 65324: return QChar(76);
|
|
case 65328: return QChar(80);
|
|
case 65332: return QChar(84);
|
|
case 65336: return QChar(88);
|
|
case 65348: return QChar(100);
|
|
case 65352: return QChar(104);
|
|
case 65356: return QChar(108);
|
|
case 65360: return QChar(112);
|
|
case 65364: return QChar(116);
|
|
case 65368: return QChar(120);
|
|
case 65299: return QChar(51);
|
|
case 65303: return QChar(55);
|
|
case 65315: return QChar(67);
|
|
case 65319: return QChar(71);
|
|
case 65323: return QChar(75);
|
|
case 65327: return QChar(79);
|
|
case 65331: return QChar(83);
|
|
case 65335: return QChar(87);
|
|
case 65347: return QChar(99);
|
|
case 65351: return QChar(103);
|
|
case 65355: return QChar(107);
|
|
case 65359: return QChar(111);
|
|
case 65363: return QChar(115);
|
|
case 65367: return QChar(119);
|
|
case 1105: return QChar(1077);
|
|
default:
|
|
break;
|
|
}
|
|
return QChar(0);
|
|
}
|
|
|
|
const QRegularExpression &RegExpWordSplit() {
|
|
static const auto result = QRegularExpression (qsl("[\\@\\s\\-\\+\\(\\)\\[\\]\\{\\}\\<\\>\\,\\.\\:\\!\\_\\;\\\"\\'\\x0]"));
|
|
return result;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
const QRegularExpression &RegExpMailNameAtEnd() {
|
|
static const auto result = CreateRegExp(ExpressionMailNameAtEnd());
|
|
return result;
|
|
}
|
|
|
|
const QRegularExpression &RegExpHashtag() {
|
|
static const auto result = CreateRegExp(ExpressionHashtag());
|
|
return result;
|
|
}
|
|
|
|
const QRegularExpression &RegExpHashtagExclude() {
|
|
static const auto result = CreateRegExp(ExpressionHashtagExclude());
|
|
return result;
|
|
}
|
|
|
|
const QRegularExpression &RegExpMention() {
|
|
static const auto result = CreateRegExp(ExpressionMention());
|
|
return result;
|
|
}
|
|
|
|
const QRegularExpression &RegExpBotCommand() {
|
|
static const auto result = CreateRegExp(ExpressionBotCommand());
|
|
return result;
|
|
}
|
|
|
|
QString MarkdownBoldGoodBefore() {
|
|
return SeparatorsBold();
|
|
}
|
|
|
|
QString MarkdownBoldBadAfter() {
|
|
return qsl("*");
|
|
}
|
|
|
|
QString MarkdownItalicGoodBefore() {
|
|
return SeparatorsItalic();
|
|
}
|
|
|
|
QString MarkdownItalicBadAfter() {
|
|
return qsl("_");
|
|
}
|
|
|
|
QString MarkdownCodeGoodBefore() {
|
|
return SeparatorsMono();
|
|
}
|
|
|
|
QString MarkdownCodeBadAfter() {
|
|
return qsl("`\n\r");
|
|
}
|
|
|
|
QString MarkdownPreGoodBefore() {
|
|
return SeparatorsMono();
|
|
}
|
|
|
|
QString MarkdownPreBadAfter() {
|
|
return qsl("`");
|
|
}
|
|
|
|
bool IsValidProtocol(const QString &protocol) {
|
|
static const auto list = CreateValidProtocols();
|
|
return list.contains(hashCrc32(protocol.constData(), protocol.size() * sizeof(QChar)));
|
|
}
|
|
|
|
bool IsValidTopDomain(const QString &protocol) {
|
|
static const auto list = CreateValidTopDomains();
|
|
return list.contains(hashCrc32(protocol.constData(), protocol.size() * sizeof(QChar)));
|
|
}
|
|
|
|
QString Clean(const QString &text) {
|
|
auto result = text;
|
|
for (auto s = text.unicode(), ch = s, e = text.unicode() + text.size(); ch != e; ++ch) {
|
|
if (*ch == TextCommand) {
|
|
result[int(ch - s)] = QChar::Space;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
QString EscapeForRichParsing(const QString &text) {
|
|
QString result;
|
|
result.reserve(text.size());
|
|
auto s = text.constData(), ch = s;
|
|
for (const QChar *e = s + text.size(); ch != e; ++ch) {
|
|
if (*ch == TextCommand) {
|
|
if (ch > s) result.append(s, ch - s);
|
|
result.append(QChar::Space);
|
|
s = ch + 1;
|
|
continue;
|
|
}
|
|
if (ch->unicode() == '\\' || ch->unicode() == '[') {
|
|
if (ch > s) result.append(s, ch - s);
|
|
result.append('\\');
|
|
s = ch;
|
|
continue;
|
|
}
|
|
}
|
|
if (ch > s) result.append(s, ch - s);
|
|
return result;
|
|
}
|
|
|
|
QString SingleLine(const QString &text) {
|
|
auto result = text;
|
|
auto s = text.unicode(), ch = s, e = text.unicode() + text.size();
|
|
|
|
// Trim.
|
|
while (s < e && chIsTrimmed(*s)) {
|
|
++s;
|
|
}
|
|
while (s < e && chIsTrimmed(*(e - 1))) {
|
|
--e;
|
|
}
|
|
if (e - s != text.size()) {
|
|
result = text.mid(s - text.unicode(), e - s);
|
|
}
|
|
|
|
for (auto ch = s; ch != e; ++ch) {
|
|
if (chIsNewline(*ch) || *ch == TextCommand) {
|
|
result[int(ch - s)] = QChar::Space;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
QString RemoveAccents(const QString &text) {
|
|
auto result = text;
|
|
auto copying = false;
|
|
auto i = 0;
|
|
for (auto s = text.unicode(), ch = s, e = text.unicode() + text.size(); ch != e; ++ch, ++i) {
|
|
if (ch->unicode() < 128) {
|
|
if (copying) result[i] = *ch;
|
|
continue;
|
|
}
|
|
if (chIsDiac(*ch)) {
|
|
copying = true;
|
|
--i;
|
|
continue;
|
|
}
|
|
if (ch->isHighSurrogate() && ch + 1 < e && (ch + 1)->isLowSurrogate()) {
|
|
auto noAccent = RemoveOneAccent(QChar::surrogateToUcs4(*ch, *(ch + 1)));
|
|
if (noAccent.unicode() > 0) {
|
|
copying = true;
|
|
result[i] = noAccent;
|
|
} else {
|
|
if (copying) result[i] = *ch;
|
|
++ch, ++i;
|
|
if (copying) result[i] = *ch;
|
|
}
|
|
} else {
|
|
auto noAccent = RemoveOneAccent(ch->unicode());
|
|
if (noAccent.unicode() > 0 && noAccent != *ch) {
|
|
result[i] = noAccent;
|
|
} else if (copying) {
|
|
result[i] = *ch;
|
|
}
|
|
}
|
|
}
|
|
return (i < result.size()) ? result.mid(0, i) : result;
|
|
}
|
|
|
|
QStringList PrepareSearchWords(const QString &query, const QRegularExpression *SplitterOverride) {
|
|
auto clean = RemoveAccents(query.trimmed().toLower());
|
|
auto result = QStringList();
|
|
if (!clean.isEmpty()) {
|
|
auto list = clean.split(SplitterOverride ? *SplitterOverride : RegExpWordSplit(), QString::SkipEmptyParts);
|
|
auto size = list.size();
|
|
result.reserve(list.size());
|
|
for_const (auto &word, list) {
|
|
auto trimmed = word.trimmed();
|
|
if (!trimmed.isEmpty()) {
|
|
result.push_back(trimmed);
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
bool CutPart(TextWithEntities &sending, TextWithEntities &left, int32 limit) {
|
|
if (left.text.isEmpty() || !limit) return false;
|
|
|
|
int32 currentEntity = 0, goodEntity = currentEntity, entityCount = left.entities.size();
|
|
bool goodInEntity = false, goodCanBreakEntity = false;
|
|
|
|
int32 s = 0, half = limit / 2, goodLevel = 0;
|
|
for (const QChar *start = left.text.constData(), *ch = start, *end = left.text.constEnd(), *good = ch; ch != end; ++ch, ++s) {
|
|
while (currentEntity < entityCount && ch >= start + left.entities[currentEntity].offset() + left.entities[currentEntity].length()) {
|
|
++currentEntity;
|
|
}
|
|
|
|
if (s > half) {
|
|
bool inEntity = (currentEntity < entityCount) && (ch > start + left.entities[currentEntity].offset()) && (ch < start + left.entities[currentEntity].offset() + left.entities[currentEntity].length());
|
|
EntityType entityType = (currentEntity < entityCount) ? left.entities[currentEntity].type() : EntityType::Invalid;
|
|
bool canBreakEntity = (entityType == EntityType::Pre || entityType == EntityType::Code);
|
|
int32 noEntityLevel = inEntity ? 0 : 1;
|
|
|
|
auto markGoodAsLevel = [&](int newLevel) {
|
|
if (goodLevel > newLevel) {
|
|
return;
|
|
}
|
|
goodLevel = newLevel;
|
|
good = ch;
|
|
goodEntity = currentEntity;
|
|
goodInEntity = inEntity;
|
|
goodCanBreakEntity = canBreakEntity;
|
|
};
|
|
|
|
if (inEntity && !canBreakEntity) {
|
|
markGoodAsLevel(0);
|
|
} else {
|
|
if (chIsNewline(*ch)) {
|
|
if (inEntity) {
|
|
if (ch + 1 < end && chIsNewline(*(ch + 1))) {
|
|
markGoodAsLevel(12);
|
|
} else {
|
|
markGoodAsLevel(11);
|
|
}
|
|
} else if (ch + 1 < end && chIsNewline(*(ch + 1))) {
|
|
markGoodAsLevel(15);
|
|
} else if (currentEntity < entityCount && ch + 1 == start + left.entities[currentEntity].offset() && left.entities[currentEntity].type() == EntityType::Pre) {
|
|
markGoodAsLevel(14);
|
|
} else if (currentEntity > 0 && ch == start + left.entities[currentEntity - 1].offset() + left.entities[currentEntity - 1].length() && left.entities[currentEntity - 1].type() == EntityType::Pre) {
|
|
markGoodAsLevel(14);
|
|
} else {
|
|
markGoodAsLevel(13);
|
|
}
|
|
} else if (chIsSpace(*ch)) {
|
|
if (chIsSentenceEnd(*(ch - 1))) {
|
|
markGoodAsLevel(9 + noEntityLevel);
|
|
} else if (chIsSentencePartEnd(*(ch - 1))) {
|
|
markGoodAsLevel(7 + noEntityLevel);
|
|
} else {
|
|
markGoodAsLevel(5 + noEntityLevel);
|
|
}
|
|
} else if (chIsWordSeparator(*(ch - 1))) {
|
|
markGoodAsLevel(3 + noEntityLevel);
|
|
} else {
|
|
markGoodAsLevel(1 + noEntityLevel);
|
|
}
|
|
}
|
|
}
|
|
|
|
int elen = 0;
|
|
if (auto e = Ui::Emoji::Find(ch, end, &elen)) {
|
|
for (int i = 0; i < elen; ++i, ++ch, ++s) {
|
|
if (ch->isHighSurrogate() && i + 1 < elen && (ch + 1)->isLowSurrogate()) {
|
|
++ch;
|
|
++i;
|
|
}
|
|
}
|
|
--ch;
|
|
--s;
|
|
} else if (ch->isHighSurrogate() && ch + 1 < end && (ch + 1)->isLowSurrogate()) {
|
|
++ch;
|
|
}
|
|
if (s >= limit) {
|
|
sending.text = left.text.mid(0, good - start);
|
|
left.text = left.text.mid(good - start);
|
|
if (goodInEntity) {
|
|
if (goodCanBreakEntity) {
|
|
sending.entities = left.entities.mid(0, goodEntity + 1);
|
|
sending.entities.back().updateTextEnd(good - start);
|
|
left.entities = left.entities.mid(goodEntity);
|
|
for (auto &entity : left.entities) {
|
|
entity.shiftLeft(good - start);
|
|
}
|
|
} else {
|
|
sending.entities = left.entities.mid(0, goodEntity);
|
|
left.entities = left.entities.mid(goodEntity + 1);
|
|
}
|
|
} else {
|
|
sending.entities = left.entities.mid(0, goodEntity);
|
|
left.entities = left.entities.mid(goodEntity);
|
|
for (auto &entity : left.entities) {
|
|
entity.shiftLeft(good - start);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
sending.text = left.text;
|
|
left.text = QString();
|
|
sending.entities = left.entities;
|
|
left.entities = EntitiesInText();
|
|
return true;
|
|
}
|
|
|
|
bool textcmdStartsLink(const QChar *start, int32 len, int32 commandOffset) {
|
|
if (commandOffset + 2 < len) {
|
|
if (*(start + commandOffset + 1) == TextCommandLinkIndex) {
|
|
return (*(start + commandOffset + 2) != 0);
|
|
}
|
|
return (*(start + commandOffset + 1) != TextCommandLinkText);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool checkTagStartInCommand(const QChar *start, int32 len, int32 tagStart, int32 &commandOffset, bool &commandIsLink, bool &inLink) {
|
|
bool inCommand = false;
|
|
const QChar *commandEnd = start + commandOffset;
|
|
while (commandOffset < len && tagStart > commandOffset) { // skip commands, evaluating are we in link or not
|
|
commandEnd = textSkipCommand(start + commandOffset, start + len);
|
|
if (commandEnd > start + commandOffset) {
|
|
if (tagStart < (commandEnd - start)) {
|
|
inCommand = true;
|
|
break;
|
|
}
|
|
for (commandOffset = commandEnd - start; commandOffset < len; ++commandOffset) {
|
|
if (*(start + commandOffset) == TextCommand) {
|
|
inLink = commandIsLink;
|
|
commandIsLink = textcmdStartsLink(start, len, commandOffset);
|
|
break;
|
|
}
|
|
}
|
|
if (commandOffset >= len) {
|
|
inLink = commandIsLink;
|
|
commandIsLink = false;
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
if (inCommand) {
|
|
commandOffset = commandEnd - start;
|
|
}
|
|
return inCommand;
|
|
}
|
|
|
|
EntitiesInText EntitiesFromMTP(const QVector<MTPMessageEntity> &entities) {
|
|
auto result = EntitiesInText();
|
|
if (!entities.isEmpty()) {
|
|
result.reserve(entities.size());
|
|
for_const (auto &entity, entities) {
|
|
switch (entity.type()) {
|
|
case mtpc_messageEntityUrl: { auto &d = entity.c_messageEntityUrl(); result.push_back({ EntityType::Url, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityTextUrl: { auto &d = entity.c_messageEntityTextUrl(); result.push_back({ EntityType::CustomUrl, d.voffset.v, d.vlength.v, Clean(qs(d.vurl)) }); } break;
|
|
case mtpc_messageEntityEmail: { auto &d = entity.c_messageEntityEmail(); result.push_back({ EntityType::Email, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityHashtag: { auto &d = entity.c_messageEntityHashtag(); result.push_back({ EntityType::Hashtag, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityCashtag: { auto &d = entity.c_messageEntityCashtag(); result.push_back({ EntityType::Cashtag, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityPhone: break; // Skipping phones.
|
|
case mtpc_messageEntityMention: { auto &d = entity.c_messageEntityMention(); result.push_back({ EntityType::Mention, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityMentionName: {
|
|
auto &d = entity.c_messageEntityMentionName();
|
|
auto data = [&d] {
|
|
if (auto user = Auth().data().userLoaded(d.vuser_id.v)) {
|
|
return MentionNameDataFromFields({
|
|
d.vuser_id.v,
|
|
user->accessHash() });
|
|
}
|
|
return MentionNameDataFromFields(d.vuser_id.v);
|
|
};
|
|
result.push_back({ EntityType::MentionName, d.voffset.v, d.vlength.v, data() });
|
|
} break;
|
|
case mtpc_inputMessageEntityMentionName: {
|
|
auto &d = entity.c_inputMessageEntityMentionName();
|
|
auto data = ([&d]() -> QString {
|
|
if (d.vuser_id.type() == mtpc_inputUserSelf) {
|
|
return MentionNameDataFromFields(Auth().userId());
|
|
} else if (d.vuser_id.type() == mtpc_inputUser) {
|
|
auto &user = d.vuser_id.c_inputUser();
|
|
return MentionNameDataFromFields({ user.vuser_id.v, user.vaccess_hash.v });
|
|
}
|
|
return QString();
|
|
})();
|
|
if (!data.isEmpty()) {
|
|
result.push_back({ EntityType::MentionName, d.voffset.v, d.vlength.v, data });
|
|
}
|
|
} break;
|
|
case mtpc_messageEntityBotCommand: { auto &d = entity.c_messageEntityBotCommand(); result.push_back({ EntityType::BotCommand, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityBold: { auto &d = entity.c_messageEntityBold(); result.push_back({ EntityType::Bold, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityItalic: { auto &d = entity.c_messageEntityItalic(); result.push_back({ EntityType::Italic, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityCode: { auto &d = entity.c_messageEntityCode(); result.push_back({ EntityType::Code, d.voffset.v, d.vlength.v }); } break;
|
|
case mtpc_messageEntityPre: { auto &d = entity.c_messageEntityPre(); result.push_back({ EntityType::Pre, d.voffset.v, d.vlength.v, Clean(qs(d.vlanguage)) }); } break;
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
MTPVector<MTPMessageEntity> EntitiesToMTP(const EntitiesInText &entities, ConvertOption option) {
|
|
auto v = QVector<MTPMessageEntity>();
|
|
v.reserve(entities.size());
|
|
for_const (auto &entity, entities) {
|
|
if (entity.length() <= 0) continue;
|
|
if (option == ConvertOption::SkipLocal
|
|
&& entity.type() != EntityType::Bold
|
|
&& entity.type() != EntityType::Italic
|
|
&& entity.type() != EntityType::Code
|
|
&& entity.type() != EntityType::Pre
|
|
&& entity.type() != EntityType::MentionName
|
|
&& entity.type() != EntityType::CustomUrl) {
|
|
continue;
|
|
}
|
|
|
|
auto offset = MTP_int(entity.offset());
|
|
auto length = MTP_int(entity.length());
|
|
switch (entity.type()) {
|
|
case EntityType::Url: v.push_back(MTP_messageEntityUrl(offset, length)); break;
|
|
case EntityType::CustomUrl: v.push_back(MTP_messageEntityTextUrl(offset, length, MTP_string(entity.data()))); break;
|
|
case EntityType::Email: v.push_back(MTP_messageEntityEmail(offset, length)); break;
|
|
case EntityType::Hashtag: v.push_back(MTP_messageEntityHashtag(offset, length)); break;
|
|
case EntityType::Cashtag: v.push_back(MTP_messageEntityCashtag(offset, length)); break;
|
|
case EntityType::Mention: v.push_back(MTP_messageEntityMention(offset, length)); break;
|
|
case EntityType::MentionName: {
|
|
auto inputUser = ([](const QString &data) -> MTPInputUser {
|
|
auto fields = MentionNameDataToFields(data);
|
|
if (fields.userId == Auth().userId()) {
|
|
return MTP_inputUserSelf();
|
|
} else if (fields.userId) {
|
|
return MTP_inputUser(MTP_int(fields.userId), MTP_long(fields.accessHash));
|
|
}
|
|
return MTP_inputUserEmpty();
|
|
})(entity.data());
|
|
if (inputUser.type() != mtpc_inputUserEmpty) {
|
|
v.push_back(MTP_inputMessageEntityMentionName(offset, length, inputUser));
|
|
}
|
|
} break;
|
|
case EntityType::BotCommand: v.push_back(MTP_messageEntityBotCommand(offset, length)); break;
|
|
case EntityType::Bold: v.push_back(MTP_messageEntityBold(offset, length)); break;
|
|
case EntityType::Italic: v.push_back(MTP_messageEntityItalic(offset, length)); break;
|
|
case EntityType::Code: v.push_back(MTP_messageEntityCode(offset, length)); break;
|
|
case EntityType::Pre: v.push_back(MTP_messageEntityPre(offset, length, MTP_string(entity.data()))); break;
|
|
}
|
|
}
|
|
return MTP_vector<MTPMessageEntity>(std::move(v));
|
|
}
|
|
|
|
TextWithEntities ParseEntities(const QString &text, int32 flags) {
|
|
const auto rich = ((flags & TextParseRichText) != 0);
|
|
auto result = TextWithEntities{ text, EntitiesInText() };
|
|
ParseEntities(result, flags, rich);
|
|
return result;
|
|
}
|
|
|
|
// Some code is duplicated in message_field.cpp!
|
|
void ParseEntities(TextWithEntities &result, int32 flags, bool rich) {
|
|
constexpr auto kNotFound = std::numeric_limits<int>::max();
|
|
|
|
auto newEntities = EntitiesInText();
|
|
bool withHashtags = (flags & TextParseHashtags);
|
|
bool withMentions = (flags & TextParseMentions);
|
|
bool withBotCommands = (flags & TextParseBotCommands);
|
|
|
|
int existingEntityIndex = 0, existingEntitiesCount = result.entities.size();
|
|
int existingEntityEnd = 0;
|
|
|
|
int32 len = result.text.size(), commandOffset = rich ? 0 : len;
|
|
bool inLink = false, commandIsLink = false;
|
|
const QChar *start = result.text.constData(), *end = start + result.text.size();
|
|
for (int32 offset = 0, matchOffset = offset, mentionSkip = 0; offset < len;) {
|
|
if (commandOffset <= offset) {
|
|
for (commandOffset = offset; commandOffset < len; ++commandOffset) {
|
|
if (*(start + commandOffset) == TextCommand) {
|
|
inLink = commandIsLink;
|
|
commandIsLink = textcmdStartsLink(start, len, commandOffset);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
auto mDomain = qthelp::RegExpDomain().match(result.text, matchOffset);
|
|
auto mExplicitDomain = qthelp::RegExpDomainExplicit().match(result.text, matchOffset);
|
|
auto mHashtag = withHashtags ? RegExpHashtag().match(result.text, matchOffset) : QRegularExpressionMatch();
|
|
auto mMention = withMentions ? RegExpMention().match(result.text, qMax(mentionSkip, matchOffset)) : QRegularExpressionMatch();
|
|
auto mBotCommand = withBotCommands ? RegExpBotCommand().match(result.text, matchOffset) : QRegularExpressionMatch();
|
|
|
|
auto lnkType = EntityType::Url;
|
|
int32 lnkStart = 0, lnkLength = 0;
|
|
auto domainStart = mDomain.hasMatch() ? mDomain.capturedStart() : kNotFound,
|
|
domainEnd = mDomain.hasMatch() ? mDomain.capturedEnd() : kNotFound,
|
|
explicitDomainStart = mExplicitDomain.hasMatch() ? mExplicitDomain.capturedStart() : kNotFound,
|
|
explicitDomainEnd = mExplicitDomain.hasMatch() ? mExplicitDomain.capturedEnd() : kNotFound,
|
|
hashtagStart = mHashtag.hasMatch() ? mHashtag.capturedStart() : kNotFound,
|
|
hashtagEnd = mHashtag.hasMatch() ? mHashtag.capturedEnd() : kNotFound,
|
|
mentionStart = mMention.hasMatch() ? mMention.capturedStart() : kNotFound,
|
|
mentionEnd = mMention.hasMatch() ? mMention.capturedEnd() : kNotFound,
|
|
botCommandStart = mBotCommand.hasMatch() ? mBotCommand.capturedStart() : kNotFound,
|
|
botCommandEnd = mBotCommand.hasMatch() ? mBotCommand.capturedEnd() : kNotFound;
|
|
auto hashtagIgnore = false;
|
|
auto mentionIgnore = false;
|
|
|
|
if (mHashtag.hasMatch()) {
|
|
if (!mHashtag.capturedRef(1).isEmpty()) {
|
|
++hashtagStart;
|
|
}
|
|
if (!mHashtag.capturedRef(2).isEmpty()) {
|
|
--hashtagEnd;
|
|
}
|
|
if (RegExpHashtagExclude().match(
|
|
result.text.mid(
|
|
hashtagStart + 1,
|
|
hashtagEnd - hashtagStart - 1)).hasMatch()) {
|
|
hashtagIgnore = true;
|
|
}
|
|
}
|
|
while (mMention.hasMatch()) {
|
|
if (!mMention.capturedRef(1).isEmpty()) {
|
|
++mentionStart;
|
|
}
|
|
if (!mMention.capturedRef(2).isEmpty()) {
|
|
--mentionEnd;
|
|
}
|
|
if (!(start + mentionStart + 1)->isLetter() || !(start + mentionEnd - 1)->isLetterOrNumber()) {
|
|
mentionSkip = mentionEnd;
|
|
mMention = RegExpMention().match(result.text, qMax(mentionSkip, matchOffset));
|
|
if (mMention.hasMatch()) {
|
|
mentionStart = mMention.capturedStart();
|
|
mentionEnd = mMention.capturedEnd();
|
|
} else {
|
|
mentionIgnore = true;
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
if (mBotCommand.hasMatch()) {
|
|
if (!mBotCommand.capturedRef(1).isEmpty()) {
|
|
++botCommandStart;
|
|
}
|
|
if (!mBotCommand.capturedRef(3).isEmpty()) {
|
|
--botCommandEnd;
|
|
}
|
|
}
|
|
if (!mDomain.hasMatch()
|
|
&& !mExplicitDomain.hasMatch()
|
|
&& !mHashtag.hasMatch()
|
|
&& !mMention.hasMatch()
|
|
&& !mBotCommand.hasMatch()) {
|
|
break;
|
|
}
|
|
|
|
if (explicitDomainStart < domainStart) {
|
|
domainStart = explicitDomainStart;
|
|
domainEnd = explicitDomainEnd;
|
|
mDomain = mExplicitDomain;
|
|
}
|
|
if (mentionStart < hashtagStart
|
|
&& mentionStart < domainStart
|
|
&& mentionStart < botCommandStart) {
|
|
if (mentionIgnore) {
|
|
offset = matchOffset = mentionEnd;
|
|
continue;
|
|
}
|
|
const auto inCommand = checkTagStartInCommand(
|
|
start,
|
|
len,
|
|
mentionStart,
|
|
commandOffset,
|
|
commandIsLink,
|
|
inLink);
|
|
if (inCommand || inLink) {
|
|
offset = matchOffset = commandOffset;
|
|
continue;
|
|
}
|
|
|
|
lnkType = EntityType::Mention;
|
|
lnkStart = mentionStart;
|
|
lnkLength = mentionEnd - mentionStart;
|
|
} else if (hashtagStart < domainStart
|
|
&& hashtagStart < botCommandStart) {
|
|
if (hashtagIgnore) {
|
|
offset = matchOffset = hashtagEnd;
|
|
continue;
|
|
}
|
|
const auto inCommand = checkTagStartInCommand(
|
|
start,
|
|
len,
|
|
hashtagStart,
|
|
commandOffset,
|
|
commandIsLink,
|
|
inLink);
|
|
if (inCommand || inLink) {
|
|
offset = matchOffset = commandOffset;
|
|
continue;
|
|
}
|
|
|
|
lnkType = EntityType::Hashtag;
|
|
lnkStart = hashtagStart;
|
|
lnkLength = hashtagEnd - hashtagStart;
|
|
} else if (botCommandStart < domainStart) {
|
|
const auto inCommand = checkTagStartInCommand(
|
|
start,
|
|
len,
|
|
botCommandStart,
|
|
commandOffset,
|
|
commandIsLink,
|
|
inLink);
|
|
if (inCommand || inLink) {
|
|
offset = matchOffset = commandOffset;
|
|
continue;
|
|
}
|
|
|
|
lnkType = EntityType::BotCommand;
|
|
lnkStart = botCommandStart;
|
|
lnkLength = botCommandEnd - botCommandStart;
|
|
} else {
|
|
const auto inCommand = checkTagStartInCommand(
|
|
start,
|
|
len,
|
|
domainStart,
|
|
commandOffset,
|
|
commandIsLink,
|
|
inLink);
|
|
if (inCommand || inLink) {
|
|
offset = matchOffset = commandOffset;
|
|
continue;
|
|
}
|
|
|
|
auto protocol = mDomain.captured(1).toLower();
|
|
auto topDomain = mDomain.captured(3).toLower();
|
|
auto isProtocolValid = protocol.isEmpty() || IsValidProtocol(protocol);
|
|
auto isTopDomainValid = !protocol.isEmpty() || IsValidTopDomain(topDomain);
|
|
|
|
if (protocol.isEmpty() && domainStart > offset + 1 && *(start + domainStart - 1) == QChar('@')) {
|
|
auto forMailName = result.text.mid(offset, domainStart - offset - 1);
|
|
auto mMailName = RegExpMailNameAtEnd().match(forMailName);
|
|
if (mMailName.hasMatch()) {
|
|
auto mailStart = offset + mMailName.capturedStart();
|
|
if (mailStart < offset) {
|
|
mailStart = offset;
|
|
}
|
|
lnkType = EntityType::Email;
|
|
lnkStart = mailStart;
|
|
lnkLength = domainEnd - mailStart;
|
|
}
|
|
}
|
|
if (lnkType == EntityType::Url && !lnkLength) {
|
|
if (!isProtocolValid || !isTopDomainValid) {
|
|
matchOffset = domainEnd;
|
|
continue;
|
|
}
|
|
lnkStart = domainStart;
|
|
|
|
QStack<const QChar*> parenth;
|
|
const QChar *domainEnd = start + mDomain.capturedEnd(), *p = domainEnd;
|
|
for (; p < end; ++p) {
|
|
QChar ch(*p);
|
|
if (chIsLinkEnd(ch)) break; // link finished
|
|
if (chIsAlmostLinkEnd(ch)) {
|
|
const QChar *endTest = p + 1;
|
|
while (endTest < end && chIsAlmostLinkEnd(*endTest)) {
|
|
++endTest;
|
|
}
|
|
if (endTest >= end || chIsLinkEnd(*endTest)) {
|
|
break; // link finished at p
|
|
}
|
|
p = endTest;
|
|
ch = *p;
|
|
}
|
|
if (ch == '(' || ch == '[' || ch == '{' || ch == '<') {
|
|
parenth.push(p);
|
|
} else if (ch == ')' || ch == ']' || ch == '}' || ch == '>') {
|
|
if (parenth.isEmpty()) break;
|
|
const QChar *q = parenth.pop(), open(*q);
|
|
if ((ch == ')' && open != '(') || (ch == ']' && open != '[') || (ch == '}' && open != '{') || (ch == '>' && open != '<')) {
|
|
p = q;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (p > domainEnd) { // check, that domain ended
|
|
if (domainEnd->unicode() != '/' && domainEnd->unicode() != '?') {
|
|
matchOffset = domainEnd - start;
|
|
continue;
|
|
}
|
|
}
|
|
lnkLength = (p - start) - lnkStart;
|
|
}
|
|
}
|
|
for (; existingEntityIndex < existingEntitiesCount && result.entities[existingEntityIndex].offset() <= lnkStart; ++existingEntityIndex) {
|
|
auto &entity = result.entities[existingEntityIndex];
|
|
accumulate_max(existingEntityEnd, entity.offset() + entity.length());
|
|
newEntities.push_back(entity);
|
|
}
|
|
if (lnkStart >= existingEntityEnd) {
|
|
result.entities.push_back({ lnkType, lnkStart, lnkLength });
|
|
}
|
|
|
|
offset = matchOffset = lnkStart + lnkLength;
|
|
}
|
|
if (!newEntities.isEmpty()) {
|
|
for (; existingEntityIndex < existingEntitiesCount; ++existingEntityIndex) {
|
|
auto &entity = result.entities[existingEntityIndex];
|
|
newEntities.push_back(entity);
|
|
}
|
|
result.entities = newEntities;
|
|
}
|
|
}
|
|
|
|
void MoveStringPart(TextWithEntities &result, int to, int from, int count) {
|
|
if (!count) return;
|
|
if (to != from) {
|
|
auto start = result.text.data();
|
|
memmove(start + to, start + from, count * sizeof(QChar));
|
|
|
|
for (auto &entity : result.entities) {
|
|
if (entity.offset() >= from + count) break;
|
|
if (entity.offset() + entity.length() <= from) continue;
|
|
if (entity.offset() >= from) {
|
|
entity.extendToLeft(from - to);
|
|
}
|
|
if (entity.offset() + entity.length() <= from + count) {
|
|
entity.shrinkFromRight(from - to);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void MovePartAndGoForward(TextWithEntities &result, int &to, int &from, int count) {
|
|
if (!count) return;
|
|
MoveStringPart(result, to, from, count);
|
|
to += count;
|
|
from += count;
|
|
}
|
|
|
|
void PrepareForSending(TextWithEntities &result, int32 flags) {
|
|
ApplyServerCleaning(result);
|
|
|
|
if (flags) {
|
|
ParseEntities(result, flags);
|
|
}
|
|
|
|
Trim(result);
|
|
}
|
|
|
|
// Replace bad symbols with space and remove '\r'.
|
|
void ApplyServerCleaning(TextWithEntities &result) {
|
|
auto len = result.text.size();
|
|
|
|
// Replace tabs with two spaces.
|
|
if (auto tabs = std::count(result.text.cbegin(), result.text.cend(), '\t')) {
|
|
auto replacement = qsl(" ");
|
|
auto replacementLength = replacement.size();
|
|
auto shift = (replacementLength - 1);
|
|
result.text.resize(len + shift * tabs);
|
|
for (auto i = len, movedTill = len, to = result.text.size(); i > 0; --i) {
|
|
if (result.text[i - 1] == '\t') {
|
|
auto toMove = movedTill - i;
|
|
to -= toMove;
|
|
MoveStringPart(result, to, i, toMove);
|
|
to -= replacementLength;
|
|
memcpy(result.text.data() + to, replacement.constData(), replacementLength * sizeof(QChar));
|
|
movedTill = i - 1;
|
|
}
|
|
}
|
|
len = result.text.size();
|
|
}
|
|
|
|
auto to = 0;
|
|
auto from = 0;
|
|
auto start = result.text.data();
|
|
for (auto ch = start, end = start + len; ch < end; ++ch) {
|
|
if (ch->unicode() == '\r') {
|
|
MovePartAndGoForward(result, to, from, (ch - start) - from);
|
|
++from;
|
|
} else if (chReplacedBySpace(*ch)) {
|
|
*ch = ' ';
|
|
}
|
|
}
|
|
MovePartAndGoForward(result, to, from, len - from);
|
|
if (to < len) result.text.resize(to);
|
|
}
|
|
|
|
void Trim(TextWithEntities &result) {
|
|
auto foundNotTrimmedChar = false;
|
|
|
|
// right trim
|
|
for (auto s = result.text.data(), e = s + result.text.size(), ch = e; ch != s;) {
|
|
--ch;
|
|
if (!chIsTrimmed(*ch)) {
|
|
if (ch + 1 < e) {
|
|
auto l = ch + 1 - s;
|
|
for (auto &entity : result.entities) {
|
|
entity.updateTextEnd(l);
|
|
}
|
|
result.text.resize(l);
|
|
}
|
|
foundNotTrimmedChar = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!foundNotTrimmedChar) {
|
|
result = TextWithEntities();
|
|
return;
|
|
}
|
|
|
|
const auto firstMonospaceOffset = EntityInText::FirstMonospaceOffset(
|
|
result.entities,
|
|
result.text.size());
|
|
|
|
// left trim
|
|
for (auto s = result.text.data(), ch = s, e = s + result.text.size(); ch != e; ++ch) {
|
|
if (!chIsTrimmed(*ch) || (ch - s) == firstMonospaceOffset) {
|
|
if (ch > s) {
|
|
auto l = ch - s;
|
|
for (auto &entity : result.entities) {
|
|
entity.shiftLeft(l);
|
|
}
|
|
result.text = result.text.mid(l);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
QByteArray SerializeTags(const TextWithTags::Tags &tags) {
|
|
if (tags.isEmpty()) {
|
|
return QByteArray();
|
|
}
|
|
|
|
QByteArray tagsSerialized;
|
|
{
|
|
QDataStream stream(&tagsSerialized, QIODevice::WriteOnly);
|
|
stream.setVersion(QDataStream::Qt_5_1);
|
|
stream << qint32(tags.size());
|
|
for (const auto &tag : tags) {
|
|
stream << qint32(tag.offset) << qint32(tag.length) << tag.id;
|
|
}
|
|
}
|
|
return tagsSerialized;
|
|
}
|
|
|
|
TextWithTags::Tags DeserializeTags(QByteArray data, int textLength) {
|
|
auto result = TextWithTags::Tags();
|
|
if (data.isEmpty()) {
|
|
return result;
|
|
}
|
|
|
|
QDataStream stream(data);
|
|
stream.setVersion(QDataStream::Qt_5_1);
|
|
|
|
qint32 tagCount = 0;
|
|
stream >> tagCount;
|
|
if (stream.status() != QDataStream::Ok) {
|
|
return result;
|
|
}
|
|
if (tagCount <= 0 || tagCount > textLength) {
|
|
return result;
|
|
}
|
|
|
|
for (auto i = 0; i != tagCount; ++i) {
|
|
qint32 offset = 0, length = 0;
|
|
QString id;
|
|
stream >> offset >> length >> id;
|
|
if (stream.status() != QDataStream::Ok) {
|
|
return result;
|
|
}
|
|
if (offset < 0 || length <= 0 || offset + length > textLength) {
|
|
return result;
|
|
}
|
|
result.push_back({ offset, length, id });
|
|
}
|
|
return result;
|
|
}
|
|
|
|
QString TagsMimeType() {
|
|
return qsl("application/x-td-field-tags");
|
|
}
|
|
|
|
QString TagsTextMimeType() {
|
|
return qsl("application/x-td-field-text");
|
|
}
|
|
|
|
} // namespace TextUtilities
|
|
|
|
EntityInText::EntityInText(
|
|
EntityType type,
|
|
int offset,
|
|
int length,
|
|
const QString &data)
|
|
: _type(type)
|
|
, _offset(offset)
|
|
, _length(length)
|
|
, _data(data) {
|
|
}
|
|
|
|
int EntityInText::FirstMonospaceOffset(
|
|
const EntitiesInText &entities,
|
|
int textLength) {
|
|
auto &&monospace = ranges::make_iterator_range(
|
|
entities.begin(),
|
|
entities.end()
|
|
) | ranges::view::filter([](const EntityInText & entity) {
|
|
return (entity.type() == EntityType::Pre)
|
|
|| (entity.type() == EntityType::Code);
|
|
});
|
|
const auto i = ranges::max_element(
|
|
monospace,
|
|
std::greater<>(),
|
|
&EntityInText::offset);
|
|
return (i == monospace.end()) ? textLength : i->offset();
|
|
}
|
|
|
|
namespace Lang {
|
|
|
|
TextWithEntities ReplaceTag<TextWithEntities>::Call(TextWithEntities &&original, ushort tag, const TextWithEntities &replacement) {
|
|
auto replacementPosition = FindTagReplacementPosition(original.text, tag);
|
|
if (replacementPosition < 0) {
|
|
return std::move(original);
|
|
}
|
|
|
|
auto result = TextWithEntities();
|
|
result.text = ReplaceTag<QString>::Replace(std::move(original.text), replacement.text, replacementPosition);
|
|
auto originalEntitiesCount = original.entities.size();
|
|
auto replacementEntitiesCount = replacement.entities.size();
|
|
if (originalEntitiesCount != 0 || replacementEntitiesCount != 0) {
|
|
result.entities.reserve(originalEntitiesCount + replacementEntitiesCount);
|
|
|
|
auto replacementEnd = replacementPosition + replacement.text.size();
|
|
auto replacementEntity = replacement.entities.cbegin();
|
|
auto addReplacementEntitiesUntil = [&replacementEntity, &replacement, &result, replacementPosition, replacementEnd](int untilPosition) {
|
|
while (replacementEntity != replacement.entities.cend()) {
|
|
auto newOffset = replacementPosition + replacementEntity->offset();
|
|
if (newOffset >= untilPosition) {
|
|
return;
|
|
}
|
|
auto newEnd = newOffset + replacementEntity->length();
|
|
newOffset = snap(newOffset, replacementPosition, replacementEnd);
|
|
newEnd = snap(newEnd, replacementPosition, replacementEnd);
|
|
if (auto newLength = newEnd - newOffset) {
|
|
result.entities.push_back({ replacementEntity->type(), newOffset, newLength, replacementEntity->data() });
|
|
}
|
|
++replacementEntity;
|
|
}
|
|
};
|
|
|
|
for_const (auto &entity, original.entities) {
|
|
// Transform the entity by the replacement.
|
|
auto offset = entity.offset();
|
|
auto end = offset + entity.length();
|
|
if (offset > replacementPosition) {
|
|
offset = offset + replacement.text.size() - kTagReplacementSize;
|
|
}
|
|
if (end > replacementPosition) {
|
|
end = end + replacement.text.size() - kTagReplacementSize;
|
|
}
|
|
offset = snap(offset, 0, result.text.size());
|
|
end = snap(end, 0, result.text.size());
|
|
|
|
// Add all replacement entities that start before the current original entity.
|
|
addReplacementEntitiesUntil(offset);
|
|
|
|
// Add a modified original entity.
|
|
if (auto length = end - offset) {
|
|
result.entities.push_back({ entity.type(), offset, length, entity.data() });
|
|
}
|
|
}
|
|
// Add the remaining replacement entities.
|
|
addReplacementEntitiesUntil(result.text.size());
|
|
}
|
|
return result;
|
|
}
|
|
|
|
} // namespace Lang
|