Change Stickers::Set from value to object type.

This commit is contained in:
John Preston 2020-05-28 14:00:51 +04:00
parent 897e432f40
commit 803593cd8d
33 changed files with 894 additions and 671 deletions

View File

@ -301,6 +301,8 @@ PRIVATE
chat_helpers/stickers_dice_pack.h chat_helpers/stickers_dice_pack.h
chat_helpers/stickers_list_widget.cpp chat_helpers/stickers_list_widget.cpp
chat_helpers/stickers_list_widget.h chat_helpers/stickers_list_widget.h
chat_helpers/stickers_set.cpp
chat_helpers/stickers_set.h
chat_helpers/tabbed_panel.cpp chat_helpers/tabbed_panel.cpp
chat_helpers/tabbed_panel.h chat_helpers/tabbed_panel.h
chat_helpers/tabbed_section.cpp chat_helpers/tabbed_section.cpp

View File

@ -1890,16 +1890,17 @@ void ApiWrap::saveStickerSets(
auto it = sets.find(removedSetId); auto it = sets.find(removedSetId);
if (it != sets.cend()) { if (it != sets.cend()) {
const auto set = it->second.get();
for (auto i = recent.begin(); i != recent.cend();) { for (auto i = recent.begin(); i != recent.cend();) {
if (it->stickers.indexOf(i->first) >= 0) { if (set->stickers.indexOf(i->first) >= 0) {
i = recent.erase(i); i = recent.erase(i);
writeRecent = true; writeRecent = true;
} else { } else {
++i; ++i;
} }
} }
if (!(it->flags & MTPDstickerSet::Flag::f_archived)) { if (!(set->flags & MTPDstickerSet::Flag::f_archived)) {
MTPInputStickerSet setId = (it->id && it->access) ? MTP_inputStickerSetID(MTP_long(it->id), MTP_long(it->access)) : MTP_inputStickerSetShortName(MTP_string(it->shortName)); const auto setId = set->mtpInput();
auto requestId = request(MTPmessages_UninstallStickerSet(setId)).done([this](const MTPBool &result, mtpRequestId requestId) { auto requestId = request(MTPmessages_UninstallStickerSet(setId)).done([this](const MTPBool &result, mtpRequestId requestId) {
stickerSetDisenabled(requestId); stickerSetDisenabled(requestId);
@ -1909,60 +1910,69 @@ void ApiWrap::saveStickerSets(
_stickerSetDisenableRequests.insert(requestId); _stickerSetDisenableRequests.insert(requestId);
int removeIndex = _session->data().stickerSetsOrder().indexOf(it->id); int removeIndex = _session->data().stickerSetsOrder().indexOf(set->id);
if (removeIndex >= 0) _session->data().stickerSetsOrderRef().removeAt(removeIndex); if (removeIndex >= 0) _session->data().stickerSetsOrderRef().removeAt(removeIndex);
if (!(it->flags & MTPDstickerSet_ClientFlag::f_featured) if (!(set->flags & MTPDstickerSet_ClientFlag::f_featured)
&& !(it->flags & MTPDstickerSet_ClientFlag::f_special)) { && !(set->flags & MTPDstickerSet_ClientFlag::f_special)) {
sets.erase(it); sets.erase(it);
} else { } else {
if (it->flags & MTPDstickerSet::Flag::f_archived) { if (set->flags & MTPDstickerSet::Flag::f_archived) {
writeArchived = true; writeArchived = true;
} }
it->flags &= ~(MTPDstickerSet::Flag::f_installed_date | MTPDstickerSet::Flag::f_archived); set->flags &= ~(MTPDstickerSet::Flag::f_installed_date | MTPDstickerSet::Flag::f_archived);
it->installDate = TimeId(0); set->installDate = TimeId(0);
} }
} }
} }
} }
// Clear all installed flags, set only for sets from order. // Clear all installed flags, set only for sets from order.
for (auto &set : sets) { for (auto &[id, set] : sets) {
if (!(set.flags & MTPDstickerSet::Flag::f_archived)) { if (!(set->flags & MTPDstickerSet::Flag::f_archived)) {
set.flags &= ~MTPDstickerSet::Flag::f_installed_date; set->flags &= ~MTPDstickerSet::Flag::f_installed_date;
} }
} }
auto &order = _session->data().stickerSetsOrderRef(); auto &order = _session->data().stickerSetsOrderRef();
order.clear(); order.clear();
for_const (auto setId, _stickersOrder) { for (const auto setId : std::as_const(_stickersOrder)) {
auto it = sets.find(setId); auto it = sets.find(setId);
if (it != sets.cend()) { if (it != sets.cend()) {
if ((it->flags & MTPDstickerSet::Flag::f_archived) && !localRemoved.contains(it->id)) { const auto set = it->second.get();
MTPInputStickerSet mtpSetId = (it->id && it->access) ? MTP_inputStickerSetID(MTP_long(it->id), MTP_long(it->access)) : MTP_inputStickerSetShortName(MTP_string(it->shortName)); if ((set->flags & MTPDstickerSet::Flag::f_archived) && !localRemoved.contains(set->id)) {
const auto mtpSetId = set->mtpInput();
auto requestId = request(MTPmessages_InstallStickerSet(mtpSetId, MTP_boolFalse())).done([this](const MTPmessages_StickerSetInstallResult &result, mtpRequestId requestId) { const auto requestId = request(MTPmessages_InstallStickerSet(
mtpSetId,
MTP_boolFalse()
)).done([=](
const MTPmessages_StickerSetInstallResult &result,
mtpRequestId requestId) {
stickerSetDisenabled(requestId); stickerSetDisenabled(requestId);
}).fail([this](const RPCError &error, mtpRequestId requestId) { }).fail([=](
const RPCError &error,
mtpRequestId requestId) {
stickerSetDisenabled(requestId); stickerSetDisenabled(requestId);
}).afterDelay(kSmallDelayMs).send(); }).afterDelay(kSmallDelayMs).send();
_stickerSetDisenableRequests.insert(requestId); _stickerSetDisenableRequests.insert(requestId);
it->flags &= ~MTPDstickerSet::Flag::f_archived; set->flags &= ~MTPDstickerSet::Flag::f_archived;
writeArchived = true; writeArchived = true;
} }
order.push_back(setId); order.push_back(setId);
it->flags |= MTPDstickerSet::Flag::f_installed_date; set->flags |= MTPDstickerSet::Flag::f_installed_date;
if (!it->installDate) { if (!set->installDate) {
it->installDate = base::unixtime::now(); set->installDate = base::unixtime::now();
} }
} }
} }
for (auto it = sets.begin(); it != sets.cend();) { for (auto it = sets.begin(); it != sets.cend();) {
if ((it->flags & MTPDstickerSet_ClientFlag::f_featured) const auto set = it->second.get();
|| (it->flags & MTPDstickerSet::Flag::f_installed_date) if ((set->flags & MTPDstickerSet_ClientFlag::f_featured)
|| (it->flags & MTPDstickerSet::Flag::f_archived) || (set->flags & MTPDstickerSet::Flag::f_installed_date)
|| (it->flags & MTPDstickerSet_ClientFlag::f_special)) { || (set->flags & MTPDstickerSet::Flag::f_archived)
|| (set->flags & MTPDstickerSet_ClientFlag::f_special)) {
++it; ++it;
} else { } else {
it = sets.erase(it); it = sets.erase(it);
@ -3341,7 +3351,7 @@ void ApiWrap::readFeaturedSets() {
for (auto setId : _featuredSetsRead) { for (auto setId : _featuredSetsRead) {
auto it = sets.find(setId); auto it = sets.find(setId);
if (it != sets.cend()) { if (it != sets.cend()) {
it->flags &= ~MTPDstickerSet_ClientFlag::f_unread; it->second->flags &= ~MTPDstickerSet_ClientFlag::f_unread;
wrappedIds.append(MTP_long(setId)); wrappedIds.append(MTP_long(setId));
if (count) { if (count) {
--count; --count;

View File

@ -12,7 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "base/flat_map.h" #include "base/flat_map.h"
#include "base/flat_set.h" #include "base/flat_set.h"
#include "mtproto/sender.h" #include "mtproto/sender.h"
#include "chat_helpers/stickers.h" #include "chat_helpers/stickers_set.h"
#include "data/data_messages.h" #include "data/data_messages.h"
class TaskQueue; class TaskQueue;

View File

@ -19,6 +19,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "main/main_session.h" #include "main/main_session.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "data/data_user.h" #include "data/data_user.h"
#include "mtproto/facade.h"
#include "app.h" #include "app.h"
#include "styles/style_layers.h" #include "styles/style_layers.h"
#include "styles/style_boxes.h" #include "styles/style_boxes.h"

View File

@ -21,6 +21,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/widgets/buttons.h" #include "ui/widgets/buttons.h"
#include "ui/widgets/scroll_area.h" #include "ui/widgets/scroll_area.h"
#include "ui/image/image.h" #include "ui/image/image.h"
#include "ui/image/image_location_factory.h"
#include "ui/text/text_utilities.h" #include "ui/text/text_utilities.h"
#include "ui/emoji_config.h" #include "ui/emoji_config.h"
#include "lottie/lottie_multi_player.h" #include "lottie/lottie_multi_player.h"
@ -115,7 +116,7 @@ private:
int32 _setHash = 0; int32 _setHash = 0;
MTPDstickerSet::Flags _setFlags = 0; MTPDstickerSet::Flags _setFlags = 0;
TimeId _setInstallDate = TimeId(0); TimeId _setInstallDate = TimeId(0);
ImagePtr _setThumbnail; ImageWithLocation _setThumbnail;
MTPInputStickerSet _input; MTPInputStickerSet _input;
@ -299,25 +300,29 @@ void StickerSetBox::Inner::gotSet(const MTPmessages_StickerSet &set) {
_setFlags = set.vflags().v; _setFlags = set.vflags().v;
_setInstallDate = set.vinstalled_date().value_or(0); _setInstallDate = set.vinstalled_date().value_or(0);
if (const auto thumb = set.vthumb()) { if (const auto thumb = set.vthumb()) {
_setThumbnail = Images::Create(set, *thumb); _setThumbnail = Images::FromPhotoSize(
&_controller->session(),
set,
*thumb);
} else { } else {
_setThumbnail = ImagePtr(); _setThumbnail = ImageWithLocation();
} }
auto &sets = _controller->session().data().stickerSetsRef(); auto &sets = _controller->session().data().stickerSetsRef();
const auto it = sets.find(_setId); const auto it = sets.find(_setId);
if (it != sets.cend()) { if (it != sets.cend()) {
const auto set = it->second.get();
using ClientFlag = MTPDstickerSet_ClientFlag; using ClientFlag = MTPDstickerSet_ClientFlag;
const auto clientFlags = it->flags const auto clientFlags = set->flags
& (ClientFlag::f_featured & (ClientFlag::f_featured
| ClientFlag::f_not_loaded | ClientFlag::f_not_loaded
| ClientFlag::f_unread | ClientFlag::f_unread
| ClientFlag::f_special); | ClientFlag::f_special);
_setFlags |= clientFlags; _setFlags |= clientFlags;
it->flags = _setFlags; set->flags = _setFlags;
it->installDate = _setInstallDate; set->installDate = _setInstallDate;
it->stickers = _pack; set->stickers = _pack;
it->emoji = _emoji; set->emoji = _emoji;
it->thumbnail = _setThumbnail; set->setThumbnail(_setThumbnail);
} }
}); });
}); });
@ -359,9 +364,10 @@ void StickerSetBox::Inner::installDone(
_setFlags |= MTPDstickerSet::Flag::f_installed_date; _setFlags |= MTPDstickerSet::Flag::f_installed_date;
auto it = sets.find(_setId); auto it = sets.find(_setId);
if (it == sets.cend()) { if (it == sets.cend()) {
it = sets.insert( it = sets.emplace(
_setId, _setId,
Stickers::Set( std::make_unique<Stickers::Set>(
&_controller->session().data(),
_setId, _setId,
_setAccess, _setAccess,
_setTitle, _setTitle,
@ -369,14 +375,15 @@ void StickerSetBox::Inner::installDone(
_setCount, _setCount,
_setHash, _setHash,
_setFlags, _setFlags,
_setInstallDate, _setInstallDate)).first;
_setThumbnail));
} else { } else {
it->flags = _setFlags; it->second->flags = _setFlags;
it->installDate = _setInstallDate; it->second->installDate = _setInstallDate;
} }
it->stickers = _pack; const auto set = it->second.get();
it->emoji = _emoji; set->setThumbnail(_setThumbnail);
set->stickers = _pack;
set->emoji = _emoji;
auto &order = _controller->session().data().stickerSetsOrderRef(); auto &order = _controller->session().data().stickerSetsOrderRef();
int insertAtIndex = 0, currentIndex = order.indexOf(_setId); int insertAtIndex = 0, currentIndex = order.indexOf(_setId);
@ -387,14 +394,15 @@ void StickerSetBox::Inner::installDone(
order.insert(insertAtIndex, _setId); order.insert(insertAtIndex, _setId);
} }
auto custom = sets.find(Stickers::CustomSetId); const auto customIt = sets.find(Stickers::CustomSetId);
if (custom != sets.cend()) { if (customIt != sets.cend()) {
for_const (auto sticker, _pack) { const auto custom = customIt->second.get();
for (const auto sticker : std::as_const(_pack)) {
int removeIndex = custom->stickers.indexOf(sticker); int removeIndex = custom->stickers.indexOf(sticker);
if (removeIndex >= 0) custom->stickers.removeAt(removeIndex); if (removeIndex >= 0) custom->stickers.removeAt(removeIndex);
} }
if (custom->stickers.isEmpty()) { if (custom->stickers.isEmpty()) {
sets.erase(custom); sets.erase(customIt);
} }
} }
@ -669,10 +677,11 @@ bool StickerSetBox::Inner::notInstalled() const {
if (!_loaded) { if (!_loaded) {
return false; return false;
} }
const auto it = _controller->session().data().stickerSets().constFind(_setId); const auto &sets = _controller->session().data().stickerSets();
if ((it == _controller->session().data().stickerSets().cend()) const auto it = sets.find(_setId);
|| !(it->flags & MTPDstickerSet::Flag::f_installed_date) if ((it == sets.cend())
|| (it->flags & MTPDstickerSet::Flag::f_archived)) { || !(it->second->flags & MTPDstickerSet::Flag::f_installed_date)
|| (it->second->flags & MTPDstickerSet::Flag::f_archived)) {
return !_pack.empty(); return !_pack.empty();
} }
return false; return false;

View File

@ -374,17 +374,24 @@ void StickersBox::loadMoreArchived() {
} }
uint64 lastId = 0; uint64 lastId = 0;
for (auto setIt = _session->data().archivedStickerSetsOrder().cend(), e = _session->data().archivedStickerSetsOrder().cbegin(); setIt != e;) { const auto &order = _session->data().archivedStickerSetsOrder();
const auto &sets = _session->data().stickerSets();
for (auto setIt = order.cend(), e = order.cbegin(); setIt != e;) {
--setIt; --setIt;
auto it = _session->data().stickerSets().constFind(*setIt); auto it = sets.find(*setIt);
if (it != _session->data().stickerSets().cend()) { if (it != sets.cend()) {
if (it->flags & MTPDstickerSet::Flag::f_archived) { if (it->second->flags & MTPDstickerSet::Flag::f_archived) {
lastId = it->id; lastId = it->second->id;
break; break;
} }
} }
} }
_archivedRequestId = MTP::send(MTPmessages_GetArchivedStickers(MTP_flags(0), MTP_long(lastId), MTP_int(kArchivedLimitPerPage)), rpcDone(&StickersBox::getArchivedDone, lastId)); _archivedRequestId = MTP::send(
MTPmessages_GetArchivedStickers(
MTP_flags(0),
MTP_long(lastId),
MTP_int(kArchivedLimitPerPage)),
rpcDone(&StickersBox::getArchivedDone, lastId));
} }
void StickersBox::paintEvent(QPaintEvent *e) { void StickersBox::paintEvent(QPaintEvent *e) {
@ -497,6 +504,7 @@ void StickersBox::installSet(uint64 setId) {
return; return;
} }
const auto set = it->second.get();
if (_localRemoved.contains(setId)) { if (_localRemoved.contains(setId)) {
_localRemoved.removeOne(setId); _localRemoved.removeOne(setId);
if (_installed.widget()) _installed.widget()->setRemovedSets(_localRemoved); if (_installed.widget()) _installed.widget()->setRemovedSets(_localRemoved);
@ -504,11 +512,11 @@ void StickersBox::installSet(uint64 setId) {
if (_archived.widget()) _archived.widget()->setRemovedSets(_localRemoved); if (_archived.widget()) _archived.widget()->setRemovedSets(_localRemoved);
if (_attached.widget()) _attached.widget()->setRemovedSets(_localRemoved); if (_attached.widget()) _attached.widget()->setRemovedSets(_localRemoved);
} }
if (!(it->flags & MTPDstickerSet::Flag::f_installed_date) if (!(set->flags & MTPDstickerSet::Flag::f_installed_date)
|| (it->flags & MTPDstickerSet::Flag::f_archived)) { || (set->flags & MTPDstickerSet::Flag::f_archived)) {
MTP::send( MTP::send(
MTPmessages_InstallStickerSet( MTPmessages_InstallStickerSet(
Stickers::inputSetId(*it), set->mtpInput(),
MTP_boolFalse()), MTP_boolFalse()),
rpcDone(&StickersBox::installDone), rpcDone(&StickersBox::installDone),
rpcFail(&StickersBox::installFail, setId)); rpcFail(&StickersBox::installFail, setId));
@ -551,12 +559,14 @@ void StickersBox::requestArchivedSets() {
preloadArchivedSets(); preloadArchivedSets();
} }
auto &sets = _session->data().stickerSets(); const auto &sets = _session->data().stickerSets();
for_const (auto setId, _session->data().archivedStickerSetsOrder()) { const auto &order = _session->data().archivedStickerSetsOrder();
auto it = sets.constFind(setId); for (const auto setId : order) {
auto it = sets.find(setId);
if (it != sets.cend()) { if (it != sets.cend()) {
if (it->stickers.isEmpty() && (it->flags & MTPDstickerSet_ClientFlag::f_not_loaded)) { const auto set = it->second.get();
_session->api().scheduleStickerSetRequest(setId, it->access); if (set->stickers.isEmpty() && (set->flags & MTPDstickerSet_ClientFlag::f_not_loaded)) {
_session->api().scheduleStickerSetRequest(setId, set->access);
} }
} }
} }
@ -658,6 +668,10 @@ StickersBox::Inner::Row::Row(
StickersBox::Inner::Row::~Row() = default; StickersBox::Inner::Row::~Row() = default;
bool StickersBox::Inner::Row::isRecentSet() const {
return (id == Stickers::CloudRecentSetId);
}
StickersBox::Inner::Inner( StickersBox::Inner::Inner(
QWidget *parent, QWidget *parent,
not_null<Main::Session*> session, not_null<Main::Session*> session,
@ -1298,7 +1312,7 @@ void StickersBox::Inner::mouseReleaseEvent(QMouseEvent *e) {
if (!row.isRecentSet()) { if (!row.isRecentSet()) {
auto it = sets.find(row.id); auto it = sets.find(row.id);
if (it != sets.cend()) { if (it != sets.cend()) {
return &*it; return it->second.get();
} }
} }
return nullptr; return nullptr;
@ -1309,7 +1323,7 @@ void StickersBox::Inner::mouseReleaseEvent(QMouseEvent *e) {
Ui::show( Ui::show(
Box<StickerSetBox>( Box<StickerSetBox>(
App::wnd()->sessionController(), App::wnd()->sessionController(),
Stickers::inputSetId(*set)), set->mtpInput()),
Ui::LayerOption::KeepOther); Ui::LayerOption::KeepOther);
} }
}; };
@ -1470,8 +1484,9 @@ void StickersBox::Inner::handleMegagroupSetAddressChange() {
auto text = _megagroupSetField->getLastText().trimmed(); auto text = _megagroupSetField->getLastText().trimmed();
if (text.isEmpty()) { if (text.isEmpty()) {
if (_megagroupSelectedSet) { if (_megagroupSelectedSet) {
auto it = _session->data().stickerSets().constFind(_megagroupSelectedSet->id); const auto &sets = _session->data().stickerSets();
if (it != _session->data().stickerSets().cend() && !it->shortName.isEmpty()) { const auto it = sets.find(_megagroupSelectedSet->id);
if (it != sets.cend() && !it->second->shortName.isEmpty()) {
setMegagroupSelectedSet(MTP_inputStickerSetEmpty()); setMegagroupSelectedSet(MTP_inputStickerSetEmpty());
} }
} }
@ -1481,7 +1496,9 @@ void StickersBox::Inner::handleMegagroupSetAddressChange() {
)).done([=](const MTPmessages_StickerSet &result) { )).done([=](const MTPmessages_StickerSet &result) {
_megagroupSetRequestId = 0; _megagroupSetRequestId = 0;
auto set = Stickers::FeedSetFull(result); auto set = Stickers::FeedSetFull(result);
setMegagroupSelectedSet(MTP_inputStickerSetID(MTP_long(set->id), MTP_long(set->access))); setMegagroupSelectedSet(MTP_inputStickerSetID(
MTP_long(set->id),
MTP_long(set->access)));
}).fail([=](const RPCError &error) { }).fail([=](const RPCError &error) {
_megagroupSetRequestId = 0; _megagroupSetRequestId = 0;
setMegagroupSelectedSet(MTP_inputStickerSetEmpty()); setMegagroupSelectedSet(MTP_inputStickerSetEmpty());
@ -1503,31 +1520,35 @@ void StickersBox::Inner::rebuildMegagroupSet() {
_megagroupSelectedShadow.destroy(); _megagroupSelectedShadow.destroy();
return; return;
} }
auto &set = _megagroupSetInput.c_inputStickerSetID(); auto &inputId = _megagroupSetInput.c_inputStickerSetID();
auto setId = set.vid().v; auto setId = inputId.vid().v;
auto &sets = _session->data().stickerSets(); auto &sets = _session->data().stickerSets();
auto it = sets.find(setId); auto it = sets.find(setId);
if (it == sets.cend() || (it->flags & MTPDstickerSet_ClientFlag::f_not_loaded)) { if (it == sets.cend()
_session->api().scheduleStickerSetRequest(set.vid().v, set.vaccess_hash().v); || (it->second->flags & MTPDstickerSet_ClientFlag::f_not_loaded)) {
_session->api().scheduleStickerSetRequest(
inputId.vid().v,
inputId.vaccess_hash().v);
return; return;
} }
const auto set = it->second.get();
auto maxNameWidth = countMaxNameWidth(); auto maxNameWidth = countMaxNameWidth();
auto titleWidth = 0; auto titleWidth = 0;
auto title = fillSetTitle(*it, maxNameWidth, &titleWidth); auto title = fillSetTitle(*set, maxNameWidth, &titleWidth);
auto count = fillSetCount(*it); auto count = fillSetCount(*set);
auto thumbnail = ImagePtr(); auto thumbnail = ImagePtr();
auto sticker = (DocumentData*)nullptr; auto sticker = (DocumentData*)nullptr;
auto pixw = 0, pixh = 0; auto pixw = 0, pixh = 0;
fillSetCover(*it, &thumbnail, &sticker, &pixw, &pixh); fillSetCover(*set, &thumbnail, &sticker, &pixw, &pixh);
auto installed = true, official = false, unread = false, archived = false, removed = false; auto installed = true, official = false, unread = false, archived = false, removed = false;
if (!_megagroupSelectedSet || _megagroupSelectedSet->id != it->id) { if (!_megagroupSelectedSet || _megagroupSelectedSet->id != set->id) {
_megagroupSetField->setText(it->shortName); _megagroupSetField->setText(set->shortName);
_megagroupSetField->finishAnimating(); _megagroupSetField->finishAnimating();
} }
_megagroupSelectedSet = std::make_unique<Row>( _megagroupSelectedSet = std::make_unique<Row>(
it->id, set->id,
it->access, set->access,
thumbnail, thumbnail,
sticker, sticker,
count, count,
@ -1565,7 +1586,7 @@ void StickersBox::Inner::rebuild() {
auto maxNameWidth = countMaxNameWidth(); auto maxNameWidth = countMaxNameWidth();
clear(); clear();
auto &order = ([&]() -> const Stickers::Order & { const auto &order = ([&]() -> const Stickers::Order & {
if (_section == Section::Installed) { if (_section == Section::Installed) {
auto &result = _session->data().stickerSetsOrder(); auto &result = _session->data().stickerSetsOrder();
if (_megagroupSet && result.empty()) { if (_megagroupSet && result.empty()) {
@ -1588,21 +1609,23 @@ void StickersBox::Inner::rebuild() {
: tr::lng_stickers_group_from_your(tr::now)); : tr::lng_stickers_group_from_your(tr::now));
updateControlsGeometry(); updateControlsGeometry();
} else if (_section == Section::Installed) { } else if (_section == Section::Installed) {
auto cloudIt = sets.constFind(Stickers::CloudRecentSetId); auto cloudIt = sets.find(Stickers::CloudRecentSetId);
if (cloudIt != sets.cend() && !cloudIt->stickers.isEmpty()) { if (cloudIt != sets.cend() && !cloudIt->second->stickers.isEmpty()) {
rebuildAppendSet(cloudIt.value(), maxNameWidth); rebuildAppendSet(*cloudIt->second, maxNameWidth);
} }
} }
for_const (auto setId, order) { for (const auto setId : order) {
auto it = sets.constFind(setId); auto it = sets.find(setId);
if (it == sets.cend()) { if (it == sets.cend()) {
continue; continue;
} }
rebuildAppendSet(it.value(), maxNameWidth); const auto set = it->second.get();
rebuildAppendSet(*set, maxNameWidth);
if (it->stickers.isEmpty() || (it->flags & MTPDstickerSet_ClientFlag::f_not_loaded)) { if (set->stickers.isEmpty()
_session->api().scheduleStickerSetRequest(it->id, it->access); || (set->flags & MTPDstickerSet_ClientFlag::f_not_loaded)) {
_session->api().scheduleStickerSetRequest(set->id, set->access);
} }
} }
_session->api().requestStickerSets(); _session->api().requestStickerSets();
@ -1632,16 +1655,16 @@ void StickersBox::Inner::updateRows() {
int maxNameWidth = countMaxNameWidth(); int maxNameWidth = countMaxNameWidth();
auto &sets = _session->data().stickerSets(); auto &sets = _session->data().stickerSets();
for (const auto &row : _rows) { for (const auto &row : _rows) {
const auto it = sets.constFind(row->id); const auto it = sets.find(row->id);
if (it == sets.cend()) { if (it == sets.cend()) {
continue; continue;
} }
const auto &set = it.value(); const auto set = it->second.get();
if (!row->sticker) { if (!row->sticker) {
auto thumbnail = ImagePtr(); auto thumbnail = ImagePtr();
auto sticker = (DocumentData*)nullptr; auto sticker = (DocumentData*)nullptr;
auto pixw = 0, pixh = 0; auto pixw = 0, pixh = 0;
fillSetCover(set, &thumbnail, &sticker, &pixw, &pixh); fillSetCover(*set, &thumbnail, &sticker, &pixw, &pixh);
if (sticker) { if (sticker) {
if ((row->thumbnail.get() != thumbnail.get()) if ((row->thumbnail.get() != thumbnail.get())
|| (!thumbnail && row->sticker != sticker)) { || (!thumbnail && row->sticker != sticker)) {
@ -1659,7 +1682,7 @@ void StickersBox::Inner::updateRows() {
if (!row->isRecentSet()) { if (!row->isRecentSet()) {
auto wasInstalled = row->installed; auto wasInstalled = row->installed;
auto wasArchived = row->archived; auto wasArchived = row->archived;
fillSetFlags(set, &row->installed, &row->official, &row->unread, &row->archived); fillSetFlags(*set, &row->installed, &row->official, &row->unread, &row->archived);
if (_section == Section::Installed) { if (_section == Section::Installed) {
row->archived = false; row->archived = false;
} }
@ -1667,8 +1690,8 @@ void StickersBox::Inner::updateRows() {
row->ripple.reset(); row->ripple.reset();
} }
} }
row->title = fillSetTitle(set, maxNameWidth, &row->titleWidth); row->title = fillSetTitle(*set, maxNameWidth, &row->titleWidth);
row->count = fillSetCount(set); row->count = fillSetCount(*set);
} }
update(); update();
} }
@ -1739,7 +1762,7 @@ void StickersBox::Inner::rebuildAppendSet(const Stickers::Set &set, int maxNameW
} }
void StickersBox::Inner::fillSetCover(const Stickers::Set &set, ImagePtr *thumbnail, DocumentData **outSticker, int *outWidth, int *outHeight) const { void StickersBox::Inner::fillSetCover(const Stickers::Set &set, ImagePtr *thumbnail, DocumentData **outSticker, int *outWidth, int *outHeight) const {
*thumbnail = set.thumbnail; //*thumbnail = set.thumbnail; // #TODO optimize stickers
if (set.stickers.isEmpty()) { if (set.stickers.isEmpty()) {
*outSticker = nullptr; *outSticker = nullptr;
*outWidth = *outHeight = 0; *outWidth = *outHeight = 0;
@ -1747,8 +1770,10 @@ void StickersBox::Inner::fillSetCover(const Stickers::Set &set, ImagePtr *thumbn
} }
auto sticker = *outSticker = set.stickers.front(); auto sticker = *outSticker = set.stickers.front();
const auto size = set.thumbnail const auto size = set.hasThumbnail()
? set.thumbnail->size() ? QSize(
set.thumbnailLocation().width(),
set.thumbnailLocation().height())
: sticker->hasThumbnail() : sticker->hasThumbnail()
? QSize( ? QSize(
sticker->thumbnailLocation().width(), sticker->thumbnailLocation().width(),
@ -1775,11 +1800,13 @@ void StickersBox::Inner::fillSetCover(const Stickers::Set &set, ImagePtr *thumbn
int StickersBox::Inner::fillSetCount(const Stickers::Set &set) const { int StickersBox::Inner::fillSetCount(const Stickers::Set &set) const {
int result = set.stickers.isEmpty() ? set.count : set.stickers.size(), added = 0; int result = set.stickers.isEmpty() ? set.count : set.stickers.size(), added = 0;
if (set.id == Stickers::CloudRecentSetId) { if (set.id == Stickers::CloudRecentSetId) {
auto customIt = _session->data().stickerSets().constFind(Stickers::CustomSetId); const auto &sets = _session->data().stickerSets();
if (customIt != _session->data().stickerSets().cend()) { auto customIt = sets.find(Stickers::CustomSetId);
added = customIt->stickers.size(); if (customIt != sets.cend()) {
for_const (auto &sticker, Stickers::GetRecentPack()) { added = customIt->second->stickers.size();
if (customIt->stickers.indexOf(sticker.first) < 0) { const auto &recent = Stickers::GetRecentPack();
for (const auto &sticker : recent) {
if (customIt->second->stickers.indexOf(sticker.first) < 0) {
++added; ++added;
} }
} }

View File

@ -10,7 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "boxes/abstract_box.h" #include "boxes/abstract_box.h"
#include "base/timer.h" #include "base/timer.h"
#include "mtproto/sender.h" #include "mtproto/sender.h"
#include "chat_helpers/stickers.h" #include "chat_helpers/stickers_set.h"
#include "ui/effects/animations.h" #include "ui/effects/animations.h"
#include "ui/special_fields.h" #include "ui/special_fields.h"
@ -33,6 +33,14 @@ namespace Main {
class Session; class Session;
} // namespace Main } // namespace Main
namespace Data {
class DocumentMedia;
} // namespace Data
namespace Lottie {
class SinglePlayer;
} // namespace Lottie
class StickersBox final class StickersBox final
: public Ui::BoxContent : public Ui::BoxContent
, public RPCSender , public RPCSender
@ -235,9 +243,7 @@ private:
int32 pixh); int32 pixh);
~Row(); ~Row();
bool isRecentSet() const { bool isRecentSet() const;
return (id == Stickers::CloudRecentSetId);
}
uint64 id = 0; uint64 id = 0;
uint64 accessHash = 0; uint64 accessHash = 0;

View File

@ -17,6 +17,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "main/main_session.h" #include "main/main_session.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "data/data_user.h" #include "data/data_user.h"
#include "mtproto/facade.h"
#include "styles/style_layers.h" #include "styles/style_layers.h"
#include "styles/style_boxes.h" #include "styles/style_boxes.h"

View File

@ -5,12 +5,13 @@ the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link: For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/ */
#include "stickers.h" #include "chat_helpers/stickers.h"
#include "data/data_document.h" #include "data/data_document.h"
#include "data/data_document_media.h" #include "data/data_document_media.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "data/data_file_origin.h" #include "data/data_file_origin.h"
#include "chat_helpers/stickers_set.h"
#include "boxes/stickers_box.h" #include "boxes/stickers_box.h"
#include "boxes/confirm_box.h" #include "boxes/confirm_box.h"
#include "lang/lang_keys.h" #include "lang/lang_keys.h"
@ -21,6 +22,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "mainwindow.h" #include "mainwindow.h"
#include "ui/toast/toast.h" #include "ui/toast/toast.h"
#include "ui/emoji_config.h" #include "ui/emoji_config.h"
#include "ui/image/image_location_factory.h"
#include "base/unixtime.h" #include "base/unixtime.h"
#include "lottie/lottie_single_player.h" #include "lottie/lottie_single_player.h"
#include "lottie/lottie_multi_player.h" #include "lottie/lottie_multi_player.h"
@ -41,7 +43,7 @@ void ApplyArchivedResult(const MTPDmessages_stickerSetInstallResultArchive &d) {
Order archived; Order archived;
archived.reserve(v.size()); archived.reserve(v.size());
QMap<uint64, uint64> setsToRequest; QMap<uint64, uint64> setsToRequest;
for_const (auto &stickerSet, v) { for (const auto &stickerSet : v) {
const MTPDstickerSet *setData = nullptr; const MTPDstickerSet *setData = nullptr;
switch (stickerSet.type()) { switch (stickerSet.type()) {
case mtpc_stickerSetCovered: { case mtpc_stickerSetCovered: {
@ -91,21 +93,22 @@ void ApplyArchivedResult(const MTPDmessages_stickerSetInstallResultArchive &d) {
// For testing: Just apply random subset or your sticker sets as archived. // For testing: Just apply random subset or your sticker sets as archived.
bool ApplyArchivedResultFake() { bool ApplyArchivedResultFake() {
auto sets = QVector<MTPStickerSetCovered>(); auto sets = QVector<MTPStickerSetCovered>();
for (auto &set : Auth().data().stickerSetsRef()) { for (const auto &[id, set] : Auth().data().stickerSetsRef()) {
if ((set.flags & MTPDstickerSet::Flag::f_installed_date) const auto raw = set.get();
&& !(set.flags & MTPDstickerSet_ClientFlag::f_special)) { if ((raw->flags & MTPDstickerSet::Flag::f_installed_date)
&& !(raw->flags & MTPDstickerSet_ClientFlag::f_special)) {
if (rand_value<uint32>() % 128 < 64) { if (rand_value<uint32>() % 128 < 64) {
const auto data = MTP_stickerSet( const auto data = MTP_stickerSet(
MTP_flags(set.flags | MTPDstickerSet::Flag::f_archived), MTP_flags(raw->flags | MTPDstickerSet::Flag::f_archived),
MTP_int(set.installDate), MTP_int(raw->installDate),
MTP_long(set.id), MTP_long(raw->id),
MTP_long(set.access), MTP_long(raw->access),
MTP_string(set.title), MTP_string(raw->title),
MTP_string(set.shortName), MTP_string(raw->shortName),
MTP_photoSizeEmpty(MTP_string()), MTP_photoSizeEmpty(MTP_string()),
MTP_int(0), MTP_int(0),
MTP_int(set.count), MTP_int(raw->count),
MTP_int(set.hash)); MTP_int(raw->hash));
sets.push_back(MTP_stickerSetCovered( sets.push_back(MTP_stickerSetCovered(
data, data,
MTP_documentEmpty(MTP_long(0)))); MTP_documentEmpty(MTP_long(0))));
@ -128,11 +131,12 @@ void InstallLocally(uint64 setId) {
return; return;
} }
auto flags = it->flags; const auto set = it->second.get();
it->flags &= ~(MTPDstickerSet::Flag::f_archived | MTPDstickerSet_ClientFlag::f_unread); auto flags = set->flags;
it->flags |= MTPDstickerSet::Flag::f_installed_date; set->flags &= ~(MTPDstickerSet::Flag::f_archived | MTPDstickerSet_ClientFlag::f_unread);
it->installDate = base::unixtime::now(); set->flags |= MTPDstickerSet::Flag::f_installed_date;
auto changedFlags = flags ^ it->flags; set->installDate = base::unixtime::now();
auto changedFlags = flags ^ set->flags;
auto &order = Auth().data().stickerSetsOrderRef(); auto &order = Auth().data().stickerSetsOrderRef();
int insertAtIndex = 0, currentIndex = order.indexOf(setId); int insertAtIndex = 0, currentIndex = order.indexOf(setId);
@ -143,14 +147,15 @@ void InstallLocally(uint64 setId) {
order.insert(insertAtIndex, setId); order.insert(insertAtIndex, setId);
} }
auto custom = sets.find(CustomSetId); auto customIt = sets.find(CustomSetId);
if (custom != sets.cend()) { if (customIt != sets.cend()) {
for_const (auto sticker, it->stickers) { const auto custom = customIt->second.get();
for (const auto sticker : std::as_const(set->stickers)) {
int removeIndex = custom->stickers.indexOf(sticker); int removeIndex = custom->stickers.indexOf(sticker);
if (removeIndex >= 0) custom->stickers.removeAt(removeIndex); if (removeIndex >= 0) custom->stickers.removeAt(removeIndex);
} }
if (custom->stickers.isEmpty()) { if (custom->stickers.isEmpty()) {
sets.erase(custom); sets.erase(customIt);
} }
} }
Local::writeInstalledStickers(); Local::writeInstalledStickers();
@ -174,8 +179,9 @@ void UndoInstallLocally(uint64 setId) {
return; return;
} }
it->flags &= ~MTPDstickerSet::Flag::f_installed_date; const auto set = it->second.get();
it->installDate = TimeId(0); set->flags &= ~MTPDstickerSet::Flag::f_installed_date;
set->installDate = TimeId(0);
auto &order = Auth().data().stickerSetsOrderRef(); auto &order = Auth().data().stickerSetsOrderRef();
int currentIndex = order.indexOf(setId); int currentIndex = order.indexOf(setId);
@ -192,11 +198,12 @@ void UndoInstallLocally(uint64 setId) {
} }
bool IsFaved(not_null<const DocumentData*> document) { bool IsFaved(not_null<const DocumentData*> document) {
const auto it = Auth().data().stickerSets().constFind(FavedSetId); const auto &sets = Auth().data().stickerSets();
if (it == Auth().data().stickerSets().cend()) { const auto it = sets.find(FavedSetId);
if (it == sets.cend()) {
return false; return false;
} }
for (const auto sticker : it->stickers) { for (const auto sticker : it->second->stickers) {
if (sticker == document) { if (sticker == document) {
return true; return true;
} }
@ -254,11 +261,14 @@ void MoveFavedToFront(Set &set, int index) {
void RequestSetToPushFaved(not_null<DocumentData*> document); void RequestSetToPushFaved(not_null<DocumentData*> document);
void SetIsFaved(not_null<DocumentData*> document, std::optional<std::vector<not_null<EmojiPtr>>> emojiList = std::nullopt) { void SetIsFaved(
auto &sets = Auth().data().stickerSetsRef(); not_null<DocumentData*> document,
std::optional<std::vector<not_null<EmojiPtr>>> emojiList = std::nullopt) {
auto &sets = document->owner().stickerSetsRef();
auto it = sets.find(FavedSetId); auto it = sets.find(FavedSetId);
if (it == sets.end()) { if (it == sets.end()) {
it = sets.insert(FavedSetId, Set( it = sets.emplace(FavedSetId, std::make_unique<Set>(
&document->owner(),
FavedSetId, FavedSetId,
uint64(0), uint64(0),
Lang::Hard::FavedSetTitle(), Lang::Hard::FavedSetTitle(),
@ -266,19 +276,19 @@ void SetIsFaved(not_null<DocumentData*> document, std::optional<std::vector<not_
0, // count 0, // count
0, // hash 0, // hash
MTPDstickerSet_ClientFlag::f_special | 0, MTPDstickerSet_ClientFlag::f_special | 0,
TimeId(0), TimeId(0))).first;
ImagePtr()));
} }
auto index = it->stickers.indexOf(document); const auto set = it->second.get();
auto index = set->stickers.indexOf(document);
if (index == 0) { if (index == 0) {
return; return;
} }
if (index > 0) { if (index > 0) {
MoveFavedToFront(*it, index); MoveFavedToFront(*set, index);
} else if (emojiList) { } else if (emojiList) {
PushFavedToFront(*it, document, *emojiList); PushFavedToFront(*set, document, *emojiList);
} else if (auto list = GetEmojiListFromSet(document)) { } else if (auto list = GetEmojiListFromSet(document)) {
PushFavedToFront(*it, document, *list); PushFavedToFront(*set, document, *list);
} else { } else {
RequestSetToPushFaved(document); RequestSetToPushFaved(document);
return; return;
@ -304,9 +314,9 @@ void RequestSetToPushFaved(not_null<DocumentData*> document) {
auto list = std::vector<not_null<EmojiPtr>>(); auto list = std::vector<not_null<EmojiPtr>>();
auto &d = result.c_messages_stickerSet(); auto &d = result.c_messages_stickerSet();
list.reserve(d.vpacks().v.size()); list.reserve(d.vpacks().v.size());
for_const (auto &mtpPack, d.vpacks().v) { for (const auto &mtpPack : d.vpacks().v) {
auto &pack = mtpPack.c_stickerPack(); auto &pack = mtpPack.c_stickerPack();
for_const (auto &documentId, pack.vdocuments().v) { for (const auto &documentId : pack.vdocuments().v) {
if (documentId.v == document->id) { if (documentId.v == document->id) {
if (const auto emoji = Ui::Emoji::Find(qs(mtpPack.c_stickerPack().vemoticon()))) { if (const auto emoji = Ui::Emoji::Find(qs(mtpPack.c_stickerPack().vemoticon()))) {
list.emplace_back(emoji); list.emplace_back(emoji);
@ -332,23 +342,24 @@ void SetIsNotFaved(not_null<DocumentData*> document) {
if (it == sets.end()) { if (it == sets.end()) {
return; return;
} }
auto index = it->stickers.indexOf(document); const auto set = it->second.get();
auto index = set->stickers.indexOf(document);
if (index < 0) { if (index < 0) {
return; return;
} }
it->stickers.removeAt(index); set->stickers.removeAt(index);
for (auto i = it->emoji.begin(); i != it->emoji.end();) { for (auto i = set->emoji.begin(); i != set->emoji.end();) {
auto index = i->indexOf(document); auto index = i->indexOf(document);
if (index >= 0) { if (index >= 0) {
i->removeAt(index); i->removeAt(index);
if (i->empty()) { if (i->empty()) {
i = it->emoji.erase(i); i = set->emoji.erase(i);
continue; continue;
} }
} }
++i; ++i;
} }
if (it->stickers.empty()) { if (set->stickers.empty()) {
sets.erase(it); sets.erase(it);
} }
Local::writeFavedStickers(); Local::writeFavedStickers();
@ -369,14 +380,14 @@ void SetsReceived(const QVector<MTPStickerSet> &data, int32 hash) {
auto &sets = Auth().data().stickerSetsRef(); auto &sets = Auth().data().stickerSetsRef();
QMap<uint64, uint64> setsToRequest; QMap<uint64, uint64> setsToRequest;
for (auto &set : sets) { for (auto &[id, set] : sets) {
if (!(set.flags & MTPDstickerSet::Flag::f_archived)) { if (!(set->flags & MTPDstickerSet::Flag::f_archived)) {
// Mark for removing. // Mark for removing.
set.flags &= ~MTPDstickerSet::Flag::f_installed_date; set->flags &= ~MTPDstickerSet::Flag::f_installed_date;
set.installDate = 0; set->installDate = 0;
} }
} }
for_const (auto &setData, data) { for (const auto &setData : data) {
if (setData.type() == mtpc_stickerSet) { if (setData.type() == mtpc_stickerSet) {
auto set = FeedSet(setData.c_stickerSet()); auto set = FeedSet(setData.c_stickerSet());
if (!(set->flags & MTPDstickerSet::Flag::f_archived) || (set->flags & MTPDstickerSet::Flag::f_official)) { if (!(set->flags & MTPDstickerSet::Flag::f_archived) || (set->flags & MTPDstickerSet::Flag::f_official)) {
@ -390,13 +401,14 @@ void SetsReceived(const QVector<MTPStickerSet> &data, int32 hash) {
auto writeRecent = false; auto writeRecent = false;
auto &recent = GetRecentPack(); auto &recent = GetRecentPack();
for (auto it = sets.begin(), e = sets.end(); it != e;) { for (auto it = sets.begin(), e = sets.end(); it != e;) {
bool installed = (it->flags & MTPDstickerSet::Flag::f_installed_date); const auto set = it->second.get();
bool featured = (it->flags & MTPDstickerSet_ClientFlag::f_featured); bool installed = (set->flags & MTPDstickerSet::Flag::f_installed_date);
bool special = (it->flags & MTPDstickerSet_ClientFlag::f_special); bool featured = (set->flags & MTPDstickerSet_ClientFlag::f_featured);
bool archived = (it->flags & MTPDstickerSet::Flag::f_archived); bool special = (set->flags & MTPDstickerSet_ClientFlag::f_special);
bool archived = (set->flags & MTPDstickerSet::Flag::f_archived);
if (!installed) { // remove not mine sets from recent stickers if (!installed) { // remove not mine sets from recent stickers
for (auto i = recent.begin(); i != recent.cend();) { for (auto i = recent.begin(); i != recent.cend();) {
if (it->stickers.indexOf(i->first) >= 0) { if (set->stickers.indexOf(i->first) >= 0) {
i = recent.erase(i); i = recent.erase(i);
writeRecent = true; writeRecent = true;
} else { } else {
@ -437,7 +449,7 @@ void SetPackAndEmoji(
set.stickers = std::move(pack); set.stickers = std::move(pack);
set.dates = std::move(dates); set.dates = std::move(dates);
set.emoji.clear(); set.emoji.clear();
for_const (auto &mtpPack, packs) { for (const auto &mtpPack : packs) {
Assert(mtpPack.type() == mtpc_stickerPack); Assert(mtpPack.type() == mtpc_stickerPack);
auto &pack = mtpPack.c_stickerPack(); auto &pack = mtpPack.c_stickerPack();
if (auto emoji = Ui::Emoji::Find(qs(pack.vemoticon()))) { if (auto emoji = Ui::Emoji::Find(qs(pack.vemoticon()))) {
@ -473,7 +485,8 @@ void SpecialSetReceived(
} }
} else { } else {
if (it == sets.cend()) { if (it == sets.cend()) {
it = sets.insert(setId, Set( it = sets.emplace(setId, std::make_unique<Set>(
&Auth().data(),
setId, setId,
uint64(0), uint64(0),
setTitle, setTitle,
@ -481,19 +494,19 @@ void SpecialSetReceived(
0, // count 0, // count
0, // hash 0, // hash
MTPDstickerSet_ClientFlag::f_special | 0, MTPDstickerSet_ClientFlag::f_special | 0,
TimeId(0), TimeId(0))).first;
ImagePtr()));
} else { } else {
it->title = setTitle; it->second->title = setTitle;
} }
it->hash = hash; const auto set = it->second.get();
set->hash = hash;
auto dates = std::vector<TimeId>(); auto dates = std::vector<TimeId>();
auto dateIndex = 0; auto dateIndex = 0;
auto datesAvailable = (items.size() == usageDates.size()) auto datesAvailable = (items.size() == usageDates.size())
&& (setId == CloudRecentSetId); && (setId == CloudRecentSetId);
auto custom = sets.find(CustomSetId); auto customIt = sets.find(CustomSetId);
auto pack = Pack(); auto pack = Pack();
pack.reserve(items.size()); pack.reserve(items.size());
for (const auto &item : items) { for (const auto &item : items) {
@ -507,22 +520,24 @@ void SpecialSetReceived(
if (datesAvailable) { if (datesAvailable) {
dates.push_back(TimeId(usageDates[dateIndex - 1].v)); dates.push_back(TimeId(usageDates[dateIndex - 1].v));
} }
if (custom != sets.cend()) { if (customIt != sets.cend()) {
const auto custom = customIt->second.get();
auto index = custom->stickers.indexOf(document); auto index = custom->stickers.indexOf(document);
if (index >= 0) { if (index >= 0) {
custom->stickers.removeAt(index); custom->stickers.removeAt(index);
} }
} }
} }
if (custom != sets.cend() && custom->stickers.isEmpty()) { if (customIt != sets.cend()
sets.erase(custom); && customIt->second->stickers.isEmpty()) {
custom = sets.end(); sets.erase(customIt);
customIt = sets.end();
} }
auto writeRecent = false; auto writeRecent = false;
auto &recent = GetRecentPack(); auto &recent = GetRecentPack();
for (auto i = recent.begin(); i != recent.cend();) { for (auto i = recent.begin(); i != recent.cend();) {
if (it->stickers.indexOf(i->first) >= 0 && pack.indexOf(i->first) < 0) { if (set->stickers.indexOf(i->first) >= 0 && pack.indexOf(i->first) < 0) {
i = recent.erase(i); i = recent.erase(i);
writeRecent = true; writeRecent = true;
} else { } else {
@ -533,7 +548,7 @@ void SpecialSetReceived(
if (pack.isEmpty()) { if (pack.isEmpty()) {
sets.erase(it); sets.erase(it);
} else { } else {
SetPackAndEmoji(*it, std::move(pack), std::move(dates), packs); SetPackAndEmoji(*set, std::move(pack), std::move(dates), packs);
} }
if (writeRecent) { if (writeRecent) {
@ -561,7 +576,7 @@ void SpecialSetReceived(
} }
void FeaturedSetsReceived( void FeaturedSetsReceived(
const QVector<MTPStickerSetCovered> &data, const QVector<MTPStickerSetCovered> &list,
const QVector<MTPlong> &unread, const QVector<MTPlong> &unread,
int32 hash) { int32 hash) {
auto &&unreadIds = ranges::view::all( auto &&unreadIds = ranges::view::all(
@ -579,87 +594,76 @@ void FeaturedSetsReceived(
auto &sets = Auth().data().stickerSetsRef(); auto &sets = Auth().data().stickerSetsRef();
auto setsToRequest = base::flat_map<uint64, uint64>(); auto setsToRequest = base::flat_map<uint64, uint64>();
for (auto &set : sets) { for (auto &[id, set] : sets) {
// Mark for removing. // Mark for removing.
set.flags &= ~MTPDstickerSet_ClientFlag::f_featured; set->flags &= ~MTPDstickerSet_ClientFlag::f_featured;
} }
for (int i = 0, l = data.size(); i != l; ++i) { for (const auto &entry : list) {
auto &setData = data[i]; const auto data = entry.match([&](const auto &data) {
const MTPDstickerSet *set = nullptr; return data.vset().match([&](const MTPDstickerSet &data) {
switch (setData.type()) { return &data;
case mtpc_stickerSetCovered: { });
auto &d = setData.c_stickerSetCovered(); });
if (d.vset().type() == mtpc_stickerSet) { auto it = sets.find(data->vid().v);
set = &d.vset().c_stickerSet(); const auto title = GetSetTitle(*data);
const auto installDate = data->vinstalled_date().value_or_empty();
const auto thumb = data->vthumb();
if (it == sets.cend()) {
auto setClientFlags = MTPDstickerSet_ClientFlag::f_featured
| MTPDstickerSet_ClientFlag::f_not_loaded;
if (unreadMap.contains(data->vid().v)) {
setClientFlags |= MTPDstickerSet_ClientFlag::f_unread;
} }
} break; it = sets.emplace(data->vid().v, std::make_unique<Set>(
case mtpc_stickerSetMultiCovered: { &Auth().data(),
auto &d = setData.c_stickerSetMultiCovered(); data->vid().v,
if (d.vset().type() == mtpc_stickerSet) { data->vaccess_hash().v,
set = &d.vset().c_stickerSet(); title,
} qs(data->vshort_name()),
} break; data->vcount().v,
} data->vhash().v,
data->vflags().v | setClientFlags,
if (set) { installDate)).first;
auto it = sets.find(set->vid().v); it->second->setThumbnail(
const auto title = GetSetTitle(*set); Images::FromPhotoSize(&Auth(), *data, *thumb));
const auto installDate = set->vinstalled_date().value_or_empty(); } else {
const auto thumb = set->vthumb(); const auto set = it->second.get();
const auto thumbnail = thumb set->access = data->vaccess_hash().v;
? Images::Create(*set, *thumb) set->title = title;
: ImagePtr(); set->shortName = qs(data->vshort_name());
if (it == sets.cend()) { auto clientFlags = set->flags & (MTPDstickerSet_ClientFlag::f_featured | MTPDstickerSet_ClientFlag::f_unread | MTPDstickerSet_ClientFlag::f_not_loaded | MTPDstickerSet_ClientFlag::f_special);
auto setClientFlags = MTPDstickerSet_ClientFlag::f_featured set->flags = data->vflags().v | clientFlags;
| MTPDstickerSet_ClientFlag::f_not_loaded; set->flags |= MTPDstickerSet_ClientFlag::f_featured;
if (unreadMap.contains(set->vid().v)) { set->installDate = installDate;
setClientFlags |= MTPDstickerSet_ClientFlag::f_unread; set->setThumbnail(
} Images::FromPhotoSize(&Auth(), *data, *thumb));
it = sets.insert(set->vid().v, Set( if (unreadMap.contains(set->id)) {
set->vid().v, set->flags |= MTPDstickerSet_ClientFlag::f_unread;
set->vaccess_hash().v,
title,
qs(set->vshort_name()),
set->vcount().v,
set->vhash().v,
set->vflags().v | setClientFlags,
installDate,
thumbnail));
} else { } else {
it->access = set->vaccess_hash().v; set->flags &= ~MTPDstickerSet_ClientFlag::f_unread;
it->title = title;
it->shortName = qs(set->vshort_name());
auto clientFlags = it->flags & (MTPDstickerSet_ClientFlag::f_featured | MTPDstickerSet_ClientFlag::f_unread | MTPDstickerSet_ClientFlag::f_not_loaded | MTPDstickerSet_ClientFlag::f_special);
it->flags = set->vflags().v | clientFlags;
it->flags |= MTPDstickerSet_ClientFlag::f_featured;
it->installDate = installDate;
it->thumbnail = thumbnail;
if (unreadMap.contains(it->id)) {
it->flags |= MTPDstickerSet_ClientFlag::f_unread;
} else {
it->flags &= ~MTPDstickerSet_ClientFlag::f_unread;
}
if (it->count != set->vcount().v || it->hash != set->vhash().v || it->emoji.isEmpty()) {
it->count = set->vcount().v;
it->hash = set->vhash().v;
it->flags |= MTPDstickerSet_ClientFlag::f_not_loaded; // need to request this set
}
} }
setsOrder.push_back(set->vid().v); if (set->count != data->vcount().v || set->hash != data->vhash().v || set->emoji.isEmpty()) {
if (it->stickers.isEmpty() || (it->flags & MTPDstickerSet_ClientFlag::f_not_loaded)) { set->count = data->vcount().v;
setsToRequest.emplace(set->vid().v, set->vaccess_hash().v); set->hash = data->vhash().v;
set->flags |= MTPDstickerSet_ClientFlag::f_not_loaded; // need to request this set
} }
} }
setsOrder.push_back(data->vid().v);
if (it->second->stickers.isEmpty()
|| (it->second->flags & MTPDstickerSet_ClientFlag::f_not_loaded)) {
setsToRequest.emplace(data->vid().v, data->vaccess_hash().v);
}
} }
auto unreadCount = 0; auto unreadCount = 0;
for (auto it = sets.begin(), e = sets.end(); it != e;) { for (auto it = sets.begin(), e = sets.end(); it != e;) {
bool installed = (it->flags & MTPDstickerSet::Flag::f_installed_date); const auto set = it->second.get();
bool featured = (it->flags & MTPDstickerSet_ClientFlag::f_featured); bool installed = (set->flags & MTPDstickerSet::Flag::f_installed_date);
bool special = (it->flags & MTPDstickerSet_ClientFlag::f_special); bool featured = (set->flags & MTPDstickerSet_ClientFlag::f_featured);
bool archived = (it->flags & MTPDstickerSet::Flag::f_archived); bool special = (set->flags & MTPDstickerSet_ClientFlag::f_special);
bool archived = (set->flags & MTPDstickerSet::Flag::f_archived);
if (installed || featured || special || archived) { if (installed || featured || special || archived) {
if (featured && (it->flags & MTPDstickerSet_ClientFlag::f_unread)) { if (featured && (set->flags & MTPDstickerSet_ClientFlag::f_unread)) {
++unreadCount; ++unreadCount;
} }
++it; ++it;
@ -773,7 +777,7 @@ std::vector<not_null<DocumentData*>> GetListByEmoji(
const auto setId = sticker->set.c_inputStickerSetID().vid().v; const auto setId = sticker->set.c_inputStickerSetID().vid().v;
const auto setIt = sets.find(setId); const auto setIt = sets.find(setId);
if (setIt != sets.end()) { if (setIt != sets.end()) {
return InstallDateAdjusted(setIt->installDate, document); return InstallDateAdjusted(setIt->second->installDate, document);
} }
} }
return TimeId(0); return TimeId(0);
@ -781,20 +785,21 @@ std::vector<not_null<DocumentData*>> GetListByEmoji(
auto recentIt = sets.find(Stickers::CloudRecentSetId); auto recentIt = sets.find(Stickers::CloudRecentSetId);
if (recentIt != sets.cend()) { if (recentIt != sets.cend()) {
auto i = recentIt->emoji.constFind(original); const auto recent = recentIt->second.get();
if (i != recentIt->emoji.cend()) { auto i = recent->emoji.constFind(original);
if (i != recent->emoji.cend()) {
result.reserve(i->size()); result.reserve(i->size());
for (const auto document : *i) { for (const auto document : *i) {
const auto usageDate = [&] { const auto usageDate = [&] {
if (recentIt->dates.empty()) { if (recent->dates.empty()) {
return TimeId(0); return TimeId(0);
} }
const auto index = recentIt->stickers.indexOf(document); const auto index = recent->stickers.indexOf(document);
if (index < 0) { if (index < 0) {
return TimeId(0); return TimeId(0);
} }
Assert(index < recentIt->dates.size()); Assert(index < recent->dates.size());
return recentIt->dates[index]; return recent->dates[index];
}(); }();
const auto date = usageDate const auto date = usageDate
? usageDate ? usageDate
@ -808,22 +813,23 @@ std::vector<not_null<DocumentData*>> GetListByEmoji(
const auto addList = [&](const Order &order, MTPDstickerSet::Flag skip) { const auto addList = [&](const Order &order, MTPDstickerSet::Flag skip) {
for (const auto setId : order) { for (const auto setId : order) {
auto it = sets.find(setId); auto it = sets.find(setId);
if (it == sets.cend() || (it->flags & skip)) { if (it == sets.cend() || (it->second->flags & skip)) {
continue; continue;
} }
if (it->emoji.isEmpty()) { const auto set = it->second.get();
setsToRequest.emplace(it->id, it->access); if (set->emoji.isEmpty()) {
it->flags |= MTPDstickerSet_ClientFlag::f_not_loaded; setsToRequest.emplace(set->id, set->access);
set->flags |= MTPDstickerSet_ClientFlag::f_not_loaded;
continue; continue;
} }
auto i = it->emoji.constFind(original); auto i = set->emoji.constFind(original);
if (i == it->emoji.cend()) { if (i == set->emoji.cend()) {
continue; continue;
} }
const auto my = (it->flags & MTPDstickerSet::Flag::f_installed_date); const auto my = (set->flags & MTPDstickerSet::Flag::f_installed_date);
result.reserve(result.size() + i->size()); result.reserve(result.size() + i->size());
for (const auto document : *i) { for (const auto document : *i) {
const auto installDate = my ? it->installDate : TimeId(0); const auto installDate = my ? set->installDate : TimeId(0);
const auto date = (installDate > 1) const auto date = (installDate > 1)
? InstallDateAdjusted(installDate, document) ? InstallDateAdjusted(installDate, document)
: my : my
@ -879,12 +885,13 @@ std::optional<std::vector<not_null<EmojiPtr>>> GetEmojiListFromSet(
return std::nullopt; return std::nullopt;
} }
auto &sets = Auth().data().stickerSets(); auto &sets = Auth().data().stickerSets();
auto it = sets.constFind(inputSet.c_inputStickerSetID().vid().v); auto it = sets.find(inputSet.c_inputStickerSetID().vid().v);
if (it == sets.cend()) { if (it == sets.cend()) {
return std::nullopt; return std::nullopt;
} }
const auto set = it->second.get();
auto result = std::vector<not_null<EmojiPtr>>(); auto result = std::vector<not_null<EmojiPtr>>();
for (auto i = it->emoji.cbegin(), e = it->emoji.cend(); i != e; ++i) { for (auto i = set->emoji.cbegin(), e = set->emoji.cend(); i != e; ++i) {
if (i->contains(document)) { if (i->contains(document)) {
result.emplace_back(i.key()); result.emplace_back(i.key());
} }
@ -897,61 +904,65 @@ std::optional<std::vector<not_null<EmojiPtr>>> GetEmojiListFromSet(
return std::nullopt; return std::nullopt;
} }
Set *FeedSet(const MTPDstickerSet &set) { Set *FeedSet(const MTPDstickerSet &data) {
auto &sets = Auth().data().stickerSetsRef(); auto &sets = Auth().data().stickerSetsRef();
auto it = sets.find(set.vid().v); auto it = sets.find(data.vid().v);
auto title = GetSetTitle(set); auto title = GetSetTitle(data);
auto flags = MTPDstickerSet::Flags(0); auto flags = MTPDstickerSet::Flags(0);
const auto thumb = set.vthumb(); const auto thumb = data.vthumb();
const auto thumbnail = thumb ? Images::Create(set, *thumb) : ImagePtr();
if (it == sets.cend()) { if (it == sets.cend()) {
it = sets.insert(set.vid().v, Stickers::Set( it = sets.emplace(data.vid().v, std::make_unique<Set>(
set.vid().v, &Auth().data(),
set.vaccess_hash().v, data.vid().v,
data.vaccess_hash().v,
title, title,
qs(set.vshort_name()), qs(data.vshort_name()),
set.vcount().v, data.vcount().v,
set.vhash().v, data.vhash().v,
set.vflags().v | MTPDstickerSet_ClientFlag::f_not_loaded, data.vflags().v | MTPDstickerSet_ClientFlag::f_not_loaded,
set.vinstalled_date().value_or_empty(), data.vinstalled_date().value_or_empty())).first;
thumbnail)); it->second->setThumbnail(
Images::FromPhotoSize(&Auth(), data, *thumb));
} else { } else {
it->access = set.vaccess_hash().v; const auto set = it->second.get();
it->title = title; set->access = data.vaccess_hash().v;
it->shortName = qs(set.vshort_name()); set->title = title;
flags = it->flags; set->shortName = qs(data.vshort_name());
auto clientFlags = it->flags flags = set->flags;
auto clientFlags = set->flags
& (MTPDstickerSet_ClientFlag::f_featured & (MTPDstickerSet_ClientFlag::f_featured
| MTPDstickerSet_ClientFlag::f_unread | MTPDstickerSet_ClientFlag::f_unread
| MTPDstickerSet_ClientFlag::f_not_loaded | MTPDstickerSet_ClientFlag::f_not_loaded
| MTPDstickerSet_ClientFlag::f_special); | MTPDstickerSet_ClientFlag::f_special);
it->flags = set.vflags().v | clientFlags; set->flags = data.vflags().v | clientFlags;
const auto installDate = set.vinstalled_date(); const auto installDate = data.vinstalled_date();
it->installDate = installDate set->installDate = installDate
? (installDate->v ? installDate->v : base::unixtime::now()) ? (installDate->v ? installDate->v : base::unixtime::now())
: TimeId(0); : TimeId(0);
it->thumbnail = thumbnail; it->second->setThumbnail(
if (it->count != set.vcount().v Images::FromPhotoSize(&Auth(), data, *thumb));
|| it->hash != set.vhash().v if (set->count != data.vcount().v
|| it->emoji.isEmpty()) { || set->hash != data.vhash().v
// Need to request this set. || set->emoji.isEmpty()) {
it->count = set.vcount().v; // Need to request this data.
it->hash = set.vhash().v; set->count = data.vcount().v;
it->flags |= MTPDstickerSet_ClientFlag::f_not_loaded; set->hash = data.vhash().v;
set->flags |= MTPDstickerSet_ClientFlag::f_not_loaded;
} }
} }
auto changedFlags = (flags ^ it->flags); const auto set = it->second.get();
auto changedFlags = (flags ^ set->flags);
if (changedFlags & MTPDstickerSet::Flag::f_archived) { if (changedFlags & MTPDstickerSet::Flag::f_archived) {
auto index = Auth().data().archivedStickerSetsOrder().indexOf(it->id); auto index = Auth().data().archivedStickerSetsOrder().indexOf(set->id);
if (it->flags & MTPDstickerSet::Flag::f_archived) { if (set->flags & MTPDstickerSet::Flag::f_archived) {
if (index < 0) { if (index < 0) {
Auth().data().archivedStickerSetsOrderRef().push_front(it->id); Auth().data().archivedStickerSetsOrderRef().push_front(set->id);
} }
} else if (index >= 0) { } else if (index >= 0) {
Auth().data().archivedStickerSetsOrderRef().removeAt(index); Auth().data().archivedStickerSetsOrderRef().removeAt(index);
} }
} }
return &it.value(); return it->second.get();
} }
Set *FeedSetFull(const MTPmessages_StickerSet &data) { Set *FeedSetFull(const MTPmessages_StickerSet &data) {
@ -963,14 +974,14 @@ Set *FeedSetFull(const MTPmessages_StickerSet &data) {
auto &sets = Auth().data().stickerSetsRef(); auto &sets = Auth().data().stickerSetsRef();
auto it = sets.find(s.vid().v); auto it = sets.find(s.vid().v);
const auto wasArchived = (it->flags & MTPDstickerSet::Flag::f_archived); const auto wasArchived = (it->second->flags & MTPDstickerSet::Flag::f_archived);
auto set = FeedSet(s); auto set = FeedSet(s);
set->flags &= ~MTPDstickerSet_ClientFlag::f_not_loaded; set->flags &= ~MTPDstickerSet_ClientFlag::f_not_loaded;
auto &d_docs = d.vdocuments().v; auto &d_docs = d.vdocuments().v;
auto custom = sets.find(Stickers::CustomSetId); auto customIt = sets.find(Stickers::CustomSetId);
auto inputSet = MTP_inputStickerSetID( auto inputSet = MTP_inputStickerSetID(
MTP_long(set->id), MTP_long(set->id),
MTP_long(set->access)); MTP_long(set->access));
@ -985,16 +996,17 @@ Set *FeedSetFull(const MTPmessages_StickerSet &data) {
if (document->sticker()->set.type() != mtpc_inputStickerSetID) { if (document->sticker()->set.type() != mtpc_inputStickerSetID) {
document->sticker()->set = inputSet; document->sticker()->set = inputSet;
} }
if (custom != sets.cend()) { if (customIt != sets.cend()) {
const auto custom = customIt->second.get();
const auto index = custom->stickers.indexOf(document); const auto index = custom->stickers.indexOf(document);
if (index >= 0) { if (index >= 0) {
custom->stickers.removeAt(index); custom->stickers.removeAt(index);
} }
} }
} }
if (custom != sets.cend() && custom->stickers.isEmpty()) { if (customIt != sets.cend() && customIt->second->stickers.isEmpty()) {
sets.erase(custom); sets.erase(customIt);
custom = sets.end(); customIt = sets.end();
} }
auto writeRecent = false; auto writeRecent = false;
@ -1276,31 +1288,4 @@ std::unique_ptr<Lottie::SinglePlayer> LottieThumbnail(
box); box);
} }
ThumbnailSource::ThumbnailSource(
const StorageImageLocation &location,
int size)
: StorageSource(location, size) {
}
QImage ThumbnailSource::takeLoaded() {
const auto loader = currentLoader();
if (_bytesForAnimated.isEmpty()
&& loader
&& loader->finished()
&& !loader->cancelled()) {
_bytesForAnimated = loader->bytes();
}
auto result = StorageSource::takeLoaded();
if (!_bytesForAnimated.isEmpty()
&& !result.isNull()
&& result.size() != Image::Empty()->original().size()) {
_bytesForAnimated = QByteArray();
}
return result;
}
QByteArray ThumbnailSource::bytesForCache() {
return _bytesForAnimated;
}
} // namespace Stickers } // namespace Stickers

View File

@ -8,7 +8,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#pragma once #pragma once
#include "mtproto/sender.h" #include "mtproto/sender.h"
#include "ui/image/image_source.h"
class DocumentData; class DocumentData;
@ -46,53 +45,7 @@ constexpr auto FeaturedSetId = 0xFFFFFFFFFFFFFFFBULL; // for emoji/stickers pane
constexpr auto FavedSetId = 0xFFFFFFFFFFFFFFFAULL; // for cloud-stored faved stickers constexpr auto FavedSetId = 0xFFFFFFFFFFFFFFFAULL; // for cloud-stored faved stickers
constexpr auto MegagroupSetId = 0xFFFFFFFFFFFFFFEFULL; // for setting up megagroup sticker set constexpr auto MegagroupSetId = 0xFFFFFFFFFFFFFFEFULL; // for setting up megagroup sticker set
using Order = QList<uint64>; class Set;
using SavedGifs = QVector<DocumentData*>;
using Pack = QVector<DocumentData*>;
using ByEmojiMap = QMap<EmojiPtr, Pack>;
struct Set {
Set(
uint64 id,
uint64 access,
const QString &title,
const QString &shortName,
int count,
int32 hash,
MTPDstickerSet::Flags flags,
TimeId installDate,
ImagePtr thumbnail)
: id(id)
, access(access)
, title(title)
, shortName(shortName)
, count(count)
, hash(hash)
, flags(flags)
, installDate(installDate)
, thumbnail(thumbnail) {
}
uint64 id = 0;
uint64 access = 0;
QString title, shortName;
int count = 0;
int32 hash = 0;
MTPDstickerSet::Flags flags;
TimeId installDate = 0;
ImagePtr thumbnail;
Pack stickers;
std::vector<TimeId> dates;
Pack covers;
ByEmojiMap emoji;
};
using Sets = QMap<uint64, Set>;
inline MTPInputStickerSet inputSetId(const Set &set) {
if (set.id && set.access) {
return MTP_inputStickerSetID(MTP_long(set.id), MTP_long(set.access));
}
return MTP_inputStickerSetShortName(MTP_string(set.shortName));
}
void ApplyArchivedResult(const MTPDmessages_stickerSetInstallResultArchive &d); void ApplyArchivedResult(const MTPDmessages_stickerSetInstallResultArchive &d);
bool ApplyArchivedResultFake(); // For testing. bool ApplyArchivedResultFake(); // For testing.
@ -110,7 +63,7 @@ void SpecialSetReceived(
const QVector<MTPStickerPack> &packs = QVector<MTPStickerPack>(), const QVector<MTPStickerPack> &packs = QVector<MTPStickerPack>(),
const QVector<MTPint> &usageDates = QVector<MTPint>()); const QVector<MTPint> &usageDates = QVector<MTPint>());
void FeaturedSetsReceived( void FeaturedSetsReceived(
const QVector<MTPStickerSetCovered> &data, const QVector<MTPStickerSetCovered> &list,
const QVector<MTPlong> &unread, const QVector<MTPlong> &unread,
int32 hash); int32 hash);
void GifsReceived(const QVector<MTPDocument> &items, int32 hash); void GifsReceived(const QVector<MTPDocument> &items, int32 hash);
@ -168,19 +121,4 @@ enum class LottieSize : uchar {
QSize box, QSize box,
std::shared_ptr<Lottie::FrameRenderer> renderer = nullptr); std::shared_ptr<Lottie::FrameRenderer> renderer = nullptr);
class ThumbnailSource : public Images::StorageSource {
public:
ThumbnailSource(
const StorageImageLocation &location,
int size);
QImage takeLoaded() override;
QByteArray bytesForCache() override;
private:
QByteArray _bytesForAnimated;
};
} // namespace Stickers } // namespace Stickers

View File

@ -184,7 +184,8 @@ private:
}; };
auto StickersListWidget::PrepareStickers(const Stickers::Pack &pack) auto StickersListWidget::PrepareStickers(
const QVector<DocumentData*> &pack)
-> std::vector<Sticker> { -> std::vector<Sticker> {
return ranges::view::all( return ranges::view::all(
pack pack
@ -1288,7 +1289,7 @@ void StickersListWidget::fillLocalSearchRows(const QString &query) {
for (const auto &[setId, titleWords] : _searchIndex) { for (const auto &[setId, titleWords] : _searchIndex) {
if (allSearchWordsInTitle(titleWords)) { if (allSearchWordsInTitle(titleWords)) {
if (const auto it = sets.find(setId); it != sets.end()) { if (const auto it = sets.find(setId); it != sets.end()) {
addSearchRow(&*it); addSearchRow(it->second.get());
} }
} }
} }
@ -1299,7 +1300,7 @@ void StickersListWidget::fillCloudSearchRows(
const auto &sets = session().data().stickerSets(); const auto &sets = session().data().stickerSets();
for (const auto setId : cloudSets) { for (const auto setId : cloudSets) {
if (const auto it = sets.find(setId); it != sets.end()) { if (const auto it = sets.find(setId); it != sets.end()) {
addSearchRow(&*it); addSearchRow(it->second.get());
} }
} }
} }
@ -1310,7 +1311,7 @@ void StickersListWidget::addSearchRow(not_null<const Stickers::Set*> set) {
set->flags, set->flags,
set->title, set->title,
set->shortName, set->shortName,
set->thumbnail, ImagePtr(),// set->thumbnail, #TODO optimize stickers
set->count, set->count,
!SetInMyList(set->flags), !SetInMyList(set->flags),
PrepareStickers(set->stickers.empty() PrepareStickers(set->stickers.empty()
@ -2123,10 +2124,11 @@ void StickersListWidget::removeRecentSticker(int section, int index) {
auto &sets = session().data().stickerSetsRef(); auto &sets = session().data().stickerSetsRef();
auto it = sets.find(Stickers::CustomSetId); auto it = sets.find(Stickers::CustomSetId);
if (it != sets.cend()) { if (it != sets.cend()) {
for (int i = 0, l = it->stickers.size(); i < l; ++i) { const auto set = it->second.get();
if (it->stickers.at(i) == document) { for (int i = 0, l = set->stickers.size(); i < l; ++i) {
it->stickers.removeAt(i); if (set->stickers.at(i) == document) {
if (it->stickers.isEmpty()) { set->stickers.removeAt(i);
if (set->stickers.isEmpty()) {
sets.erase(it); sets.erase(it);
} }
Local::writeInstalledStickers(); Local::writeInstalledStickers();
@ -2295,14 +2297,14 @@ void StickersListWidget::refreshFeaturedSets() {
const auto till = end(wereOfficial); const auto till = end(wereOfficial);
for (auto i = from; i != till; ++i) { for (auto i = from; i != till; ++i) {
auto &set = *i; auto &set = *i;
auto it = sets.constFind(set.id); auto it = sets.find(set.id);
if (it == sets.cend() if (it == sets.cend()
|| ((it->flags & MTPDstickerSet::Flag::f_installed_date) || ((it->second->flags & MTPDstickerSet::Flag::f_installed_date)
&& !(it->flags & MTPDstickerSet::Flag::f_archived) && !(it->second->flags & MTPDstickerSet::Flag::f_archived)
&& !_installedLocallySets.contains(set.id))) { && !_installedLocallySets.contains(set.id))) {
continue; continue;
} }
set.flags = it->flags; set.flags = it->second->flags;
_officialSets.push_back(std::move(set)); _officialSets.push_back(std::move(set));
} }
} }
@ -2312,16 +2314,17 @@ void StickersListWidget::refreshSearchSets() {
refreshSearchIndex(); refreshSearchIndex();
const auto &sets = session().data().stickerSets(); const auto &sets = session().data().stickerSets();
for (auto &set : _searchSets) { for (auto &entry : _searchSets) {
if (const auto it = sets.find(set.id); it != sets.end()) { if (const auto it = sets.find(entry.id); it != sets.end()) {
set.flags = it->flags; const auto set = it->second.get();
if (!it->stickers.empty()) { entry.flags = set->flags;
set.lottiePlayer = nullptr; if (!set->stickers.empty()) {
set.stickers = PrepareStickers(it->stickers); entry.lottiePlayer = nullptr;
entry.stickers = PrepareStickers(set->stickers);
} }
if (!SetInMyList(set.flags)) { if (!SetInMyList(entry.flags)) {
_installedLocallySets.remove(set.id); _installedLocallySets.remove(entry.id);
set.externalLayout = true; entry.externalLayout = true;
} }
} }
} }
@ -2373,32 +2376,34 @@ bool StickersListWidget::appendSet(
bool externalLayout, bool externalLayout,
AppendSkip skip) { AppendSkip skip) {
auto &sets = session().data().stickerSets(); auto &sets = session().data().stickerSets();
auto it = sets.constFind(setId); auto it = sets.find(setId);
if (it == sets.cend() || (!externalLayout && it->stickers.isEmpty())) { if (it == sets.cend()
|| (!externalLayout && it->second->stickers.isEmpty())) {
return false; return false;
} }
const auto set = it->second.get();
if ((skip == AppendSkip::Archived) if ((skip == AppendSkip::Archived)
&& (it->flags & MTPDstickerSet::Flag::f_archived)) { && (set->flags & MTPDstickerSet::Flag::f_archived)) {
return false; return false;
} }
if ((skip == AppendSkip::Installed) if ((skip == AppendSkip::Installed)
&& (it->flags & MTPDstickerSet::Flag::f_installed_date) && (set->flags & MTPDstickerSet::Flag::f_installed_date)
&& !(it->flags & MTPDstickerSet::Flag::f_archived)) { && !(set->flags & MTPDstickerSet::Flag::f_archived)) {
if (!_installedLocallySets.contains(setId)) { if (!_installedLocallySets.contains(setId)) {
return false; return false;
} }
} }
to.emplace_back( to.emplace_back(
it->id, set->id,
it->flags, set->flags,
it->title, set->title,
it->shortName, set->shortName,
it->thumbnail, ImagePtr(), // set->thumbnail, #TODO optimize stickers
it->count, set->count,
externalLayout, externalLayout,
PrepareStickers((it->stickers.empty() && externalLayout) PrepareStickers((set->stickers.empty() && externalLayout)
? it->covers ? set->covers
: it->stickers)); : set->stickers));
return true; return true;
} }
@ -2417,13 +2422,13 @@ auto StickersListWidget::collectRecentStickers() -> std::vector<Sticker> {
const auto &sets = session().data().stickerSets(); const auto &sets = session().data().stickerSets();
const auto &recent = Stickers::GetRecentPack(); const auto &recent = Stickers::GetRecentPack();
const auto customIt = sets.constFind(Stickers::CustomSetId); const auto customIt = sets.find(Stickers::CustomSetId);
const auto cloudIt = sets.constFind(Stickers::CloudRecentSetId); const auto cloudIt = sets.find(Stickers::CloudRecentSetId);
const auto customCount = (customIt != sets.cend()) const auto customCount = (customIt != sets.cend())
? customIt->stickers.size() ? customIt->second->stickers.size()
: 0; : 0;
const auto cloudCount = (cloudIt != sets.cend()) const auto cloudCount = (cloudIt != sets.cend())
? cloudIt->stickers.size() ? cloudIt->second->stickers.size()
: 0; : 0;
result.reserve(cloudCount + recent.size() + customCount); result.reserve(cloudCount + recent.size() + customCount);
_custom.reserve(cloudCount + recent.size() + customCount); _custom.reserve(cloudCount + recent.size() + customCount);
@ -2448,7 +2453,7 @@ auto StickersListWidget::collectRecentStickers() -> std::vector<Sticker> {
}; };
if (cloudCount > 0) { if (cloudCount > 0) {
for (const auto document : cloudIt->stickers) { for (const auto document : cloudIt->second->stickers) {
add(document, false); add(document, false);
} }
} }
@ -2456,7 +2461,7 @@ auto StickersListWidget::collectRecentStickers() -> std::vector<Sticker> {
add(recentSticker.first, false); add(recentSticker.first, false);
} }
if (customCount > 0) { if (customCount > 0) {
for (const auto document : customIt->stickers) { for (const auto document : customIt->second->stickers) {
add(document, true); add(document, true);
} }
} }
@ -2502,11 +2507,12 @@ void StickersListWidget::refreshRecentStickers(bool performResize) {
void StickersListWidget::refreshFavedStickers() { void StickersListWidget::refreshFavedStickers() {
clearSelection(); clearSelection();
auto &sets = session().data().stickerSets(); const auto &sets = session().data().stickerSets();
auto it = sets.constFind(Stickers::FavedSetId); const auto it = sets.find(Stickers::FavedSetId);
if (it == sets.cend() || it->stickers.isEmpty()) { if (it == sets.cend() || it->second->stickers.isEmpty()) {
return; return;
} }
const auto set = it->second.get();
const auto externalLayout = false; const auto externalLayout = false;
const auto shortName = QString(); const auto shortName = QString();
const auto thumbnail = ImagePtr(); const auto thumbnail = ImagePtr();
@ -2517,12 +2523,12 @@ void StickersListWidget::refreshFavedStickers() {
Lang::Hard::FavedSetTitle(), Lang::Hard::FavedSetTitle(),
shortName, shortName,
thumbnail, thumbnail,
it->count, set->count,
externalLayout, externalLayout,
PrepareStickers(it->stickers)); PrepareStickers(set->stickers));
_favedStickersMap = base::flat_set<not_null<DocumentData*>> { _favedStickersMap = base::flat_set<not_null<DocumentData*>> {
it->stickers.begin(), set->stickers.begin(),
it->stickers.end() set->stickers.end()
}; };
} }
@ -2569,11 +2575,12 @@ void StickersListWidget::refreshMegagroupStickers(GroupStickersPlace place) {
return; return;
} }
auto &set = _megagroupSet->mgInfo->stickerSet.c_inputStickerSetID(); auto &set = _megagroupSet->mgInfo->stickerSet.c_inputStickerSetID();
auto &sets = session().data().stickerSets(); const auto &sets = session().data().stickerSets();
auto it = sets.constFind(set.vid().v); const auto it = sets.find(set.vid().v);
if (it != sets.cend()) { if (it != sets.cend()) {
auto isInstalled = (it->flags & MTPDstickerSet::Flag::f_installed_date) const auto set = it->second.get();
&& !(it->flags & MTPDstickerSet::Flag::f_archived); auto isInstalled = (set->flags & MTPDstickerSet::Flag::f_installed_date)
&& !(set->flags & MTPDstickerSet::Flag::f_archived);
if (isInstalled && !canEdit) { if (isInstalled && !canEdit) {
removeHiddenForGroup(); removeHiddenForGroup();
} else if (isShownHere(hidden)) { } else if (isShownHere(hidden)) {
@ -2586,9 +2593,9 @@ void StickersListWidget::refreshMegagroupStickers(GroupStickersPlace place) {
tr::lng_group_stickers(tr::now), tr::lng_group_stickers(tr::now),
shortName, shortName,
thumbnail, thumbnail,
it->count, set->count,
externalLayout, externalLayout,
PrepareStickers(it->stickers)); PrepareStickers(set->stickers));
} }
return; return;
} else if (!isShownHere(hidden) } else if (!isShownHere(hidden)
@ -2909,11 +2916,11 @@ void StickersListWidget::displaySet(uint64 setId) {
} }
} }
auto &sets = session().data().stickerSets(); auto &sets = session().data().stickerSets();
auto it = sets.constFind(setId); auto it = sets.find(setId);
if (it != sets.cend()) { if (it != sets.cend()) {
_displayingSet = true; _displayingSet = true;
checkHideWithBox(Ui::show( checkHideWithBox(Ui::show(
Box<StickerSetBox>(controller(), Stickers::inputSetId(*it)), Box<StickerSetBox>(controller(), it->second->mtpInput()),
Ui::LayerOption::KeepOther).data()); Ui::LayerOption::KeepOther).data());
} }
} }
@ -2929,12 +2936,13 @@ void StickersListWidget::checkHideWithBox(QPointer<Ui::BoxContent> box) {
} }
void StickersListWidget::installSet(uint64 setId) { void StickersListWidget::installSet(uint64 setId) {
auto &sets = session().data().stickerSets(); const auto &sets = session().data().stickerSets();
auto it = sets.constFind(setId); const auto it = sets.find(setId);
if (it != sets.cend()) { if (it != sets.cend()) {
const auto input = Stickers::inputSetId(*it); const auto set = it->second.get();
if ((it->flags & MTPDstickerSet_ClientFlag::f_not_loaded) const auto input = set->mtpInput();
|| it->stickers.empty()) { if ((set->flags & MTPDstickerSet_ClientFlag::f_not_loaded)
|| set->stickers.empty()) {
_api.request(MTPmessages_GetStickerSet( _api.request(MTPmessages_GetStickerSet(
input input
)).done([=](const MTPmessages_StickerSet &result) { )).done([=](const MTPmessages_StickerSet &result) {
@ -2990,38 +2998,40 @@ void StickersListWidget::removeMegagroupSet(bool locally) {
} }
void StickersListWidget::removeSet(uint64 setId) { void StickersListWidget::removeSet(uint64 setId) {
auto &sets = session().data().stickerSets(); const auto &sets = session().data().stickerSets();
auto it = sets.constFind(setId); const auto it = sets.find(setId);
if (it != sets.cend()) { if (it != sets.cend()) {
_removingSetId = it->id; const auto set = it->second.get();
auto text = tr::lng_stickers_remove_pack(tr::now, lt_sticker_pack, it->title); _removingSetId = set->id;
auto text = tr::lng_stickers_remove_pack(tr::now, lt_sticker_pack, set->title);
Ui::show(Box<ConfirmBox>(text, tr::lng_stickers_remove_pack_confirm(tr::now), crl::guard(this, [=] { Ui::show(Box<ConfirmBox>(text, tr::lng_stickers_remove_pack_confirm(tr::now), crl::guard(this, [=] {
Ui::hideLayer(); Ui::hideLayer();
auto &sets = session().data().stickerSetsRef(); const auto &sets = session().data().stickerSetsRef();
auto it = sets.find(_removingSetId); const auto it = sets.find(_removingSetId);
if (it != sets.cend()) { if (it != sets.cend()) {
if (it->id && it->access) { const auto set = it->second.get();
_api.request(MTPmessages_UninstallStickerSet(MTP_inputStickerSetID(MTP_long(it->id), MTP_long(it->access)))).send(); if (set->id && set->access) {
} else if (!it->shortName.isEmpty()) { _api.request(MTPmessages_UninstallStickerSet(MTP_inputStickerSetID(MTP_long(set->id), MTP_long(set->access)))).send();
_api.request(MTPmessages_UninstallStickerSet(MTP_inputStickerSetShortName(MTP_string(it->shortName)))).send(); } else if (!set->shortName.isEmpty()) {
_api.request(MTPmessages_UninstallStickerSet(MTP_inputStickerSetShortName(MTP_string(set->shortName)))).send();
} }
auto writeRecent = false; auto writeRecent = false;
auto &recent = Stickers::GetRecentPack(); auto &recent = Stickers::GetRecentPack();
for (auto i = recent.begin(); i != recent.cend();) { for (auto i = recent.begin(); i != recent.cend();) {
if (it->stickers.indexOf(i->first) >= 0) { if (set->stickers.indexOf(i->first) >= 0) {
i = recent.erase(i); i = recent.erase(i);
writeRecent = true; writeRecent = true;
} else { } else {
++i; ++i;
} }
} }
it->flags &= ~MTPDstickerSet::Flag::f_installed_date; set->flags &= ~MTPDstickerSet::Flag::f_installed_date;
it->installDate = TimeId(0); set->installDate = TimeId(0);
// //
// Set can be in search results. // Set can be in search results.
// //
//if (!(it->flags & MTPDstickerSet_ClientFlag::f_featured) //if (!(set->flags & MTPDstickerSet_ClientFlag::f_featured)
// && !(it->flags & MTPDstickerSet_ClientFlag::f_special)) { // && !(set->flags & MTPDstickerSet_ClientFlag::f_special)) {
// sets.erase(it); // sets.erase(it);
//} //}
int removeIndex = session().data().stickerSetsOrder().indexOf(_removingSetId); int removeIndex = session().data().stickerSetsOrder().indexOf(_removingSetId);

View File

@ -196,7 +196,8 @@ private:
std::vector<Sticker> stickers; std::vector<Sticker> stickers;
}; };
static std::vector<Sticker> PrepareStickers(const Stickers::Pack &pack); static std::vector<Sticker> PrepareStickers(
const QVector<DocumentData*> &pack);
void preloadMoreOfficial(); void preloadMoreOfficial();
QSize boundingBoxSize() const; QSize boundingBoxSize() const;

View File

@ -0,0 +1,134 @@
/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "chat_helpers/stickers_set.h"
#include "main/main_session.h"
#include "data/data_session.h"
#include "data/data_file_origin.h"
#include "storage/file_download.h"
#include "ui/image/image_source.h"
#include "app.h"
namespace Stickers {
void SetThumbnailView::set(
not_null<Main::Session*> session,
QByteArray content) {
auto image = App::readImage(content, nullptr, false);
if (image.isNull()) {
_content = std::move(content);
} else {
_image = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(std::move(image), "PNG"));
}
session->downloaderTaskFinished().notify();
}
Image *SetThumbnailView::image() const {
return _image.get();
}
QByteArray SetThumbnailView::content() const {
return _content;
}
Set::Set(
not_null<Data::Session*> owner,
uint64 id,
uint64 access,
const QString &title,
const QString &shortName,
int count,
int32 hash,
MTPDstickerSet::Flags flags,
TimeId installDate)
: id(id)
, access(access)
, title(title)
, shortName(shortName)
, count(count)
, hash(hash)
, flags(flags)
, installDate(installDate)
, _owner(owner) {
}
MTPInputStickerSet Set::mtpInput() const {
return (id && access)
? MTP_inputStickerSetID(MTP_long(id), MTP_long(access))
: MTP_inputStickerSetShortName(MTP_string(shortName));
}
void Set::setThumbnail(const ImageWithLocation &data) {
Data::UpdateCloudFile(
_thumbnail,
data,
_owner->cache(),
Data::kImageCacheTag,
[=](Data::FileOrigin origin) { loadThumbnail(origin); });
if (!data.bytes.isEmpty()) {
if (_thumbnail.loader) {
_thumbnail.loader->cancel();
}
if (const auto view = activeThumbnailView()) {
view->set(&_owner->session(), data.bytes);
}
}
}
bool Set::hasThumbnail() const {
return _thumbnail.location.valid();
}
bool Set::thumbnailLoading() const {
return (_thumbnail.loader != nullptr);
}
bool Set::thumbnailFailed() const {
return (_thumbnail.flags & Data::CloudFile::Flag::Failed);
}
void Set::loadThumbnail(Data::FileOrigin origin) {
auto &file = _thumbnail;
const auto fromCloud = LoadFromCloudOrLocal;
const auto cacheTag = Data::kImageCacheTag;
const auto autoLoading = false;
Data::LoadCloudFile(file, origin, fromCloud, autoLoading, cacheTag, [=] {
if (const auto active = activeThumbnailView()) {
return !active->image() && active->content().isEmpty();
}
return true;
}, [=](QByteArray result) {
if (const auto active = activeThumbnailView()) {
active->set(&_owner->session(), std::move(result));
}
});
}
const ImageLocation &Set::thumbnailLocation() const {
return _thumbnail.location;
}
int Set::thumbnailByteSize() const {
return _thumbnail.byteSize;
}
std::shared_ptr<SetThumbnailView> Set::createThumbnailView() {
if (auto active = activeThumbnailView()) {
return active;
}
auto view = std::make_shared<SetThumbnailView>();
_thumbnailView = view;
return view;
}
std::shared_ptr<SetThumbnailView> Set::activeThumbnailView() {
return _thumbnailView.lock();
}
} // namespace Stickers

View File

@ -0,0 +1,92 @@
/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#pragma once
#include "data/data_cloud_file.h"
class DocumentData;
namespace Data {
class Session;
} // namespace Data
namespace Main {
class Session;
} // namespace Main
namespace Stickers {
using Order = QList<uint64>;
using SavedGifs = QVector<DocumentData*>;
using Pack = QVector<DocumentData*>;
using ByEmojiMap = QMap<EmojiPtr, Pack>;
class Set;
using Sets = base::flat_map<uint64, std::unique_ptr<Set>>;
class SetThumbnailView final {
public:
void set(not_null<Main::Session*> session, QByteArray content);
[[nodiscard]] Image *image() const;
[[nodiscard]] QByteArray content() const;
private:
std::unique_ptr<Image> _image;
QByteArray _content;
};
class Set final {
public:
Set(
not_null<Data::Session*> owner,
uint64 id,
uint64 access,
const QString &title,
const QString &shortName,
int count,
int32 hash,
MTPDstickerSet::Flags flags,
TimeId installDate);
[[nodiscard]] MTPInputStickerSet mtpInput() const;
void setThumbnail(const ImageWithLocation &data);
[[nodiscard]] bool hasThumbnail() const;
[[nodiscard]] bool thumbnailLoading() const;
[[nodiscard]] bool thumbnailFailed() const;
void loadThumbnail(Data::FileOrigin origin);
[[nodiscard]] const ImageLocation &thumbnailLocation() const;
[[nodiscard]] int thumbnailByteSize() const;
[[nodiscard]] std::shared_ptr<SetThumbnailView> createThumbnailView();
[[nodiscard]] std::shared_ptr<SetThumbnailView> activeThumbnailView();
uint64 id = 0;
uint64 access = 0;
QString title, shortName;
int count = 0;
int32 hash = 0;
MTPDstickerSet::Flags flags;
TimeId installDate = 0;
Pack stickers;
std::vector<TimeId> dates;
Pack covers;
ByEmojiMap emoji;
private:
const not_null<Data::Session*> _owner;
Data::CloudFile _thumbnail;
std::weak_ptr<SetThumbnailView> _thumbnailView;
};
} // namespace Stickers

View File

@ -11,7 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_session.h" #include "data/data_session.h"
#include "storage/cache/storage_cache_database.h" #include "storage/cache/storage_cache_database.h"
#include "storage/file_download.h" #include "storage/file_download.h"
#include "ui/image/image.h" #include "ui/image/image_source.h"
#include "main/main_session.h" #include "main/main_session.h"
#include <compare> #include <compare>
@ -74,7 +74,6 @@ void CloudImage::load(FileOrigin origin) {
active->set(_session, std::move(result)); active->set(_session, std::move(result));
} }
}); });
} }
const ImageLocation &CloudImage::location() const { const ImageLocation &CloudImage::location() const {

View File

@ -18,6 +18,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "core/file_utilities.h" #include "core/file_utilities.h"
#include "core/media_active_cache.h" #include "core/media_active_cache.h"
#include "core/mime_type.h" #include "core/mime_type.h"
#include "chat_helpers/stickers.h"
#include "chat_helpers/stickers_set.h"
#include "media/audio/media_audio.h" #include "media/audio/media_audio.h"
#include "media/player/media_player_instance.h" #include "media/player/media_player_instance.h"
#include "media/streaming/media_streaming_loader_mtproto.h" #include "media/streaming/media_streaming_loader_mtproto.h"
@ -1182,16 +1184,16 @@ bool DocumentData::isStickerSetInstalled() const {
const auto &sets = _owner->stickerSets(); const auto &sets = _owner->stickerSets();
return sticker()->set.match([&](const MTPDinputStickerSetID &data) { return sticker()->set.match([&](const MTPDinputStickerSetID &data) {
const auto i = sets.constFind(data.vid().v); const auto i = sets.find(data.vid().v);
return (i != sets.cend()) return (i != sets.cend())
&& !(i->flags & MTPDstickerSet::Flag::f_archived) && !(i->second->flags & MTPDstickerSet::Flag::f_archived)
&& (i->flags & MTPDstickerSet::Flag::f_installed_date); && (i->second->flags & MTPDstickerSet::Flag::f_installed_date);
}, [&](const MTPDinputStickerSetShortName &data) { }, [&](const MTPDinputStickerSetShortName &data) {
const auto name = qs(data.vshort_name()).toLower(); const auto name = qs(data.vshort_name()).toLower();
for (const auto &set : sets) { for (const auto &[id, set] : sets) {
if (set.shortName.toLower() == name) { if (set->shortName.toLower() == name) {
return !(set.flags & MTPDstickerSet::Flag::f_archived) return !(set->flags & MTPDstickerSet::Flag::f_archived)
&& (set.flags & MTPDstickerSet::Flag::f_installed_date); && (set->flags & MTPDstickerSet::Flag::f_installed_date);
} }
} }
return false; return false;

View File

@ -19,7 +19,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/history.h" #include "history/history.h"
#include "window/themes/window_theme_preview.h" #include "window/themes/window_theme_preview.h"
#include "storage/file_download.h" #include "storage/file_download.h"
#include "ui/image/image.h" #include "ui/image/image_source.h"
#include "facades.h" #include "facades.h"
#include "app.h" #include "app.h"

View File

@ -15,7 +15,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/history_item.h" #include "history/history_item.h"
#include "history/history.h" #include "history/history.h"
#include "storage/file_download.h" #include "storage/file_download.h"
#include "ui/image/image.h" #include "ui/image/image_source.h"
namespace Data { namespace Data {

View File

@ -8,7 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#pragma once #pragma once
#include "storage/storage_databases.h" #include "storage/storage_databases.h"
#include "chat_helpers/stickers.h" #include "chat_helpers/stickers_set.h"
#include "dialogs/dialogs_key.h" #include "dialogs/dialogs_key.h"
#include "dialogs/dialogs_indexed_list.h" #include "dialogs/dialogs_indexed_list.h"
#include "dialogs/dialogs_main_list.h" #include "dialogs/dialogs_main_list.h"

View File

@ -362,17 +362,6 @@ struct Data {
HiddenPinnedMessagesMap HiddenPinnedMessages; HiddenPinnedMessagesMap HiddenPinnedMessages;
Stickers::Sets StickerSets;
Stickers::Order StickerSetsOrder;
crl::time LastStickersUpdate = 0;
crl::time LastRecentStickersUpdate = 0;
crl::time LastFavedStickersUpdate = 0;
Stickers::Order FeaturedStickerSetsOrder;
int FeaturedStickerSetsUnreadCount = 0;
base::Observable<void> FeaturedStickerSetsUnreadCountChanged;
crl::time LastFeaturedStickersUpdate = 0;
Stickers::Order ArchivedStickerSetsOrder;
bool AskDownloadPath = false; bool AskDownloadPath = false;
QString DownloadPath; QString DownloadPath;
QByteArray DownloadPathBookmark; QByteArray DownloadPathBookmark;
@ -491,17 +480,6 @@ DefineRefVar(Global, base::Observable<void>, PhoneCallsEnabledChanged);
DefineVar(Global, HiddenPinnedMessagesMap, HiddenPinnedMessages); DefineVar(Global, HiddenPinnedMessagesMap, HiddenPinnedMessages);
DefineVar(Global, Stickers::Sets, StickerSets);
DefineVar(Global, Stickers::Order, StickerSetsOrder);
DefineVar(Global, crl::time, LastStickersUpdate);
DefineVar(Global, crl::time, LastRecentStickersUpdate);
DefineVar(Global, crl::time, LastFavedStickersUpdate);
DefineVar(Global, Stickers::Order, FeaturedStickerSetsOrder);
DefineVar(Global, int, FeaturedStickerSetsUnreadCount);
DefineRefVar(Global, base::Observable<void>, FeaturedStickerSetsUnreadCountChanged);
DefineVar(Global, crl::time, LastFeaturedStickersUpdate);
DefineVar(Global, Stickers::Order, ArchivedStickerSetsOrder);
DefineVar(Global, bool, AskDownloadPath); DefineVar(Global, bool, AskDownloadPath);
DefineVar(Global, QString, DownloadPath); DefineVar(Global, QString, DownloadPath);
DefineVar(Global, QByteArray, DownloadPathBookmark); DefineVar(Global, QByteArray, DownloadPathBookmark);

View File

@ -26,7 +26,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/view/media/history_view_media_common.h" #include "history/view/media/history_view_media_common.h"
#include "window/window_session_controller.h" #include "window/window_session_controller.h"
#include "core/application.h" // Application::showDocument. #include "core/application.h" // Application::showDocument.
#include "ui/image/image.h" #include "ui/image/image_source.h"
#include "ui/grouped_layout.h" #include "ui/grouped_layout.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "data/data_streaming.h" #include "data/data_streaming.h"

View File

@ -25,6 +25,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/history_location_manager.h" #include "history/history_location_manager.h"
#include "history/view/history_view_cursor_state.h" #include "history/view/history_view_cursor_state.h"
#include "storage/localstorage.h" #include "storage/localstorage.h"
#include "chat_helpers/stickers.h"
#include "ui/image/image.h" #include "ui/image/image.h"
#include "main/main_session.h" #include "main/main_session.h"
#include "apiwrap.h" #include "apiwrap.h"

View File

@ -16,6 +16,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_user.h" #include "data/data_user.h"
#include "window/window_controller.h" #include "window/window_controller.h"
#include "media/audio/media_audio.h" #include "media/audio/media_audio.h"
#include "ui/image/image.h"
#include "mainwidget.h" #include "mainwidget.h"
#include "observer_peer.h" #include "observer_peer.h"
#include "main/main_app_config.h" #include "main/main_app_config.h"

View File

@ -3360,7 +3360,8 @@ void MainWidget::incrementSticker(DocumentData *sticker) {
auto it = sets.find(Stickers::CloudRecentSetId); auto it = sets.find(Stickers::CloudRecentSetId);
if (it == sets.cend()) { if (it == sets.cend()) {
if (it == sets.cend()) { if (it == sets.cend()) {
it = sets.insert(Stickers::CloudRecentSetId, Stickers::Set( it = sets.emplace(Stickers::CloudRecentSetId, std::make_unique<Stickers::Set>(
&session().data(),
Stickers::CloudRecentSetId, Stickers::CloudRecentSetId,
uint64(0), uint64(0),
tr::lng_recent_stickers(tr::now), tr::lng_recent_stickers(tr::now),
@ -3368,28 +3369,28 @@ void MainWidget::incrementSticker(DocumentData *sticker) {
0, // count 0, // count
0, // hash 0, // hash
MTPDstickerSet_ClientFlag::f_special | 0, MTPDstickerSet_ClientFlag::f_special | 0,
TimeId(0), TimeId(0))).first;
ImagePtr()));
} else { } else {
it->title = tr::lng_recent_stickers(tr::now); it->second->title = tr::lng_recent_stickers(tr::now);
} }
} }
const auto set = it->second.get();
auto removedFromEmoji = std::vector<not_null<EmojiPtr>>(); auto removedFromEmoji = std::vector<not_null<EmojiPtr>>();
auto index = it->stickers.indexOf(sticker); auto index = set->stickers.indexOf(sticker);
if (index > 0) { if (index > 0) {
if (it->dates.empty()) { if (set->dates.empty()) {
session().api().requestRecentStickersForce(); session().api().requestRecentStickersForce();
} else { } else {
Assert(it->dates.size() == it->stickers.size()); Assert(set->dates.size() == set->stickers.size());
it->dates.erase(it->dates.begin() + index); set->dates.erase(set->dates.begin() + index);
} }
it->stickers.removeAt(index); set->stickers.removeAt(index);
for (auto i = it->emoji.begin(); i != it->emoji.end();) { for (auto i = set->emoji.begin(); i != set->emoji.end();) {
if (const auto index = i->indexOf(sticker); index >= 0) { if (const auto index = i->indexOf(sticker); index >= 0) {
removedFromEmoji.emplace_back(i.key()); removedFromEmoji.emplace_back(i.key());
i->removeAt(index); i->removeAt(index);
if (i->isEmpty()) { if (i->isEmpty()) {
i = it->emoji.erase(i); i = set->emoji.erase(i);
continue; continue;
} }
} }
@ -3397,17 +3398,17 @@ void MainWidget::incrementSticker(DocumentData *sticker) {
} }
} }
if (index) { if (index) {
if (it->dates.size() == it->stickers.size()) { if (set->dates.size() == set->stickers.size()) {
it->dates.insert(it->dates.begin(), base::unixtime::now()); set->dates.insert(set->dates.begin(), base::unixtime::now());
} }
it->stickers.push_front(sticker); set->stickers.push_front(sticker);
if (const auto emojiList = Stickers::GetEmojiListFromSet(sticker)) { if (const auto emojiList = Stickers::GetEmojiListFromSet(sticker)) {
for (const auto emoji : *emojiList) { for (const auto emoji : *emojiList) {
it->emoji[emoji].push_front(sticker); set->emoji[emoji].push_front(sticker);
} }
} else if (!removedFromEmoji.empty()) { } else if (!removedFromEmoji.empty()) {
for (const auto emoji : removedFromEmoji) { for (const auto emoji : removedFromEmoji) {
it->emoji[emoji].push_front(sticker); set->emoji[emoji].push_front(sticker);
} }
} else { } else {
session().api().requestRecentStickersForce(); session().api().requestRecentStickersForce();
@ -3426,7 +3427,7 @@ void MainWidget::incrementSticker(DocumentData *sticker) {
break; break;
} }
} }
while (!recent.isEmpty() && it->stickers.size() + recent.size() > Global::StickersRecentLimit()) { while (!recent.isEmpty() && set->stickers.size() + recent.size() > Global::StickersRecentLimit()) {
writeOldRecent = true; writeOldRecent = true;
recent.pop_back(); recent.pop_back();
} }
@ -3437,13 +3438,14 @@ void MainWidget::incrementSticker(DocumentData *sticker) {
// Remove that sticker from custom stickers, now it is in cloud recent stickers. // Remove that sticker from custom stickers, now it is in cloud recent stickers.
bool writeInstalledStickers = false; bool writeInstalledStickers = false;
auto custom = sets.find(Stickers::CustomSetId); auto customIt = sets.find(Stickers::CustomSetId);
if (custom != sets.cend()) { if (customIt != sets.cend()) {
const auto custom = customIt->second.get();
int removeIndex = custom->stickers.indexOf(sticker); int removeIndex = custom->stickers.indexOf(sticker);
if (removeIndex >= 0) { if (removeIndex >= 0) {
custom->stickers.removeAt(removeIndex); custom->stickers.removeAt(removeIndex);
if (custom->stickers.isEmpty()) { if (custom->stickers.isEmpty()) {
sets.erase(custom); sets.erase(customIt);
} }
writeInstalledStickers = true; writeInstalledStickers = true;
} }
@ -4520,12 +4522,13 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
auto &sets = session().data().stickerSets(); auto &sets = session().data().stickerSets();
Stickers::Order result; Stickers::Order result;
for (const auto &item : order) { for (const auto &item : order) {
if (sets.constFind(item.v) == sets.cend()) { if (sets.find(item.v) == sets.cend()) {
break; break;
} }
result.push_back(item.v); result.push_back(item.v);
} }
if (result.size() != session().data().stickerSetsOrder().size() || result.size() != order.size()) { if (result.size() != session().data().stickerSetsOrder().size()
|| result.size() != order.size()) {
session().data().setLastStickersUpdate(0); session().data().setLastStickersUpdate(0);
session().api().updateStickers(); session().api().updateStickers();
} else { } else {

View File

@ -27,6 +27,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "storage/localstorage.h" #include "storage/localstorage.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "main/main_session.h" #include "main/main_session.h"
#include "mtproto/facade.h"
#include "layout.h" #include "layout.h"
#include "facades.h" #include "facades.h"
#include "app.h" #include "app.h"

View File

@ -15,6 +15,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "base/unixtime.h" #include "base/unixtime.h"
#include "media/audio/media_audio.h" #include "media/audio/media_audio.h"
#include "media/clip/media_clip_reader.h" #include "media/clip/media_clip_reader.h"
#include "mtproto/facade.h"
#include "lottie/lottie_animation.h" #include "lottie/lottie_animation.h"
#include "history/history_item.h" #include "history/history_item.h"
#include "boxes/send_files_box.h" #include "boxes/send_files_box.h"

View File

@ -3438,9 +3438,7 @@ void _writeStickerSet(QDataStream &stream, const Stickers::Set &set) {
<< qint32(set.hash) << qint32(set.hash)
<< qint32(set.flags) << qint32(set.flags)
<< qint32(set.installDate); << qint32(set.installDate);
Serialize::writeStorageImageLocation( Serialize::writeImageLocation(stream, set.thumbnailLocation());
stream,
set.thumbnail ? set.thumbnail->location() : StorageImageLocation());
}; };
if (set.flags & MTPDstickerSet_ClientFlag::f_not_loaded) { if (set.flags & MTPDstickerSet_ClientFlag::f_not_loaded) {
writeInfo(-set.count); writeInfo(-set.count);
@ -3483,7 +3481,7 @@ void _writeStickerSets(FileKey &stickersKey, CheckSet checkSet, const Stickers::
if (!_working()) return; if (!_working()) return;
const auto &sets = Auth().data().stickerSets(); const auto &sets = Auth().data().stickerSets();
if (sets.isEmpty()) { if (sets.empty()) {
if (stickersKey) { if (stickersKey) {
ClearKey(stickersKey); ClearKey(stickersKey);
stickersKey = 0; stickersKey = 0;
@ -3497,8 +3495,9 @@ void _writeStickerSets(FileKey &stickersKey, CheckSet checkSet, const Stickers::
quint32 size = sizeof(quint32) + sizeof(qint32) + sizeof(qint32); quint32 size = sizeof(quint32) + sizeof(qint32) + sizeof(qint32);
int32 setsCount = 0; int32 setsCount = 0;
for (const auto &set : sets) { for (const auto &[id, set] : sets) {
auto result = checkSet(set); const auto raw = set.get();
auto result = checkSet(*raw);
if (result == StickerSetCheckResult::Abort) { if (result == StickerSetCheckResult::Abort) {
return; return;
} else if (result == StickerSetCheckResult::Skip) { } else if (result == StickerSetCheckResult::Skip) {
@ -3507,28 +3506,26 @@ void _writeStickerSets(FileKey &stickersKey, CheckSet checkSet, const Stickers::
// id + access + title + shortName + stickersCount + hash + flags + installDate // id + access + title + shortName + stickersCount + hash + flags + installDate
size += sizeof(quint64) * 2 size += sizeof(quint64) * 2
+ Serialize::stringSize(set.title) + Serialize::stringSize(raw->title)
+ Serialize::stringSize(set.shortName) + Serialize::stringSize(raw->shortName)
+ sizeof(qint32) * 4 + sizeof(qint32) * 4
+ Serialize::storageImageLocationSize(set.thumbnail + Serialize::imageLocationSize(raw->thumbnailLocation());
? set.thumbnail->location() if (raw->flags & MTPDstickerSet_ClientFlag::f_not_loaded) {
: StorageImageLocation());
if (set.flags & MTPDstickerSet_ClientFlag::f_not_loaded) {
continue; continue;
} }
for (const auto sticker : set.stickers) { for (const auto sticker : raw->stickers) {
size += Serialize::Document::sizeInStream(sticker); size += Serialize::Document::sizeInStream(sticker);
} }
size += sizeof(qint32); // datesCount size += sizeof(qint32); // datesCount
if (!set.dates.empty()) { if (!raw->dates.empty()) {
Assert(set.stickers.size() == set.dates.size()); Assert(raw->stickers.size() == raw->dates.size());
size += set.dates.size() * sizeof(qint32); size += raw->dates.size() * sizeof(qint32);
} }
size += sizeof(qint32); // emojiCount size += sizeof(qint32); // emojiCount
for (auto j = set.emoji.cbegin(), e = set.emoji.cend(); j != e; ++j) { for (auto j = raw->emoji.cbegin(), e = raw->emoji.cend(); j != e; ++j) {
size += Serialize::stringSize(j.key()->id()) + sizeof(qint32) + (j->size() * sizeof(quint64)); size += Serialize::stringSize(j.key()->id()) + sizeof(qint32) + (j->size() * sizeof(quint64));
} }
@ -3555,14 +3552,14 @@ void _writeStickerSets(FileKey &stickersKey, CheckSet checkSet, const Stickers::
<< quint32(kStickersVersionTag) << quint32(kStickersVersionTag)
<< qint32(kStickersSerializeVersion) << qint32(kStickersSerializeVersion)
<< qint32(setsCount); << qint32(setsCount);
for (const auto &set : sets) { for (const auto &[id, set] : sets) {
auto result = checkSet(set); auto result = checkSet(*set);
if (result == StickerSetCheckResult::Abort) { if (result == StickerSetCheckResult::Abort) {
return; return;
} else if (result == StickerSetCheckResult::Skip) { } else if (result == StickerSetCheckResult::Skip) {
continue; continue;
} }
_writeStickerSet(data.stream, set); _writeStickerSet(data.stream, *set);
} }
data.stream << order; data.stream << order;
@ -3612,7 +3609,7 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
qint32 setHash = 0; qint32 setHash = 0;
MTPDstickerSet::Flags setFlags = 0; MTPDstickerSet::Flags setFlags = 0;
qint32 setFlagsValue = 0; qint32 setFlagsValue = 0;
StorageImageLocation setThumbnail; ImageLocation setThumbnail;
stickers.stream stickers.stream
>> setId >> setId
@ -3623,13 +3620,12 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
>> setHash >> setHash
>> setFlagsValue >> setFlagsValue
>> setInstallDate; >> setInstallDate;
const auto thumbnail = Serialize::readStorageImageLocation( const auto thumbnail = Serialize::readImageLocation(
stickers.version, stickers.version,
stickers.stream); stickers.stream);
if (!thumbnail || !_checkStreamStatus(stickers.stream)) { if (!thumbnail || !_checkStreamStatus(stickers.stream)) {
return failed(); return failed();
} else if (thumbnail->valid() } else if (thumbnail->valid() && thumbnail->isLegacy()) {
&& thumbnail->type() == LocationType::Legacy) {
setThumbnail = thumbnail->convertToModern( setThumbnail = thumbnail->convertToModern(
LocationType::StickerSetThumb, LocationType::StickerSetThumb,
setId, setId,
@ -3659,7 +3655,8 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
if (it == sets.cend()) { if (it == sets.cend()) {
// We will set this flags from order lists when reading those stickers. // We will set this flags from order lists when reading those stickers.
setFlags &= ~(MTPDstickerSet::Flag::f_installed_date | MTPDstickerSet_ClientFlag::f_featured); setFlags &= ~(MTPDstickerSet::Flag::f_installed_date | MTPDstickerSet_ClientFlag::f_featured);
it = sets.insert(setId, Stickers::Set( it = sets.emplace(setId, std::make_unique<Stickers::Set>(
&Auth().data(),
setId, setId,
setAccess, setAccess,
setTitle, setTitle,
@ -3667,23 +3664,24 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
0, 0,
setHash, setHash,
MTPDstickerSet::Flags(setFlags), MTPDstickerSet::Flags(setFlags),
setInstallDate, setInstallDate)).first;
Images::CreateStickerSetThumbnail(setThumbnail))); it->second->setThumbnail(
ImageWithLocation{ .location = setThumbnail });
} }
auto &set = it.value(); const auto set = it->second.get();
auto inputSet = MTP_inputStickerSetID(MTP_long(set.id), MTP_long(set.access)); auto inputSet = MTP_inputStickerSetID(MTP_long(set->id), MTP_long(set->access));
const auto fillStickers = set.stickers.isEmpty(); const auto fillStickers = set->stickers.isEmpty();
if (scnt < 0) { // disabled not loaded set if (scnt < 0) { // disabled not loaded set
if (!set.count || fillStickers) { if (!set->count || fillStickers) {
set.count = -scnt; set->count = -scnt;
} }
continue; continue;
} }
if (fillStickers) { if (fillStickers) {
set.stickers.reserve(scnt); set->stickers.reserve(scnt);
set.count = 0; set->count = 0;
} }
Serialize::Document::StickerSetInfo info(setId, setAccess, setShortName); Serialize::Document::StickerSetInfo info(setId, setAccess, setShortName);
@ -3699,13 +3697,13 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
} }
read.emplace(document->id); read.emplace(document->id);
if (fillStickers) { if (fillStickers) {
set.stickers.push_back(document); set->stickers.push_back(document);
if (!(set.flags & MTPDstickerSet_ClientFlag::f_special)) { if (!(set->flags & MTPDstickerSet_ClientFlag::f_special)) {
if (document->sticker()->set.type() != mtpc_inputStickerSetID) { if (document->sticker()->set.type() != mtpc_inputStickerSetID) {
document->sticker()->set = inputSet; document->sticker()->set = inputSet;
} }
} }
++set.count; ++set->count;
} }
} }
@ -3715,17 +3713,17 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
if (datesCount != scnt) { if (datesCount != scnt) {
return failed(); return failed();
} }
const auto fillDates = (set.id == Stickers::CloudRecentSetId) const auto fillDates = (set->id == Stickers::CloudRecentSetId)
&& (set.stickers.size() == datesCount); && (set->stickers.size() == datesCount);
if (fillDates) { if (fillDates) {
set.dates.clear(); set->dates.clear();
set.dates.reserve(datesCount); set->dates.reserve(datesCount);
} }
for (auto i = 0; i != datesCount; ++i) { for (auto i = 0; i != datesCount; ++i) {
qint32 date = 0; qint32 date = 0;
stickers.stream >> date; stickers.stream >> date;
if (fillDates) { if (fillDates) {
set.dates.push_back(TimeId(date)); set->dates.push_back(TimeId(date));
} }
} }
} }
@ -3752,7 +3750,7 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
if (fillStickers) { if (fillStickers) {
if (auto emoji = Ui::Emoji::Find(emojiString)) { if (auto emoji = Ui::Emoji::Find(emojiString)) {
emoji = emoji->original(); emoji = emoji->original();
set.emoji.insert(emoji, pack); set->emoji.insert(emoji, pack);
} }
} }
} }
@ -3785,10 +3783,11 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
for (const auto setId : std::as_const(*outOrder)) { for (const auto setId : std::as_const(*outOrder)) {
auto it = sets.find(setId); auto it = sets.find(setId);
if (it != sets.cend()) { if (it != sets.cend()) {
it->flags |= readingFlags; const auto set = it->second.get();
set->flags |= readingFlags;
if ((readingFlags == MTPDstickerSet::Flag::f_installed_date) if ((readingFlags == MTPDstickerSet::Flag::f_installed_date)
&& !it->installDate) { && !set->installDate) {
it->installDate = kDefaultStickerInstallDate; set->installDate = kDefaultStickerInstallDate;
} }
} }
} }
@ -3888,7 +3887,8 @@ void importOldRecentStickers() {
auto &recent = cRefRecentStickers(); auto &recent = cRefRecentStickers();
recent.clear(); recent.clear();
auto &def = sets.insert(Stickers::DefaultSetId, Stickers::Set( const auto def = sets.emplace(Stickers::DefaultSetId, std::make_unique<Stickers::Set>(
&Auth().data(),
Stickers::DefaultSetId, Stickers::DefaultSetId,
uint64(0), uint64(0),
tr::lng_stickers_default_set(tr::now), tr::lng_stickers_default_set(tr::now),
@ -3898,9 +3898,9 @@ void importOldRecentStickers() {
(MTPDstickerSet::Flag::f_official (MTPDstickerSet::Flag::f_official
| MTPDstickerSet::Flag::f_installed_date | MTPDstickerSet::Flag::f_installed_date
| MTPDstickerSet_ClientFlag::f_special), | MTPDstickerSet_ClientFlag::f_special),
kDefaultStickerInstallDate, kDefaultStickerInstallDate)).first->second.get();
ImagePtr())).value(); const auto custom = sets.emplace(Stickers::CustomSetId, std::make_unique<Stickers::Set>(
auto &custom = sets.insert(Stickers::CustomSetId, Stickers::Set( &Auth().data(),
Stickers::CustomSetId, Stickers::CustomSetId,
uint64(0), uint64(0),
qsl("Custom stickers"), qsl("Custom stickers"),
@ -3909,8 +3909,7 @@ void importOldRecentStickers() {
0, // hash 0, // hash
(MTPDstickerSet::Flag::f_installed_date (MTPDstickerSet::Flag::f_installed_date
| MTPDstickerSet_ClientFlag::f_special), | MTPDstickerSet_ClientFlag::f_special),
kDefaultStickerInstallDate, kDefaultStickerInstallDate)).first->second.get();
ImagePtr())).value();
QMap<uint64, bool> read; QMap<uint64, bool> read;
while (!stickers.stream.atEnd()) { while (!stickers.stream.atEnd()) {
@ -3953,22 +3952,24 @@ void importOldRecentStickers() {
} }
if (value > 0) { if (value > 0) {
def.stickers.push_back(doc); def->stickers.push_back(doc);
++def.count; ++def->count;
} else { } else {
custom.stickers.push_back(doc); custom->stickers.push_back(doc);
++custom.count; ++custom->count;
} }
if (recent.size() < Global::StickersRecentLimit() && qAbs(value) > 1) { if (recent.size() < Global::StickersRecentLimit() && qAbs(value) > 1) {
recent.push_back(qMakePair(doc, qAbs(value))); recent.push_back(qMakePair(doc, qAbs(value)));
} }
} }
if (def.stickers.isEmpty()) { if (def->stickers.isEmpty()) {
sets.remove(Stickers::DefaultSetId); sets.remove(Stickers::DefaultSetId);
} else { } else {
order.push_front(Stickers::DefaultSetId); order.push_front(Stickers::DefaultSetId);
} }
if (custom.stickers.isEmpty()) sets.remove(Stickers::CustomSetId); if (custom->stickers.isEmpty()) {
sets.remove(Stickers::CustomSetId);
}
writeInstalledStickers(); writeInstalledStickers();
writeUserSettings(); writeUserSettings();
@ -3996,11 +3997,13 @@ void readFeaturedStickers() {
&Auth().data().featuredStickerSetsOrderRef(), &Auth().data().featuredStickerSetsOrderRef(),
MTPDstickerSet::Flags() | MTPDstickerSet_ClientFlag::f_featured); MTPDstickerSet::Flags() | MTPDstickerSet_ClientFlag::f_featured);
auto &sets = Auth().data().stickerSets(); const auto &sets = Auth().data().stickerSets();
const auto &order = Auth().data().featuredStickerSetsOrder();
int unreadCount = 0; int unreadCount = 0;
for_const (auto setId, Auth().data().featuredStickerSetsOrder()) { for (const auto setId : order) {
auto it = sets.constFind(setId); auto it = sets.find(setId);
if (it != sets.cend() && (it->flags & MTPDstickerSet_ClientFlag::f_unread)) { if (it != sets.cend()
&& (it->second->flags & MTPDstickerSet_ClientFlag::f_unread)) {
++unreadCount; ++unreadCount;
} }
} }
@ -4033,9 +4036,9 @@ int32 countDocumentVectorHash(const QVector<DocumentData*> vector) {
int32 countSpecialStickerSetHash(uint64 setId) { int32 countSpecialStickerSetHash(uint64 setId) {
auto &sets = Auth().data().stickerSets(); auto &sets = Auth().data().stickerSets();
auto it = sets.constFind(setId); auto it = sets.find(setId);
if (it != sets.cend()) { if (it != sets.cend()) {
return countDocumentVectorHash(it->stickers); return countDocumentVectorHash(it->second->stickers);
} }
return 0; return 0;
} }
@ -4043,16 +4046,17 @@ int32 countSpecialStickerSetHash(uint64 setId) {
int32 countStickersHash(bool checkOutdatedInfo) { int32 countStickersHash(bool checkOutdatedInfo) {
auto result = Api::HashInit(); auto result = Api::HashInit();
bool foundOutdated = false; bool foundOutdated = false;
auto &sets = Auth().data().stickerSets(); const auto &sets = Auth().data().stickerSets();
auto &order = Auth().data().stickerSetsOrder(); const auto &order = Auth().data().stickerSetsOrder();
for (auto i = order.cbegin(), e = order.cend(); i != e; ++i) { for (auto i = order.cbegin(), e = order.cend(); i != e; ++i) {
auto j = sets.constFind(*i); auto it = sets.find(*i);
if (j != sets.cend()) { if (it != sets.cend()) {
if (j->id == Stickers::DefaultSetId) { const auto set = it->second.get();
if (set->id == Stickers::DefaultSetId) {
foundOutdated = true; foundOutdated = true;
} else if (!(j->flags & MTPDstickerSet_ClientFlag::f_special) } else if (!(set->flags & MTPDstickerSet_ClientFlag::f_special)
&& !(j->flags & MTPDstickerSet::Flag::f_archived)) { && !(set->flags & MTPDstickerSet::Flag::f_archived)) {
Api::HashUpdate(result, j->hash); Api::HashUpdate(result, set->hash);
} }
} }
} }
@ -4076,8 +4080,9 @@ int32 countFeaturedStickersHash() {
for (const auto setId : featured) { for (const auto setId : featured) {
Api::HashUpdate(result, setId); Api::HashUpdate(result, setId);
auto it = sets.constFind(setId); const auto it = sets.find(setId);
if (it != sets.cend() && (it->flags & MTPDstickerSet_ClientFlag::f_unread)) { if (it != sets.cend()
&& (it->second->flags & MTPDstickerSet_ClientFlag::f_unread)) {
Api::HashUpdate(result, 1); Api::HashUpdate(result, 1);
} }
} }

View File

@ -175,74 +175,6 @@ ImagePtr Create(const StorageImageLocation &location, int size) {
return Create<StorageSource>(location, size, QByteArray()); return Create<StorageSource>(location, size, QByteArray());
} }
struct CreateSetThumbnail {
using Source = Stickers::ThumbnailSource;
ImagePtr operator()(
const StorageImageLocation &location,
int size) {
return Create<Source>(location, size, QByteArray());
}
ImagePtr operator()(
const StorageImageLocation &location,
const QByteArray &bytes) {
return Create<Source>(location, bytes.size(), bytes);
}
};
template <typename CreateLocation, typename Method>
ImagePtr CreateFromPhotoSize(
CreateLocation &&createLocation,
const MTPPhotoSize &size,
Method method = Method()) {
return size.match([&](const MTPDphotoSize &data) {
const auto &location = data.vlocation().c_fileLocationToBeDeprecated();
return method(
StorageImageLocation(
createLocation(data.vtype(), location),
data.vw().v,
data.vh().v),
data.vsize().v);
}, [&](const MTPDphotoCachedSize &data) {
const auto bytes = qba(data.vbytes());
const auto &location = data.vlocation().c_fileLocationToBeDeprecated();
return method(
StorageImageLocation(
createLocation(data.vtype(), location),
data.vw().v,
data.vh().v),
bytes);
}, [&](const MTPDphotoStrippedSize &data) {
auto image = FromInlineBytes(qba(data.vbytes()));
return !image.isNull()
? Images::Create(std::move(image), "JPG")
: ImagePtr();
}, [&](const MTPDphotoSizeEmpty &) {
return ImagePtr();
});
}
ImagePtr Create(const MTPDstickerSet &set, const MTPPhotoSize &size) {
const auto thumbDcId = set.vthumb_dc_id();
const auto create = [&](
const MTPstring &thumbSize,
const MTPDfileLocationToBeDeprecated &location) {
return StorageFileLocation(
thumbDcId->v,
Auth().userId(),
MTP_inputStickerSetThumb(
MTP_inputStickerSetID(set.vid(), set.vaccess_hash()),
location.vvolume_id(),
location.vlocal_id()));
};
return thumbDcId
? CreateFromPhotoSize(create, size, CreateSetThumbnail())
: ImagePtr();
}
ImagePtr CreateStickerSetThumbnail(const StorageImageLocation &location) {
return CreateSetThumbnail()(location, 0);
}
QSize GetSizeForDocument(const QVector<MTPDocumentAttribute> &attributes) { QSize GetSizeForDocument(const QVector<MTPDocumentAttribute> &attributes) {
for (const auto &attribute : attributes) { for (const auto &attribute : attributes) {
if (attribute.type() == mtpc_documentAttributeImageSize) { if (attribute.type() == mtpc_documentAttributeImageSize) {

View File

@ -378,6 +378,10 @@ bool StorageFileLocation::valid() const {
return false; return false;
} }
bool StorageFileLocation::isLegacy() const {
return (_type == Type::Legacy);
}
bool StorageFileLocation::isDocumentThumbnail() const { bool StorageFileLocation::isDocumentThumbnail() const {
return (_type == Type::Document) && (_sizeLetter != 0); return (_type == Type::Document) && (_sizeLetter != 0);
} }
@ -810,6 +814,12 @@ bool DownloadLocation::valid() const {
}); });
} }
bool DownloadLocation::isLegacy() const {
return data.is<StorageFileLocation>()
? data.get_unchecked<StorageFileLocation>().isLegacy()
: false;
}
QByteArray DownloadLocation::fileReference() const { QByteArray DownloadLocation::fileReference() const {
if (!data.is<StorageFileLocation>()) { if (!data.is<StorageFileLocation>()) {
return QByteArray(); return QByteArray();

View File

@ -83,6 +83,7 @@ public:
[[nodiscard]] Type type() const; [[nodiscard]] Type type() const;
[[nodiscard]] bool valid() const; [[nodiscard]] bool valid() const;
[[nodiscard]] bool isLegacy() const;
[[nodiscard]] Storage::Cache::Key cacheKey() const; [[nodiscard]] Storage::Cache::Key cacheKey() const;
[[nodiscard]] Storage::Cache::Key bigFileBaseCacheKey() const; [[nodiscard]] Storage::Cache::Key bigFileBaseCacheKey() const;
@ -184,6 +185,9 @@ public:
[[nodiscard]] bool valid() const { [[nodiscard]] bool valid() const {
return _file.valid(); return _file.valid();
} }
[[nodiscard]] bool isLegacy() const {
return _file.isLegacy();
}
[[nodiscard]] QByteArray fileReference() const { [[nodiscard]] QByteArray fileReference() const {
return _file.fileReference(); return _file.fileReference();
} }
@ -413,6 +417,7 @@ public:
[[nodiscard]] Storage::Cache::Key cacheKey() const; [[nodiscard]] Storage::Cache::Key cacheKey() const;
[[nodiscard]] bool valid() const; [[nodiscard]] bool valid() const;
[[nodiscard]] bool isLegacy() const;
[[nodiscard]] QByteArray fileReference() const; [[nodiscard]] QByteArray fileReference() const;
bool refreshFileReference(const QByteArray &data); bool refreshFileReference(const QByteArray &data);
bool refreshFileReference(const Data::UpdatedFileReferences &updates); bool refreshFileReference(const Data::UpdatedFileReferences &updates);
@ -472,6 +477,9 @@ public:
[[nodiscard]] bool valid() const { [[nodiscard]] bool valid() const {
return _file.valid(); return _file.valid();
} }
[[nodiscard]] bool isLegacy() const {
return _file.isLegacy();
}
[[nodiscard]] QByteArray fileReference() const { [[nodiscard]] QByteArray fileReference() const {
return _file.fileReference(); return _file.fileReference();
} }

View File

@ -136,6 +136,68 @@ ImageWithLocation FromPhotoSize(
}); });
} }
ImageWithLocation FromPhotoSize(
not_null<Main::Session*> session,
const MTPDstickerSet &set,
const MTPPhotoSize &size) {
if (!set.vthumb_dc_id()) {
return ImageWithLocation();
}
return size.match([&](const MTPDphotoSize &data) {
const auto &location = data.vlocation().c_fileLocationToBeDeprecated();
return ImageWithLocation{
.location = ImageLocation(
DownloadLocation{ StorageFileLocation(
set.vthumb_dc_id()->v,
session->userId(),
MTP_inputStickerSetThumb(
MTP_inputStickerSetID(set.vid(), set.vaccess_hash()),
location.vvolume_id(),
location.vlocal_id())) },
data.vw().v,
data.vh().v),
.bytesCount = data.vsize().v
};
}, [&](const MTPDphotoCachedSize &data) {
const auto &location = data.vlocation().c_fileLocationToBeDeprecated();
const auto bytes = qba(data.vbytes());
return ImageWithLocation{
.location = ImageLocation(
DownloadLocation{ StorageFileLocation(
set.vthumb_dc_id()->v,
session->userId(),
MTP_inputStickerSetThumb(
MTP_inputStickerSetID(set.vid(), set.vaccess_hash()),
location.vvolume_id(),
location.vlocal_id())) },
data.vw().v,
data.vh().v),
.bytesCount = bytes.size(),
.bytes = bytes
};
}, [&](const MTPDphotoStrippedSize &data) {
return ImageWithLocation();
//const auto bytes = ExpandInlineBytes(qba(data.vbytes()));
//return ImageWithLocation{
// .location = ImageLocation(
// DownloadLocation{ StorageFileLocation(
// document.vdc_id().v,
// session->userId(),
// MTP_inputDocumentFileLocation(
// document.vid(),
// document.vaccess_hash(),
// document.vfile_reference(),
// data.vtype())) },
// width, // ???
// height), // ???
// .bytesCount = bytes.size(),
// .bytes = bytes
//};
}, [&](const MTPDphotoSizeEmpty &) {
return ImageWithLocation();
});
}
ImageWithLocation FromImageInMemory( ImageWithLocation FromImageInMemory(
const QImage &image, const QImage &image,
const char *format) { const char *format) {

View File

@ -23,6 +23,10 @@ namespace Images {
not_null<Main::Session*> session, not_null<Main::Session*> session,
const MTPDdocument &document, const MTPDdocument &document,
const MTPPhotoSize &size); const MTPPhotoSize &size);
[[nodiscard]] ImageWithLocation FromPhotoSize(
not_null<Main::Session*> session,
const MTPDstickerSet &set,
const MTPPhotoSize &size);
[[nodiscard]] ImageWithLocation FromVideoSize( [[nodiscard]] ImageWithLocation FromVideoSize(
not_null<Main::Session*> session, not_null<Main::Session*> session,
const MTPDdocument &document, const MTPDdocument &document,

View File

@ -445,7 +445,7 @@ void EditorBlock::enumerateRows(Callback callback) const {
} }
} }
} else { } else {
for_const (auto &row, _data) { for (const auto &row : _data) {
if (!callback(row)) { if (!callback(row)) {
break; break;
} }