mirror of https://github.com/procxx/kepka.git
Reuse global emoji large images.
This commit is contained in:
parent
e479daca03
commit
1b1b1780db
|
@ -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) {
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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();
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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'
|
||||||
|
|
Loading…
Reference in New Issue