/* This file is part of Telegram Desktop, the official desktop version of Telegram messaging app, see https://telegram.org Telegram Desktop is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. It is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE Copyright (c) 2014 John Preston, https://desktop.telegram.org */ #pragma once #include "gui/text.h" void emojiInit(); EmojiPtr emojiGet(uint32 code); EmojiPtr emojiGet(uint32 code, uint32 code2); EmojiPtr emojiGet(EmojiPtr emoji, uint32 color); EmojiPtr emojiGet(const QChar *from, const QChar *end); QString emojiGetSequence(int index); inline QString emojiString(EmojiPtr emoji) { if ((emoji->code & 0xFFFF0000U) == 0xFFFF0000U) { // sequence return emojiGetSequence(emoji->code & 0xFFFFU); } QString result; result.reserve(emoji->len + (emoji->postfix ? 1 : 0)); if (!(emoji->code >> 16)) { result.append(QChar(emoji->code & 0xFFFF)); } else { result.append(QChar((emoji->code >> 16) & 0xFFFF)); result.append(QChar(emoji->code & 0xFFFF)); if (emoji->code2) { result.append(QChar((emoji->code2 >> 16) & 0xFFFF)); result.append(QChar(emoji->code2 & 0xFFFF)); } } if (emoji->color && ((emoji->color & 0xFFFF0000U) != 0xFFFF0000U)) { result.append(QChar((emoji->color >> 16) & 0xFFFF)); result.append(QChar(emoji->color & 0xFFFF)); } if (emoji->postfix) result.append(QChar(emoji->postfix)); return result; } inline uint64 emojiKey(EmojiPtr emoji) { uint64 key = emoji->code; if (emoji->code2) { key = (key << 32) | uint64(emoji->code2); } else if (emoji->color && ((emoji->color & 0xFFFF0000U) != 0xFFFF0000U)) { key = (key << 32) | uint64(emoji->color); } return key; } inline EmojiPtr emojiFromKey(uint64 key) { uint32 code = uint32(key >> 32), code2 = uint32(key & 0xFFFFFFFFLLU); if (!code && code2) { code = code2; code2 = 0; } EmojiPtr emoji = emojiGet(code); if (emoji == TwoSymbolEmoji) { return emojiGet(code, code2); } else if (emoji && emoji->color && code2) { return emojiGet(emoji, code2); } return emoji; } inline EmojiPtr emojiFromUrl(const QString &url) { return emojiFromKey(url.midRef(10).toULongLong(0, 16)); // skip emoji://e. } inline EmojiPtr emojiFromText(const QChar *ch, const QChar *e, int &len) { EmojiPtr emoji = 0; if (ch + 1 < e && ((ch->isHighSurrogate() && (ch + 1)->isLowSurrogate()) || (((ch->unicode() >= 48 && ch->unicode() < 58) || ch->unicode() == 35) && (ch + 1)->unicode() == 0x20E3))) { uint32 code = (ch->unicode() << 16) | (ch + 1)->unicode(); emoji = emojiGet(code); if (emoji) { if (emoji == TwoSymbolEmoji) { // check two symbol if (ch + 3 >= e) { emoji = 0; } else { uint32 code2 = ((uint32((ch + 2)->unicode()) << 16) | uint32((ch + 3)->unicode())); emoji = emojiGet(code, code2); } } else { if (ch + 2 < e && (ch + 2)->unicode() == 0x200D) { // check sequence EmojiPtr seq = emojiGet(ch, e); if (seq) { emoji = seq; } } } } } else if (ch < e) { emoji = emojiGet(ch->unicode()); Q_ASSERT(emoji != TwoSymbolEmoji); } if (emoji) { len = emoji->len + ((ch + emoji->len < e && (ch + emoji->len)->unicode() == 0xFE0F) ? 1 : 0); if (emoji->color && (ch + len + 1 < e && (ch + len)->isHighSurrogate() && (ch + len + 1)->isLowSurrogate())) { // color uint32 color = ((uint32((ch + len)->unicode()) << 16) | uint32((ch + len + 1)->unicode())); EmojiPtr col = emojiGet(emoji, color); if (col && col != emoji) { len += col->len - emoji->len; emoji = col; if (ch + len < e && (ch + len)->unicode() == 0xFE0F) { ++len; } } } } return emoji; } extern int EmojiSizes[5], EIndex, ESize; extern const char *EmojiNames[5], *EName; void emojiFind(const QChar *ch, const QChar *e, const QChar *&newEmojiEnd, uint32 &emojiCode); inline bool emojiEdge(const QChar *ch) { return true; switch (ch->unicode()) { case '.': case ',': case ':': case ';': case '!': case '?': case '#': case '@': case '(': case ')': case '[': case ']': case '{': case '}': case '<': case '>': case '+': case '=': case '-': case '_': case '*': case '/': case '\\': case '^': case '$': case '"': case '\'': case 8212: case 171: case 187: // --, <<, >> return true; } return false; } inline QString replaceEmojis(const QString &text) { QString result; LinkRanges lnkRanges = textParseLinks(text, TextParseLinks | TextParseMentions | TextParseHashtags); int32 currentLink = 0, lnkCount = lnkRanges.size(); const QChar *emojiStart = text.unicode(), *emojiEnd = emojiStart, *e = text.cend(); bool canFindEmoji = true, consumePrevious = false; for (const QChar *ch = emojiEnd; ch != e;) { uint32 emojiCode = 0; const QChar *newEmojiEnd = 0; if (canFindEmoji) { emojiFind(ch, e, newEmojiEnd, emojiCode); } while (currentLink < lnkCount && ch >= lnkRanges[currentLink].from + lnkRanges[currentLink].len) { ++currentLink; } EmojiPtr emoji = emojiCode ? emojiGet(emojiCode) : 0; if (emoji && emoji != TwoSymbolEmoji && (ch == emojiStart || !ch->isLetterOrNumber() || !(ch - 1)->isLetterOrNumber()) && (newEmojiEnd == e || !newEmojiEnd->isLetterOrNumber() || newEmojiEnd == emojiStart || !(newEmojiEnd - 1)->isLetterOrNumber()) && (currentLink >= lnkCount || (ch < lnkRanges[currentLink].from && newEmojiEnd <= lnkRanges[currentLink].from) || (ch >= lnkRanges[currentLink].from + lnkRanges[currentLink].len && newEmojiEnd > lnkRanges[currentLink].from + lnkRanges[currentLink].len)) ) { // if (newEmojiEnd < e && newEmojiEnd->unicode() == ' ') ++newEmojiEnd; if (result.isEmpty()) result.reserve(text.size()); if (ch > emojiEnd + (consumePrevious ? 1 : 0)) { result.append(emojiEnd, ch - emojiEnd - (consumePrevious ? 1 : 0)); } if (emoji->color) { EmojiColorVariants::const_iterator it = cEmojiVariants().constFind(emoji->code); if (it != cEmojiVariants().cend()) { EmojiPtr replace = emojiFromKey(it.value()); if (replace) { if (replace != TwoSymbolEmoji && replace->code == emoji->code && replace->code2 == emoji->code2) { emoji = replace; } } } } result.append(emojiString(emoji)); ch = emojiEnd = newEmojiEnd; canFindEmoji = true; consumePrevious = false; } else { if (false && (ch->unicode() == QChar::Space || ch->unicode() == QChar::Nbsp)) { canFindEmoji = true; consumePrevious = true; } else if (emojiEdge(ch)) { canFindEmoji = true; consumePrevious = false; } else { canFindEmoji = false; } ++ch; } } if (result.isEmpty()) return text; if (emojiEnd < e) result.append(emojiEnd, e - emojiEnd); return result; } int emojiPackCount(DBIEmojiTab tab); EmojiPack emojiPack(DBIEmojiTab tab);