Reuse global emoji large images.

This commit is contained in:
John Preston 2019-08-05 15:08:20 +01:00
parent e479daca03
commit 1b1b1780db
7 changed files with 98 additions and 24 deletions

View File

@ -15,34 +15,35 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_file_origin.h" #include "data/data_file_origin.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "data/data_document.h" #include "data/data_document.h"
#include "base/concurrent_timer.h"
#include "apiwrap.h" #include "apiwrap.h"
#include "styles/style_history.h" #include "styles/style_history.h"
namespace Stickers { namespace Stickers {
namespace details { namespace details {
using UniversalImages = Ui::Emoji::UniversalImages;
class EmojiImageLoader { class EmojiImageLoader {
public: public:
EmojiImageLoader( EmojiImageLoader(
crl::weak_on_queue<EmojiImageLoader> weak, crl::weak_on_queue<EmojiImageLoader> weak,
int id); std::shared_ptr<UniversalImages> images,
bool largeEnabled);
[[nodiscard]] QImage prepare(EmojiPtr emoji); [[nodiscard]] QImage prepare(EmojiPtr emoji);
void switchTo(int id); void switchTo(std::shared_ptr<UniversalImages> images);
std::shared_ptr<UniversalImages> releaseImages();
private: private:
crl::weak_on_queue<EmojiImageLoader> _weak; crl::weak_on_queue<EmojiImageLoader> _weak;
std::optional<Ui::Emoji::UniversalImages> _images; std::shared_ptr<UniversalImages> _images;
base::ConcurrentTimer _unloadTimer;
}; };
namespace { namespace {
constexpr auto kRefreshTimeout = TimeId(7200); constexpr auto kRefreshTimeout = TimeId(7200);
constexpr auto kUnloadTimeout = 86400 * crl::time(1000); constexpr auto kClearSourceTimeout = 10 * crl::time(1000);
[[nodiscard]] QSize SingleSize() { [[nodiscard]] QSize SingleSize() {
const auto single = st::largeEmojiSize; const auto single = st::largeEmojiSize;
@ -245,15 +246,18 @@ QByteArray ImageSource::bytesForCache() {
EmojiImageLoader::EmojiImageLoader( EmojiImageLoader::EmojiImageLoader(
crl::weak_on_queue<EmojiImageLoader> weak, crl::weak_on_queue<EmojiImageLoader> weak,
int id) std::shared_ptr<UniversalImages> images,
bool largeEnabled)
: _weak(std::move(weak)) : _weak(std::move(weak))
, _images(std::in_place, id) , _images(std::move(images)) {
, _unloadTimer(_weak.runner(), [=] { _images->clear(); }) { Expects(_images != nullptr);
if (largeEnabled) {
_images->ensureLoaded();
}
} }
QImage EmojiImageLoader::prepare(EmojiPtr emoji) { QImage EmojiImageLoader::prepare(EmojiPtr emoji) {
Expects(_images.has_value());
_images->ensureLoaded(); _images->ensureLoaded();
const auto factor = cIntRetinaFactor(); const auto factor = cIntRetinaFactor();
const auto side = st::largeEmojiSize + 2 * st::largeEmojiOutline; const auto side = st::largeEmojiSize + 2 * st::largeEmojiOutline;
@ -300,19 +304,25 @@ QImage EmojiImageLoader::prepare(EmojiPtr emoji) {
delta, delta,
delta); delta);
} }
_unloadTimer.callOnce(kUnloadTimeout);
return result; return result;
} }
void EmojiImageLoader::switchTo(int id) { void EmojiImageLoader::switchTo(std::shared_ptr<UniversalImages> images) {
_images.emplace(id); _images = std::move(images);
}
std::shared_ptr<UniversalImages> EmojiImageLoader::releaseImages() {
return std::exchange(
_images,
std::make_shared<UniversalImages>(_images->id()));
} }
} // namespace details } // namespace details
EmojiPack::EmojiPack(not_null<Main::Session*> session) EmojiPack::EmojiPack(not_null<Main::Session*> session)
: _session(session) : _session(session)
, _imageLoader(Ui::Emoji::CurrentSetId()) { , _imageLoader(prepareSourceImages(), session->settings().largeEmoji())
, _clearTimer([=] { clearSourceImages(); }) {
refresh(); refresh();
session->data().itemRemoved( session->data().itemRemoved(
@ -322,17 +332,23 @@ EmojiPack::EmojiPack(not_null<Main::Session*> session)
remove(item); remove(item);
}, _lifetime); }, _lifetime);
session->settings().largeEmojiChanges( _session->settings().largeEmojiChanges(
) | rpl::start_with_next([=] { ) | rpl::start_with_next([=](bool large) {
if (large) {
_clearTimer.cancel();
} else {
_clearTimer.callOnce(details::kClearSourceTimeout);
}
refreshAll(); refreshAll();
}, _lifetime); }, _lifetime);
Ui::Emoji::Updated( Ui::Emoji::Updated(
) | rpl::start_with_next([=] { ) | rpl::start_with_next([=] {
const auto id = Ui::Emoji::CurrentSetId();
_images.clear(); _images.clear();
_imageLoader.with([=](details::EmojiImageLoader &loader) { _imageLoader.with([
loader.switchTo(id); source = prepareSourceImages()
](details::EmojiImageLoader &loader) mutable {
loader.switchTo(std::move(source));
}); });
refreshAll(); refreshAll();
}, _lifetime); }, _lifetime);
@ -450,6 +466,24 @@ void EmojiPack::refreshItems(
} }
} }
auto EmojiPack::prepareSourceImages()
-> std::shared_ptr<Ui::Emoji::UniversalImages> {
const auto &images = Ui::Emoji::SourceImages();
if (_session->settings().largeEmoji()) {
return images;
}
Ui::Emoji::ClearSourceImages(images);
return std::make_shared<Ui::Emoji::UniversalImages>(images->id());
}
void EmojiPack::clearSourceImages() {
_imageLoader.with([](details::EmojiImageLoader &loader) {
crl::on_main([images = loader.releaseImages()]{
Ui::Emoji::ClearSourceImages(images);
});
});
}
void EmojiPack::applyPack( void EmojiPack::applyPack(
const MTPDstickerPack &data, const MTPDstickerPack &data,
const base::flat_map<uint64, not_null<DocumentData*>> &map) { const base::flat_map<uint64, not_null<DocumentData*>> &map) {

View File

@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#pragma once #pragma once
#include "ui/text/text_isolated_emoji.h" #include "ui/text/text_isolated_emoji.h"
#include "base/timer.h"
#include <crl/crl_object_on_queue.h> #include <crl/crl_object_on_queue.h>
@ -22,6 +23,9 @@ namespace Ui {
namespace Text { namespace Text {
class String; class String;
} // namespace Text } // namespace Text
namespace Emoji {
class UniversalImages;
} // namespace Emoji
} // namespace Ui } // namespace Ui
namespace Stickers { namespace Stickers {
@ -56,6 +60,8 @@ private:
void refreshAll(); void refreshAll();
void refreshItems(EmojiPtr emoji); void refreshItems(EmojiPtr emoji);
void refreshItems(const base::flat_set<not_null<HistoryItem*>> &list); void refreshItems(const base::flat_set<not_null<HistoryItem*>> &list);
std::shared_ptr<Ui::Emoji::UniversalImages> prepareSourceImages();
void clearSourceImages();
not_null<Main::Session*> _session; not_null<Main::Session*> _session;
base::flat_map<EmojiPtr, not_null<DocumentData*>> _map; base::flat_map<EmojiPtr, not_null<DocumentData*>> _map;
@ -66,6 +72,7 @@ private:
mtpRequestId _requestId = 0; mtpRequestId _requestId = 0;
crl::object_on_queue<details::EmojiImageLoader> _imageLoader; crl::object_on_queue<details::EmojiImageLoader> _imageLoader;
base::Timer _clearTimer;
rpl::lifetime _lifetime; rpl::lifetime _lifetime;

View File

@ -481,6 +481,10 @@ bool Settings::largeEmoji() const {
return _variables.largeEmoji.current(); return _variables.largeEmoji.current();
} }
rpl::producer<bool> Settings::largeEmojiValue() const {
return _variables.largeEmoji.value();
}
rpl::producer<bool> Settings::largeEmojiChanges() const { rpl::producer<bool> Settings::largeEmojiChanges() const {
return _variables.largeEmoji.changes(); return _variables.largeEmoji.changes();
} }

View File

@ -249,6 +249,7 @@ public:
} }
void setLargeEmoji(bool value); void setLargeEmoji(bool value);
[[nodiscard]] bool largeEmoji() const; [[nodiscard]] bool largeEmoji() const;
[[nodiscard]] rpl::producer<bool> largeEmojiValue() const;
[[nodiscard]] rpl::producer<bool> largeEmojiChanges() const; [[nodiscard]] rpl::producer<bool> largeEmojiChanges() const;
void setReplaceEmoji(bool value); void setReplaceEmoji(bool value);
[[nodiscard]] bool replaceEmoji() const; [[nodiscard]] bool replaceEmoji() const;

View File

@ -68,6 +68,7 @@ auto SpritesCount = -1;
auto InstanceNormal = std::unique_ptr<Instance>(); auto InstanceNormal = std::unique_ptr<Instance>();
auto InstanceLarge = std::unique_ptr<Instance>(); auto InstanceLarge = std::unique_ptr<Instance>();
auto Universal = std::shared_ptr<UniversalImages>(); auto Universal = std::shared_ptr<UniversalImages>();
auto CanClearUniversal = false;
auto Updates = rpl::event_stream<>(); auto Updates = rpl::event_stream<>();
auto UpdatesRecent = rpl::event_stream<>(); auto UpdatesRecent = rpl::event_stream<>();
@ -148,6 +149,7 @@ void SwitchToSetPrepared(int id, std::shared_ptr<UniversalImages> images) {
stream << qint32(id); stream << qint32(id);
} }
Universal = std::move(images); Universal = std::move(images);
CanClearUniversal = false;
MainEmojiMap.clear(); MainEmojiMap.clear();
OtherEmojiMap.clear(); OtherEmojiMap.clear();
Updates.fire({}); Updates.fire({});
@ -383,7 +385,10 @@ EmojiPtr FindReplacement(const QChar *start, const QChar *end, int *outLength) {
void ClearUniversalChecked() { void ClearUniversalChecked() {
Expects(InstanceNormal != nullptr && InstanceLarge != nullptr); Expects(InstanceNormal != nullptr && InstanceLarge != nullptr);
if (InstanceNormal->cached() && InstanceLarge->cached() && Universal) { if (CanClearUniversal
&& Universal
&& InstanceNormal->cached()
&& InstanceLarge->cached()) {
Universal->clear(); Universal->clear();
} }
} }
@ -507,6 +512,7 @@ void Init() {
SizeNormal = ConvertScale(18, cScale() * cIntRetinaFactor()); SizeNormal = ConvertScale(18, cScale() * cIntRetinaFactor());
SizeLarge = int(ConvertScale(18 * 4 / 3., cScale() * cIntRetinaFactor())); SizeLarge = int(ConvertScale(18 * 4 / 3., cScale() * cIntRetinaFactor()));
Universal = std::make_shared<UniversalImages>(ReadCurrentSetId()); Universal = std::make_shared<UniversalImages>(ReadCurrentSetId());
CanClearUniversal = false;
InstanceNormal = std::make_unique<Instance>(SizeNormal); InstanceNormal = std::make_unique<Instance>(SizeNormal);
InstanceLarge = std::make_unique<Instance>(SizeLarge); InstanceLarge = std::make_unique<Instance>(SizeLarge);
@ -1007,5 +1013,24 @@ void Instance::pushSprite(QImage &&data) {
_sprites.back().setDevicePixelRatio(cRetinaFactor()); _sprites.back().setDevicePixelRatio(cRetinaFactor());
} }
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 Emoji
} // namespace Ui } // namespace Ui

View File

@ -182,10 +182,13 @@ public:
QImage generate(int size, int index) const; QImage generate(int size, int index) const;
private: private:
int _id = 0; const int _id = 0;
std::vector<QImage> _sprites; std::vector<QImage> _sprites;
}; };
const std::shared_ptr<UniversalImages> &SourceImages();
void ClearSourceImages(const std::shared_ptr<UniversalImages> &images);
} // namespace Emoji } // namespace Emoji
} // namespace Ui } // namespace Ui

View File

@ -96,7 +96,7 @@ if sys.platform == 'win32':
elif sys.platform == 'darwin': elif sys.platform == 'darwin':
# use patched gyp with Xcode project generator # use patched gyp with Xcode project generator
gypScript = '../../../Libraries/gyp/gyp' gypScript = '../../../Libraries/gyp/gyp'
gypArguments.append('-Gxcode_upgrade_check_project_version=1020') gypArguments.append('-Gxcode_upgrade_check_project_version=1030')
gypFormats.append('xcode') gypFormats.append('xcode')
else: else:
gypScript = '../../../Libraries/gyp/gyp' gypScript = '../../../Libraries/gyp/gyp'