| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917 |
- // 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 "emoji_config.h"
- #include "emoji_suggestions_helper.h"
- #include "base/bytes.h"
- #include "base/openssl_help.h"
- #include "base/parse_helper.h"
- #include "base/debug_log.h"
- #include "ui/style/style_core.h"
- #include "ui/integration.h"
- #include "ui/painter.h"
- #include "ui/ui_utility.h"
- #include "styles/style_basic.h"
- #include <QtCore/QJsonDocument>
- #include <QtCore/QJsonObject>
- #include <QtCore/QFile>
- #include <QtCore/QDir>
- #include <crl/crl_async.h>
- namespace Ui {
- namespace Emoji {
- namespace {
- constexpr auto kUniversalSize = 72;
- constexpr auto kImagesPerRow = 32;
- constexpr auto kImageRowsPerSprite = 16;
- constexpr auto kSetVersion = uint32(6);
- constexpr auto kCacheVersion = uint32(8);
- constexpr auto kMaxId = uint32(1 << 8);
- #ifdef Q_OS_MAC
- constexpr auto kScaleForTouchBar = 150;
- #endif
- enum class ConfigResult {
- Invalid,
- BadVersion,
- Good,
- };
- // Right now we can't allow users of Ui::Emoji to create custom sizes.
- // Any Instance::Instance() can invalidate Universal.id() and sprites.
- // So all Instance::Instance() should happen before async generations.
- class Instance {
- public:
- explicit Instance(int size);
- bool cached() const;
- void draw(QPainter &p, EmojiPtr emoji, int x, int y);
- private:
- void readCache();
- void generateCache();
- void checkUniversalImages();
- void pushSprite(QImage &&data);
- int _id = 0;
- int _size = 0;
- std::vector<QPixmap> _sprites;
- base::binary_guard _generating;
- bool _unsupported = false;
- };
- auto SizeNormal = -1;
- auto SizeLarge = -1;
- auto SpritesCount = -1;
- auto InstanceNormal = std::unique_ptr<Instance>();
- auto InstanceLarge = std::unique_ptr<Instance>();
- auto Universal = std::shared_ptr<UniversalImages>();
- auto CanClearUniversal = false;
- auto WaitingToSwitchBackToId = 0;
- auto Updates = rpl::event_stream<>();
- #ifdef Q_OS_MAC
- auto TouchbarSize = -1;
- auto TouchbarInstance = std::unique_ptr<Instance>();
- auto TouchbarEmoji = (Instance*)nullptr;
- #endif
- auto MainEmojiMap = std::map<int, QPixmap>();
- auto OtherEmojiMap = base::flat_map<int, std::map<int, QPixmap>>();
- int RowsCount(int index) {
- if (index + 1 < SpritesCount) {
- return kImageRowsPerSprite;
- }
- const auto count = internal::FullCount()
- - (index * kImagesPerRow * kImageRowsPerSprite);
- return (count / kImagesPerRow)
- + ((count % kImagesPerRow) ? 1 : 0);
- }
- QString CacheFileNameMask(int size) {
- return "cache_" + QString::number(size) + '_';
- }
- QString CacheFilePath(int size, int index) {
- return internal::CacheFileFolder()
- + '/'
- + CacheFileNameMask(size)
- + QString::number(index);
- }
- QString CurrentSettingPath() {
- return internal::CacheFileFolder() + "/current";
- }
- bool IsValidSetId(int id) {
- return (id == 0) || (id > 0 && id < kMaxId);
- }
- uint32 ComputeVersion(int id) {
- Expects(IsValidSetId(id));
- static_assert(kCacheVersion > 0 && kCacheVersion < (1 << 16));
- static_assert(kSetVersion > 0 && kSetVersion < (1 << 8));
- return uint32(kCacheVersion)
- | (uint32(kSetVersion) << 16)
- | (uint32(id) << 24);
- }
- int ReadCurrentSetId() {
- const auto path = CurrentSettingPath();
- auto file = QFile(path);
- if (!file.open(QIODevice::ReadOnly)) {
- return 0;
- }
- auto stream = QDataStream(&file);
- stream.setVersion(QDataStream::Qt_5_1);
- auto id = qint32(0);
- stream >> id;
- return (stream.status() == QDataStream::Ok && IsValidSetId(id))
- ? id
- : 0;
- }
- void ApplyUniversalImages(std::shared_ptr<UniversalImages> images) {
- Universal = std::move(images);
- CanClearUniversal = false;
- MainEmojiMap.clear();
- OtherEmojiMap.clear();
- Updates.fire({});
- }
- void SwitchToSetPrepared(int id, std::shared_ptr<UniversalImages> images) {
- WaitingToSwitchBackToId = 0;
- auto setting = QFile(CurrentSettingPath());
- if (!id) {
- setting.remove();
- } else if (setting.open(QIODevice::WriteOnly)) {
- auto stream = QDataStream(&setting);
- stream.setVersion(QDataStream::Qt_5_1);
- stream << qint32(id);
- }
- ApplyUniversalImages(std::move(images));
- }
- [[nodiscard]] ConfigResult ValidateConfig(int id) {
- Expects(IsValidSetId(id));
- if (!id) {
- return ConfigResult::Good;
- }
- constexpr auto kSizeLimit = 65536;
- auto config = QFile(internal::SetDataPath(id) + "/config.json");
- if (!config.open(QIODevice::ReadOnly) || config.size() > kSizeLimit) {
- return ConfigResult::Invalid;
- }
- auto error = QJsonParseError{ 0, QJsonParseError::NoError };
- const auto document = QJsonDocument::fromJson(
- base::parse::stripComments(config.readAll()),
- &error);
- config.close();
- if (error.error != QJsonParseError::NoError) {
- return ConfigResult::Invalid;
- }
- if (document.object()["id"].toInt() != id) {
- return ConfigResult::Invalid;
- } else if (document.object()["version"].toInt() != kSetVersion) {
- return ConfigResult::BadVersion;
- }
- return ConfigResult::Good;
- }
- void ClearCurrentSetIdSync() {
- Expects(Universal != nullptr);
- const auto id = Universal->id();
- if (!id) {
- return;
- }
- const auto newId = 0;
- auto universal = std::make_shared<UniversalImages>(newId);
- universal->ensureLoaded();
- // Start loading the set when possible.
- ApplyUniversalImages(std::move(universal));
- WaitingToSwitchBackToId = id;
- }
- void SaveToFile(int id, const QImage &image, int size, int index) {
- Expects(image.bytesPerLine() == image.width() * 4);
- QFile f(CacheFilePath(size, index));
- if (!f.open(QIODevice::WriteOnly)) {
- if (!QDir::current().mkpath(internal::CacheFileFolder())
- || !f.open(QIODevice::WriteOnly)) {
- LOG(("App Error: Could not open emoji cache '%1' for size %2_%3"
- ).arg(f.fileName()
- ).arg(size
- ).arg(index));
- return;
- }
- }
- const auto write = [&](bytes::const_span data) {
- return f.write(
- reinterpret_cast<const char*>(data.data()),
- data.size()
- ) == data.size();
- };
- const uint32 header[] = {
- uint32(ComputeVersion(id)),
- uint32(size),
- uint32(image.width()),
- uint32(image.height()),
- };
- const auto data = bytes::const_span(
- reinterpret_cast<const bytes::type*>(image.bits()),
- image.width() * image.height() * 4);
- if (!write(bytes::make_span(header))
- || !write(data)
- || !write(openssl::Sha256(bytes::make_span(header), data))
- || false) {
- LOG(("App Error: Could not write emoji cache '%1' for size %2"
- ).arg(f.fileName()
- ).arg(size));
- }
- }
- QImage LoadFromFile(int id, int size, int index) {
- const auto rows = RowsCount(index);
- const auto width = kImagesPerRow * size;
- const auto height = rows * size;
- const auto fileSize = 4 * sizeof(uint32)
- + (width * height * 4)
- + openssl::kSha256Size;
- QFile f(CacheFilePath(size, index));
- if (!f.exists()
- || f.size() != fileSize
- || !f.open(QIODevice::ReadOnly)) {
- return QImage();
- }
- const auto read = [&](bytes::span data) {
- return f.read(
- reinterpret_cast<char*>(data.data()),
- data.size()
- ) == data.size();
- };
- uint32 header[4] = { 0 };
- if (!read(bytes::make_span(header))
- || header[0] != ComputeVersion(id)
- || header[1] != size
- || header[2] != width
- || header[3] != height) {
- return QImage();
- }
- auto result = QImage(
- width,
- height,
- QImage::Format_ARGB32_Premultiplied);
- Assert(result.bytesPerLine() == width * 4);
- const auto data = bytes::make_span(
- reinterpret_cast<bytes::type*>(result.bits()),
- width * height * 4);
- auto signature = bytes::vector(openssl::kSha256Size);
- if (!read(data)
- || !read(signature)
- //|| (bytes::compare(
- // signature,
- // openssl::Sha256(bytes::make_span(header), data)) != 0)
- || false) {
- return QImage();
- }
- // This should remove a non necessary detach on Retina screens later.
- result.setDevicePixelRatio(style::DevicePixelRatio());
- crl::async([=, signature = std::move(signature)] {
- // This should not happen (invalid signature),
- // so we delay this check and fix only the next launch.
- const auto data = bytes::make_span(
- reinterpret_cast<const bytes::type*>(result.bits()),
- width * height * 4);
- const auto result = bytes::compare(
- signature,
- openssl::Sha256(bytes::make_span(header), data));
- if (result != 0) {
- QFile(CacheFilePath(size, index)).remove();
- }
- });
- return result;
- }
- std::vector<QImage> LoadSprites(int id) {
- Expects(IsValidSetId(id));
- Expects(SpritesCount > 0);
- auto result = std::vector<QImage>();
- const auto folder = (id != 0)
- ? internal::SetDataPath(id) + '/'
- : QStringLiteral(":/gui/emoji/");
- const auto base = folder + "emoji_";
- return ranges::views::ints(
- 0,
- SpritesCount
- ) | ranges::views::transform([&](int index) {
- return base + QString::number(index + 1) + ".webp";
- }) | ranges::views::transform([](const QString &path) {
- return QImage(path, "WEBP").convertToFormat(
- QImage::Format_ARGB32_Premultiplied);
- }) | ranges::to_vector;
- }
- std::vector<QImage> LoadAndValidateSprites(int id) {
- Expects(IsValidSetId(id));
- Expects(SpritesCount > 0);
- const auto config = ValidateConfig(id);
- if (config != ConfigResult::Good) {
- return {};
- }
- auto result = LoadSprites(id);
- const auto sizes = ranges::views::ints(
- 0,
- SpritesCount
- ) | ranges::views::transform([](int index) {
- return QSize(
- kImagesPerRow * kUniversalSize,
- RowsCount(index) * kUniversalSize);
- });
- const auto good = ranges::views::zip_with(
- [](const QImage &data, QSize size) { return data.size() == size; },
- result,
- sizes);
- if (ranges::find(good, false) != end(good)) {
- return {};
- }
- return result;
- }
- void ClearUniversalChecked() {
- Expects(InstanceNormal != nullptr && InstanceLarge != nullptr);
- if (Universal
- && InstanceNormal->cached()
- && InstanceLarge->cached()) {
- if (CanClearUniversal) {
- Universal->clear();
- }
- ClearIrrelevantCache();
- }
- }
- } // namespace
- namespace internal {
- QString CacheFileFolder() {
- return Integration::Instance().emojiCacheFolder();
- }
- QString SetDataPath(int id) {
- Expects(IsValidSetId(id) && id != 0);
- return CacheFileFolder() + "/set" + QString::number(id);
- }
- } // namespace internal
- UniversalImages::UniversalImages(int id) : _id(id) {
- Expects(IsValidSetId(id));
- }
- int UniversalImages::id() const {
- return _id;
- }
- bool UniversalImages::ensureLoaded() {
- Expects(SpritesCount > 0);
- if (!_sprites.empty()) {
- return true;
- }
- _sprites = LoadAndValidateSprites(_id);
- return !_sprites.empty();
- }
- void UniversalImages::clear() {
- _sprites.clear();
- }
- void UniversalImages::draw(
- QPainter &p,
- EmojiPtr emoji,
- int size,
- int x,
- int y) const {
- Expects(emoji->sprite() < _sprites.size());
- const auto large = kUniversalSize;
- const auto &original = _sprites[emoji->sprite()];
- const auto data = original.bits();
- const auto stride = original.bytesPerLine();
- const auto format = original.format();
- const auto row = emoji->row();
- const auto column = emoji->column();
- auto single = QImage(
- data + (row * kImagesPerRow * large + column) * large * 4,
- large,
- large,
- stride,
- format
- ).scaled(
- size,
- size,
- Qt::IgnoreAspectRatio,
- Qt::SmoothTransformation);
- single.setDevicePixelRatio(p.device()->devicePixelRatio());
- p.drawImage(x, y, single);
- }
- QImage UniversalImages::generate(int size, int index) const {
- Expects(size > 0);
- Expects(index < _sprites.size());
- const auto rows = RowsCount(index);
- const auto large = kUniversalSize;
- const auto &original = _sprites[index];
- const auto data = original.bits();
- const auto stride = original.bytesPerLine();
- const auto format = original.format();
- auto result = QImage(
- size * kImagesPerRow,
- size * rows,
- QImage::Format_ARGB32_Premultiplied);
- result.fill(Qt::transparent);
- {
- QPainter p(&result);
- for (auto y = 0; y != rows; ++y) {
- for (auto x = 0; x != kImagesPerRow; ++x) {
- const auto single = QImage(
- data + (y * kImagesPerRow * large + x) * large * 4,
- large,
- large,
- stride,
- format
- ).scaled(
- size,
- size,
- Qt::IgnoreAspectRatio,
- Qt::SmoothTransformation);
- p.drawImage(
- x * size,
- y * size,
- single);
- }
- }
- }
- SaveToFile(_id, result, size, index);
- return result;
- }
- void Init() {
- internal::Init();
- const auto count = internal::FullCount();
- const auto persprite = kImagesPerRow * kImageRowsPerSprite;
- SpritesCount = (count / persprite) + ((count % persprite) ? 1 : 0);
- SizeNormal = st::emojiSize * style::DevicePixelRatio();
- SizeLarge = int(style::ConvertScale(18 * 4 / 3., style::Scale())) * style::DevicePixelRatio();
- Universal = std::make_shared<UniversalImages>(ReadCurrentSetId());
- CanClearUniversal = false;
- InstanceNormal = std::make_unique<Instance>(SizeNormal);
- InstanceLarge = std::make_unique<Instance>(SizeLarge);
- #ifdef Q_OS_MAC
- if (style::Scale() != kScaleForTouchBar) {
- TouchbarSize = int(style::ConvertScale(18 * 4 / 3.,
- kScaleForTouchBar * style::DevicePixelRatio()));
- TouchbarInstance = std::make_unique<Instance>(TouchbarSize);
- TouchbarEmoji = TouchbarInstance.get();
- } else {
- TouchbarEmoji = InstanceLarge.get();
- }
- #endif
- }
- void Clear() {
- MainEmojiMap.clear();
- OtherEmojiMap.clear();
- InstanceNormal = nullptr;
- InstanceLarge = nullptr;
- #ifdef Q_OS_MAC
- TouchbarInstance = nullptr;
- TouchbarEmoji = nullptr;
- #endif
- }
- void ClearIrrelevantCache() {
- Expects(SizeNormal > 0);
- Expects(SizeLarge > 0);
- crl::async([] {
- const auto folder = internal::CacheFileFolder();
- const auto list = QDir(folder).entryList(QDir::Files);
- const auto good1 = CacheFileNameMask(SizeNormal);
- const auto good2 = CacheFileNameMask(SizeLarge);
- const auto good3full = CurrentSettingPath();
- for (const auto &name : list) {
- if (!name.startsWith(good1) && !name.startsWith(good2)) {
- const auto full = folder + '/' + name;
- if (full != good3full) {
- QFile(full).remove();
- }
- }
- }
- });
- }
- int CurrentSetId() {
- Expects(Universal != nullptr);
- return Universal->id();
- }
- int NeedToSwitchBackToId() {
- return WaitingToSwitchBackToId;
- }
- void ClearNeedSwitchToId() {
- if (!WaitingToSwitchBackToId) {
- return;
- }
- WaitingToSwitchBackToId = 0;
- QFile(CurrentSettingPath()).remove();
- }
- void SwitchToSet(int id, Fn<void(bool)> callback) {
- Expects(IsValidSetId(id));
- if (Universal && Universal->id() == id) {
- callback(true);
- return;
- }
- crl::async([=] {
- auto universal = std::make_shared<UniversalImages>(id);
- if (!universal->ensureLoaded()) {
- crl::on_main([=] {
- callback(false);
- });
- } else {
- crl::on_main([=, universal = std::move(universal)]() mutable {
- SwitchToSetPrepared(id, std::move(universal));
- callback(true);
- });
- }
- });
- }
- bool SetIsReady(int id) {
- Expects(IsValidSetId(id));
- if (!id) {
- return true;
- }
- const auto folder = internal::SetDataPath(id) + '/';
- auto names = ranges::views::ints(
- 0,
- SpritesCount + 1
- ) | ranges::views::transform([](int index) {
- return index
- ? "emoji_" + QString::number(index) + ".webp"
- : QString("config.json");
- });
- const auto bad = ranges::find_if(names, [&](const QString &name) {
- return !QFile(folder + name).exists();
- });
- return (bad == names.end());
- }
- rpl::producer<> Updated() {
- return Updates.events();
- }
- int GetSizeNormal() {
- Expects(SizeNormal > 0);
- return SizeNormal;
- }
- int GetSizeLarge() {
- Expects(SizeLarge > 0);
- return SizeLarge;
- }
- #ifdef Q_OS_MAC
- int GetSizeTouchbar() {
- return (style::Scale() == kScaleForTouchBar)
- ? GetSizeLarge()
- : TouchbarSize;
- }
- #endif
- One::One(
- const QString &id,
- EmojiPtr original,
- uint32 index,
- bool hasPostfix,
- bool colorizable,
- const CreationTag &)
- : _id(id)
- , _original(original)
- , _index(index)
- , _hasPostfix(hasPostfix)
- , _colorizable(colorizable) {
- Expects(!_colorizable || !colored());
- }
- int One::variantsCount() const {
- return hasVariants() ? 5 : 0;
- }
- int One::variantIndex(EmojiPtr variant) const {
- return (variant - original());
- }
- EmojiPtr One::variant(int index) const {
- return (index >= 0 && index <= variantsCount()) ? (original() + index) : this;
- }
- QString IdFromOldKey(uint64 oldKey) {
- auto code = uint32(oldKey >> 32);
- auto code2 = uint32(oldKey & 0xFFFFFFFFLLU);
- if (!code && code2) {
- code = base::take(code2);
- }
- if ((code & 0xFFFF0000U) != 0xFFFF0000U) { // code and code2 contain the whole id
- auto result = QString();
- result.reserve(4);
- auto addCode = [&result](uint32 code) {
- if (auto high = (code >> 16)) {
- result.append(QChar(static_cast<ushort>(high & 0xFFFFU)));
- }
- result.append(QChar(static_cast<ushort>(code & 0xFFFFU)));
- };
- addCode(code);
- if (code2) addCode(code2);
- return result;
- }
- // old sequence
- auto sequenceIndex = int(code & 0xFFFFU);
- switch (sequenceIndex) {
- case 0: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa7");
- case 1: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa7\xe2\x80\x8d\xf0\x9f\x91\xa6");
- case 2: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa6\xe2\x80\x8d\xf0\x9f\x91\xa6");
- case 3: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa7\xe2\x80\x8d\xf0\x9f\x91\xa7");
- case 4: return QString::fromUtf8("\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa6");
- case 5: return QString::fromUtf8("\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa7");
- case 6: return QString::fromUtf8("\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa7\xe2\x80\x8d\xf0\x9f\x91\xa6");
- case 7: return QString::fromUtf8("\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa6\xe2\x80\x8d\xf0\x9f\x91\xa6");
- case 8: return QString::fromUtf8("\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa9\xe2\x80\x8d\xf0\x9f\x91\xa7\xe2\x80\x8d\xf0\x9f\x91\xa7");
- case 9: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa6");
- case 10: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa7");
- case 11: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa7\xe2\x80\x8d\xf0\x9f\x91\xa6");
- case 12: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa6\xe2\x80\x8d\xf0\x9f\x91\xa6");
- case 13: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa8\xe2\x80\x8d\xf0\x9f\x91\xa7\xe2\x80\x8d\xf0\x9f\x91\xa7");
- case 14: return QString::fromUtf8("\xf0\x9f\x91\xa9\xe2\x80\x8d\xe2\x9d\xa4\xef\xb8\x8f\xe2\x80\x8d\xf0\x9f\x91\xa9");
- case 15: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xe2\x9d\xa4\xef\xb8\x8f\xe2\x80\x8d\xf0\x9f\x91\xa8");
- case 16: return QString::fromUtf8("\xf0\x9f\x91\xa9\xe2\x80\x8d\xe2\x9d\xa4\xef\xb8\x8f\xe2\x80\x8d\xf0\x9f\x92\x8b\xe2\x80\x8d\xf0\x9f\x91\xa9");
- case 17: return QString::fromUtf8("\xf0\x9f\x91\xa8\xe2\x80\x8d\xe2\x9d\xa4\xef\xb8\x8f\xe2\x80\x8d\xf0\x9f\x92\x8b\xe2\x80\x8d\xf0\x9f\x91\xa8");
- case 18: return QString::fromUtf8("\xf0\x9f\x91\x81\xe2\x80\x8d\xf0\x9f\x97\xa8");
- }
- return QString();
- }
- QVector<EmojiPtr> GetDefaultRecent() {
- const auto defaultRecent = {
- 0xD83DDE02LLU,
- 0xD83DDE18LLU,
- 0x2764LLU,
- 0xD83DDE0DLLU,
- 0xD83DDE0ALLU,
- 0xD83DDE01LLU,
- 0xD83DDC4DLLU,
- 0x263ALLU,
- 0xD83DDE14LLU,
- 0xD83DDE04LLU,
- 0xD83DDE2DLLU,
- 0xD83DDC8BLLU,
- 0xD83DDE12LLU,
- 0xD83DDE33LLU,
- 0xD83DDE1CLLU,
- 0xD83DDE48LLU,
- 0xD83DDE09LLU,
- 0xD83DDE03LLU,
- 0xD83DDE22LLU,
- 0xD83DDE1DLLU,
- 0xD83DDE31LLU,
- 0xD83DDE21LLU,
- 0xD83DDE0FLLU,
- 0xD83DDE1ELLU,
- 0xD83DDE05LLU,
- 0xD83DDE1ALLU,
- 0xD83DDE4ALLU,
- 0xD83DDE0CLLU,
- 0xD83DDE00LLU,
- 0xD83DDE0BLLU,
- 0xD83DDE06LLU,
- 0xD83DDC4CLLU,
- 0xD83DDE10LLU,
- 0xD83DDE15LLU,
- };
- auto result = QVector<EmojiPtr>();
- for (const auto oldKey : defaultRecent) {
- if (const auto emoji = FromOldKey(oldKey)) {
- result.push_back(emoji);
- }
- }
- return result;
- }
- const QPixmap &SinglePixmap(EmojiPtr emoji, int fontHeight) {
- const auto factor = style::DevicePixelRatio();
- auto &map = (fontHeight == st::normalFont->height * factor)
- ? MainEmojiMap
- : OtherEmojiMap[fontHeight];
- auto i = map.find(emoji->index());
- if (i != end(map)) {
- return i->second;
- }
- auto image = QImage(
- SizeNormal + st::emojiPadding * factor * 2,
- fontHeight,
- QImage::Format_ARGB32_Premultiplied);
- image.setDevicePixelRatio(factor);
- image.fill(Qt::transparent);
- {
- QPainter p(&image);
- PainterHighQualityEnabler hq(p);
- Draw(
- p,
- emoji,
- SizeNormal,
- st::emojiPadding,
- (fontHeight - SizeNormal) / (2 * factor));
- }
- return map.emplace(
- emoji->index(),
- PixmapFromImage(std::move(image))
- ).first->second;
- }
- void Draw(QPainter &p, EmojiPtr emoji, int size, int x, int y) {
- #ifdef Q_OS_MAC
- const auto s = (style::Scale() == kScaleForTouchBar)
- ? SizeLarge
- : TouchbarSize;
- if (size == s) {
- TouchbarEmoji->draw(p, emoji, x, y);
- return;
- }
- #endif
- if (size == SizeNormal) {
- InstanceNormal->draw(p, emoji, x, y);
- } else if (size == SizeLarge) {
- InstanceLarge->draw(p, emoji, x, y);
- } else {
- Unexpected("Size in Ui::Emoji::Draw.");
- }
- }
- Instance::Instance(int size) : _id(Universal->id()), _size(size) {
- Expects(Universal != nullptr);
- readCache();
- if (!cached()) {
- generateCache();
- }
- }
- bool Instance::cached() const {
- Expects(Universal != nullptr);
- return (Universal->id() == _id) && (_sprites.size() == SpritesCount);
- }
- void Instance::draw(QPainter &p, EmojiPtr emoji, int x, int y) {
- if (_unsupported) {
- return;
- } else if (Universal && Universal->id() != _id) {
- generateCache();
- }
- const auto sprite = emoji->sprite();
- if (sprite >= _sprites.size()) {
- Assert(Universal != nullptr);
- Universal->draw(p, emoji, _size, x, y);
- return;
- }
- p.drawPixmap(
- QPoint(x, y),
- _sprites[sprite],
- QRect(emoji->column() * _size, emoji->row() * _size, _size, _size));
- }
- void Instance::readCache() {
- for (auto i = 0; i != SpritesCount; ++i) {
- auto image = LoadFromFile(_id, _size, i);
- if (image.isNull()) {
- return;
- }
- pushSprite(std::move(image));
- }
- }
- void Instance::checkUniversalImages() {
- Expects(Universal != nullptr);
- if (_id != Universal->id()) {
- _id = Universal->id();
- _generating = nullptr;
- _sprites.clear();
- }
- if (!Universal->ensureLoaded()) {
- if (Universal->id() != 0) {
- ClearCurrentSetIdSync();
- } else {
- _unsupported = true;
- }
- }
- }
- void Instance::generateCache() {
- checkUniversalImages();
- const auto cachePath = internal::CacheFileFolder();
- if (cachePath.isEmpty()) {
- return;
- }
- const auto size = _size;
- const auto index = _sprites.size();
- crl::async([
- =,
- universal = Universal,
- guard = _generating.make_guard()
- ]() mutable {
- auto image = universal->generate(size, index);
- crl::on_main(std::move(guard), [
- =,
- image = std::move(image)
- ]() mutable {
- if (universal != Universal) {
- return;
- }
- pushSprite(std::move(image));
- if (cached()) {
- ClearUniversalChecked();
- } else {
- generateCache();
- }
- });
- });
- }
- void Instance::pushSprite(QImage &&data) {
- _sprites.push_back(PixmapFromImage(std::move(data)));
- _sprites.back().setDevicePixelRatio(style::DevicePixelRatio());
- }
- const std::shared_ptr<UniversalImages> &SourceImages() {
- return Universal;
- }
- void ClearSourceImages(const std::shared_ptr<UniversalImages> &images) {
- if (images == Universal) {
- CanClearUniversal = true;
- ClearUniversalChecked();
- }
- }
- void ReplaceSourceImages(std::shared_ptr<UniversalImages> images) {
- Expects(images != nullptr);
- if (Universal->id() == images->id()) {
- Universal = std::move(images);
- }
- }
- } // namespace Emoji
- } // namespace Ui
|