// This file is part of Desktop App Toolkit, // a set of libraries for developing nice desktop applications. // // For license and copyright information please follow this link: // https://github.com/desktop-app/legal/blob/master/LEGAL // #include "ui/text/text_entity.h" #include "base/qthelp_url.h" #include "base/qthelp_regex.h" #include "base/crc32hash.h" #include "ui/text/text.h" #include "ui/widgets/fields/input_field.h" #include "ui/emoji_config.h" #include "ui/basic_click_handlers.h" #include "base/qt/qt_common_adapters.h" #include #include #include #include namespace TextUtilities { namespace { constexpr auto kTagSeparator = '\\'; using namespace Ui::Text; 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 QString::fromUtf8("[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 QString::fromUtf8("\\s\\.,:;<>|'\"\\[\\]\\{\\}\\~\\!\\?\\%\\^\\(\\)\\-\\+=\\x10") + quotes + additional; } QString Separators(const QString &additional) { static const auto quotes = Quotes(); return QString::fromUtf8(" \x10\n\r\t.,:;<>|'\"[]{}!?%^()-+=") + QChar(0xfdd0) // QTextBeginningOfFrame + QChar(0xfdd1) // QTextEndOfFrame + QChar(QChar::ParagraphSeparator) + QChar(QChar::LineSeparator) + quotes + additional; } QString SeparatorsBold() { return Separators(QString::fromUtf8("`~/")); } QString SeparatorsItalic() { return Separators(QString::fromUtf8("`*~/")); } QString SeparatorsStrikeOut() { return Separators(QString::fromUtf8("`*~/")); } QString SeparatorsMono() { return Separators(QString::fromUtf8("*~/")); } QString SeparatorsSpoiler() { return Separators(QString::fromUtf8("|*~/")); } QString ExpressionHashtag() { return QString::fromUtf8("(^|[") + ExpressionSeparators(QString::fromUtf8("`\\*/")) + QString::fromUtf8("])#[\\w]{2,64}([\\W]|$)"); } QString ExpressionHashtagExclude() { return QString::fromUtf8("^#?\\d+$"); } QString ExpressionMention() { return QString::fromUtf8("(^|[") + ExpressionSeparators(QString::fromUtf8("`\\*/")) + QString::fromUtf8("])@[A-Za-z_0-9]{1,32}([\\W]|$)"); } QString ExpressionBotCommand() { return QString::fromUtf8("(^|[") + ExpressionSeparators(QString::fromUtf8("`\\*")) + QString::fromUtf8("])/[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); result.optimize(); return result; } base::flat_set CreateValidProtocols() { auto result = base::flat_set(); const auto addOne = [&](const QString &string) { result.insert(base::crc32(string.constData(), string.size() * sizeof(QChar))); }; addOne(QString::fromLatin1("itmss")); // itunes addOne(QString::fromLatin1("http")); addOne(QString::fromLatin1("https")); addOne(QString::fromLatin1("ftp")); addOne(QString::fromLatin1("tg")); // local urls return result; } base::flat_set CreateValidTopDomains() { auto result = base::flat_set(); auto addOne = [&result](const QString &string) { result.insert(base::crc32(string.constData(), string.size() * sizeof(QChar))); }; addOne(QString::fromLatin1("ac")); addOne(QString::fromLatin1("ad")); addOne(QString::fromLatin1("ae")); addOne(QString::fromLatin1("af")); addOne(QString::fromLatin1("ag")); addOne(QString::fromLatin1("ai")); addOne(QString::fromLatin1("al")); addOne(QString::fromLatin1("am")); addOne(QString::fromLatin1("an")); addOne(QString::fromLatin1("ao")); addOne(QString::fromLatin1("aq")); addOne(QString::fromLatin1("ar")); addOne(QString::fromLatin1("as")); addOne(QString::fromLatin1("at")); addOne(QString::fromLatin1("au")); addOne(QString::fromLatin1("aw")); addOne(QString::fromLatin1("ax")); addOne(QString::fromLatin1("az")); addOne(QString::fromLatin1("ba")); addOne(QString::fromLatin1("bb")); addOne(QString::fromLatin1("bd")); addOne(QString::fromLatin1("be")); addOne(QString::fromLatin1("bf")); addOne(QString::fromLatin1("bg")); addOne(QString::fromLatin1("bh")); addOne(QString::fromLatin1("bi")); addOne(QString::fromLatin1("bj")); addOne(QString::fromLatin1("bm")); addOne(QString::fromLatin1("bn")); addOne(QString::fromLatin1("bo")); addOne(QString::fromLatin1("br")); addOne(QString::fromLatin1("bs")); addOne(QString::fromLatin1("bt")); addOne(QString::fromLatin1("bv")); addOne(QString::fromLatin1("bw")); addOne(QString::fromLatin1("by")); addOne(QString::fromLatin1("bz")); addOne(QString::fromLatin1("ca")); addOne(QString::fromLatin1("cc")); addOne(QString::fromLatin1("cd")); addOne(QString::fromLatin1("cf")); addOne(QString::fromLatin1("cg")); addOne(QString::fromLatin1("ch")); addOne(QString::fromLatin1("ci")); addOne(QString::fromLatin1("ck")); addOne(QString::fromLatin1("cl")); addOne(QString::fromLatin1("cm")); addOne(QString::fromLatin1("cn")); addOne(QString::fromLatin1("co")); addOne(QString::fromLatin1("cr")); addOne(QString::fromLatin1("cu")); addOne(QString::fromLatin1("cv")); addOne(QString::fromLatin1("cx")); addOne(QString::fromLatin1("cy")); addOne(QString::fromLatin1("cz")); addOne(QString::fromLatin1("de")); addOne(QString::fromLatin1("dj")); addOne(QString::fromLatin1("dk")); addOne(QString::fromLatin1("dm")); addOne(QString::fromLatin1("do")); addOne(QString::fromLatin1("dz")); addOne(QString::fromLatin1("ec")); addOne(QString::fromLatin1("ee")); addOne(QString::fromLatin1("eg")); addOne(QString::fromLatin1("eh")); addOne(QString::fromLatin1("er")); addOne(QString::fromLatin1("es")); addOne(QString::fromLatin1("et")); addOne(QString::fromLatin1("eu")); addOne(QString::fromLatin1("fi")); addOne(QString::fromLatin1("fj")); addOne(QString::fromLatin1("fk")); addOne(QString::fromLatin1("fm")); addOne(QString::fromLatin1("fo")); addOne(QString::fromLatin1("fr")); addOne(QString::fromLatin1("ga")); addOne(QString::fromLatin1("gd")); addOne(QString::fromLatin1("ge")); addOne(QString::fromLatin1("gf")); addOne(QString::fromLatin1("gg")); addOne(QString::fromLatin1("gh")); addOne(QString::fromLatin1("gi")); addOne(QString::fromLatin1("gl")); addOne(QString::fromLatin1("gm")); addOne(QString::fromLatin1("gn")); addOne(QString::fromLatin1("gp")); addOne(QString::fromLatin1("gq")); addOne(QString::fromLatin1("gr")); addOne(QString::fromLatin1("gs")); addOne(QString::fromLatin1("gt")); addOne(QString::fromLatin1("gu")); addOne(QString::fromLatin1("gw")); addOne(QString::fromLatin1("gy")); addOne(QString::fromLatin1("hk")); addOne(QString::fromLatin1("hm")); addOne(QString::fromLatin1("hn")); addOne(QString::fromLatin1("hr")); addOne(QString::fromLatin1("ht")); addOne(QString::fromLatin1("hu")); addOne(QString::fromLatin1("id")); addOne(QString::fromLatin1("ie")); addOne(QString::fromLatin1("il")); addOne(QString::fromLatin1("im")); addOne(QString::fromLatin1("in")); addOne(QString::fromLatin1("io")); addOne(QString::fromLatin1("iq")); addOne(QString::fromLatin1("ir")); addOne(QString::fromLatin1("is")); addOne(QString::fromLatin1("it")); addOne(QString::fromLatin1("je")); addOne(QString::fromLatin1("jm")); addOne(QString::fromLatin1("jo")); addOne(QString::fromLatin1("jp")); addOne(QString::fromLatin1("ke")); addOne(QString::fromLatin1("kg")); addOne(QString::fromLatin1("kh")); addOne(QString::fromLatin1("ki")); addOne(QString::fromLatin1("km")); addOne(QString::fromLatin1("kn")); addOne(QString::fromLatin1("kp")); addOne(QString::fromLatin1("kr")); addOne(QString::fromLatin1("kw")); addOne(QString::fromLatin1("ky")); addOne(QString::fromLatin1("kz")); addOne(QString::fromLatin1("la")); addOne(QString::fromLatin1("lb")); addOne(QString::fromLatin1("lc")); addOne(QString::fromLatin1("li")); addOne(QString::fromLatin1("lk")); addOne(QString::fromLatin1("lr")); addOne(QString::fromLatin1("ls")); addOne(QString::fromLatin1("lt")); addOne(QString::fromLatin1("lu")); addOne(QString::fromLatin1("lv")); addOne(QString::fromLatin1("ly")); addOne(QString::fromLatin1("ma")); addOne(QString::fromLatin1("mc")); addOne(QString::fromLatin1("md")); addOne(QString::fromLatin1("me")); addOne(QString::fromLatin1("mg")); addOne(QString::fromLatin1("mh")); addOne(QString::fromLatin1("mk")); addOne(QString::fromLatin1("ml")); addOne(QString::fromLatin1("mm")); addOne(QString::fromLatin1("mn")); addOne(QString::fromLatin1("mo")); addOne(QString::fromLatin1("mp")); addOne(QString::fromLatin1("mq")); addOne(QString::fromLatin1("mr")); addOne(QString::fromLatin1("ms")); addOne(QString::fromLatin1("mt")); addOne(QString::fromLatin1("mu")); addOne(QString::fromLatin1("mv")); addOne(QString::fromLatin1("mw")); addOne(QString::fromLatin1("mx")); addOne(QString::fromLatin1("my")); addOne(QString::fromLatin1("mz")); addOne(QString::fromLatin1("na")); addOne(QString::fromLatin1("nc")); addOne(QString::fromLatin1("ne")); addOne(QString::fromLatin1("nf")); addOne(QString::fromLatin1("ng")); addOne(QString::fromLatin1("ni")); addOne(QString::fromLatin1("nl")); addOne(QString::fromLatin1("no")); addOne(QString::fromLatin1("np")); addOne(QString::fromLatin1("nr")); addOne(QString::fromLatin1("nu")); addOne(QString::fromLatin1("nz")); addOne(QString::fromLatin1("om")); addOne(QString::fromLatin1("pa")); addOne(QString::fromLatin1("pe")); addOne(QString::fromLatin1("pf")); addOne(QString::fromLatin1("pg")); addOne(QString::fromLatin1("ph")); addOne(QString::fromLatin1("pk")); addOne(QString::fromLatin1("pl")); addOne(QString::fromLatin1("pm")); addOne(QString::fromLatin1("pn")); addOne(QString::fromLatin1("pr")); addOne(QString::fromLatin1("ps")); addOne(QString::fromLatin1("pt")); addOne(QString::fromLatin1("pw")); addOne(QString::fromLatin1("py")); addOne(QString::fromLatin1("qa")); addOne(QString::fromLatin1("re")); addOne(QString::fromLatin1("ro")); addOne(QString::fromLatin1("ru")); addOne(QString::fromLatin1("rs")); addOne(QString::fromLatin1("rw")); addOne(QString::fromLatin1("sa")); addOne(QString::fromLatin1("sb")); addOne(QString::fromLatin1("sc")); addOne(QString::fromLatin1("sd")); addOne(QString::fromLatin1("se")); addOne(QString::fromLatin1("sg")); addOne(QString::fromLatin1("sh")); addOne(QString::fromLatin1("si")); addOne(QString::fromLatin1("sj")); addOne(QString::fromLatin1("sk")); addOne(QString::fromLatin1("sl")); addOne(QString::fromLatin1("sm")); addOne(QString::fromLatin1("sn")); addOne(QString::fromLatin1("so")); addOne(QString::fromLatin1("sr")); addOne(QString::fromLatin1("ss")); addOne(QString::fromLatin1("st")); addOne(QString::fromLatin1("su")); addOne(QString::fromLatin1("sv")); addOne(QString::fromLatin1("sx")); addOne(QString::fromLatin1("sy")); addOne(QString::fromLatin1("sz")); addOne(QString::fromLatin1("tc")); addOne(QString::fromLatin1("td")); addOne(QString::fromLatin1("tf")); addOne(QString::fromLatin1("tg")); addOne(QString::fromLatin1("th")); addOne(QString::fromLatin1("tj")); addOne(QString::fromLatin1("tk")); addOne(QString::fromLatin1("tl")); addOne(QString::fromLatin1("tm")); addOne(QString::fromLatin1("tn")); addOne(QString::fromLatin1("to")); addOne(QString::fromLatin1("tp")); addOne(QString::fromLatin1("tr")); addOne(QString::fromLatin1("tt")); addOne(QString::fromLatin1("tv")); addOne(QString::fromLatin1("tw")); addOne(QString::fromLatin1("tz")); addOne(QString::fromLatin1("ua")); addOne(QString::fromLatin1("ug")); addOne(QString::fromLatin1("uk")); addOne(QString::fromLatin1("um")); addOne(QString::fromLatin1("us")); addOne(QString::fromLatin1("uy")); addOne(QString::fromLatin1("uz")); addOne(QString::fromLatin1("va")); addOne(QString::fromLatin1("vc")); addOne(QString::fromLatin1("ve")); addOne(QString::fromLatin1("vg")); addOne(QString::fromLatin1("vi")); addOne(QString::fromLatin1("vn")); addOne(QString::fromLatin1("vu")); addOne(QString::fromLatin1("wf")); addOne(QString::fromLatin1("ws")); addOne(QString::fromLatin1("ye")); addOne(QString::fromLatin1("yt")); addOne(QString::fromLatin1("yu")); addOne(QString::fromLatin1("za")); addOne(QString::fromLatin1("zm")); addOne(QString::fromLatin1("zw")); addOne(QString::fromLatin1("arpa")); addOne(QString::fromLatin1("aero")); addOne(QString::fromLatin1("asia")); addOne(QString::fromLatin1("biz")); addOne(QString::fromLatin1("cat")); addOne(QString::fromLatin1("com")); addOne(QString::fromLatin1("coop")); addOne(QString::fromLatin1("info")); addOne(QString::fromLatin1("int")); addOne(QString::fromLatin1("jobs")); addOne(QString::fromLatin1("mobi")); addOne(QString::fromLatin1("museum")); addOne(QString::fromLatin1("name")); addOne(QString::fromLatin1("net")); addOne(QString::fromLatin1("org")); addOne(QString::fromLatin1("post")); addOne(QString::fromLatin1("pro")); addOne(QString::fromLatin1("tel")); addOne(QString::fromLatin1("travel")); addOne(QString::fromLatin1("xxx")); addOne(QString::fromLatin1("edu")); addOne(QString::fromLatin1("gov")); addOne(QString::fromLatin1("mil")); addOne(QString::fromLatin1("local")); addOne(QString::fromLatin1("xn--lgbbat1ad8j")); addOne(QString::fromLatin1("xn--54b7fta0cc")); addOne(QString::fromLatin1("xn--fiqs8s")); addOne(QString::fromLatin1("xn--fiqz9s")); addOne(QString::fromLatin1("xn--wgbh1c")); addOne(QString::fromLatin1("xn--node")); addOne(QString::fromLatin1("xn--j6w193g")); addOne(QString::fromLatin1("xn--h2brj9c")); addOne(QString::fromLatin1("xn--mgbbh1a71e")); addOne(QString::fromLatin1("xn--fpcrj9c3d")); addOne(QString::fromLatin1("xn--gecrj9c")); addOne(QString::fromLatin1("xn--s9brj9c")); addOne(QString::fromLatin1("xn--xkc2dl3a5ee0h")); addOne(QString::fromLatin1("xn--45brj9c")); addOne(QString::fromLatin1("xn--mgba3a4f16a")); addOne(QString::fromLatin1("xn--mgbayh7gpa")); addOne(QString::fromLatin1("xn--80ao21a")); addOne(QString::fromLatin1("xn--mgbx4cd0ab")); addOne(QString::fromLatin1("xn--l1acc")); addOne(QString::fromLatin1("xn--mgbc0a9azcg")); addOne(QString::fromLatin1("xn--mgb9awbf")); addOne(QString::fromLatin1("xn--mgbai9azgqp6j")); addOne(QString::fromLatin1("xn--ygbi2ammx")); addOne(QString::fromLatin1("xn--wgbl6a")); addOne(QString::fromLatin1("xn--p1ai")); addOne(QString::fromLatin1("xn--mgberp4a5d4ar")); addOne(QString::fromLatin1("xn--90a3ac")); addOne(QString::fromLatin1("xn--yfro4i67o")); addOne(QString::fromLatin1("xn--clchc0ea0b2g2a9gcd")); addOne(QString::fromLatin1("xn--3e0b707e")); addOne(QString::fromLatin1("xn--fzc2c9e2c")); addOne(QString::fromLatin1("xn--xkc2al3hye2a")); addOne(QString::fromLatin1("xn--mgbtf8fl")); addOne(QString::fromLatin1("xn--kprw13d")); addOne(QString::fromLatin1("xn--kpry57d")); addOne(QString::fromLatin1("xn--o3cw4h")); addOne(QString::fromLatin1("xn--pgbs0dh")); addOne(QString::fromLatin1("xn--j1amh")); addOne(QString::fromLatin1("xn--mgbaam7a8h")); addOne(QString::fromLatin1("xn--mgb2ddes")); addOne(QString::fromLatin1("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(QString::fromLatin1("[\\@\\s\\-\\+\\(\\)\\[\\]\\{\\}\\<\\>\\,\\.\\:\\!\\_\\;\\\"\\'\\x0]")); return result; } [[nodiscard]] QString ExpandCustomLinks(const TextWithTags &text) { const auto entities = ConvertTextTagsToEntities(text.tags); auto &&urls = ranges::make_subrange( entities.begin(), entities.end() ) | ranges::views::filter([](const EntityInText &entity) { return entity.type() == EntityType::CustomUrl; }); const auto &original = text.text; if (urls.begin() == urls.end()) { return original; } auto result = QString(); auto offset = 0; for (const auto &entity : urls) { const auto till = entity.offset() + entity.length(); if (till > offset) { result.append(base::StringViewMid(original, offset, till - offset)); } result.append(qstr(" (")).append(entity.data()).append(')'); offset = till; } if (original.size() > offset) { result.append(base::StringViewMid(original, offset)); } return result; } std::unique_ptr MimeDataFromText( TextWithTags &&text, const QString &expanded) { if (expanded.isEmpty()) { return nullptr; } auto result = std::make_unique(); result->setText(expanded); if (!text.tags.isEmpty()) { for (auto &tag : text.tags) { tag.id = Ui::Integration::Instance().convertTagToMimeTag(tag.id); } result->setData( TextUtilities::TagsTextMimeType(), text.text.toUtf8()); result->setData( TextUtilities::TagsMimeType(), TextUtilities::SerializeTags(text.tags)); } return result; } bool IsSentencePartEnd(QChar ch) { return (ch == ',') || (ch == ':') || (ch == ';'); } bool IsSentenceEnd(QChar ch) { return (ch == '.') || (ch == '?') || (ch == '!'); } } // 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; } QRegularExpression RegExpDigitsExclude() { return QRegularExpression("[^\\d]"); } QString MarkdownBoldGoodBefore() { return SeparatorsBold(); } QString MarkdownBoldBadAfter() { return QString::fromLatin1("*"); } QString MarkdownItalicGoodBefore() { return SeparatorsItalic(); } QString MarkdownItalicBadAfter() { return QString::fromLatin1("_"); } QString MarkdownStrikeOutGoodBefore() { return SeparatorsStrikeOut(); } QString MarkdownStrikeOutBadAfter() { return QString::fromLatin1("~"); } QString MarkdownCodeGoodBefore() { return SeparatorsMono(); } QString MarkdownCodeBadAfter() { return QString::fromLatin1("`\n\r"); } QString MarkdownPreGoodBefore() { return SeparatorsMono(); } QString MarkdownPreBadAfter() { return QString::fromLatin1("`"); } QString MarkdownSpoilerGoodBefore() { return SeparatorsSpoiler(); } QString MarkdownSpoilerBadAfter() { return QString::fromLatin1("|"); } bool IsValidProtocol(const QString &protocol) { static const auto list = CreateValidProtocols(); return list.contains(base::crc32(protocol.constData(), protocol.size() * sizeof(QChar))); } bool IsValidTopDomain(const QString &protocol) { static const auto list = CreateValidTopDomains(); return list.contains(base::crc32(protocol.constData(), protocol.size() * sizeof(QChar))); } 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(), e = text.unicode() + text.size(); // Trim. while (s < e && IsTrimmed(*s)) { ++s; } while (s < e && IsTrimmed(*(e - 1))) { --e; } if (e - s != text.size()) { result = text.mid(s - text.unicode(), e - s); } for (auto ch = s; ch != e; ++ch) { if (IsNewline(*ch)/* || *ch == TextCommand*/) { result[int(ch - s)] = QChar::Space; } } return result; } TextWithEntities SingleLine(const TextWithEntities &text) { auto copy = text; Trim(copy); return { SingleLine(copy.text), std::move(copy.entities) }; } 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 (IsDiacritic(*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; } QString RemoveEmoji(const QString &text) { auto result = QString(); result.reserve(text.size()); auto begin = text.data(); const auto end = begin + text.size(); while (begin != end) { auto length = 0; if (Ui::Emoji::Find(begin, end, &length)) { begin += length; } else { result.append(*begin++); } } return 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(), Qt::SkipEmptyParts); result.reserve(list.size()); for (const auto &word : std::as_const(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::Blockquote) || (entityType == EntityType::Code); // #TODO entities 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 (IsNewline(*ch)) { if (inEntity) { if (ch + 1 < end && IsNewline(*(ch + 1))) { markGoodAsLevel(12); } else { markGoodAsLevel(11); } } else if (ch + 1 < end && IsNewline(*(ch + 1))) { markGoodAsLevel(15); } else if (currentEntity < entityCount && ch + 1 == start + left.entities[currentEntity].offset() && (left.entities[currentEntity].type() == EntityType::Pre || left.entities[currentEntity].type() == EntityType::Blockquote)) { 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 || left.entities[currentEntity - 1].type() == EntityType::Blockquote)) { markGoodAsLevel(14); } else { markGoodAsLevel(13); } } else if (IsSpace(*ch)) { if (IsSentenceEnd(*(ch - 1))) { markGoodAsLevel(9 + noEntityLevel); } else if (IsSentencePartEnd(*(ch - 1))) { markGoodAsLevel(7 + noEntityLevel); } else { markGoodAsLevel(5 + noEntityLevel); } } else if (IsWordSeparator(*(ch - 1))) { markGoodAsLevel(3 + noEntityLevel); } else { markGoodAsLevel(1 + noEntityLevel); } } } int elen = 0; if (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; } MentionNameFields MentionNameDataToFields(QStringView data) { const auto components = data.split('.'); if (components.size() != 2) { return {}; } const auto parts = components[1].split(':'); if (parts.size() != 2) { return {}; } return { .selfId = parts[1].toULongLong(), .userId = components[0].toULongLong(), .accessHash = parts[0].toULongLong(), }; } QString MentionNameDataFromFields(const MentionNameFields &fields) { return u"%1.%2:%3"_q .arg(fields.userId) .arg(fields.accessHash) .arg(fields.selfId); } TextWithEntities ParseEntities(const QString &text, int32 flags) { auto result = TextWithEntities{ text, EntitiesInText() }; ParseEntities(result, flags); return result; } // Some code is duplicated in message_field.cpp! void ParseEntities(TextWithEntities &result, int32 flags) { constexpr auto kNotFound = std::numeric_limits::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(); const auto start = result.text.constData(); const auto end = start + result.text.size(); for (int32 offset = 0, matchOffset = offset, mentionSkip = 0; offset < len;) { 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.capturedView(1).isEmpty()) { ++hashtagStart; } if (!mHashtag.capturedView(2).isEmpty()) { --hashtagEnd; } if (RegExpHashtagExclude().match( result.text.mid( hashtagStart + 1, hashtagEnd - hashtagStart - 1)).hasMatch()) { hashtagIgnore = true; } } while (mMention.hasMatch()) { if (!mMention.capturedView(1).isEmpty()) { ++mentionStart; } if (!mMention.capturedView(2).isEmpty()) { --mentionEnd; } if (!(start + mentionStart + 1)->isLetter() || !(start + mentionEnd - 1)->isLetterOrNumber()) { mentionSkip = mentionEnd; if (mentionSkip < len && (start + mentionSkip)->isLowSurrogate()) { ++mentionSkip; } 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.capturedView(1).isEmpty()) { ++botCommandStart; } if (!mBotCommand.capturedView(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; } lnkType = EntityType::Mention; lnkStart = mentionStart; lnkLength = mentionEnd - mentionStart; } else if (hashtagStart < domainStart && hashtagStart < botCommandStart) { if (hashtagIgnore) { offset = matchOffset = hashtagEnd; continue; } lnkType = EntityType::Hashtag; lnkStart = hashtagStart; lnkLength = hashtagEnd - hashtagStart; } else if (botCommandStart < domainStart) { lnkType = EntityType::BotCommand; lnkStart = botCommandStart; lnkLength = botCommandEnd - botCommandStart; } else { 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 parenth; const QChar *domainEnd = start + mDomain.capturedEnd(), *p = domainEnd; for (; p < end; ++p) { QChar ch(*p); if (IsLinkEnd(ch)) { break; // link finished } else if (IsAlmostLinkEnd(ch)) { const QChar *endTest = p + 1; while (endTest < end && IsAlmostLinkEnd(*endTest)) { ++endTest; } if (endTest >= end || IsLinkEnd(*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 = QString::fromLatin1(" "); 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 (IsReplacedBySpace(*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 (!IsTrimmed(*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 (!IsTrimmed(*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; } } } int SerializeTagsSize(const TextWithTags::Tags &tags) { auto result = int(sizeof(qint32)); // QByteArray size if (tags.isEmpty()) { return result; } result += sizeof(qint32); for (const auto &tag : tags) { result += 2 * sizeof(qint32) // offset, length + sizeof(quint32) // id.size + tag.id.size() * sizeof(ushort); } return result; } 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 QString::fromLatin1("application/x-td-field-tags"); } QString TagsTextMimeType() { return QString::fromLatin1("application/x-td-field-text"); } bool IsMentionLink(QStringView link) { return link.startsWith(kMentionTagStart); } QString MentionEntityData(QStringView link) { const auto match = qthelp::regex_match( "^(\\d+\\.\\d+:\\d+)(/|$)", base::StringViewMid(link, kMentionTagStart.size())); return match ? match->captured(1) : QString(); } bool IsSeparateTag(QStringView tag) { return (tag == Ui::InputField::kTagCode) || (tag == Ui::InputField::kTagPre); } QString JoinTag(const QList &list) { if (list.isEmpty()) { return QString(); } auto length = (list.size() - 1); for (const auto &entry : list) { length += entry.size(); } auto result = QString(); result.reserve(length); result.append(list.front()); for (auto i = 1, count = int(list.size()); i != count; ++i) { if (!IsSeparateTag(list[i])) { result.append(kTagSeparator).append(list[i]); } } return result; } QList SplitTags(QStringView tag) { return tag.split(kTagSeparator); } QString TagWithRemoved(const QString &tag, const QString &removed) { if (tag == removed) { return QString(); } auto list = SplitTags(tag); list.erase(ranges::remove(list, QStringView(removed)), list.end()); return JoinTag(list); } QString TagWithAdded(const QString &tag, const QString &added) { if (tag.isEmpty() || tag == added) { return added; } auto list = SplitTags(tag); const auto ref = QStringView(added); if (list.contains(ref)) { return tag; } list.push_back(ref); std::sort(list.begin(), list.end()); return JoinTag(list); } EntitiesInText ConvertTextTagsToEntities(const TextWithTags::Tags &tags) { auto result = EntitiesInText(); if (tags.isEmpty()) { return result; } constexpr auto kInMaskTypes = std::array{ EntityType::Bold, EntityType::Italic, EntityType::Underline, EntityType::StrikeOut, EntityType::Spoiler, EntityType::Code, EntityType::Pre, EntityType::Blockquote, }; struct State { QString link; QString language; uint32 mask = 0; void set(EntityType type) { mask |= (1 << int(type)); } void remove(EntityType type) { mask &= ~(1 << int(type)); } [[nodiscard]] bool has(EntityType type) const { return (mask & (1 << int(type))); } }; auto offset = 0; auto state = State(); auto notClosedEntities = std::vector(); // Stack of indices. const auto closeOne = [&] { Expects(!notClosedEntities.empty()); auto &entity = result[notClosedEntities.back()]; entity = { entity.type(), entity.offset(), offset - entity.offset(), entity.data(), }; if (ranges::contains(kInMaskTypes, entity.type())) { state.remove(entity.type()); } else { state.link = QString(); } notClosedEntities.pop_back(); }; const auto closeType = [&](EntityType type) { auto closeCount = 0; const auto notClosedCount = notClosedEntities.size(); while (closeCount < notClosedCount) { const auto index = notClosedCount - closeCount - 1; if (result[notClosedEntities[index]].type() == type) { for (auto i = 0; i != closeCount + 1; ++i) { closeOne(); } break; } ++closeCount; } }; const auto openType = [&](EntityType type, const QString &data = {}) { notClosedEntities.push_back(result.size()); result.push_back({ type, offset, -1, data }); }; const auto processState = [&](State nextState) { const auto linkChanged = (nextState.link != state.link); if (linkChanged) { if (Ui::InputField::IsCustomEmojiLink(state.link)) { closeType(EntityType::CustomEmoji); } else if (IsMentionLink(state.link)) { closeType(EntityType::MentionName); } else { closeType(EntityType::CustomUrl); } } for (const auto type : kInMaskTypes) { if (state.has(type) && !nextState.has(type)) { closeType(type); } } if (linkChanged && !nextState.link.isEmpty()) { if (Ui::InputField::IsCustomEmojiLink(nextState.link)) { const auto data = Ui::InputField::CustomEmojiEntityData( nextState.link); if (!data.isEmpty()) { openType(EntityType::CustomEmoji, data); } } else if (IsMentionLink(nextState.link)) { const auto data = MentionEntityData(nextState.link); if (!data.isEmpty()) { openType(EntityType::MentionName, data); } } else { openType(EntityType::CustomUrl, nextState.link); } } for (const auto type : kInMaskTypes) { if (nextState.has(type) && !state.has(type)) { openType(type, nextState.language); } } state = nextState; }; const auto stateForTag = [&](const QString &tag) { using Tags = Ui::InputField; auto result = State(); const auto list = SplitTags(tag); const auto languageStart = Tags::kTagPre.size(); for (const auto &single : list) { if (single == Tags::kTagBold) { result.set(EntityType::Bold); } else if (single == Tags::kTagItalic) { result.set(EntityType::Italic); } else if (single == Tags::kTagUnderline) { result.set(EntityType::Underline); } else if (single == Tags::kTagStrikeOut) { result.set(EntityType::StrikeOut); } else if (single == Tags::kTagCode) { result.set(EntityType::Code); } else if (single == Tags::kTagPre) { result.set(EntityType::Pre); } else if (single.size() > languageStart && single.startsWith(Tags::kTagPre)) { result.set(EntityType::Pre); result.language = single.mid(languageStart).toString(); } else if (single == Tags::kTagBlockquote) { result.set(EntityType::Blockquote); } else if (single == Tags::kTagSpoiler) { result.set(EntityType::Spoiler); } else { result.link = single.toString(); } } return result; }; for (const auto &tag : tags) { if (tag.offset > offset) { processState(State()); } offset = tag.offset; processState(stateForTag(tag.id)); offset += tag.length; } processState(State()); result.erase(ranges::remove_if(result, [](const EntityInText &entity) { return (entity.length() <= 0); }), result.end()); return result; } TextWithTags::Tags ConvertEntitiesToTextTags( const EntitiesInText &entities) { auto result = TextWithTags::Tags(); if (entities.isEmpty()) { return result; } auto offset = 0; auto current = QString(); const auto updateCurrent = [&](int nextOffset, const QString &next) { if (next == current) { return; } else if (nextOffset > offset) { if (!current.isEmpty()) { result.push_back({ offset, nextOffset - offset, current }); } offset = nextOffset; } current = next; }; auto toRemove = std::vector>(); const auto removeTill = [&](int nextOffset) { while (!toRemove.empty() && toRemove.front().first <= nextOffset) { updateCurrent( toRemove.front().first, TagWithRemoved(current, toRemove.front().second)); toRemove.erase(toRemove.begin()); } }; for (const auto &entity : entities) { const auto push = [&](const QString &tag) { removeTill(entity.offset()); updateCurrent(entity.offset(), TagWithAdded(current, tag)); toRemove.push_back({ offset + entity.length(), tag }); ranges::sort(toRemove); }; switch (entity.type()) { case EntityType::MentionName: { const auto match = QRegularExpression( "^(\\d+\\.\\d+:\\d+)$" ).match(entity.data()); if (match.hasMatch()) { push(kMentionTagStart + entity.data()); } } break; case EntityType::CustomUrl: { const auto url = entity.data(); if (Ui::InputField::IsValidMarkdownLink(url) && !IsMentionLink(url)) { push(url); } } break; case EntityType::CustomEmoji: { const auto match = QRegularExpression( "^(\\d+)$" ).match(entity.data()); if (match.hasMatch()) { push(Ui::InputField::CustomEmojiLink(entity.data())); } } break; case EntityType::Bold: push(Ui::InputField::kTagBold); break; //case EntityType::Semibold: // Semibold is for UI parts only. // push(Ui::InputField::kTagSemibold); // break; case EntityType::Italic: push(Ui::InputField::kTagItalic); break; case EntityType::Underline: push(Ui::InputField::kTagUnderline); break; case EntityType::StrikeOut: push(Ui::InputField::kTagStrikeOut); break; case EntityType::Code: push(Ui::InputField::kTagCode); break; case EntityType::Pre: { if (!entity.data().isEmpty()) { const auto language = QRegularExpression("^[a-z0-9\\-]+$"); if (language.match(entity.data()).hasMatch()) { push(Ui::InputField::kTagPre + entity.data()); break; } } push(Ui::InputField::kTagPre); } break; case EntityType::Blockquote: push(Ui::InputField::kTagBlockquote); break; case EntityType::Spoiler: push(Ui::InputField::kTagSpoiler); break; } } if (!toRemove.empty()) { removeTill(toRemove.back().first); } return result; } std::unique_ptr MimeDataFromText(const TextForMimeData &text) { return MimeDataFromText( { text.rich.text, ConvertEntitiesToTextTags(text.rich.entities) }, text.expanded); } std::unique_ptr MimeDataFromText(TextWithTags &&text) { const auto expanded = ExpandCustomLinks(text); return MimeDataFromText(std::move(text), expanded); } void SetClipboardText( const TextForMimeData &text, QClipboard::Mode mode) { if (auto data = MimeDataFromText(text)) { QGuiApplication::clipboard()->setMimeData(data.release(), mode); } } } // namespace TextUtilities TextForMimeData TextForMimeData::WithExpandedLinks( const TextWithEntities &text) { auto result = TextForMimeData{ .rich = text }; if (!ranges::contains( text.entities, EntityType::CustomUrl, &EntityInText::type)) { result.expanded = text.text; } else { auto from = 0; for (const auto &entity : text.entities) { if (entity.type() != EntityType::CustomUrl) { continue; } // This logic is duplicated in Ui::Text::String::toText. const auto &data = entity.data(); if (!data.startsWith(qstr("internal:")) && (data != UrlClickHandler::EncodeForOpening( text.text.mid(entity.offset(), entity.length())))) { const auto till = entity.offset() + entity.length(); if (const auto add = till - from; add > 0) { result.expanded.append(text.text.data() + from, add); from = till; } result.expanded.append(qstr(" (")).append(data).append(')'); } } const auto till = text.text.size(); if (const auto add = till - from; add > 0) { result.expanded.append(text.text.data() + from, add); from = till; } } return result; } 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_subrange( entities.begin(), entities.end() ) | ranges::views::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(); }