mirror of https://github.com/procxx/kepka.git
Move all (item/view/media) maps to Data::Session.
This commit is contained in:
parent
7425e80f05
commit
8a56ede187
|
@ -325,7 +325,7 @@ void ApiWrap::requestContacts() {
|
|||
if (contact.type() != mtpc_contact) continue;
|
||||
|
||||
const auto userId = contact.c_contact().vuser_id.v;
|
||||
if (userId == Auth().userId() && App::self()) {
|
||||
if (userId == _session->userId() && App::self()) {
|
||||
App::self()->setContactStatus(
|
||||
UserData::ContactStatus::Contact);
|
||||
}
|
||||
|
@ -527,7 +527,7 @@ void ApiWrap::gotUserFull(UserData *user, const MTPUserFull &result, mtpRequestI
|
|||
|
||||
App::feedUsers(MTP_vector<MTPUser>(1, d.vuser));
|
||||
if (d.has_profile_photo()) {
|
||||
App::feedPhoto(d.vprofile_photo);
|
||||
_session->data().photo(d.vprofile_photo);
|
||||
}
|
||||
App::feedUserLink(MTP_int(peerToUser(user->id)), d.vlink.c_contacts_link().vmy_link, d.vlink.c_contacts_link().vforeign_link);
|
||||
if (App::main()) {
|
||||
|
@ -606,6 +606,65 @@ void ApiWrap::requestPeer(PeerData *peer) {
|
|||
}
|
||||
}
|
||||
|
||||
void ApiWrap::markMediaRead(
|
||||
const base::flat_set<not_null<HistoryItem*>> &items) {
|
||||
auto markedIds = QVector<MTPint>();
|
||||
auto channelMarkedIds = base::flat_map<
|
||||
not_null<ChannelData*>,
|
||||
QVector<MTPint>>();
|
||||
markedIds.reserve(items.size());
|
||||
for (const auto item : items) {
|
||||
if (!item->isMediaUnread() || (item->out() && !item->mentionsMe())) {
|
||||
continue;
|
||||
}
|
||||
item->markMediaRead();
|
||||
if (!IsServerMsgId(item->id)) {
|
||||
continue;
|
||||
}
|
||||
if (const auto channel = item->history()->peer->asChannel()) {
|
||||
channelMarkedIds[channel].push_back(MTP_int(item->id));
|
||||
} else {
|
||||
markedIds.push_back(MTP_int(item->id));
|
||||
}
|
||||
}
|
||||
if (!markedIds.isEmpty()) {
|
||||
request(MTPmessages_ReadMessageContents(
|
||||
MTP_vector<MTPint>(markedIds)
|
||||
)).done([=](const MTPmessages_AffectedMessages &result) {
|
||||
applyAffectedMessages(result);
|
||||
}).send();
|
||||
}
|
||||
for (const auto &channelIds : channelMarkedIds) {
|
||||
request(MTPchannels_ReadMessageContents(
|
||||
channelIds.first->inputChannel,
|
||||
MTP_vector<MTPint>(channelIds.second)
|
||||
)).send();
|
||||
}
|
||||
}
|
||||
|
||||
void ApiWrap::markMediaRead(not_null<HistoryItem*> item) {
|
||||
if (!item->isMediaUnread() || (item->out() && !item->mentionsMe())) {
|
||||
return;
|
||||
}
|
||||
item->markMediaRead();
|
||||
if (!IsServerMsgId(item->id)) {
|
||||
return;
|
||||
}
|
||||
const auto ids = MTP_vector<MTPint>(1, MTP_int(item->id));
|
||||
if (const auto channel = item->history()->peer->asChannel()) {
|
||||
request(MTPchannels_ReadMessageContents(
|
||||
channel->inputChannel,
|
||||
ids
|
||||
)).send();
|
||||
} else {
|
||||
request(MTPmessages_ReadMessageContents(
|
||||
ids
|
||||
)).done([=](const MTPmessages_AffectedMessages &result) {
|
||||
applyAffectedMessages(result);
|
||||
}).send();
|
||||
}
|
||||
}
|
||||
|
||||
void ApiWrap::requestPeers(const QList<PeerData*> &peers) {
|
||||
QVector<MTPint> chats;
|
||||
QVector<MTPInputChannel> channels;
|
||||
|
@ -1612,31 +1671,29 @@ void ApiWrap::resolveWebPages() {
|
|||
using MessageIdsByChannel = QMap<ChannelData*, IndexAndMessageIds>;
|
||||
MessageIdsByChannel idsByChannel; // temp_req_id = -index - 2
|
||||
|
||||
auto &items = App::webPageItems();
|
||||
ids.reserve(_webPagesPending.size());
|
||||
int32 t = unixtime(), m = INT_MAX;
|
||||
for (auto i = _webPagesPending.begin(); i != _webPagesPending.cend(); ++i) {
|
||||
if (i.value() > 0) continue;
|
||||
if (i.key()->pendingTill <= t) {
|
||||
auto j = items.constFind(i.key());
|
||||
if (j != items.cend() && !j.value().empty()) {
|
||||
for_const (auto item, j.value()) {
|
||||
if (item->id > 0) {
|
||||
if (item->channelId() == NoChannel) {
|
||||
ids.push_back(MTP_int(item->id));
|
||||
i.value() = -1;
|
||||
} else {
|
||||
auto channel = item->history()->peer->asChannel();
|
||||
auto channelMap = idsByChannel.find(channel);
|
||||
if (channelMap == idsByChannel.cend()) {
|
||||
channelMap = idsByChannel.insert(channel, IndexAndMessageIds(idsByChannel.size(), QVector<MTPint>(1, MTP_int(item->id))));
|
||||
} else {
|
||||
channelMap.value().second.push_back(MTP_int(item->id));
|
||||
}
|
||||
i.value() = -channelMap.value().first - 2;
|
||||
}
|
||||
break;
|
||||
const auto item = _session->data().findWebPageItem(i.key());
|
||||
if (item) {
|
||||
if (item->channelId() == NoChannel) {
|
||||
ids.push_back(MTP_int(item->id));
|
||||
i.value() = -1;
|
||||
} else {
|
||||
auto channel = item->history()->peer->asChannel();
|
||||
auto channelMap = idsByChannel.find(channel);
|
||||
if (channelMap == idsByChannel.cend()) {
|
||||
channelMap = idsByChannel.insert(
|
||||
channel,
|
||||
IndexAndMessageIds(
|
||||
idsByChannel.size(),
|
||||
QVector<MTPint>(1, MTP_int(item->id))));
|
||||
} else {
|
||||
channelMap.value().second.push_back(MTP_int(item->id));
|
||||
}
|
||||
i.value() = -channelMap.value().first - 2;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
|
@ -1728,27 +1785,22 @@ void ApiWrap::gotWebPages(ChannelData *channel, const MTPmessages_Messages &msgs
|
|||
v->at(index),
|
||||
NewMessageExisting);
|
||||
if (item) {
|
||||
Auth().data().requestItemViewResize(item);
|
||||
_session->data().requestItemViewResize(item);
|
||||
}
|
||||
}
|
||||
|
||||
auto &items = App::webPageItems();
|
||||
for (auto i = _webPagesPending.begin(); i != _webPagesPending.cend();) {
|
||||
if (i.value() == req) {
|
||||
if (i.key()->pendingTill > 0) {
|
||||
i.key()->pendingTill = -1;
|
||||
auto j = items.constFind(i.key());
|
||||
if (j != items.cend()) {
|
||||
for_const (auto item, j.value()) {
|
||||
Auth().data().requestItemViewResize(item);
|
||||
}
|
||||
}
|
||||
_session->data().notifyWebPageUpdateDelayed(i.key());
|
||||
}
|
||||
i = _webPagesPending.erase(i);
|
||||
} else {
|
||||
++i;
|
||||
}
|
||||
}
|
||||
_session->data().sendWebPageGameNotifications();
|
||||
}
|
||||
|
||||
void ApiWrap::stickersSaveOrder() {
|
||||
|
@ -2081,11 +2133,11 @@ void ApiWrap::applyUpdateNoPtsCheck(const MTPUpdate &update) {
|
|||
case mtpc_updateReadMessagesContents: {
|
||||
auto &d = update.c_updateReadMessagesContents();
|
||||
auto possiblyReadMentions = base::flat_set<MsgId>();
|
||||
for_const (auto &msgId, d.vmessages.v) {
|
||||
for (const auto &msgId : d.vmessages.v) {
|
||||
if (auto item = App::histItemById(NoChannel, msgId.v)) {
|
||||
if (item->isMediaUnread()) {
|
||||
item->markMediaRead();
|
||||
_session->data().requestItemRepaint(item);
|
||||
_session->data().requestItemViewRepaint(item);
|
||||
|
||||
if (item->out() && item->history()->peer->isUser()) {
|
||||
auto when = App::main()->requestingDifference() ? 0 : unixtime();
|
||||
|
@ -2276,7 +2328,9 @@ void ApiWrap::preloadEnoughUnreadMentions(not_null<History*> history) {
|
|||
_unreadMentionsRequests.emplace(history, requestId);
|
||||
}
|
||||
|
||||
void ApiWrap::checkForUnreadMentions(const base::flat_set<MsgId> &possiblyReadMentions, ChannelData *channel) {
|
||||
void ApiWrap::checkForUnreadMentions(
|
||||
const base::flat_set<MsgId> &possiblyReadMentions,
|
||||
ChannelData *channel) {
|
||||
for (auto msgId : possiblyReadMentions) {
|
||||
requestMessageData(channel, msgId, [](ChannelData *channel, MsgId msgId) {
|
||||
if (auto item = App::histItemById(channel, msgId)) {
|
||||
|
@ -2510,7 +2564,7 @@ void ApiWrap::userPhotosDone(
|
|||
auto photoIds = std::vector<PhotoId>();
|
||||
photoIds.reserve(photos.size());
|
||||
for (auto &photo : photos) {
|
||||
if (auto photoData = App::feedPhoto(photo)) {
|
||||
if (auto photoData = _session->data().photo(photo)) {
|
||||
photoIds.push_back(photoData->id);
|
||||
}
|
||||
}
|
||||
|
@ -2774,9 +2828,7 @@ void ApiWrap::shareContact(
|
|||
not_null<UserData*> user,
|
||||
const SendOptions &options) {
|
||||
const auto userId = peerToUser(user->id);
|
||||
const auto phone = user->phone().isEmpty()
|
||||
? App::phoneFromSharedContact(userId)
|
||||
: user->phone();
|
||||
const auto phone = Auth().data().findContactPhone(user);
|
||||
if (phone.isEmpty()) {
|
||||
return;
|
||||
}
|
||||
|
|
|
@ -73,6 +73,9 @@ public:
|
|||
void processFullPeer(PeerData *peer, const MTPmessages_ChatFull &result);
|
||||
void processFullPeer(UserData *user, const MTPUserFull &result);
|
||||
|
||||
void markMediaRead(const base::flat_set<not_null<HistoryItem*>> &items);
|
||||
void markMediaRead(not_null<HistoryItem*> item);
|
||||
|
||||
void requestSelfParticipant(ChannelData *channel);
|
||||
void kickParticipant(not_null<ChatData*> chat, not_null<UserData*> user);
|
||||
void kickParticipant(
|
||||
|
|
|
@ -55,25 +55,9 @@ namespace {
|
|||
using MutedPeers = QMap<not_null<PeerData*>, bool>;
|
||||
MutedPeers mutedPeers;
|
||||
|
||||
PhotosData photosData;
|
||||
DocumentsData documentsData;
|
||||
|
||||
using LocationsData = QHash<LocationCoords, LocationData*>;
|
||||
LocationsData locationsData;
|
||||
|
||||
using WebPagesData = QHash<WebPageId, WebPageData*>;
|
||||
WebPagesData webPagesData;
|
||||
|
||||
using GamesData = QHash<GameId, GameData*>;
|
||||
GamesData gamesData;
|
||||
|
||||
PhotoItems photoItems;
|
||||
DocumentItems documentItems;
|
||||
WebPageItems webPageItems;
|
||||
GameItems gameItems;
|
||||
SharedContactItems sharedContactItems;
|
||||
GifItems gifItems;
|
||||
|
||||
using DependentItemsSet = OrderedSet<HistoryItem*>;
|
||||
using DependentItems = QMap<HistoryItem*, DependentItemsSet>;
|
||||
DependentItems dependentItems;
|
||||
|
@ -114,11 +98,6 @@ namespace {
|
|||
|
||||
int32 serviceImageCacheSize = 0;
|
||||
|
||||
using LastPhotosList = QLinkedList<PhotoData*>;
|
||||
LastPhotosList lastPhotos;
|
||||
using LastPhotosMap = QHash<PhotoData*, LastPhotosList::iterator>;
|
||||
LastPhotosMap lastPhotosMap;
|
||||
|
||||
} // namespace
|
||||
|
||||
namespace App {
|
||||
|
@ -928,8 +907,9 @@ namespace {
|
|||
auto entities = m.has_entities()
|
||||
? TextUtilities::EntitiesFromMTP(m.ventities.v)
|
||||
: EntitiesInText();
|
||||
const auto media = m.has_media() ? &m.vmedia : nullptr;
|
||||
existing->setText({ text, entities });
|
||||
existing->updateMedia(m.has_media() ? (&m.vmedia) : nullptr);
|
||||
existing->updateSentMedia(m.has_media() ? &m.vmedia : nullptr);
|
||||
existing->updateReplyMarkup(m.has_reply_markup()
|
||||
? (&m.vreply_markup)
|
||||
: nullptr);
|
||||
|
@ -1146,207 +1126,6 @@ namespace {
|
|||
}
|
||||
}
|
||||
|
||||
PhotoData *feedPhoto(const MTPPhoto &photo, PhotoData *convert) {
|
||||
switch (photo.type()) {
|
||||
case mtpc_photo: {
|
||||
return feedPhoto(photo.c_photo(), convert);
|
||||
} break;
|
||||
case mtpc_photoEmpty: {
|
||||
return App::photoSet(photo.c_photoEmpty().vid.v, convert, 0, 0, ImagePtr(), ImagePtr(), ImagePtr());
|
||||
} break;
|
||||
}
|
||||
return App::photo(0);
|
||||
}
|
||||
|
||||
PhotoData *feedPhoto(const MTPPhoto &photo, const PreparedPhotoThumbs &thumbs) {
|
||||
const QPixmap *thumb = 0, *medium = 0, *full = 0;
|
||||
int32 thumbLevel = -1, mediumLevel = -1, fullLevel = -1;
|
||||
for (PreparedPhotoThumbs::const_iterator i = thumbs.cbegin(), e = thumbs.cend(); i != e; ++i) {
|
||||
int32 newThumbLevel = -1, newMediumLevel = -1, newFullLevel = -1;
|
||||
switch (i.key()) {
|
||||
case 's': newThumbLevel = 0; newMediumLevel = 5; newFullLevel = 4; break; // box 100x100
|
||||
case 'm': newThumbLevel = 2; newMediumLevel = 0; newFullLevel = 3; break; // box 320x320
|
||||
case 'x': newThumbLevel = 5; newMediumLevel = 3; newFullLevel = 1; break; // box 800x800
|
||||
case 'y': newThumbLevel = 6; newMediumLevel = 6; newFullLevel = 0; break; // box 1280x1280
|
||||
case 'w': newThumbLevel = 8; newMediumLevel = 8; newFullLevel = 2; break; // box 2560x2560 // if loading this fix HistoryPhoto::updateFrom
|
||||
case 'a': newThumbLevel = 1; newMediumLevel = 4; newFullLevel = 8; break; // crop 160x160
|
||||
case 'b': newThumbLevel = 3; newMediumLevel = 1; newFullLevel = 7; break; // crop 320x320
|
||||
case 'c': newThumbLevel = 4; newMediumLevel = 2; newFullLevel = 6; break; // crop 640x640
|
||||
case 'd': newThumbLevel = 7; newMediumLevel = 7; newFullLevel = 5; break; // crop 1280x1280
|
||||
}
|
||||
if (newThumbLevel < 0 || newMediumLevel < 0 || newFullLevel < 0) {
|
||||
continue;
|
||||
}
|
||||
if (thumbLevel < 0 || newThumbLevel < thumbLevel) {
|
||||
thumbLevel = newThumbLevel;
|
||||
thumb = &i.value();
|
||||
}
|
||||
if (mediumLevel < 0 || newMediumLevel < mediumLevel) {
|
||||
mediumLevel = newMediumLevel;
|
||||
medium = &i.value();
|
||||
}
|
||||
if (fullLevel < 0 || newFullLevel < fullLevel) {
|
||||
fullLevel = newFullLevel;
|
||||
full = &i.value();
|
||||
}
|
||||
}
|
||||
if (!thumb || !medium || !full) {
|
||||
return App::photo(0);
|
||||
}
|
||||
switch (photo.type()) {
|
||||
case mtpc_photo: {
|
||||
const auto &ph(photo.c_photo());
|
||||
return App::photoSet(ph.vid.v, 0, ph.vaccess_hash.v, ph.vdate.v, ImagePtr(*thumb, "JPG"), ImagePtr(*medium, "JPG"), ImagePtr(*full, "JPG"));
|
||||
} break;
|
||||
case mtpc_photoEmpty: return App::photo(photo.c_photoEmpty().vid.v);
|
||||
}
|
||||
return App::photo(0);
|
||||
}
|
||||
|
||||
PhotoData *feedPhoto(const MTPDphoto &photo, PhotoData *convert) {
|
||||
auto &sizes = photo.vsizes.v;
|
||||
const MTPPhotoSize *thumb = 0, *medium = 0, *full = 0;
|
||||
int32 thumbLevel = -1, mediumLevel = -1, fullLevel = -1;
|
||||
for (QVector<MTPPhotoSize>::const_iterator i = sizes.cbegin(), e = sizes.cend(); i != e; ++i) {
|
||||
char size = 0;
|
||||
switch (i->type()) {
|
||||
case mtpc_photoSize: {
|
||||
auto &s = i->c_photoSize().vtype.v;
|
||||
if (s.size()) size = s[0];
|
||||
} break;
|
||||
|
||||
case mtpc_photoCachedSize: {
|
||||
auto &s = i->c_photoCachedSize().vtype.v;
|
||||
if (s.size()) size = s[0];
|
||||
} break;
|
||||
}
|
||||
if (!size) continue;
|
||||
|
||||
int32 newThumbLevel = -1, newMediumLevel = -1, newFullLevel = -1;
|
||||
switch (size) {
|
||||
case 's': newThumbLevel = 0; newMediumLevel = 5; newFullLevel = 4; break; // box 100x100
|
||||
case 'm': newThumbLevel = 2; newMediumLevel = 0; newFullLevel = 3; break; // box 320x320
|
||||
case 'x': newThumbLevel = 5; newMediumLevel = 3; newFullLevel = 1; break; // box 800x800
|
||||
case 'y': newThumbLevel = 6; newMediumLevel = 6; newFullLevel = 0; break; // box 1280x1280
|
||||
case 'w': newThumbLevel = 8; newMediumLevel = 8; newFullLevel = 2; break; // box 2560x2560
|
||||
case 'a': newThumbLevel = 1; newMediumLevel = 4; newFullLevel = 8; break; // crop 160x160
|
||||
case 'b': newThumbLevel = 3; newMediumLevel = 1; newFullLevel = 7; break; // crop 320x320
|
||||
case 'c': newThumbLevel = 4; newMediumLevel = 2; newFullLevel = 6; break; // crop 640x640
|
||||
case 'd': newThumbLevel = 7; newMediumLevel = 7; newFullLevel = 5; break; // crop 1280x1280
|
||||
}
|
||||
if (newThumbLevel < 0 || newMediumLevel < 0 || newFullLevel < 0) {
|
||||
continue;
|
||||
}
|
||||
if (thumbLevel < 0 || newThumbLevel < thumbLevel) {
|
||||
thumbLevel = newThumbLevel;
|
||||
thumb = &(*i);
|
||||
}
|
||||
if (mediumLevel < 0 || newMediumLevel < mediumLevel) {
|
||||
mediumLevel = newMediumLevel;
|
||||
medium = &(*i);
|
||||
}
|
||||
if (fullLevel < 0 || newFullLevel < fullLevel) {
|
||||
fullLevel = newFullLevel;
|
||||
full = &(*i);
|
||||
}
|
||||
}
|
||||
if (thumb && medium && full) {
|
||||
return App::photoSet(photo.vid.v, convert, photo.vaccess_hash.v, photo.vdate.v, App::image(*thumb), App::image(*medium), App::image(*full));
|
||||
}
|
||||
return App::photoSet(photo.vid.v, convert, 0, 0, ImagePtr(), ImagePtr(), ImagePtr());
|
||||
}
|
||||
|
||||
DocumentData *feedDocument(const MTPdocument &document, const QPixmap &thumb) {
|
||||
switch (document.type()) {
|
||||
case mtpc_document: {
|
||||
auto &d = document.c_document();
|
||||
return App::documentSet(d.vid.v, 0, d.vaccess_hash.v, d.vversion.v, d.vdate.v, d.vattributes.v, qs(d.vmime_type), ImagePtr(thumb, "JPG"), d.vdc_id.v, d.vsize.v, StorageImageLocation());
|
||||
} break;
|
||||
case mtpc_documentEmpty: return App::document(document.c_documentEmpty().vid.v);
|
||||
}
|
||||
return App::document(0);
|
||||
}
|
||||
|
||||
DocumentData *feedDocument(const MTPdocument &document, DocumentData *convert) {
|
||||
switch (document.type()) {
|
||||
case mtpc_document: {
|
||||
return feedDocument(document.c_document(), convert);
|
||||
} break;
|
||||
case mtpc_documentEmpty: {
|
||||
return App::documentSet(document.c_documentEmpty().vid.v, convert, 0, 0, 0, QVector<MTPDocumentAttribute>(), QString(), ImagePtr(), 0, 0, StorageImageLocation());
|
||||
} break;
|
||||
}
|
||||
return App::document(0);
|
||||
}
|
||||
|
||||
DocumentData *feedDocument(const MTPDdocument &document, DocumentData *convert) {
|
||||
return App::documentSet(
|
||||
document.vid.v,
|
||||
convert,
|
||||
document.vaccess_hash.v,
|
||||
document.vversion.v,
|
||||
document.vdate.v,
|
||||
document.vattributes.v,
|
||||
qs(document.vmime_type),
|
||||
App::image(document.vthumb),
|
||||
document.vdc_id.v,
|
||||
document.vsize.v,
|
||||
StorageImageLocation::FromMTP(document.vthumb));
|
||||
}
|
||||
|
||||
WebPageData *feedWebPage(const MTPDwebPage &webpage, WebPageData *convert) {
|
||||
auto description = TextWithEntities { webpage.has_description() ? TextUtilities::Clean(qs(webpage.vdescription)) : QString() };
|
||||
auto siteName = webpage.has_site_name() ? qs(webpage.vsite_name) : QString();
|
||||
auto parseFlags = TextParseLinks | TextParseMultiline | TextParseRichText;
|
||||
if (siteName == qstr("Twitter") || siteName == qstr("Instagram")) {
|
||||
parseFlags |= TextParseHashtags | TextParseMentions;
|
||||
}
|
||||
TextUtilities::ParseEntities(description, parseFlags);
|
||||
return App::webPageSet(webpage.vid.v, convert, webpage.has_type() ? qs(webpage.vtype) : qsl("article"), qs(webpage.vurl), qs(webpage.vdisplay_url), siteName, webpage.has_title() ? qs(webpage.vtitle) : QString(), description, webpage.has_photo() ? App::feedPhoto(webpage.vphoto) : nullptr, webpage.has_document() ? App::feedDocument(webpage.vdocument) : nullptr, webpage.has_duration() ? webpage.vduration.v : 0, webpage.has_author() ? qs(webpage.vauthor) : QString(), 0);
|
||||
}
|
||||
|
||||
WebPageData *feedWebPage(const MTPDwebPagePending &webpage, WebPageData *convert) {
|
||||
constexpr auto kDefaultPendingTimeout = 60;
|
||||
return App::webPageSet(
|
||||
webpage.vid.v,
|
||||
convert,
|
||||
QString(),
|
||||
QString(),
|
||||
QString(),
|
||||
QString(),
|
||||
QString(),
|
||||
TextWithEntities(),
|
||||
nullptr,
|
||||
nullptr,
|
||||
0,
|
||||
QString(),
|
||||
webpage.vdate.v
|
||||
? webpage.vdate.v
|
||||
: (unixtime() + kDefaultPendingTimeout));
|
||||
}
|
||||
|
||||
WebPageData *feedWebPage(const MTPWebPage &webpage) {
|
||||
switch (webpage.type()) {
|
||||
case mtpc_webPage: return App::feedWebPage(webpage.c_webPage());
|
||||
case mtpc_webPageEmpty: {
|
||||
WebPageData *page = App::webPage(webpage.c_webPageEmpty().vid.v);
|
||||
if (page->pendingTill > 0) page->pendingTill = -1; // failed
|
||||
return page;
|
||||
} break;
|
||||
case mtpc_webPagePending: return App::feedWebPage(webpage.c_webPagePending());
|
||||
case mtpc_webPageNotModified: LOG(("API Error: webPageNotModified is unexpected in feedWebPage().")); break;
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
WebPageData *feedWebPage(WebPageId webPageId, const QString &siteName, const TextWithEntities &content) {
|
||||
return App::webPageSet(webPageId, nullptr, qsl("article"), QString(), QString(), siteName, QString(), content, nullptr, nullptr, 0, QString(), 0);
|
||||
}
|
||||
|
||||
GameData *feedGame(const MTPDgame &game, GameData *convert) {
|
||||
return App::gameSet(game.vid.v, convert, game.vaccess_hash.v, qs(game.vshort_name), qs(game.vtitle), qs(game.vdescription), App::feedPhoto(game.vphoto), game.has_document() ? App::feedDocument(game.vdocument) : nullptr);
|
||||
}
|
||||
|
||||
PeerData *peer(const PeerId &id, PeerData::LoadedStatus restriction) {
|
||||
if (!id) return nullptr;
|
||||
|
||||
|
@ -1402,361 +1181,6 @@ namespace {
|
|||
return nullptr;
|
||||
}
|
||||
|
||||
void updateImage(ImagePtr &old, ImagePtr now) {
|
||||
if (now->isNull()) return;
|
||||
if (old->isNull()) {
|
||||
old = now;
|
||||
} else if (DelayedStorageImage *img = old->toDelayedStorageImage()) {
|
||||
StorageImageLocation loc = now->location();
|
||||
if (!loc.isNull()) {
|
||||
img->setStorageLocation(loc);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
PhotoData *photo(const PhotoId &photo) {
|
||||
auto i = ::photosData.constFind(photo);
|
||||
if (i == ::photosData.cend()) {
|
||||
i = ::photosData.insert(photo, new PhotoData(photo));
|
||||
}
|
||||
return i.value();
|
||||
}
|
||||
|
||||
PhotoData *photoSet(
|
||||
const PhotoId &photo,
|
||||
PhotoData *convert,
|
||||
const uint64 &access,
|
||||
int32 date,
|
||||
const ImagePtr &thumb,
|
||||
const ImagePtr &medium,
|
||||
const ImagePtr &full) {
|
||||
if (convert) {
|
||||
if (convert->id != photo) {
|
||||
const auto i = ::photosData.find(convert->id);
|
||||
if (i != ::photosData.cend() && i.value() == convert) {
|
||||
::photosData.erase(i);
|
||||
}
|
||||
convert->id = photo;
|
||||
convert->uploadingData = nullptr;
|
||||
}
|
||||
if (date) {
|
||||
convert->access = access;
|
||||
convert->date = date;
|
||||
updateImage(convert->thumb, thumb);
|
||||
updateImage(convert->medium, medium);
|
||||
updateImage(convert->full, full);
|
||||
}
|
||||
}
|
||||
const auto i = ::photosData.constFind(photo);
|
||||
PhotoData *result;
|
||||
auto inLastIter = lastPhotosMap.end();
|
||||
if (i == ::photosData.cend()) {
|
||||
if (convert) {
|
||||
result = convert;
|
||||
} else {
|
||||
result = new PhotoData(photo, access, date, thumb, medium, full);
|
||||
}
|
||||
::photosData.insert(photo, result);
|
||||
} else {
|
||||
result = i.value();
|
||||
if (result != convert && date) {
|
||||
result->access = access;
|
||||
result->date = date;
|
||||
updateImage(result->thumb, thumb);
|
||||
updateImage(result->medium, medium);
|
||||
updateImage(result->full, full);
|
||||
}
|
||||
inLastIter = lastPhotosMap.find(result);
|
||||
}
|
||||
if (inLastIter == lastPhotosMap.end()) { // insert new one
|
||||
if (lastPhotos.size() == MaxPhotosInMemory) {
|
||||
lastPhotos.front()->forget();
|
||||
lastPhotosMap.remove(lastPhotos.front());
|
||||
lastPhotos.pop_front();
|
||||
}
|
||||
lastPhotosMap.insert(result, lastPhotos.insert(lastPhotos.end(), result));
|
||||
} else {
|
||||
lastPhotos.erase(inLastIter.value()); // move to back
|
||||
(*inLastIter) = lastPhotos.insert(lastPhotos.end(), result);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
DocumentData *document(const DocumentId &document) {
|
||||
auto i = ::documentsData.constFind(document);
|
||||
if (i == ::documentsData.cend()) {
|
||||
i = ::documentsData.insert(document, DocumentData::create(document));
|
||||
}
|
||||
return i.value();
|
||||
}
|
||||
|
||||
DocumentData *documentSet(
|
||||
const DocumentId &document,
|
||||
DocumentData *convert,
|
||||
const uint64 &access,
|
||||
int32 version,
|
||||
int32 date,
|
||||
const QVector<MTPDocumentAttribute> &attributes,
|
||||
const QString &mime,
|
||||
const ImagePtr &thumb,
|
||||
int32 dc,
|
||||
int32 size,
|
||||
const StorageImageLocation &thumbLocation) {
|
||||
bool versionChanged = false;
|
||||
bool sentSticker = false;
|
||||
if (convert) {
|
||||
const auto oldKey = convert->mediaKey();
|
||||
const auto idChanged = (convert->id != document);
|
||||
if (idChanged) {
|
||||
const auto i = ::documentsData.find(convert->id);
|
||||
if (i != ::documentsData.cend() && i.value() == convert) {
|
||||
::documentsData.erase(i);
|
||||
}
|
||||
|
||||
convert->id = document;
|
||||
convert->status = FileReady;
|
||||
convert->uploadingData = nullptr;
|
||||
sentSticker = (convert->sticker() != 0);
|
||||
}
|
||||
if (date) {
|
||||
convert->setattributes(attributes);
|
||||
versionChanged = convert->setRemoteVersion(version);
|
||||
convert->setRemoteLocation(dc, access);
|
||||
convert->date = date;
|
||||
convert->setMimeString(mime);
|
||||
if (!thumb->isNull() && (convert->thumb->isNull() || convert->thumb->width() < thumb->width() || convert->thumb->height() < thumb->height() || versionChanged)) {
|
||||
updateImage(convert->thumb, thumb);
|
||||
}
|
||||
convert->size = size;
|
||||
convert->recountIsImage();
|
||||
if (convert->sticker() && convert->sticker()->loc.isNull() && !thumbLocation.isNull()) {
|
||||
convert->sticker()->loc = thumbLocation;
|
||||
}
|
||||
|
||||
const auto newKey = convert->mediaKey();
|
||||
if (idChanged) {
|
||||
if (convert->isVoiceMessage()) {
|
||||
Local::copyAudio(oldKey, newKey);
|
||||
} else if (convert->sticker() || convert->isAnimation()) {
|
||||
Local::copyStickerImage(oldKey, newKey);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (Auth().data().savedGifs().indexOf(convert) >= 0) { // id changed
|
||||
Local::writeSavedGifs();
|
||||
}
|
||||
}
|
||||
const auto i = ::documentsData.constFind(document);
|
||||
DocumentData *result;
|
||||
if (i == ::documentsData.cend()) {
|
||||
if (convert) {
|
||||
result = convert;
|
||||
} else {
|
||||
result = DocumentData::create(document, dc, access, version, attributes);
|
||||
result->date = date;
|
||||
result->setMimeString(mime);
|
||||
result->thumb = thumb;
|
||||
result->size = size;
|
||||
result->recountIsImage();
|
||||
if (result->sticker()) {
|
||||
result->sticker()->loc = thumbLocation;
|
||||
}
|
||||
}
|
||||
::documentsData.insert(document, result);
|
||||
} else {
|
||||
result = i.value();
|
||||
if (result != convert && date) {
|
||||
result->setattributes(attributes);
|
||||
versionChanged = result->setRemoteVersion(version);
|
||||
if (!result->isValid()) {
|
||||
result->setRemoteLocation(dc, access);
|
||||
}
|
||||
result->date = date;
|
||||
result->setMimeString(mime);
|
||||
if (!thumb->isNull() && (result->thumb->isNull() || result->thumb->width() < thumb->width() || result->thumb->height() < thumb->height() || versionChanged)) {
|
||||
result->thumb = thumb;
|
||||
}
|
||||
result->size = size;
|
||||
result->recountIsImage();
|
||||
if (result->sticker() && result->sticker()->loc.isNull() && !thumbLocation.isNull()) {
|
||||
result->sticker()->loc = thumbLocation;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (sentSticker && App::main()) {
|
||||
App::main()->incrementSticker(result);
|
||||
}
|
||||
if (versionChanged) {
|
||||
if (result->sticker() && result->sticker()->set.type() == mtpc_inputStickerSetID) {
|
||||
auto it = Auth().data().stickerSets().constFind(result->sticker()->set.c_inputStickerSetID().vid.v);
|
||||
if (it != Auth().data().stickerSets().cend()) {
|
||||
if (it->id == Stickers::CloudRecentSetId) {
|
||||
Local::writeRecentStickers();
|
||||
} else if (it->id == Stickers::FavedSetId) {
|
||||
Local::writeFavedStickers();
|
||||
} else if (it->flags & MTPDstickerSet::Flag::f_archived) {
|
||||
Local::writeArchivedStickers();
|
||||
} else if (it->flags & MTPDstickerSet::Flag::f_installed) {
|
||||
Local::writeInstalledStickers();
|
||||
}
|
||||
if (it->flags & MTPDstickerSet_ClientFlag::f_featured) {
|
||||
Local::writeFeaturedStickers();
|
||||
}
|
||||
}
|
||||
}
|
||||
auto &items = App::documentItems();
|
||||
auto i = items.constFind(result);
|
||||
if (i != items.cend()) {
|
||||
for_const (auto item, i.value()) {
|
||||
Auth().data().requestItemViewResize(item);
|
||||
}
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
WebPageData *webPage(const WebPageId &webPage) {
|
||||
auto i = webPagesData.constFind(webPage);
|
||||
if (i == webPagesData.cend()) {
|
||||
i = webPagesData.insert(webPage, new WebPageData(webPage));
|
||||
}
|
||||
return i.value();
|
||||
}
|
||||
|
||||
WebPageData *webPageSet(
|
||||
const WebPageId &webPage,
|
||||
WebPageData *convert,
|
||||
const QString &type,
|
||||
const QString &url,
|
||||
const QString &displayUrl,
|
||||
const QString &siteName,
|
||||
const QString &title,
|
||||
const TextWithEntities &description,
|
||||
PhotoData *photo,
|
||||
DocumentData *document,
|
||||
int duration,
|
||||
const QString &author,
|
||||
int pendingTill) {
|
||||
if (convert) {
|
||||
if (convert->id != webPage) {
|
||||
const auto i = webPagesData.find(convert->id);
|
||||
if (i != webPagesData.cend() && i.value() == convert) {
|
||||
webPagesData.erase(i);
|
||||
}
|
||||
convert->id = webPage;
|
||||
}
|
||||
convert->applyChanges(
|
||||
type,
|
||||
url,
|
||||
displayUrl,
|
||||
siteName,
|
||||
title,
|
||||
description,
|
||||
photo,
|
||||
document,
|
||||
duration,
|
||||
author,
|
||||
pendingTill);
|
||||
}
|
||||
const auto i = webPagesData.constFind(webPage);
|
||||
WebPageData *result;
|
||||
if (i == webPagesData.cend()) {
|
||||
if (convert) {
|
||||
result = convert;
|
||||
} else {
|
||||
result = new WebPageData(
|
||||
webPage,
|
||||
toWebPageType(type),
|
||||
url,
|
||||
displayUrl,
|
||||
siteName,
|
||||
title,
|
||||
description,
|
||||
document,
|
||||
photo,
|
||||
duration,
|
||||
author,
|
||||
(pendingTill >= -1) ? pendingTill : -1);
|
||||
if (pendingTill > 0) {
|
||||
Auth().api().requestWebPageDelayed(result);
|
||||
}
|
||||
}
|
||||
webPagesData.insert(webPage, result);
|
||||
} else {
|
||||
result = i.value();
|
||||
if (result != convert) {
|
||||
result->applyChanges(
|
||||
type,
|
||||
url,
|
||||
displayUrl,
|
||||
siteName,
|
||||
title,
|
||||
description,
|
||||
photo,
|
||||
document,
|
||||
duration,
|
||||
author,
|
||||
pendingTill);
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
GameData *game(const GameId &game) {
|
||||
auto i = gamesData.constFind(game);
|
||||
if (i == gamesData.cend()) {
|
||||
i = gamesData.insert(game, new GameData(game));
|
||||
}
|
||||
return i.value();
|
||||
}
|
||||
|
||||
GameData *gameSet(const GameId &game, GameData *convert, const uint64 &accessHash, const QString &shortName, const QString &title, const QString &description, PhotoData *photo, DocumentData *document) {
|
||||
if (convert) {
|
||||
if (convert->id != game) {
|
||||
auto i = gamesData.find(convert->id);
|
||||
if (i != gamesData.cend() && i.value() == convert) {
|
||||
gamesData.erase(i);
|
||||
}
|
||||
convert->id = game;
|
||||
convert->accessHash = 0;
|
||||
}
|
||||
if (!convert->accessHash && accessHash) {
|
||||
convert->accessHash = accessHash;
|
||||
convert->shortName = TextUtilities::Clean(shortName);
|
||||
convert->title = TextUtilities::SingleLine(title);
|
||||
convert->description = TextUtilities::Clean(description);
|
||||
convert->photo = photo;
|
||||
convert->document = document;
|
||||
if (App::main()) App::main()->gameUpdated(convert);
|
||||
}
|
||||
}
|
||||
auto i = gamesData.constFind(game);
|
||||
GameData *result;
|
||||
if (i == gamesData.cend()) {
|
||||
if (convert) {
|
||||
result = convert;
|
||||
} else {
|
||||
result = new GameData(game, accessHash, shortName, title, description, photo, document);
|
||||
}
|
||||
gamesData.insert(game, result);
|
||||
} else {
|
||||
result = i.value();
|
||||
if (result != convert) {
|
||||
if (!result->accessHash && accessHash) {
|
||||
result->accessHash = accessHash;
|
||||
result->shortName = TextUtilities::Clean(shortName);
|
||||
result->title = TextUtilities::SingleLine(title);
|
||||
result->description = TextUtilities::Clean(description);
|
||||
result->photo = photo;
|
||||
result->document = document;
|
||||
if (App::main()) App::main()->gameUpdated(result);
|
||||
}
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
LocationData *location(const LocationCoords &coords) {
|
||||
auto i = locationsData.constFind(coords);
|
||||
if (i == locationsData.cend()) {
|
||||
|
@ -1766,14 +1190,6 @@ namespace {
|
|||
}
|
||||
|
||||
void forgetMedia() {
|
||||
lastPhotos.clear();
|
||||
lastPhotosMap.clear();
|
||||
for_const (auto photo, ::photosData) {
|
||||
photo->forget();
|
||||
}
|
||||
for_const (auto document, ::documentsData) {
|
||||
document->forget();
|
||||
}
|
||||
for_const (auto location, ::locationsData) {
|
||||
location->thumb->forget();
|
||||
}
|
||||
|
@ -1914,22 +1330,6 @@ namespace {
|
|||
delete peer;
|
||||
}
|
||||
::peersData.clear();
|
||||
for_const (auto game, ::gamesData) {
|
||||
delete game;
|
||||
}
|
||||
::gamesData.clear();
|
||||
for_const (auto webpage, ::webPagesData) {
|
||||
delete webpage;
|
||||
}
|
||||
::webPagesData.clear();
|
||||
for_const (auto photo, ::photosData) {
|
||||
delete photo;
|
||||
}
|
||||
::photosData.clear();
|
||||
for_const (auto document, ::documentsData) {
|
||||
delete document;
|
||||
}
|
||||
::documentsData.clear();
|
||||
|
||||
if (AuthSession::Exists()) {
|
||||
Auth().api().clearWebPageRequests();
|
||||
|
@ -1939,14 +1339,6 @@ namespace {
|
|||
cSetAutoDownloadPhoto(0);
|
||||
cSetAutoDownloadAudio(0);
|
||||
cSetAutoDownloadGif(0);
|
||||
::photoItems.clear();
|
||||
::documentItems.clear();
|
||||
::webPageItems.clear();
|
||||
::gameItems.clear();
|
||||
::sharedContactItems.clear();
|
||||
::gifItems.clear();
|
||||
lastPhotos.clear();
|
||||
lastPhotosMap.clear();
|
||||
::self = nullptr;
|
||||
Global::RefSelfChanged().notify(true);
|
||||
}
|
||||
|
@ -2367,118 +1759,6 @@ namespace {
|
|||
return QPixmap::fromImage(std::move(image), Qt::ColorOnly);
|
||||
}
|
||||
|
||||
void regPhotoItem(PhotoData *data, HistoryItem *item) {
|
||||
::photoItems[data].insert(item);
|
||||
}
|
||||
|
||||
void unregPhotoItem(PhotoData *data, HistoryItem *item) {
|
||||
::photoItems[data].remove(item);
|
||||
}
|
||||
|
||||
const PhotoItems &photoItems() {
|
||||
return ::photoItems;
|
||||
}
|
||||
|
||||
const PhotosData &photosData() {
|
||||
return ::photosData;
|
||||
}
|
||||
|
||||
void regDocumentItem(DocumentData *data, HistoryItem *item) {
|
||||
::documentItems[data].insert(item);
|
||||
}
|
||||
|
||||
void unregDocumentItem(DocumentData *data, HistoryItem *item) {
|
||||
::documentItems[data].remove(item);
|
||||
}
|
||||
|
||||
const DocumentItems &documentItems() {
|
||||
return ::documentItems;
|
||||
}
|
||||
|
||||
const DocumentsData &documentsData() {
|
||||
return ::documentsData;
|
||||
}
|
||||
|
||||
void regWebPageItem(WebPageData *data, HistoryItem *item) {
|
||||
::webPageItems[data].insert(item);
|
||||
}
|
||||
|
||||
void unregWebPageItem(WebPageData *data, HistoryItem *item) {
|
||||
::webPageItems[data].remove(item);
|
||||
}
|
||||
|
||||
const WebPageItems &webPageItems() {
|
||||
return ::webPageItems;
|
||||
}
|
||||
|
||||
void regGameItem(GameData *data, HistoryItem *item) {
|
||||
::gameItems[data].insert(item);
|
||||
}
|
||||
|
||||
void unregGameItem(GameData *data, HistoryItem *item) {
|
||||
::gameItems[data].remove(item);
|
||||
}
|
||||
|
||||
const GameItems &gameItems() {
|
||||
return ::gameItems;
|
||||
}
|
||||
|
||||
void regSharedContactItem(int32 userId, HistoryItem *item) {
|
||||
auto user = App::userLoaded(userId);
|
||||
auto canShareThisContact = user ? user->canShareThisContact() : false;
|
||||
::sharedContactItems[userId].insert(item);
|
||||
if (canShareThisContact != (user ? user->canShareThisContact() : false)) {
|
||||
Notify::peerUpdatedDelayed(user, Notify::PeerUpdate::Flag::UserCanShareContact);
|
||||
}
|
||||
}
|
||||
|
||||
void unregSharedContactItem(int32 userId, HistoryItem *item) {
|
||||
auto user = App::userLoaded(userId);
|
||||
auto canShareThisContact = user ? user->canShareThisContact() : false;
|
||||
::sharedContactItems[userId].remove(item);
|
||||
if (canShareThisContact != (user ? user->canShareThisContact() : false)) {
|
||||
Notify::peerUpdatedDelayed(user, Notify::PeerUpdate::Flag::UserCanShareContact);
|
||||
}
|
||||
}
|
||||
|
||||
const SharedContactItems &sharedContactItems() {
|
||||
return ::sharedContactItems;
|
||||
}
|
||||
|
||||
void regGifItem(Media::Clip::Reader *reader, HistoryItem *item) {
|
||||
::gifItems.insert(reader, item);
|
||||
}
|
||||
|
||||
void unregGifItem(Media::Clip::Reader *reader) {
|
||||
::gifItems.remove(reader);
|
||||
}
|
||||
|
||||
void stopGifItems() {
|
||||
if (!::gifItems.isEmpty()) {
|
||||
auto gifs = ::gifItems;
|
||||
for_const (auto item, gifs) {
|
||||
// #TODO GIFs
|
||||
//if (auto media = item->getMedia()) {
|
||||
// if (!media->isRoundVideoPlaying()) {
|
||||
// media->stopInline();
|
||||
// }
|
||||
//}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
QString phoneFromSharedContact(int32 userId) {
|
||||
auto i = ::sharedContactItems.constFind(userId);
|
||||
if (i != ::sharedContactItems.cend() && !i->empty()) {
|
||||
if (const auto media = (*i->cbegin())->media()) {
|
||||
if (const auto contact = media->sharedContact()) {
|
||||
return contact->phoneNumber;
|
||||
}
|
||||
}
|
||||
}
|
||||
return QString();
|
||||
}
|
||||
|
||||
void regMuted(not_null<PeerData*> peer, TimeMs changeIn) {
|
||||
::mutedPeers.insert(peer, true);
|
||||
App::main()->updateMutedIn(changeIn);
|
||||
|
|
|
@ -25,16 +25,8 @@ class Element;
|
|||
} // namespace HistoryView
|
||||
|
||||
using HistoryItemsMap = base::flat_set<not_null<HistoryItem*>>;
|
||||
using PhotoItems = QHash<PhotoData*, HistoryItemsMap>;
|
||||
using DocumentItems = QHash<DocumentData*, HistoryItemsMap>;
|
||||
using WebPageItems = QHash<WebPageData*, HistoryItemsMap>;
|
||||
using GameItems = QHash<GameData*, HistoryItemsMap>;
|
||||
using SharedContactItems = QHash<int32, HistoryItemsMap>;
|
||||
using GifItems = QHash<Media::Clip::Reader*, HistoryItem*>;
|
||||
|
||||
using PhotosData = QHash<PhotoId, PhotoData*>;
|
||||
using DocumentsData = QHash<DocumentId, DocumentData*>;
|
||||
|
||||
enum RoundCorners {
|
||||
SmallMaskCorners = 0x00, // for images
|
||||
LargeMaskCorners,
|
||||
|
@ -103,18 +95,6 @@ namespace App {
|
|||
|
||||
ImagePtr image(const MTPPhotoSize &size);
|
||||
|
||||
PhotoData *feedPhoto(const MTPPhoto &photo, const PreparedPhotoThumbs &thumbs);
|
||||
PhotoData *feedPhoto(const MTPPhoto &photo, PhotoData *convert = nullptr);
|
||||
PhotoData *feedPhoto(const MTPDphoto &photo, PhotoData *convert = nullptr);
|
||||
DocumentData *feedDocument(const MTPdocument &document, const QPixmap &thumb);
|
||||
DocumentData *feedDocument(const MTPdocument &document, DocumentData *convert = nullptr);
|
||||
DocumentData *feedDocument(const MTPDdocument &document, DocumentData *convert = nullptr);
|
||||
WebPageData *feedWebPage(const MTPDwebPage &webpage, WebPageData *convert = nullptr);
|
||||
WebPageData *feedWebPage(const MTPDwebPagePending &webpage, WebPageData *convert = nullptr);
|
||||
WebPageData *feedWebPage(const MTPWebPage &webpage);
|
||||
WebPageData *feedWebPage(WebPageId webPageId, const QString &siteName, const TextWithEntities &content);
|
||||
GameData *feedGame(const MTPDgame &game, GameData *convert = nullptr);
|
||||
|
||||
PeerData *peer(const PeerId &id, PeerData::LoadedStatus restriction = PeerData::NotLoaded);
|
||||
inline UserData *user(const PeerId &id, PeerData::LoadedStatus restriction = PeerData::NotLoaded) {
|
||||
return asUser(peer(id, restriction));
|
||||
|
@ -160,53 +140,7 @@ namespace App {
|
|||
UserData *self();
|
||||
PeerData *peerByName(const QString &username);
|
||||
QString peerName(const PeerData *peer, bool forDialogs = false);
|
||||
PhotoData *photo(const PhotoId &photo);
|
||||
PhotoData *photoSet(
|
||||
const PhotoId &photo,
|
||||
PhotoData *convert,
|
||||
const uint64 &access,
|
||||
int32 date,
|
||||
const ImagePtr &thumb,
|
||||
const ImagePtr &medium,
|
||||
const ImagePtr &full);
|
||||
DocumentData *document(const DocumentId &document);
|
||||
DocumentData *documentSet(
|
||||
const DocumentId &document,
|
||||
DocumentData *convert,
|
||||
const uint64 &access,
|
||||
int32 version,
|
||||
int32 date,
|
||||
const QVector<MTPDocumentAttribute> &attributes,
|
||||
const QString &mime,
|
||||
const ImagePtr &thumb,
|
||||
int32 dc,
|
||||
int32 size,
|
||||
const StorageImageLocation &thumbLocation);
|
||||
WebPageData *webPage(const WebPageId &webPage);
|
||||
WebPageData *webPageSet(
|
||||
const WebPageId &webPage,
|
||||
WebPageData *convert,
|
||||
const QString &type,
|
||||
const QString &url,
|
||||
const QString &displayUrl,
|
||||
const QString &siteName,
|
||||
const QString &title,
|
||||
const TextWithEntities &description,
|
||||
PhotoData *photo,
|
||||
DocumentData *document,
|
||||
int duration,
|
||||
const QString &author,
|
||||
int pendingTill);
|
||||
GameData *game(const GameId &game);
|
||||
GameData *gameSet(
|
||||
const GameId &game,
|
||||
GameData *convert,
|
||||
const uint64 &accessHash,
|
||||
const QString &shortName,
|
||||
const QString &title,
|
||||
const QString &description,
|
||||
PhotoData *photo,
|
||||
DocumentData *document);
|
||||
|
||||
LocationData *location(const LocationCoords &coords);
|
||||
void forgetMedia();
|
||||
|
||||
|
@ -288,34 +222,6 @@ namespace App {
|
|||
QImage readImage(const QString &file, QByteArray *format = nullptr, bool opaque = true, bool *animated = nullptr, QByteArray *content = 0);
|
||||
QPixmap pixmapFromImageInPlace(QImage &&image);
|
||||
|
||||
void regPhotoItem(PhotoData *data, HistoryItem *item);
|
||||
void unregPhotoItem(PhotoData *data, HistoryItem *item);
|
||||
const PhotoItems &photoItems();
|
||||
const PhotosData &photosData();
|
||||
|
||||
void regDocumentItem(DocumentData *data, HistoryItem *item);
|
||||
void unregDocumentItem(DocumentData *data, HistoryItem *item);
|
||||
const DocumentItems &documentItems();
|
||||
const DocumentsData &documentsData();
|
||||
|
||||
void regWebPageItem(WebPageData *data, HistoryItem *item);
|
||||
void unregWebPageItem(WebPageData *data, HistoryItem *item);
|
||||
const WebPageItems &webPageItems();
|
||||
|
||||
void regGameItem(GameData *data, HistoryItem *item);
|
||||
void unregGameItem(GameData *data, HistoryItem *item);
|
||||
const GameItems &gameItems();
|
||||
|
||||
void regSharedContactItem(int32 userId, HistoryItem *item);
|
||||
void unregSharedContactItem(int32 userId, HistoryItem *item);
|
||||
const SharedContactItems &sharedContactItems();
|
||||
QString phoneFromSharedContact(int32 userId);
|
||||
|
||||
void regGifItem(Media::Clip::Reader *reader, HistoryItem *item);
|
||||
void unregGifItem(Media::Clip::Reader *reader);
|
||||
void stopRoundVideoPlayback();
|
||||
void stopGifItems();
|
||||
|
||||
void regMuted(not_null<PeerData*> peer, TimeMs changeIn);
|
||||
void unregMuted(not_null<PeerData*> peer);
|
||||
void updateMuted();
|
||||
|
|
|
@ -268,7 +268,7 @@ AuthSession &Auth() {
|
|||
AuthSession::AuthSession(UserId userId)
|
||||
: _userId(userId)
|
||||
, _autoLockTimer([this] { checkAutoLock(); })
|
||||
, _data(std::make_unique<Data::Session>())
|
||||
, _data(std::make_unique<Data::Session>(this))
|
||||
, _api(std::make_unique<ApiWrap>(this))
|
||||
, _calls(std::make_unique<Calls::Instance>())
|
||||
, _downloader(std::make_unique<Storage::Downloader>())
|
||||
|
|
|
@ -182,7 +182,9 @@ private:
|
|||
class AuthSession;
|
||||
AuthSession &Auth();
|
||||
|
||||
class AuthSession final : private base::Subscriber {
|
||||
class AuthSession final
|
||||
: public base::has_weak_ptr
|
||||
, private base::Subscriber {
|
||||
public:
|
||||
AuthSession(UserId userId);
|
||||
|
||||
|
|
|
@ -14,6 +14,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "storage/localstorage.h"
|
||||
#include "mainwidget.h"
|
||||
#include "mainwindow.h"
|
||||
#include "auth_session.h"
|
||||
#include "data/data_session.h"
|
||||
#include "ui/widgets/checkbox.h"
|
||||
#include "ui/widgets/buttons.h"
|
||||
#include "ui/widgets/input_fields.h"
|
||||
|
@ -291,55 +293,63 @@ void AutoDownloadBox::resizeEvent(QResizeEvent *e) {
|
|||
}
|
||||
|
||||
void AutoDownloadBox::onSave() {
|
||||
bool changed = false;
|
||||
int32 autoDownloadPhoto = (_photoPrivate->checked() ? 0 : dbiadNoPrivate) | (_photoGroups->checked() ? 0 : dbiadNoGroups);
|
||||
auto photosChanged = false;
|
||||
auto documentsChanged = false;
|
||||
auto autoplayChanged = false;
|
||||
auto photosEnabled = false;
|
||||
auto voiceEnabled = false;
|
||||
auto animationsEnabled = false;
|
||||
auto autoDownloadPhoto = (_photoPrivate->checked() ? 0 : dbiadNoPrivate)
|
||||
| (_photoGroups->checked() ? 0 : dbiadNoGroups);
|
||||
if (cAutoDownloadPhoto() != autoDownloadPhoto) {
|
||||
bool enabledPrivate = ((cAutoDownloadPhoto() & dbiadNoPrivate) && !(autoDownloadPhoto & dbiadNoPrivate));
|
||||
bool enabledGroups = ((cAutoDownloadPhoto() & dbiadNoGroups) && !(autoDownloadPhoto & dbiadNoGroups));
|
||||
const auto enabledPrivate = (cAutoDownloadPhoto() & dbiadNoPrivate)
|
||||
&& !(autoDownloadPhoto & dbiadNoPrivate);
|
||||
const auto enabledGroups = (cAutoDownloadPhoto() & dbiadNoGroups)
|
||||
&& !(autoDownloadPhoto & dbiadNoGroups);
|
||||
photosEnabled = enabledPrivate || enabledGroups;
|
||||
photosChanged = true;
|
||||
cSetAutoDownloadPhoto(autoDownloadPhoto);
|
||||
if (enabledPrivate || enabledGroups) {
|
||||
const PhotosData &data(App::photosData());
|
||||
for (PhotosData::const_iterator i = data.cbegin(), e = data.cend(); i != e; ++i) {
|
||||
i.value()->automaticLoadSettingsChanged();
|
||||
}
|
||||
}
|
||||
changed = true;
|
||||
}
|
||||
int32 autoDownloadAudio = (_audioPrivate->checked() ? 0 : dbiadNoPrivate) | (_audioGroups->checked() ? 0 : dbiadNoGroups);
|
||||
auto autoDownloadAudio = (_audioPrivate->checked() ? 0 : dbiadNoPrivate)
|
||||
| (_audioGroups->checked() ? 0 : dbiadNoGroups);
|
||||
if (cAutoDownloadAudio() != autoDownloadAudio) {
|
||||
bool enabledPrivate = ((cAutoDownloadAudio() & dbiadNoPrivate) && !(autoDownloadAudio & dbiadNoPrivate));
|
||||
bool enabledGroups = ((cAutoDownloadAudio() & dbiadNoGroups) && !(autoDownloadAudio & dbiadNoGroups));
|
||||
const auto enabledPrivate = (cAutoDownloadAudio() & dbiadNoPrivate)
|
||||
&& !(autoDownloadAudio & dbiadNoPrivate);
|
||||
const auto enabledGroups = (cAutoDownloadAudio() & dbiadNoGroups)
|
||||
&& !(autoDownloadAudio & dbiadNoGroups);
|
||||
voiceEnabled = enabledPrivate || enabledGroups;
|
||||
documentsChanged = true;
|
||||
cSetAutoDownloadAudio(autoDownloadAudio);
|
||||
if (enabledPrivate || enabledGroups) {
|
||||
for (auto document : App::documentsData()) {
|
||||
if (document->isVoiceMessage()) {
|
||||
document->automaticLoadSettingsChanged();
|
||||
}
|
||||
}
|
||||
}
|
||||
changed = true;
|
||||
}
|
||||
int32 autoDownloadGif = (_gifPrivate->checked() ? 0 : dbiadNoPrivate) | (_gifGroups->checked() ? 0 : dbiadNoGroups);
|
||||
auto autoDownloadGif = (_gifPrivate->checked() ? 0 : dbiadNoPrivate)
|
||||
| (_gifGroups->checked() ? 0 : dbiadNoGroups);
|
||||
if (cAutoDownloadGif() != autoDownloadGif) {
|
||||
bool enabledPrivate = ((cAutoDownloadGif() & dbiadNoPrivate) && !(autoDownloadGif & dbiadNoPrivate));
|
||||
bool enabledGroups = ((cAutoDownloadGif() & dbiadNoGroups) && !(autoDownloadGif & dbiadNoGroups));
|
||||
const auto enabledPrivate = (cAutoDownloadGif() & dbiadNoPrivate)
|
||||
&& !(autoDownloadGif & dbiadNoPrivate);
|
||||
const auto enabledGroups = (cAutoDownloadGif() & dbiadNoGroups)
|
||||
&& !(autoDownloadGif & dbiadNoGroups);
|
||||
animationsEnabled = enabledPrivate || enabledGroups;
|
||||
documentsChanged = true;
|
||||
cSetAutoDownloadGif(autoDownloadGif);
|
||||
if (enabledPrivate || enabledGroups) {
|
||||
for (auto document : App::documentsData()) {
|
||||
if (document->isAnimation()) {
|
||||
document->automaticLoadSettingsChanged();
|
||||
}
|
||||
}
|
||||
}
|
||||
changed = true;
|
||||
}
|
||||
if (cAutoPlayGif() != _gifPlay->checked()) {
|
||||
cSetAutoPlayGif(_gifPlay->checked());
|
||||
if (!cAutoPlayGif()) {
|
||||
App::stopGifItems();
|
||||
Auth().data().stopAutoplayAnimations();
|
||||
}
|
||||
changed = true;
|
||||
autoplayChanged = true;
|
||||
}
|
||||
if (photosChanged || documentsChanged || autoplayChanged) {
|
||||
Local::writeUserSettings();
|
||||
}
|
||||
if (photosEnabled) {
|
||||
Auth().data().photoLoadSettingsChanged();
|
||||
}
|
||||
if (voiceEnabled) {
|
||||
Auth().data().voiceLoadSettingsChanged();
|
||||
}
|
||||
if (animationsEnabled) {
|
||||
Auth().data().animationLoadSettingsChanged();
|
||||
}
|
||||
if (changed) Local::writeUserSettings();
|
||||
closeBox();
|
||||
}
|
||||
|
|
|
@ -122,8 +122,8 @@ void StickerSetBox::Inner::gotSet(const MTPmessages_StickerSet &set) {
|
|||
_pack.reserve(v.size());
|
||||
_packOvers.reserve(v.size());
|
||||
for (int i = 0, l = v.size(); i < l; ++i) {
|
||||
auto doc = App::feedDocument(v.at(i));
|
||||
if (!doc || !doc->sticker()) continue;
|
||||
auto doc = Auth().data().document(v.at(i));
|
||||
if (!doc->sticker()) continue;
|
||||
|
||||
_pack.push_back(doc);
|
||||
_packOvers.push_back(Animation());
|
||||
|
@ -139,7 +139,7 @@ void StickerSetBox::Inner::gotSet(const MTPmessages_StickerSet &set) {
|
|||
Stickers::Pack p;
|
||||
p.reserve(stickers.size());
|
||||
for (auto j = 0, c = stickers.size(); j != c; ++j) {
|
||||
auto doc = App::document(stickers[j].v);
|
||||
auto doc = Auth().data().document(stickers[j].v);
|
||||
if (!doc || !doc->sticker()) continue;
|
||||
|
||||
p.push_back(doc);
|
||||
|
|
|
@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "calls/calls_panel.h"
|
||||
|
||||
#include "data/data_photo.h"
|
||||
#include "data/data_session.h"
|
||||
#include "calls/calls_emoji_fingerprint.h"
|
||||
#include "styles/style_calls.h"
|
||||
#include "styles/style_history.h"
|
||||
|
@ -430,7 +431,7 @@ void Panel::processUserPhoto() {
|
|||
_user->loadUserpic(true);
|
||||
}
|
||||
const auto photo = _user->userpicPhotoId()
|
||||
? App::photo(_user->userpicPhotoId())
|
||||
? Auth().data().photo(_user->userpicPhotoId()).get()
|
||||
: nullptr;
|
||||
if (isGoodUserPhoto(photo)) {
|
||||
photo->full->load(true);
|
||||
|
@ -442,7 +443,7 @@ void Panel::processUserPhoto() {
|
|||
|
||||
void Panel::refreshUserPhoto() {
|
||||
const auto photo = _user->userpicPhotoId()
|
||||
? App::photo(_user->userpicPhotoId())
|
||||
? Auth().data().photo(_user->userpicPhotoId()).get()
|
||||
: nullptr;
|
||||
const auto isNewPhoto = [&](not_null<PhotoData*> photo) {
|
||||
return photo->full->loaded()
|
||||
|
|
|
@ -15,6 +15,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "apiwrap.h"
|
||||
#include "storage/localstorage.h"
|
||||
#include "mainwidget.h"
|
||||
#include "auth_session.h"
|
||||
#include "mainwindow.h"
|
||||
#include "ui/toast/toast.h"
|
||||
#include "styles/style_chat_helpers.h"
|
||||
|
@ -389,7 +390,7 @@ void SetPackAndEmoji(Set &set, Pack &&pack, const QVector<MTPStickerPack> &packs
|
|||
auto p = Pack();
|
||||
p.reserve(stickers.size());
|
||||
for (auto j = 0, c = stickers.size(); j != c; ++j) {
|
||||
auto document = App::document(stickers[j].v);
|
||||
auto document = Auth().data().document(stickers[j].v);
|
||||
if (!document || !document->sticker()) continue;
|
||||
|
||||
p.push_back(document);
|
||||
|
@ -420,8 +421,8 @@ void SpecialSetReceived(uint64 setId, const QString &setTitle, const QVector<MTP
|
|||
auto pack = Pack();
|
||||
pack.reserve(d_docs.size());
|
||||
for_const (auto &mtpDocument, d_docs) {
|
||||
auto document = App::feedDocument(mtpDocument);
|
||||
if (!document || !document->sticker()) continue;
|
||||
auto document = Auth().data().document(mtpDocument);
|
||||
if (!document->sticker()) continue;
|
||||
|
||||
pack.push_back(document);
|
||||
if (custom != sets.cend()) {
|
||||
|
@ -583,8 +584,8 @@ void GifsReceived(const QVector<MTPDocument> &items, int32 hash) {
|
|||
|
||||
saved.reserve(items.size());
|
||||
for_const (auto &gif, items) {
|
||||
auto document = App::feedDocument(gif);
|
||||
if (!document || !document->isGifv()) {
|
||||
auto document = Auth().data().document(gif);
|
||||
if (!document->isGifv()) {
|
||||
LOG(("API Error: bad document returned in HistoryWidget::savedGifsGot!"));
|
||||
continue;
|
||||
}
|
||||
|
@ -719,8 +720,8 @@ Set *FeedSetFull(const MTPmessages_StickerSet &data) {
|
|||
auto pack = Pack();
|
||||
pack.reserve(d_docs.size());
|
||||
for (auto i = 0, l = d_docs.size(); i != l; ++i) {
|
||||
auto doc = App::feedDocument(d_docs.at(i));
|
||||
if (!doc || !doc->sticker()) continue;
|
||||
auto doc = Auth().data().document(d_docs.at(i));
|
||||
if (!doc->sticker()) continue;
|
||||
|
||||
pack.push_back(doc);
|
||||
if (custom != sets.cend()) {
|
||||
|
@ -766,7 +767,7 @@ Set *FeedSetFull(const MTPmessages_StickerSet &data) {
|
|||
Pack p;
|
||||
p.reserve(stickers.size());
|
||||
for (auto j = 0, c = stickers.size(); j != c; ++j) {
|
||||
auto doc = App::document(stickers[j].v);
|
||||
auto doc = Auth().data().document(stickers[j].v);
|
||||
if (!doc || !doc->sticker()) continue;
|
||||
|
||||
p.push_back(doc);
|
||||
|
@ -812,7 +813,7 @@ RecentStickerPack &GetRecentPack() {
|
|||
auto &recent = cRefRecentStickers();
|
||||
recent.reserve(p.size());
|
||||
for (const auto &preloaded : p) {
|
||||
const auto document = App::document(preloaded.first);
|
||||
const auto document = Auth().data().document(preloaded.first);
|
||||
if (!document || !document->sticker()) continue;
|
||||
|
||||
recent.push_back(qMakePair(document, preloaded.second));
|
||||
|
|
|
@ -248,7 +248,7 @@ void DocumentOpenClickHandler::doOpen(
|
|||
Media::Player::mixer()->play(audio);
|
||||
Media::Player::Updated().notify(audio);
|
||||
if (App::main()) {
|
||||
App::main()->mediaMarkRead(data);
|
||||
Auth().data().markMediaRead(data);
|
||||
}
|
||||
}
|
||||
} else if (playMusic) {
|
||||
|
@ -276,13 +276,13 @@ void DocumentOpenClickHandler::doOpen(
|
|||
File::Launch(filepath);
|
||||
}
|
||||
}
|
||||
if (App::main()) App::main()->mediaMarkRead(data);
|
||||
Auth().data().markMediaRead(data);
|
||||
} else if (data->isVoiceMessage() || data->isAudioFile() || data->isVideoFile()) {
|
||||
auto filepath = location.name();
|
||||
if (documentIsValidMediaFile(filepath)) {
|
||||
File::Launch(filepath);
|
||||
}
|
||||
if (App::main()) App::main()->mediaMarkRead(data);
|
||||
Auth().data().markMediaRead(data);
|
||||
} else if (data->size < App::kImageSizeLimit) {
|
||||
if (!data->data().isEmpty() && playAnimation) {
|
||||
if (action == ActionOnLoadPlayInline && context) {
|
||||
|
@ -571,7 +571,7 @@ void DocumentData::performActionOnLoad() {
|
|||
}
|
||||
} else if (Media::Player::IsStopped(state.state)) {
|
||||
Media::Player::mixer()->play(AudioMsgId(this, _actionOnLoadMsgId));
|
||||
if (App::main()) App::main()->mediaMarkRead(this);
|
||||
Auth().data().markMediaRead(this);
|
||||
}
|
||||
}
|
||||
} else if (playMusic) {
|
||||
|
@ -607,7 +607,7 @@ void DocumentData::performActionOnLoad() {
|
|||
if (documentIsValidMediaFile(already)) {
|
||||
File::Launch(already);
|
||||
}
|
||||
if (App::main()) App::main()->mediaMarkRead(this);
|
||||
Auth().data().markMediaRead(this);
|
||||
} else if (loc.accessEnable()) {
|
||||
if (showImage && QImageReader(loc.name()).canRead()) {
|
||||
if (_actionOnLoad == ActionOnLoadPlayInline && item) {
|
||||
|
@ -640,7 +640,7 @@ bool DocumentData::loaded(FilePathResolveType type) const {
|
|||
}
|
||||
destroyLoaderDelayed();
|
||||
}
|
||||
notifyLayoutChanged();
|
||||
Auth().data().notifyDocumentLayoutChanged(this);
|
||||
}
|
||||
return !data().isEmpty() || !filepath(type).isEmpty();
|
||||
}
|
||||
|
@ -695,7 +695,12 @@ bool DocumentData::waitingForAlbum() const {
|
|||
return uploading() && uploadingData->waitingForAlbum;
|
||||
}
|
||||
|
||||
void DocumentData::save(const QString &toFile, ActionOnLoad action, const FullMsgId &actionMsgId, LoadFromCloudSetting fromCloud, bool autoLoading) {
|
||||
void DocumentData::save(
|
||||
const QString &toFile,
|
||||
ActionOnLoad action,
|
||||
const FullMsgId &actionMsgId,
|
||||
LoadFromCloudSetting fromCloud,
|
||||
bool autoLoading) {
|
||||
if (loaded(FilePathResolveChecked)) {
|
||||
auto &l = location(true);
|
||||
if (!toFile.isEmpty()) {
|
||||
|
@ -745,18 +750,20 @@ void DocumentData::save(const QString &toFile, ActionOnLoad action, const FullMs
|
|||
_loader->connect(_loader, SIGNAL(failed(FileLoader*,bool)), App::main(), SLOT(documentLoadFailed(FileLoader*,bool)));
|
||||
_loader->start();
|
||||
}
|
||||
notifyLayoutChanged();
|
||||
Auth().data().notifyDocumentLayoutChanged(this);
|
||||
}
|
||||
|
||||
void DocumentData::cancel() {
|
||||
if (!loading()) return;
|
||||
if (!loading()) {
|
||||
return;
|
||||
}
|
||||
|
||||
auto loader = std::unique_ptr<FileLoader>(std::exchange(_loader, CancelledMtpFileLoader));
|
||||
loader->cancel();
|
||||
loader->stop();
|
||||
Auth().downloader().delayedDestroyLoader(std::move(loader));
|
||||
|
||||
notifyLayoutChanged();
|
||||
Auth().data().notifyDocumentLayoutChanged(this);
|
||||
if (auto main = App::main()) {
|
||||
main->documentLoadProgress(this);
|
||||
}
|
||||
|
@ -764,19 +771,6 @@ void DocumentData::cancel() {
|
|||
_actionOnLoad = ActionOnLoadNone;
|
||||
}
|
||||
|
||||
void DocumentData::notifyLayoutChanged() const {
|
||||
auto &items = App::documentItems();
|
||||
for (auto item : items.value(const_cast<DocumentData*>(this))) {
|
||||
Auth().data().markItemLayoutChange(item);
|
||||
}
|
||||
|
||||
if (auto items = InlineBots::Layout::documentItems()) {
|
||||
for (auto item : items->value(const_cast<DocumentData*>(this))) {
|
||||
item->layoutChanged();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
VoiceWaveform documentWaveformDecode(const QByteArray &encoded5bit) {
|
||||
auto bitsCount = static_cast<int>(encoded5bit.size() * 8);
|
||||
auto valuesCount = bitsCount / 5;
|
||||
|
|
|
@ -290,8 +290,6 @@ private:
|
|||
FullMsgId _actionOnLoadMsgId;
|
||||
mutable FileLoader *_loader = nullptr;
|
||||
|
||||
void notifyLayoutChanged() const;
|
||||
|
||||
void destroyLoaderDelayed(
|
||||
mtpFileLoader *newValue = nullptr) const;
|
||||
|
||||
|
|
|
@ -8,16 +8,32 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "data/data_groups.h"
|
||||
|
||||
#include "history/history_item.h"
|
||||
#include "data/data_media_types.h"
|
||||
#include "data/data_session.h"
|
||||
|
||||
namespace Data {
|
||||
|
||||
Groups::Groups(not_null<Session*> data) : _data(data) {
|
||||
}
|
||||
|
||||
bool Groups::isGrouped(not_null<HistoryItem*> item) const {
|
||||
if (!item->groupId()) {
|
||||
return false;
|
||||
}
|
||||
const auto media = item->media();
|
||||
return media && media->canBeGrouped();
|
||||
}
|
||||
|
||||
void Groups::registerMessage(not_null<HistoryItem*> item) {
|
||||
const auto groupId = item->groupId();
|
||||
if (!groupId) {
|
||||
if (!isGrouped(item)) {
|
||||
return;
|
||||
}
|
||||
const auto i = _data.emplace(groupId, Group()).first;
|
||||
i->second.items.push_back(item);
|
||||
const auto i = _groups.emplace(item->groupId(), Group()).first;
|
||||
auto &items = i->second.items;
|
||||
items.insert(findPositionForItem(items, item), item);
|
||||
if (items.size() > 1) {
|
||||
refreshViews(items);
|
||||
}
|
||||
}
|
||||
|
||||
void Groups::unregisterMessage(not_null<HistoryItem*> item) {
|
||||
|
@ -25,25 +41,88 @@ void Groups::unregisterMessage(not_null<HistoryItem*> item) {
|
|||
if (!groupId) {
|
||||
return;
|
||||
}
|
||||
const auto i = _data.find(groupId);
|
||||
if (i != _data.end()) {
|
||||
auto &group = i->second;
|
||||
group.items.erase(
|
||||
ranges::remove(group.items, item),
|
||||
group.items.end());
|
||||
if (group.items.empty()) {
|
||||
_data.erase(i);
|
||||
const auto i = _groups.find(groupId);
|
||||
if (i != end(_groups)) {
|
||||
auto &items = i->second.items;
|
||||
const auto removed = ranges::remove(items, item);
|
||||
const auto last = end(items);
|
||||
if (removed != last) {
|
||||
items.erase(removed, last);
|
||||
if (!items.empty()) {
|
||||
refreshViews(items);
|
||||
} else {
|
||||
_groups.erase(i);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void Groups::refreshMessage(not_null<HistoryItem*> item) {
|
||||
if (!isGrouped(item)) {
|
||||
unregisterMessage(item);
|
||||
return;
|
||||
}
|
||||
if (!IsServerMsgId(item->id)) {
|
||||
return;
|
||||
}
|
||||
const auto groupId = item->groupId();
|
||||
const auto i = _groups.find(groupId);
|
||||
if (i == end(_groups)) {
|
||||
registerMessage(item);
|
||||
return;
|
||||
}
|
||||
auto &items = i->second.items;
|
||||
const auto position = findPositionForItem(items, item);
|
||||
auto current = ranges::find(items, item);
|
||||
if (current == end(items)) {
|
||||
items.insert(position, item);
|
||||
} else if (position == current + 1) {
|
||||
return;
|
||||
} else if (position > current + 1) {
|
||||
for (++current; current != position; ++current) {
|
||||
std::swap(*(current - 1), *current);
|
||||
}
|
||||
} else if (position < current) {
|
||||
for (; current != position; --current) {
|
||||
std::swap(*(current - 1), *current);
|
||||
}
|
||||
} else {
|
||||
Unexpected("Position of item in Groups::refreshMessage().");
|
||||
}
|
||||
refreshViews(items);
|
||||
}
|
||||
|
||||
HistoryItemsList::const_iterator Groups::findPositionForItem(
|
||||
const HistoryItemsList &group,
|
||||
not_null<HistoryItem*> item) {
|
||||
const auto itemId = item->id;
|
||||
const auto last = end(group);
|
||||
if (!IsServerMsgId(itemId)) {
|
||||
return last;
|
||||
}
|
||||
auto result = begin(group);
|
||||
while (result != last) {
|
||||
const auto alreadyId = (*result)->id;
|
||||
if (IsServerMsgId(alreadyId) && alreadyId > itemId) {
|
||||
return result;
|
||||
}
|
||||
}
|
||||
return last;
|
||||
}
|
||||
|
||||
const Group *Groups::find(not_null<HistoryItem*> item) const {
|
||||
const auto groupId = item->groupId();
|
||||
if (!groupId) {
|
||||
return nullptr;
|
||||
}
|
||||
const auto i = _data.find(groupId);
|
||||
return (i != _data.end()) ? &i->second : nullptr;
|
||||
const auto i = _groups.find(groupId);
|
||||
return (i != _groups.end()) ? &i->second : nullptr;
|
||||
}
|
||||
|
||||
void Groups::refreshViews(const HistoryItemsList &items) {
|
||||
for (const auto item : items) {
|
||||
_data->requestItemViewRefresh(item);
|
||||
}
|
||||
}
|
||||
|
||||
} // namespace Data
|
||||
|
|
|
@ -11,6 +11,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
|
||||
namespace Data {
|
||||
|
||||
class Session;
|
||||
|
||||
struct Group {
|
||||
HistoryItemsList items;
|
||||
|
||||
|
@ -18,13 +20,23 @@ struct Group {
|
|||
|
||||
class Groups {
|
||||
public:
|
||||
Groups(not_null<Session*> data);
|
||||
|
||||
bool isGrouped(not_null<HistoryItem*> item) const;
|
||||
void registerMessage(not_null<HistoryItem*> item);
|
||||
void unregisterMessage(not_null<HistoryItem*> item);
|
||||
void refreshMessage(not_null<HistoryItem*> item);
|
||||
|
||||
const Group *find(not_null<HistoryItem*> item) const;
|
||||
|
||||
private:
|
||||
std::map<MessageGroupId, Group> _data;
|
||||
HistoryItemsList::const_iterator findPositionForItem(
|
||||
const HistoryItemsList &group,
|
||||
not_null<HistoryItem*> item);
|
||||
void refreshViews(const HistoryItemsList &items);
|
||||
|
||||
not_null<Session*> _data;
|
||||
std::map<MessageGroupId, Group> _groups;
|
||||
std::map<MessageGroupId, MessageGroupId> _alias;
|
||||
|
||||
};
|
||||
|
|
|
@ -10,9 +10,12 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_media_types.h"
|
||||
#include "history/history_item.h"
|
||||
#include "history/history_location_manager.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
#include "storage/storage_shared_media.h"
|
||||
#include "storage/localstorage.h"
|
||||
#include "data/data_session.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "auth_session.h"
|
||||
#include "layout.h"
|
||||
|
||||
namespace Data {
|
||||
|
@ -70,9 +73,22 @@ Invoice ComputeInvoiceData(const MTPDmessageMediaInvoice &data) {
|
|||
// We don't use size from WebDocument, because it is not reliable.
|
||||
// It can be > 0 and different from the real size that we get in upload.WebFile result.
|
||||
auto filesize = 0; // doc.vsize.v;
|
||||
auto full = ImagePtr(WebFileImageLocation(imageSize.width(), imageSize.height(), doc.vdc_id.v, doc.vurl.v, doc.vaccess_hash.v), filesize);
|
||||
auto full = ImagePtr(
|
||||
WebFileImageLocation(
|
||||
imageSize.width(),
|
||||
imageSize.height(),
|
||||
doc.vdc_id.v,
|
||||
doc.vurl.v,
|
||||
doc.vaccess_hash.v),
|
||||
filesize);
|
||||
auto photoId = rand_value<PhotoId>();
|
||||
result.photo = App::photoSet(photoId, 0, 0, unixtime(), thumb, medium, full);
|
||||
result.photo = Auth().data().photo(
|
||||
photoId,
|
||||
uint64(0),
|
||||
unixtime(),
|
||||
thumb,
|
||||
medium,
|
||||
full);
|
||||
}
|
||||
}
|
||||
return result;
|
||||
|
@ -162,6 +178,10 @@ Storage::SharedMediaTypesMask Media::sharedMediaTypes() const {
|
|||
return {};
|
||||
}
|
||||
|
||||
bool Media::canBeGrouped() const {
|
||||
return false;
|
||||
}
|
||||
|
||||
QString Media::caption() const {
|
||||
return QString();
|
||||
}
|
||||
|
@ -209,6 +229,11 @@ bool Media::consumeMessageText(const TextWithEntities &text) {
|
|||
return false;
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> Media::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
return createView(message, message->data());
|
||||
}
|
||||
|
||||
MediaPhoto::MediaPhoto(
|
||||
not_null<HistoryItem*> parent,
|
||||
not_null<PhotoData*> photo,
|
||||
|
@ -216,7 +241,6 @@ MediaPhoto::MediaPhoto(
|
|||
: Media(parent)
|
||||
, _photo(photo)
|
||||
, _caption(caption) {
|
||||
App::regPhotoItem(_photo, parent);
|
||||
}
|
||||
|
||||
MediaPhoto::MediaPhoto(
|
||||
|
@ -226,11 +250,9 @@ MediaPhoto::MediaPhoto(
|
|||
: Media(parent)
|
||||
, _photo(photo)
|
||||
, _chat(chat) {
|
||||
App::regPhotoItem(_photo, parent);
|
||||
}
|
||||
|
||||
MediaPhoto::~MediaPhoto() {
|
||||
App::unregPhotoItem(_photo, parent());
|
||||
}
|
||||
|
||||
std::unique_ptr<Media> MediaPhoto::clone(not_null<HistoryItem*> parent) {
|
||||
|
@ -257,6 +279,10 @@ Storage::SharedMediaTypesMask MediaPhoto::sharedMediaTypes() const {
|
|||
.added(Type::PhotoVideo);
|
||||
}
|
||||
|
||||
bool MediaPhoto::canBeGrouped() const {
|
||||
return true;
|
||||
}
|
||||
|
||||
QString MediaPhoto::caption() const {
|
||||
return _caption;
|
||||
}
|
||||
|
@ -293,7 +319,7 @@ bool MediaPhoto::updateInlineResultMedia(const MTPMessageMedia &media) {
|
|||
}
|
||||
auto &photo = media.c_messageMediaPhoto();
|
||||
if (photo.has_photo() && !photo.has_ttl_seconds()) {
|
||||
if (auto existing = App::feedPhoto(photo.vphoto)) {
|
||||
if (auto existing = Auth().data().photo(photo.vphoto)) {
|
||||
if (existing == _photo) {
|
||||
return true;
|
||||
} else {
|
||||
|
@ -321,7 +347,7 @@ bool MediaPhoto::updateSentMedia(const MTPMessageMedia &media) {
|
|||
return false;
|
||||
}
|
||||
const auto &photo = mediaPhoto.vphoto;
|
||||
App::feedPhoto(photo, _photo);
|
||||
Auth().data().photoConvert(_photo, photo);
|
||||
|
||||
if (photo.type() != mtpc_photo) {
|
||||
return false;
|
||||
|
@ -370,7 +396,8 @@ bool MediaPhoto::updateSentMedia(const MTPMessageMedia &media) {
|
|||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> MediaPhoto::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) {
|
||||
if (_chat) {
|
||||
return std::make_unique<HistoryPhoto>(
|
||||
message,
|
||||
|
@ -378,7 +405,11 @@ std::unique_ptr<HistoryMedia> MediaPhoto::createView(
|
|||
_photo,
|
||||
st::msgServicePhotoWidth);
|
||||
}
|
||||
return std::make_unique<HistoryPhoto>(message, _photo, _caption);
|
||||
return std::make_unique<HistoryPhoto>(
|
||||
message,
|
||||
realParent,
|
||||
_photo,
|
||||
_caption);
|
||||
}
|
||||
|
||||
MediaFile::MediaFile(
|
||||
|
@ -389,7 +420,7 @@ MediaFile::MediaFile(
|
|||
, _document(document)
|
||||
, _caption(caption)
|
||||
, _emoji(document->sticker() ? document->sticker()->alt : QString()) {
|
||||
App::regDocumentItem(_document, parent);
|
||||
Auth().data().registerDocumentItem(_document, parent);
|
||||
|
||||
if (!_emoji.isEmpty()) {
|
||||
if (const auto emoji = Ui::Emoji::Find(_emoji)) {
|
||||
|
@ -399,7 +430,7 @@ MediaFile::MediaFile(
|
|||
}
|
||||
|
||||
MediaFile::~MediaFile() {
|
||||
App::unregDocumentItem(_document, parent());
|
||||
Auth().data().unregisterDocumentItem(_document, parent());
|
||||
}
|
||||
|
||||
std::unique_ptr<Media> MediaFile::clone(not_null<HistoryItem*> parent) {
|
||||
|
@ -438,6 +469,10 @@ Storage::SharedMediaTypesMask MediaFile::sharedMediaTypes() const {
|
|||
return Type::File;
|
||||
}
|
||||
|
||||
bool MediaFile::canBeGrouped() const {
|
||||
return _document->isVideoFile();
|
||||
}
|
||||
|
||||
QString MediaFile::chatsListText() const {
|
||||
if (const auto sticker = _document->sticker()) {
|
||||
return Media::chatsListText();
|
||||
|
@ -545,12 +580,11 @@ bool MediaFile::updateInlineResultMedia(const MTPMessageMedia &media) {
|
|||
}
|
||||
auto &data = media.c_messageMediaDocument();
|
||||
if (data.has_document() && !data.has_ttl_seconds()) {
|
||||
if (const auto document = App::feedDocument(data.vdocument)) {
|
||||
if (document == _document) {
|
||||
return false;
|
||||
} else {
|
||||
document->collectLocalData(_document);
|
||||
}
|
||||
const auto document = Auth().data().document(data.vdocument);
|
||||
if (document == _document) {
|
||||
return false;
|
||||
} else {
|
||||
document->collectLocalData(_document);
|
||||
}
|
||||
} else {
|
||||
LOG(("API Error: "
|
||||
|
@ -571,25 +605,30 @@ bool MediaFile::updateSentMedia(const MTPMessageMedia &media) {
|
|||
"or with ttl_seconds in updateSentMedia()"));
|
||||
return false;
|
||||
}
|
||||
const auto changed = App::feedDocument(data.vdocument, _document);
|
||||
if (!changed->data().isEmpty()) {
|
||||
if (changed->isVoiceMessage()) {
|
||||
Local::writeAudio(changed->mediaKey(), changed->data());
|
||||
Auth().data().documentConvert(_document, data.vdocument);
|
||||
if (!_document->data().isEmpty()) {
|
||||
if (_document->isVoiceMessage()) {
|
||||
Local::writeAudio(_document->mediaKey(), _document->data());
|
||||
} else {
|
||||
Local::writeStickerImage(changed->mediaKey(), changed->data());
|
||||
Local::writeStickerImage(_document->mediaKey(), _document->data());
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> MediaFile::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) {
|
||||
if (_document->sticker()) {
|
||||
return std::make_unique<HistorySticker>(message, _document);
|
||||
} else if (_document->isAnimation()) {
|
||||
return std::make_unique<HistoryGif>(message, _document, _caption);
|
||||
} else if (_document->isVideoFile()) {
|
||||
return std::make_unique<HistoryVideo>(message, _document, _caption);
|
||||
return std::make_unique<HistoryVideo>(
|
||||
message,
|
||||
realParent,
|
||||
_document,
|
||||
_caption);
|
||||
}
|
||||
return std::make_unique<HistoryDocument>(message, _document, _caption);
|
||||
}
|
||||
|
@ -601,12 +640,18 @@ MediaContact::MediaContact(
|
|||
const QString &lastName,
|
||||
const QString &phoneNumber)
|
||||
: Media(parent) {
|
||||
Auth().data().registerContactItem(userId, parent);
|
||||
|
||||
_contact.userId = userId;
|
||||
_contact.firstName = firstName;
|
||||
_contact.lastName = lastName;
|
||||
_contact.phoneNumber = phoneNumber;
|
||||
}
|
||||
|
||||
MediaContact::~MediaContact() {
|
||||
Auth().data().unregisterContactItem(_contact.userId, parent());
|
||||
}
|
||||
|
||||
std::unique_ptr<Media> MediaContact::clone(not_null<HistoryItem*> parent) {
|
||||
return std::make_unique<MediaContact>(
|
||||
parent,
|
||||
|
@ -637,15 +682,16 @@ bool MediaContact::updateSentMedia(const MTPMessageMedia &media) {
|
|||
return false;
|
||||
}
|
||||
if (_contact.userId != media.c_messageMediaContact().vuser_id.v) {
|
||||
//detachFromParent(); // #TODO contacts
|
||||
Auth().data().unregisterContactItem(_contact.userId, parent());
|
||||
_contact.userId = media.c_messageMediaContact().vuser_id.v;
|
||||
//attachToParent();
|
||||
Auth().data().registerContactItem(_contact.userId, parent());
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> MediaContact::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) {
|
||||
return std::make_unique<HistoryContact>(
|
||||
message,
|
||||
_contact.userId,
|
||||
|
@ -704,7 +750,8 @@ bool MediaLocation::updateSentMedia(const MTPMessageMedia &media) {
|
|||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> MediaLocation::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) {
|
||||
return std::make_unique<HistoryLocation>(
|
||||
message,
|
||||
_location,
|
||||
|
@ -760,7 +807,8 @@ bool MediaCall::updateSentMedia(const MTPMessageMedia &media) {
|
|||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> MediaCall::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) {
|
||||
return std::make_unique<HistoryCall>(message, &_call);
|
||||
}
|
||||
|
||||
|
@ -784,6 +832,11 @@ MediaWebPage::MediaWebPage(
|
|||
not_null<WebPageData*> page)
|
||||
: Media(parent)
|
||||
, _page(page) {
|
||||
Auth().data().registerWebPageItem(_page, parent);
|
||||
}
|
||||
|
||||
MediaWebPage::~MediaWebPage() {
|
||||
Auth().data().unregisterWebPageItem(_page, parent());
|
||||
}
|
||||
|
||||
std::unique_ptr<Media> MediaWebPage::clone(not_null<HistoryItem*> parent) {
|
||||
|
@ -815,7 +868,8 @@ bool MediaWebPage::updateSentMedia(const MTPMessageMedia &media) {
|
|||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> MediaWebPage::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) {
|
||||
return std::make_unique<HistoryWebPage>(message, _page);
|
||||
}
|
||||
|
||||
|
@ -874,15 +928,13 @@ bool MediaGame::updateSentMedia(const MTPMessageMedia &media) {
|
|||
if (media.type() != mtpc_messageMediaGame) {
|
||||
return false;
|
||||
}
|
||||
const auto &game = media.c_messageMediaGame().vgame;
|
||||
if (game.type() == mtpc_game) {
|
||||
App::feedGame(game.c_game(), _game);
|
||||
}
|
||||
Auth().data().gameConvert(_game, media.c_messageMediaGame().vgame);
|
||||
return true;
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> MediaGame::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) {
|
||||
return std::make_unique<HistoryGame>(message, _game, _consumedText);
|
||||
}
|
||||
|
||||
|
@ -925,7 +977,8 @@ bool MediaInvoice::updateSentMedia(const MTPMessageMedia &media) {
|
|||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> MediaInvoice::createView(
|
||||
not_null<HistoryView::Element*> message) {
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) {
|
||||
return std::make_unique<HistoryInvoice>(message, &_invoice);
|
||||
}
|
||||
|
||||
|
|
|
@ -81,6 +81,7 @@ public:
|
|||
|
||||
virtual bool uploading() const;
|
||||
virtual Storage::SharedMediaTypesMask sharedMediaTypes() const;
|
||||
virtual bool canBeGrouped() const;
|
||||
virtual QString caption() const;
|
||||
virtual bool hasReplyPreview() const;
|
||||
virtual ImagePtr replyPreview() const;
|
||||
|
@ -105,7 +106,10 @@ public:
|
|||
virtual bool updateInlineResultMedia(const MTPMessageMedia &media) = 0;
|
||||
virtual bool updateSentMedia(const MTPMessageMedia &media) = 0;
|
||||
virtual std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) = 0;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) = 0;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message);
|
||||
|
||||
private:
|
||||
const not_null<HistoryItem*> _parent;
|
||||
|
@ -130,6 +134,7 @@ public:
|
|||
|
||||
bool uploading() const override;
|
||||
Storage::SharedMediaTypesMask sharedMediaTypes() const override;
|
||||
bool canBeGrouped() const override;
|
||||
QString caption() const override;
|
||||
QString chatsListText() const override;
|
||||
QString notificationText() const override;
|
||||
|
@ -141,7 +146,8 @@ public:
|
|||
bool updateInlineResultMedia(const MTPMessageMedia &media) override;
|
||||
bool updateSentMedia(const MTPMessageMedia &media) override;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) override;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) override;
|
||||
|
||||
private:
|
||||
not_null<PhotoData*> _photo;
|
||||
|
@ -164,6 +170,7 @@ public:
|
|||
|
||||
bool uploading() const override;
|
||||
Storage::SharedMediaTypesMask sharedMediaTypes() const override;
|
||||
bool canBeGrouped() const override;
|
||||
QString chatsListText() const override;
|
||||
QString notificationText() const override;
|
||||
QString pinnedTextSubstring() const override;
|
||||
|
@ -175,7 +182,8 @@ public:
|
|||
bool updateInlineResultMedia(const MTPMessageMedia &media) override;
|
||||
bool updateSentMedia(const MTPMessageMedia &media) override;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) override;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) override;
|
||||
|
||||
private:
|
||||
not_null<DocumentData*> _document;
|
||||
|
@ -192,6 +200,7 @@ public:
|
|||
const QString &firstName,
|
||||
const QString &lastName,
|
||||
const QString &phoneNumber);
|
||||
~MediaContact();
|
||||
|
||||
std::unique_ptr<Media> clone(not_null<HistoryItem*> parent) override;
|
||||
|
||||
|
@ -202,7 +211,8 @@ public:
|
|||
bool updateInlineResultMedia(const MTPMessageMedia &media) override;
|
||||
bool updateSentMedia(const MTPMessageMedia &media) override;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) override;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) override;
|
||||
|
||||
private:
|
||||
SharedContact _contact;
|
||||
|
@ -230,7 +240,8 @@ public:
|
|||
bool updateInlineResultMedia(const MTPMessageMedia &media) override;
|
||||
bool updateSentMedia(const MTPMessageMedia &media) override;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) override;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) override;
|
||||
|
||||
private:
|
||||
not_null<LocationData*> _location;
|
||||
|
@ -256,7 +267,8 @@ public:
|
|||
bool updateInlineResultMedia(const MTPMessageMedia &media) override;
|
||||
bool updateSentMedia(const MTPMessageMedia &media) override;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) override;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) override;
|
||||
|
||||
static QString Text(
|
||||
not_null<HistoryItem*> item,
|
||||
|
@ -272,6 +284,7 @@ public:
|
|||
MediaWebPage(
|
||||
not_null<HistoryItem*> parent,
|
||||
not_null<WebPageData*> page);
|
||||
~MediaWebPage();
|
||||
|
||||
std::unique_ptr<Media> clone(not_null<HistoryItem*> parent) override;
|
||||
|
||||
|
@ -283,7 +296,8 @@ public:
|
|||
bool updateInlineResultMedia(const MTPMessageMedia &media) override;
|
||||
bool updateSentMedia(const MTPMessageMedia &media) override;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) override;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) override;
|
||||
|
||||
private:
|
||||
not_null<WebPageData*> _page;
|
||||
|
@ -310,7 +324,8 @@ public:
|
|||
bool updateInlineResultMedia(const MTPMessageMedia &media) override;
|
||||
bool updateSentMedia(const MTPMessageMedia &media) override;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) override;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) override;
|
||||
|
||||
private:
|
||||
not_null<GameData*> _game;
|
||||
|
@ -337,7 +352,8 @@ public:
|
|||
bool updateInlineResultMedia(const MTPMessageMedia &media) override;
|
||||
bool updateSentMedia(const MTPMessageMedia &media) override;
|
||||
std::unique_ptr<HistoryMedia> createView(
|
||||
not_null<HistoryView::Element*> message) override;
|
||||
not_null<HistoryView::Element*> message,
|
||||
not_null<HistoryItem*> realParent) override;
|
||||
|
||||
private:
|
||||
Invoice _invoice;
|
||||
|
|
|
@ -171,7 +171,7 @@ void PeerData::setUserpic(
|
|||
|
||||
void PeerData::setUserpicPhoto(const MTPPhoto &data) {
|
||||
auto photoId = [&]() -> PhotoId {
|
||||
if (auto photo = App::feedPhoto(data)) {
|
||||
if (const auto photo = Auth().data().photo(data)) {
|
||||
photo->peer = this;
|
||||
return photo->id;
|
||||
}
|
||||
|
@ -353,7 +353,7 @@ const Text &BotCommand::descriptionText() const {
|
|||
|
||||
bool UserData::canShareThisContact() const {
|
||||
return canShareThisContactFast()
|
||||
|| !App::phoneFromSharedContact(peerToUser(id)).isEmpty();
|
||||
|| !Auth().data().findContactPhone(peerToUser(id)).isEmpty();
|
||||
}
|
||||
|
||||
void UserData::setContactStatus(ContactStatus status) {
|
||||
|
|
|
@ -32,14 +32,14 @@ void PhotoData::automaticLoadSettingsChanged() {
|
|||
|
||||
void PhotoData::download() {
|
||||
full->loadEvenCancelled();
|
||||
notifyLayoutChanged();
|
||||
Auth().data().notifyPhotoLayoutChanged(this);
|
||||
}
|
||||
|
||||
bool PhotoData::loaded() const {
|
||||
bool wasLoading = loading();
|
||||
if (full->loaded()) {
|
||||
if (wasLoading) {
|
||||
notifyLayoutChanged();
|
||||
Auth().data().notifyPhotoLayoutChanged(this);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
@ -58,17 +58,7 @@ bool PhotoData::displayLoading() const {
|
|||
|
||||
void PhotoData::cancel() {
|
||||
full->cancel();
|
||||
notifyLayoutChanged();
|
||||
}
|
||||
|
||||
void PhotoData::notifyLayoutChanged() const {
|
||||
auto &items = App::photoItems();
|
||||
auto i = items.constFind(const_cast<PhotoData*>(this));
|
||||
if (i != items.cend()) {
|
||||
for_const (auto item, i.value()) {
|
||||
Auth().data().markItemLayoutChange(item);
|
||||
}
|
||||
}
|
||||
Auth().data().notifyPhotoLayoutChanged(this);
|
||||
}
|
||||
|
||||
float64 PhotoData::progress() const {
|
||||
|
|
|
@ -49,9 +49,6 @@ public:
|
|||
|
||||
std::unique_ptr<Data::UploadState> uploadingData;
|
||||
|
||||
private:
|
||||
void notifyLayoutChanged() const;
|
||||
|
||||
};
|
||||
|
||||
class PhotoClickHandler : public FileClickHandler {
|
||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -18,6 +18,14 @@ struct Group;
|
|||
class Element;
|
||||
} // namespace HistoryView
|
||||
|
||||
class AuthSession;
|
||||
|
||||
namespace Media {
|
||||
namespace Clip {
|
||||
class Reader;
|
||||
} // namespace Clip
|
||||
} // namespace Media
|
||||
|
||||
namespace Data {
|
||||
|
||||
class Feed;
|
||||
|
@ -26,7 +34,7 @@ class Session final {
|
|||
public:
|
||||
using ViewElement = HistoryView::Element;
|
||||
|
||||
Session();
|
||||
explicit Session(not_null<AuthSession*> session);
|
||||
~Session();
|
||||
|
||||
base::Variable<bool> &contactsLoaded() {
|
||||
|
@ -53,26 +61,14 @@ public:
|
|||
not_null<HistoryItem*> item;
|
||||
MsgId oldId = 0;
|
||||
};
|
||||
|
||||
void registerItemView(not_null<ViewElement*> view);
|
||||
void unregisterItemView(not_null<ViewElement*> view);
|
||||
template <typename Method>
|
||||
void enumerateItemViews(not_null<HistoryItem*> item, Method method) {
|
||||
if (const auto i = _views.find(item); i != _views.end()) {
|
||||
for (const auto view : i->second) {
|
||||
method(view);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void markItemIdChange(IdChange event);
|
||||
rpl::producer<IdChange> itemIdChanged() const;
|
||||
void markItemLayoutChange(not_null<const HistoryItem*> item);
|
||||
rpl::producer<not_null<const HistoryItem*>> itemLayoutChanged() const;
|
||||
void markViewLayoutChange(not_null<const ViewElement*> view);
|
||||
rpl::producer<not_null<const ViewElement*>> viewLayoutChanged() const;
|
||||
void requestItemRepaint(not_null<const HistoryItem*> item);
|
||||
rpl::producer<not_null<const HistoryItem*>> itemRepaintRequest() const;
|
||||
void requestItemViewRepaint(not_null<const HistoryItem*> item);
|
||||
rpl::producer<not_null<const HistoryItem*>> itemViewRepaintRequest() const;
|
||||
void requestViewRepaint(not_null<const ViewElement*> view);
|
||||
rpl::producer<not_null<const ViewElement*>> viewRepaintRequest() const;
|
||||
void requestItemViewResize(not_null<const HistoryItem*> item);
|
||||
|
@ -194,9 +190,162 @@ public:
|
|||
const Dialogs::Key &key1,
|
||||
const Dialogs::Key &key2);
|
||||
|
||||
void photoLoadSettingsChanged();
|
||||
void voiceLoadSettingsChanged();
|
||||
void animationLoadSettingsChanged();
|
||||
|
||||
void notifyPhotoLayoutChanged(not_null<const PhotoData*> photo);
|
||||
void notifyDocumentLayoutChanged(
|
||||
not_null<const DocumentData*> document);
|
||||
void requestDocumentViewRepaint(not_null<const DocumentData*> document);
|
||||
void markMediaRead(not_null<const DocumentData*> document);
|
||||
|
||||
not_null<PhotoData*> photo(PhotoId id);
|
||||
not_null<PhotoData*> photo(const MTPPhoto &data);
|
||||
not_null<PhotoData*> photo(const MTPDphoto &data);
|
||||
not_null<PhotoData*> photo(
|
||||
const MTPPhoto &data,
|
||||
const PreparedPhotoThumbs &thumbs);
|
||||
not_null<PhotoData*> photo(
|
||||
PhotoId id,
|
||||
const uint64 &access,
|
||||
TimeId date,
|
||||
const ImagePtr &thumb,
|
||||
const ImagePtr &medium,
|
||||
const ImagePtr &full);
|
||||
void photoConvert(
|
||||
not_null<PhotoData*> original,
|
||||
const MTPPhoto &data);
|
||||
|
||||
not_null<DocumentData*> document(DocumentId id);
|
||||
not_null<DocumentData*> document(const MTPDocument &data);
|
||||
not_null<DocumentData*> document(const MTPDdocument &data);
|
||||
not_null<DocumentData*> document(
|
||||
const MTPdocument &data,
|
||||
const QPixmap &thumb);
|
||||
not_null<DocumentData*> document(
|
||||
DocumentId id,
|
||||
const uint64 &access,
|
||||
int32 version,
|
||||
TimeId date,
|
||||
const QVector<MTPDocumentAttribute> &attributes,
|
||||
const QString &mime,
|
||||
const ImagePtr &thumb,
|
||||
int32 dc,
|
||||
int32 size,
|
||||
const StorageImageLocation &thumbLocation);
|
||||
void documentConvert(
|
||||
not_null<DocumentData*> original,
|
||||
const MTPDocument &data);
|
||||
|
||||
not_null<WebPageData*> webpage(WebPageId id);
|
||||
not_null<WebPageData*> webpage(const MTPWebPage &data);
|
||||
not_null<WebPageData*> webpage(const MTPDwebPage &data);
|
||||
not_null<WebPageData*> webpage(const MTPDwebPagePending &data);
|
||||
not_null<WebPageData*> webpage(
|
||||
WebPageId id,
|
||||
const QString &siteName,
|
||||
const TextWithEntities &content);
|
||||
not_null<WebPageData*> webpage(
|
||||
WebPageId id,
|
||||
const QString &type,
|
||||
const QString &url,
|
||||
const QString &displayUrl,
|
||||
const QString &siteName,
|
||||
const QString &title,
|
||||
const TextWithEntities &description,
|
||||
PhotoData *photo,
|
||||
DocumentData *document,
|
||||
int duration,
|
||||
const QString &author,
|
||||
TimeId pendingTill);
|
||||
|
||||
not_null<GameData*> game(GameId id);
|
||||
not_null<GameData*> game(const MTPDgame &data);
|
||||
not_null<GameData*> game(
|
||||
GameId id,
|
||||
const uint64 &accessHash,
|
||||
const QString &shortName,
|
||||
const QString &title,
|
||||
const QString &description,
|
||||
PhotoData *photo,
|
||||
DocumentData *document);
|
||||
void gameConvert(
|
||||
not_null<GameData*> original,
|
||||
const MTPGame &data);
|
||||
|
||||
void registerPhotoView(
|
||||
not_null<const PhotoData*> photo,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void unregisterPhotoView(
|
||||
not_null<const PhotoData*> photo,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void registerDocumentView(
|
||||
not_null<const DocumentData*> document,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void unregisterDocumentView(
|
||||
not_null<const DocumentData*> document,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void registerDocumentItem(
|
||||
not_null<const DocumentData*> document,
|
||||
not_null<HistoryItem*> item);
|
||||
void unregisterDocumentItem(
|
||||
not_null<const DocumentData*> document,
|
||||
not_null<HistoryItem*> item);
|
||||
void registerWebPageView(
|
||||
not_null<const WebPageData*> page,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void unregisterWebPageView(
|
||||
not_null<const WebPageData*> page,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void registerWebPageItem(
|
||||
not_null<const WebPageData*> page,
|
||||
not_null<HistoryItem*> item);
|
||||
void unregisterWebPageItem(
|
||||
not_null<const WebPageData*> page,
|
||||
not_null<HistoryItem*> item);
|
||||
void registerGameView(
|
||||
not_null<const GameData*> game,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void unregisterGameView(
|
||||
not_null<const GameData*> game,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void registerContactView(
|
||||
UserId contactId,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void unregisterContactView(
|
||||
UserId contactId,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void registerContactItem(
|
||||
UserId contactId,
|
||||
not_null<HistoryItem*> item);
|
||||
void unregisterContactItem(
|
||||
UserId contactId,
|
||||
not_null<HistoryItem*> item);
|
||||
void registerAutoplayAnimation(
|
||||
not_null<::Media::Clip::Reader*> reader,
|
||||
not_null<HistoryView::Element*> view);
|
||||
void unregisterAutoplayAnimation(
|
||||
not_null<::Media::Clip::Reader*> reader);
|
||||
|
||||
HistoryItem *findWebPageItem(not_null<WebPageData*> page) const;
|
||||
QString findContactPhone(not_null<UserData*> contact) const;
|
||||
QString findContactPhone(UserId contactId) const;
|
||||
|
||||
void notifyWebPageUpdateDelayed(not_null<WebPageData*> page);
|
||||
void notifyGameUpdateDelayed(not_null<GameData*> game);
|
||||
void sendWebPageGameNotifications();
|
||||
|
||||
void stopAutoplayAnimations();
|
||||
|
||||
void registerItemView(not_null<ViewElement*> view);
|
||||
void unregisterItemView(not_null<ViewElement*> view);
|
||||
|
||||
not_null<Data::Feed*> feed(FeedId id);
|
||||
Data::Feed *feedLoaded(FeedId id);
|
||||
|
||||
void forgetMedia();
|
||||
|
||||
void setMimeForwardIds(MessageIdsList &&list);
|
||||
MessageIdsList takeMimeForwardIds();
|
||||
|
||||
|
@ -208,6 +357,67 @@ public:
|
|||
}
|
||||
|
||||
private:
|
||||
void photoApplyFields(
|
||||
not_null<PhotoData*> photo,
|
||||
const MTPPhoto &data);
|
||||
void photoApplyFields(
|
||||
not_null<PhotoData*> photo,
|
||||
const MTPDphoto &data);
|
||||
void photoApplyFields(
|
||||
not_null<PhotoData*> photo,
|
||||
const uint64 &access,
|
||||
TimeId date,
|
||||
const ImagePtr &thumb,
|
||||
const ImagePtr &medium,
|
||||
const ImagePtr &full);
|
||||
|
||||
void documentApplyFields(
|
||||
not_null<DocumentData*> document,
|
||||
const MTPDocument &data);
|
||||
void documentApplyFields(
|
||||
not_null<DocumentData*> document,
|
||||
const MTPDdocument &data);
|
||||
void documentApplyFields(
|
||||
not_null<DocumentData*> document,
|
||||
const uint64 &access,
|
||||
int32 version,
|
||||
TimeId date,
|
||||
const QVector<MTPDocumentAttribute> &attributes,
|
||||
const QString &mime,
|
||||
const ImagePtr &thumb,
|
||||
int32 dc,
|
||||
int32 size,
|
||||
const StorageImageLocation &thumbLocation);
|
||||
|
||||
void webpageApplyFields(
|
||||
not_null<WebPageData*> page,
|
||||
const MTPDwebPage &data);
|
||||
void webpageApplyFields(
|
||||
not_null<WebPageData*> page,
|
||||
const QString &type,
|
||||
const QString &url,
|
||||
const QString &displayUrl,
|
||||
const QString &siteName,
|
||||
const QString &title,
|
||||
const TextWithEntities &description,
|
||||
PhotoData *photo,
|
||||
DocumentData *document,
|
||||
int duration,
|
||||
const QString &author,
|
||||
TimeId pendingTill);
|
||||
|
||||
void gameApplyFields(
|
||||
not_null<GameData*> game,
|
||||
const MTPDgame &data);
|
||||
void gameApplyFields(
|
||||
not_null<GameData*> game,
|
||||
const uint64 &accessHash,
|
||||
const QString &shortName,
|
||||
const QString &title,
|
||||
const QString &description,
|
||||
PhotoData *photo,
|
||||
DocumentData *document);
|
||||
|
||||
bool stickersUpdateNeeded(TimeMs lastUpdate, TimeMs now) const {
|
||||
constexpr auto kStickersUpdateTimeout = TimeMs(3600'000);
|
||||
return (lastUpdate == 0)
|
||||
|
@ -218,6 +428,11 @@ private:
|
|||
void clearPinnedDialogs();
|
||||
void setIsPinned(const Dialogs::Key &key, bool pinned);
|
||||
|
||||
template <typename Method>
|
||||
void enumerateItemViews(not_null<HistoryItem*> item, Method method);
|
||||
|
||||
not_null<AuthSession*> _session;
|
||||
|
||||
base::Variable<bool> _contactsLoaded = { false };
|
||||
base::Variable<bool> _allChatsLoaded = { false };
|
||||
base::Observable<void> _moreChatsLoaded;
|
||||
|
@ -226,7 +441,7 @@ private:
|
|||
rpl::event_stream<IdChange> _itemIdChanges;
|
||||
rpl::event_stream<not_null<const HistoryItem*>> _itemLayoutChanges;
|
||||
rpl::event_stream<not_null<const ViewElement*>> _viewLayoutChanges;
|
||||
rpl::event_stream<not_null<const HistoryItem*>> _itemRepaintRequest;
|
||||
rpl::event_stream<not_null<const HistoryItem*>> _itemViewRepaintRequest;
|
||||
rpl::event_stream<not_null<const ViewElement*>> _viewRepaintRequest;
|
||||
rpl::event_stream<not_null<const HistoryItem*>> _itemViewResizeRequest;
|
||||
rpl::event_stream<not_null<const ViewElement*>> _viewResizeRequest;
|
||||
|
@ -252,6 +467,49 @@ private:
|
|||
Stickers::Order _archivedStickerSetsOrder;
|
||||
Stickers::SavedGifs _savedGifs;
|
||||
|
||||
std::unordered_map<
|
||||
PhotoId,
|
||||
std::unique_ptr<PhotoData>> _photos;
|
||||
std::map<
|
||||
not_null<const PhotoData*>,
|
||||
base::flat_set<not_null<HistoryView::Element*>>> _photoViews;
|
||||
std::unordered_map<
|
||||
DocumentId,
|
||||
std::unique_ptr<DocumentData>> _documents;
|
||||
std::map<
|
||||
not_null<const DocumentData*>,
|
||||
base::flat_set<not_null<HistoryItem*>>> _documentItems;
|
||||
std::map<
|
||||
not_null<const DocumentData*>,
|
||||
base::flat_set<not_null<HistoryView::Element*>>> _documentViews;
|
||||
std::unordered_map<
|
||||
WebPageId,
|
||||
std::unique_ptr<WebPageData>> _webpages;
|
||||
std::map<
|
||||
not_null<const WebPageData*>,
|
||||
base::flat_set<not_null<HistoryItem*>>> _webpageItems;
|
||||
std::map<
|
||||
not_null<const WebPageData*>,
|
||||
base::flat_set<not_null<HistoryView::Element*>>> _webpageViews;
|
||||
std::unordered_map<
|
||||
GameId,
|
||||
std::unique_ptr<GameData>> _games;
|
||||
std::map<
|
||||
not_null<const GameData*>,
|
||||
base::flat_set<not_null<HistoryView::Element*>>> _gameViews;
|
||||
std::map<
|
||||
UserId,
|
||||
base::flat_set<not_null<HistoryItem*>>> _contactItems;
|
||||
std::map<
|
||||
UserId,
|
||||
base::flat_set<not_null<HistoryView::Element*>>> _contactViews;
|
||||
base::flat_map<
|
||||
not_null<::Media::Clip::Reader*>,
|
||||
not_null<HistoryView::Element*>> _autoplayAnimations;
|
||||
|
||||
base::flat_set<not_null<WebPageData*>> _webpagesUpdated;
|
||||
base::flat_set<not_null<GameData*>> _gamesUpdated;
|
||||
|
||||
std::deque<Dialogs::Key> _pinnedDialogs;
|
||||
base::flat_map<FeedId, std::unique_ptr<Data::Feed>> _feeds;
|
||||
Groups _groups;
|
||||
|
|
|
@ -17,6 +17,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_media_types.h"
|
||||
#include "data/data_media_types.h"
|
||||
#include "data/data_sparse_ids.h"
|
||||
#include "data/data_session.h"
|
||||
#include "info/info_memento.h"
|
||||
#include "info/info_controller.h"
|
||||
#include "window/window_controller.h"
|
||||
|
@ -284,7 +285,7 @@ SharedMediaWithLastSlice::Value SharedMediaWithLastSlice::operator[](int index)
|
|||
}
|
||||
return (index < _slice.size())
|
||||
? Value(_slice[index])
|
||||
: Value(App::photo(*_lastPhotoId));
|
||||
: Value(Auth().data().photo(*_lastPhotoId));
|
||||
}
|
||||
|
||||
base::optional<int> SharedMediaWithLastSlice::distance(
|
||||
|
|
|
@ -44,7 +44,7 @@ bool WebPageData::applyChanges(
|
|||
const QString &newAuthor,
|
||||
int newPendingTill) {
|
||||
if (newPendingTill != 0
|
||||
&& (!url.isEmpty() || newUrl.isEmpty())
|
||||
&& (!url.isEmpty() || pendingTill < 0)
|
||||
&& (!pendingTill
|
||||
|| pendingTill == newPendingTill
|
||||
|| newPendingTill < -1)) {
|
||||
|
@ -103,7 +103,5 @@ bool WebPageData::applyChanges(
|
|||
author = resultAuthor;
|
||||
pendingTill = newPendingTill;
|
||||
++version;
|
||||
if (App::main()) App::main()->webPageUpdated(this);
|
||||
|
||||
return true;
|
||||
}
|
|
@ -94,7 +94,7 @@ DialogsInner::DialogsInner(QWidget *parent, not_null<Window::Controller*> contro
|
|||
) | rpl::start_with_next(
|
||||
[this](auto item) { itemRemoved(item); },
|
||||
lifetime());
|
||||
Auth().data().itemRepaintRequest(
|
||||
Auth().data().itemViewRepaintRequest(
|
||||
) | rpl::start_with_next([this](auto item) {
|
||||
const auto history = item->history();
|
||||
if (history->textCachedFor == item) {
|
||||
|
@ -2316,7 +2316,9 @@ bool DialogsInner::chooseRow() {
|
|||
if (const auto history = chosen.key.history()) {
|
||||
App::main()->choosePeer(history->peer->id, chosen.messageId);
|
||||
} else if (const auto feed = chosen.key.feed()) {
|
||||
_controller->showSection(HistoryFeed::Memento(feed));
|
||||
_controller->showSection(
|
||||
HistoryFeed::Memento(feed),
|
||||
Window::SectionShow::Way::ClearStack);
|
||||
}
|
||||
if (openSearchResult) {
|
||||
emit searchResultChosen();
|
||||
|
|
|
@ -349,8 +349,8 @@ void handlePendingHistoryUpdate() {
|
|||
}
|
||||
Auth().data().pendingHistoryResize().notify(true);
|
||||
|
||||
for (const auto item : base::take(Global::RefPendingRepaintItems())) {
|
||||
Auth().data().requestItemRepaint(item);
|
||||
//for (const auto item : base::take(Global::RefPendingRepaintItems())) {
|
||||
// Auth().data().requestItemViewRepaint(item);
|
||||
|
||||
// Start the video if it is waiting for that.
|
||||
//if (item->pendingInitDimensions()) { // #TODO floating player video
|
||||
|
@ -372,7 +372,7 @@ void handlePendingHistoryUpdate() {
|
|||
// }
|
||||
// }
|
||||
//}
|
||||
}
|
||||
//}
|
||||
}
|
||||
|
||||
void unreadCounterUpdated() {
|
||||
|
@ -572,8 +572,6 @@ struct Data {
|
|||
|
||||
HiddenPinnedMessagesMap HiddenPinnedMessages;
|
||||
|
||||
PendingItemsMap PendingRepaintItems;
|
||||
|
||||
Stickers::Sets StickerSets;
|
||||
Stickers::Order StickerSetsOrder;
|
||||
TimeMs LastStickersUpdate = 0;
|
||||
|
@ -695,8 +693,6 @@ DefineRefVar(Global, base::Observable<void>, PhoneCallsEnabledChanged);
|
|||
|
||||
DefineVar(Global, HiddenPinnedMessagesMap, HiddenPinnedMessages);
|
||||
|
||||
DefineRefVar(Global, PendingItemsMap, PendingRepaintItems);
|
||||
|
||||
DefineVar(Global, Stickers::Sets, StickerSets);
|
||||
DefineVar(Global, Stickers::Order, StickerSetsOrder);
|
||||
DefineVar(Global, TimeMs, LastStickersUpdate);
|
||||
|
|
|
@ -355,9 +355,6 @@ DeclareRefVar(base::Observable<void>, PhoneCallsEnabledChanged);
|
|||
typedef QMap<PeerId, MsgId> HiddenPinnedMessagesMap;
|
||||
DeclareVar(HiddenPinnedMessagesMap, HiddenPinnedMessages);
|
||||
|
||||
typedef OrderedSet<HistoryItem*> PendingItemsMap;
|
||||
DeclareRefVar(PendingItemsMap, PendingRepaintItems);
|
||||
|
||||
typedef QMap<uint64, QPixmap> CircleMasksMap;
|
||||
DeclareRefVar(CircleMasksMap, CircleMasks);
|
||||
|
||||
|
|
|
@ -14,8 +14,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_item_components.h"
|
||||
#include "history/admin_log/history_admin_log_section.h"
|
||||
#include "history/admin_log/history_admin_log_filter.h"
|
||||
#include "history/view/history_view_message.h"
|
||||
#include "history/view/history_view_service_message.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
#include "chat_helpers/message_field.h"
|
||||
#include "mainwindow.h"
|
||||
#include "mainwidget.h"
|
||||
|
@ -213,7 +213,7 @@ InnerWidget::InnerWidget(
|
|||
, _emptyText(st::historyAdminLogEmptyWidth - st::historyAdminLogEmptyPadding.left() - st::historyAdminLogEmptyPadding.left()) {
|
||||
setMouseTracking(true);
|
||||
_scrollDateHideTimer.setCallback([this] { scrollDateHideByTimer(); });
|
||||
Auth().data().itemRepaintRequest(
|
||||
Auth().data().itemViewRepaintRequest(
|
||||
) | rpl::start_with_next([this](auto item) {
|
||||
if (item->isLogEntry() && _history == item->history()) {
|
||||
repaintItem(viewForItem(item));
|
||||
|
@ -441,6 +441,20 @@ QPoint InnerWidget::tooltipPos() const {
|
|||
return _mousePosition;
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> InnerWidget::elementCreate(
|
||||
not_null<HistoryMessage*> message) {
|
||||
return std::make_unique<HistoryView::Message>(
|
||||
message,
|
||||
HistoryView::Context::AdminLog);
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> InnerWidget::elementCreate(
|
||||
not_null<HistoryService*> message) {
|
||||
return std::make_unique<HistoryView::Service>(
|
||||
message,
|
||||
HistoryView::Context::AdminLog);
|
||||
}
|
||||
|
||||
void InnerWidget::saveState(not_null<SectionMemento*> memento) {
|
||||
memento->setFilter(std::move(_filter));
|
||||
memento->setAdmins(std::move(_admins));
|
||||
|
@ -544,7 +558,7 @@ void InnerWidget::addEvents(Direction direction, const QVector<MTPChannelAdminLo
|
|||
++count;
|
||||
};
|
||||
GenerateItems(
|
||||
_controller,
|
||||
this,
|
||||
_history,
|
||||
_idManager,
|
||||
data,
|
||||
|
|
|
@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#pragma once
|
||||
|
||||
#include "history/view/history_view_cursor_state.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
#include "history/admin_log/history_admin_log_item.h"
|
||||
#include "history/admin_log/history_admin_log_section.h"
|
||||
#include "ui/widgets/tooltip.h"
|
||||
|
@ -23,10 +24,6 @@ namespace Window {
|
|||
class Controller;
|
||||
} // namespace Window
|
||||
|
||||
namespace HistoryView {
|
||||
class Element;
|
||||
} // namespace HistoryView
|
||||
|
||||
namespace AdminLog {
|
||||
|
||||
class SectionMemento;
|
||||
|
@ -34,6 +31,7 @@ class SectionMemento;
|
|||
class InnerWidget final
|
||||
: public Ui::RpWidget
|
||||
, public Ui::AbstractTooltipShower
|
||||
, public HistoryView::ElementDelegate
|
||||
, private MTP::Sender
|
||||
, private base::Subscriber {
|
||||
public:
|
||||
|
@ -66,10 +64,16 @@ public:
|
|||
void applySearch(const QString &query);
|
||||
void showFilter(base::lambda<void(FilterValue &&filter)> callback);
|
||||
|
||||
// AbstractTooltipShower interface
|
||||
// Ui::AbstractTooltipShower interface.
|
||||
QString tooltipText() const override;
|
||||
QPoint tooltipPos() const override;
|
||||
|
||||
// HistoryView::ElementDelegate interface.
|
||||
std::unique_ptr<HistoryView::Element> elementCreate(
|
||||
not_null<HistoryMessage*> message) override;
|
||||
std::unique_ptr<HistoryView::Element> elementCreate(
|
||||
not_null<HistoryService*> message) override;
|
||||
|
||||
~InnerWidget();
|
||||
|
||||
protected:
|
||||
|
|
|
@ -12,11 +12,13 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_service.h"
|
||||
#include "history/history_message.h"
|
||||
#include "history/history.h"
|
||||
#include "data/data_session.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "boxes/sticker_set_box.h"
|
||||
#include "core/tl_help.h"
|
||||
#include "base/overload.h"
|
||||
#include "messenger.h"
|
||||
#include "auth_session.h"
|
||||
|
||||
namespace AdminLog {
|
||||
namespace {
|
||||
|
@ -119,7 +121,12 @@ PhotoData *GenerateChatPhoto(ChannelId channelId, uint64 logEntryId, MTPint date
|
|||
photoSizes.reserve(2);
|
||||
photoSizes.push_back(MTP_photoSize(MTP_string("a"), photo.vphoto_small, MTP_int(160), MTP_int(160), MTP_int(0)));
|
||||
photoSizes.push_back(MTP_photoSize(MTP_string("c"), photo.vphoto_big, MTP_int(640), MTP_int(640), MTP_int(0)));
|
||||
return App::feedPhoto(MTP_photo(MTP_flags(0), MTP_long(photoId), MTP_long(0), date, MTP_vector<MTPPhotoSize>(photoSizes)));
|
||||
return Auth().data().photo(MTP_photo(
|
||||
MTP_flags(0),
|
||||
MTP_long(photoId),
|
||||
MTP_long(0),
|
||||
date,
|
||||
MTP_vector<MTPPhotoSize>(photoSizes)));
|
||||
}
|
||||
|
||||
const auto CollectChanges = [](auto &phraseMap, auto plusFlags, auto minusFlags) {
|
||||
|
@ -284,10 +291,10 @@ TextWithEntities GenerateParticipantChangeText(not_null<ChannelData*> channel, c
|
|||
} // namespace
|
||||
|
||||
OwnedItem::OwnedItem(
|
||||
not_null<Window::Controller*> controller,
|
||||
not_null<HistoryView::ElementDelegate*> delegate,
|
||||
not_null<HistoryItem*> data)
|
||||
: _data(data)
|
||||
, _view(_data->createView(controller, HistoryView::Context::AdminLog)) {
|
||||
, _view(_data->createView(delegate)) {
|
||||
}
|
||||
|
||||
OwnedItem::OwnedItem(OwnedItem &&other)
|
||||
|
@ -309,7 +316,7 @@ OwnedItem::~OwnedItem() {
|
|||
}
|
||||
|
||||
void GenerateItems(
|
||||
not_null<Window::Controller*> controller,
|
||||
not_null<HistoryView::ElementDelegate*> delegate,
|
||||
not_null<History*> history,
|
||||
LocalIdManager &idManager,
|
||||
const MTPDchannelAdminLogEvent &event,
|
||||
|
@ -322,7 +329,7 @@ void GenerateItems(
|
|||
auto &action = event.vaction;
|
||||
auto date = event.vdate;
|
||||
auto addPart = [&](not_null<HistoryItem*> item) {
|
||||
return callback(OwnedItem(controller, item));
|
||||
return callback(OwnedItem(delegate, item));
|
||||
};
|
||||
|
||||
using Flag = MTPDmessage::Flag;
|
||||
|
|
|
@ -7,11 +7,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
*/
|
||||
#pragma once
|
||||
|
||||
namespace Window {
|
||||
class Controller;
|
||||
} // namespace Window
|
||||
|
||||
namespace HistoryView {
|
||||
class ElementDelegate;
|
||||
class Element;
|
||||
} // namespace HistoryView
|
||||
|
||||
|
@ -21,7 +18,7 @@ class OwnedItem;
|
|||
class LocalIdManager;
|
||||
|
||||
void GenerateItems(
|
||||
not_null<Window::Controller*> controller,
|
||||
not_null<HistoryView::ElementDelegate*> delegate,
|
||||
not_null<History*> history,
|
||||
LocalIdManager &idManager,
|
||||
const MTPDchannelAdminLogEvent &event,
|
||||
|
@ -31,7 +28,7 @@ void GenerateItems(
|
|||
class OwnedItem {
|
||||
public:
|
||||
OwnedItem(
|
||||
not_null<Window::Controller*> controller,
|
||||
not_null<HistoryView::ElementDelegate*> delegate,
|
||||
not_null<HistoryItem*> data);
|
||||
OwnedItem(const OwnedItem &other) = delete;
|
||||
OwnedItem &operator=(const OwnedItem &other) = delete;
|
||||
|
|
|
@ -10,6 +10,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/view/history_view_top_bar_widget.h"
|
||||
#include "history/view/history_view_list_widget.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
#include "history/view/history_view_message.h"
|
||||
#include "history/view/history_view_service_message.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "ui/widgets/buttons.h"
|
||||
#include "ui/widgets/shadow.h"
|
||||
|
@ -162,6 +164,20 @@ rpl::producer<Data::MessagesSlice> Widget::listSource(
|
|||
limitAfter);
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> Widget::elementCreate(
|
||||
not_null<HistoryMessage*> message) {
|
||||
return std::make_unique<HistoryView::Message>(
|
||||
message,
|
||||
HistoryView::Context::Feed);
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> Widget::elementCreate(
|
||||
not_null<HistoryService*> message) {
|
||||
return std::make_unique<HistoryView::Service>(
|
||||
message,
|
||||
HistoryView::Context::Feed);
|
||||
}
|
||||
|
||||
std::unique_ptr<Window::SectionMemento> Widget::createMemento() {
|
||||
auto result = std::make_unique<Memento>(_feed);
|
||||
saveState(result.get());
|
||||
|
|
|
@ -67,6 +67,11 @@ public:
|
|||
int limitBefore,
|
||||
int limitAfter) override;
|
||||
|
||||
std::unique_ptr<HistoryView::Element> elementCreate(
|
||||
not_null<HistoryMessage*> message) override;
|
||||
std::unique_ptr<HistoryView::Element> elementCreate(
|
||||
not_null<HistoryService*> message) override;
|
||||
|
||||
protected:
|
||||
void resizeEvent(QResizeEvent *e) override;
|
||||
void paintEvent(QPaintEvent *e) override;
|
||||
|
|
|
@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_media_types.h"
|
||||
#include "history/history_service.h"
|
||||
#include "history/history_item_components.h"
|
||||
#include "history/history_inner_widget.h"
|
||||
#include "dialogs/dialogs_indexed_list.h"
|
||||
#include "styles/style_dialogs.h"
|
||||
#include "data/data_drafts.h"
|
||||
|
@ -59,7 +60,7 @@ History::History(const PeerId &peerId)
|
|||
, _sendActionText(st::dialogsTextWidthMin) {
|
||||
if (const auto user = peer->asUser()) {
|
||||
if (user->botInfo) {
|
||||
outboxReadBefore = INT_MAX;
|
||||
outboxReadBefore = std::numeric_limits<MsgId>::max();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -736,13 +737,26 @@ HistoryItem *History::createItem(
|
|||
const auto media = message.c_message().has_media()
|
||||
? &message.c_message().vmedia
|
||||
: nullptr;
|
||||
result->updateMedia(media);
|
||||
result->updateSentMedia(media);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
return HistoryItem::Create(this, message);
|
||||
}
|
||||
|
||||
std::vector<not_null<HistoryItem*>> History::createItems(
|
||||
const QVector<MTPMessage> &data) {
|
||||
auto result = std::vector<not_null<HistoryItem*>>();
|
||||
result.reserve(data.size());
|
||||
for (auto i = data.cend(), e = data.cbegin(); i != e;) {
|
||||
const auto detachExistingItem = true;
|
||||
if (const auto item = createItem(*--i, detachExistingItem)) {
|
||||
result.push_back(item);
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
not_null<HistoryItem*> History::createItemForwarded(
|
||||
MsgId id,
|
||||
MTPDmessage::Flags flags,
|
||||
|
@ -1081,10 +1095,6 @@ not_null<HistoryItem*> History::addNewItem(
|
|||
|
||||
addItemToBlock(item);
|
||||
|
||||
const auto [groupFrom, groupTill] = recountGroupingFromTill(item->mainView());
|
||||
if (groupFrom != groupTill || groupFrom->data()->groupId()) {
|
||||
recountGrouping(groupFrom, groupTill);
|
||||
}
|
||||
if (!unread && IsServerMsgId(item->id)) {
|
||||
if (const auto sharedMediaTypes = item->sharedMediaTypes()) {
|
||||
auto from = loadedAtTop() ? 0 : minMsgId();
|
||||
|
@ -1308,7 +1318,7 @@ void History::applyServiceChanges(
|
|||
if (d.vphoto.type() == mtpc_photo) {
|
||||
auto &sizes = d.vphoto.c_photo().vsizes.v;
|
||||
if (!sizes.isEmpty()) {
|
||||
auto photo = App::feedPhoto(d.vphoto.c_photo());
|
||||
auto photo = Auth().data().photo(d.vphoto.c_photo());
|
||||
if (photo) photo->peer = peer;
|
||||
auto &smallSize = sizes.front();
|
||||
auto &bigSize = sizes.back();
|
||||
|
@ -1446,8 +1456,7 @@ void History::addItemToBlock(not_null<HistoryItem*> item) {
|
|||
auto block = prepareBlockForAddingItem();
|
||||
|
||||
block->messages.push_back(item->createView(
|
||||
App::wnd()->controller(),
|
||||
HistoryView::Context::History));
|
||||
HistoryInner::ElementDelegate()));
|
||||
const auto view = block->messages.back().get();
|
||||
view->attachToBlock(block, block->messages.size() - 1);
|
||||
view->previousInBlocksChanged();
|
||||
|
@ -1457,21 +1466,16 @@ void History::addItemToBlock(not_null<HistoryItem*> item) {
|
|||
}
|
||||
}
|
||||
|
||||
template <int kSharedMediaTypeCount>
|
||||
void History::addToSharedMedia(
|
||||
std::vector<MsgId> (&medias)[kSharedMediaTypeCount],
|
||||
bool force) {
|
||||
void History::addEdgesToSharedMedia() {
|
||||
auto from = loadedAtTop() ? 0 : minMsgId();
|
||||
auto till = loadedAtBottom() ? ServerMaxMsgId : maxMsgId();
|
||||
for (auto i = 0; i != Storage::kSharedMediaTypeCount; ++i) {
|
||||
if (force || !medias[i].empty()) {
|
||||
auto type = static_cast<Storage::SharedMediaType>(i);
|
||||
Auth().storage().add(Storage::SharedMediaAddSlice(
|
||||
peer->id,
|
||||
type,
|
||||
std::move(medias[i]),
|
||||
{ from, till }));
|
||||
}
|
||||
const auto type = static_cast<Storage::SharedMediaType>(i);
|
||||
Auth().storage().add(Storage::SharedMediaAddSlice(
|
||||
peer->id,
|
||||
type,
|
||||
{},
|
||||
{ from, till }));
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1492,119 +1496,40 @@ void History::addOlderSlice(const QVector<MTPMessage> &slice) {
|
|||
logged.push_back(QString::number(minMsgId()));
|
||||
logged.push_back(QString::number(maxMsgId()));
|
||||
|
||||
auto minAdded = -1;
|
||||
auto maxAdded = -1;
|
||||
if (const auto added = createItems(slice); !added.empty()) {
|
||||
auto minAdded = -1;
|
||||
auto maxAdded = -1;
|
||||
|
||||
startBuildingFrontBlock(slice.size());
|
||||
|
||||
for (auto i = slice.cend(), e = slice.cbegin(); i != e;) {
|
||||
--i;
|
||||
const auto detachExistingItem = true;
|
||||
const auto adding = createItem(*i, detachExistingItem);
|
||||
if (!adding) continue;
|
||||
|
||||
if (!firstAdded) firstAdded = adding;
|
||||
lastAdded = adding;
|
||||
|
||||
if (minAdded < 0 || minAdded > adding->id) {
|
||||
minAdded = adding->id;
|
||||
startBuildingFrontBlock(added.size());
|
||||
for (const auto item : added) {
|
||||
addItemToBlock(item);
|
||||
if (minAdded < 0 || minAdded > item->id) {
|
||||
minAdded = item->id;
|
||||
}
|
||||
if (maxAdded < 0 || maxAdded < item->id) {
|
||||
maxAdded = item->id;
|
||||
}
|
||||
}
|
||||
if (maxAdded < 0 || maxAdded < adding->id) {
|
||||
maxAdded = adding->id;
|
||||
auto block = finishBuildingFrontBlock();
|
||||
|
||||
if (loadedAtBottom()) {
|
||||
// Add photos to overview and authors to lastAuthors.
|
||||
addItemsToLists(added);
|
||||
}
|
||||
|
||||
addItemToBlock(adding);
|
||||
}
|
||||
logged.push_back(QString::number(minAdded));
|
||||
logged.push_back(QString::number(maxAdded));
|
||||
CrashReports::SetAnnotation(
|
||||
"old_minmaxwas_minmaxadd",
|
||||
logged.join(";"));
|
||||
|
||||
auto block = finishBuildingFrontBlock();
|
||||
if (!block) {
|
||||
addToSharedMedia(added);
|
||||
|
||||
CrashReports::ClearAnnotation("old_minmaxwas_minmaxadd");
|
||||
} else {
|
||||
// If no items were added it means we've loaded everything old.
|
||||
oldLoaded = true;
|
||||
} else if (loadedAtBottom()) { // add photos to overview and authors to lastAuthors
|
||||
bool channel = isChannel();
|
||||
std::deque<not_null<UserData*>> *lastAuthors = nullptr;
|
||||
base::flat_set<not_null<PeerData*>> *markupSenders = nullptr;
|
||||
if (peer->isChat()) {
|
||||
lastAuthors = &peer->asChat()->lastAuthors;
|
||||
markupSenders = &peer->asChat()->markupSenders;
|
||||
} else if (peer->isMegagroup()) {
|
||||
// We don't add users to mgInfo->lastParticipants here.
|
||||
// We're scrolling back and we see messages from users that
|
||||
// could be gone from the megagroup already. It is fine for
|
||||
// chat->lastAuthors, because they're used only for field
|
||||
// autocomplete, but this is bad for megagroups, because its
|
||||
// lastParticipants are displayed in Profile as members list.
|
||||
markupSenders = &peer->asChannel()->mgInfo->markupSenders;
|
||||
}
|
||||
for (auto i = block->messages.size(); i > 0; --i) {
|
||||
const auto item = block->messages[i - 1]->data();
|
||||
item->addToUnreadMentions(UnreadMentionType::Existing);
|
||||
if (item->from()->id) {
|
||||
if (lastAuthors) { // chats
|
||||
if (auto user = item->from()->asUser()) {
|
||||
if (!base::contains(*lastAuthors, user)) {
|
||||
lastAuthors->push_back(user);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (item->author()->id) {
|
||||
if (markupSenders) { // chats with bots
|
||||
if (!lastKeyboardInited && item->definesReplyKeyboard() && !item->out()) {
|
||||
auto markupFlags = item->replyKeyboardFlags();
|
||||
if (!(markupFlags & MTPDreplyKeyboardMarkup::Flag::f_selective) || item->mentionsMe()) {
|
||||
bool wasKeyboardHide = markupSenders->contains(item->author());
|
||||
if (!wasKeyboardHide) {
|
||||
markupSenders->insert(item->author());
|
||||
}
|
||||
if (!(markupFlags & MTPDreplyKeyboardMarkup_ClientFlag::f_zero)) {
|
||||
if (!lastKeyboardInited) {
|
||||
bool botNotInChat = false;
|
||||
if (peer->isChat()) {
|
||||
botNotInChat = (!peer->canWrite() || !peer->asChat()->participants.empty()) && item->author()->isUser() && !peer->asChat()->participants.contains(item->author()->asUser());
|
||||
} else if (peer->isMegagroup()) {
|
||||
botNotInChat = (!peer->canWrite() || peer->asChannel()->mgInfo->botStatus != 0) && item->author()->isUser() && !peer->asChannel()->mgInfo->bots.contains(item->author()->asUser());
|
||||
}
|
||||
if (wasKeyboardHide || botNotInChat) {
|
||||
clearLastKeyboard();
|
||||
} else {
|
||||
lastKeyboardInited = true;
|
||||
lastKeyboardId = item->id;
|
||||
lastKeyboardFrom = item->author()->id;
|
||||
lastKeyboardUsed = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (!lastKeyboardInited && item->definesReplyKeyboard() && !item->out()) { // conversations with bots
|
||||
MTPDreplyKeyboardMarkup::Flags markupFlags = item->replyKeyboardFlags();
|
||||
if (!(markupFlags & MTPDreplyKeyboardMarkup::Flag::f_selective) || item->mentionsMe()) {
|
||||
if (markupFlags & MTPDreplyKeyboardMarkup_ClientFlag::f_zero) {
|
||||
clearLastKeyboard();
|
||||
} else {
|
||||
lastKeyboardInited = true;
|
||||
lastKeyboardId = item->id;
|
||||
lastKeyboardFrom = item->author()->id;
|
||||
lastKeyboardUsed = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
logged.push_back(QString::number(minAdded));
|
||||
logged.push_back(QString::number(maxAdded));
|
||||
CrashReports::SetAnnotation("old_minmaxwas_minmaxadd", logged.join(";"));
|
||||
|
||||
addBlockToSharedMedia(block);
|
||||
|
||||
CrashReports::ClearAnnotation("old_minmaxwas_minmaxadd");
|
||||
|
||||
if (lastAdded) {
|
||||
const auto [from, till] = recountGroupingFromTill(lastAdded->mainView());
|
||||
recountGrouping(firstAdded->mainView(), till);
|
||||
addEdgesToSharedMedia();
|
||||
}
|
||||
|
||||
if (isChannel()) {
|
||||
|
@ -1624,11 +1549,9 @@ void History::addNewerSlice(const QVector<MTPMessage> &slice) {
|
|||
}
|
||||
}
|
||||
|
||||
auto firstAdded = (HistoryItem*)nullptr;
|
||||
auto lastAdded = (HistoryItem*)nullptr;
|
||||
if (const auto added = createItems(slice); !added.empty()) {
|
||||
Assert(!isBuildingFrontBlock());
|
||||
|
||||
Assert(!isBuildingFrontBlock());
|
||||
if (!slice.isEmpty()) {
|
||||
auto logged = QStringList();
|
||||
logged.push_back(QString::number(minMsgId()));
|
||||
logged.push_back(QString::number(maxMsgId()));
|
||||
|
@ -1636,58 +1559,35 @@ void History::addNewerSlice(const QVector<MTPMessage> &slice) {
|
|||
auto minAdded = -1;
|
||||
auto maxAdded = -1;
|
||||
|
||||
std::vector<MsgId> medias[Storage::kSharedMediaTypeCount];
|
||||
for (auto i = slice.cend(), e = slice.cbegin(); i != e;) {
|
||||
--i;
|
||||
const auto detachExistingItem = true;
|
||||
const auto adding = createItem(*i, detachExistingItem);
|
||||
if (!adding) continue;
|
||||
|
||||
if (!firstAdded) firstAdded = adding;
|
||||
lastAdded = adding;
|
||||
|
||||
if (minAdded < 0 || minAdded > adding->id) {
|
||||
minAdded = adding->id;
|
||||
for (const auto item : added) {
|
||||
addItemToBlock(item);
|
||||
if (minAdded < 0 || minAdded > item->id) {
|
||||
minAdded = item->id;
|
||||
}
|
||||
if (maxAdded < 0 || maxAdded < adding->id) {
|
||||
maxAdded = adding->id;
|
||||
}
|
||||
|
||||
addItemToBlock(adding);
|
||||
if (auto types = adding->sharedMediaTypes()) {
|
||||
for (auto i = 0; i != Storage::kSharedMediaTypeCount; ++i) {
|
||||
auto type = static_cast<Storage::SharedMediaType>(i);
|
||||
if (types.test(type)) {
|
||||
if (medias[i].empty()) {
|
||||
medias[i].reserve(slice.size());
|
||||
}
|
||||
medias[i].push_back(adding->id);
|
||||
}
|
||||
}
|
||||
if (maxAdded < 0 || maxAdded < item->id) {
|
||||
maxAdded = item->id;
|
||||
}
|
||||
}
|
||||
|
||||
logged.push_back(QString::number(minAdded));
|
||||
logged.push_back(QString::number(maxAdded));
|
||||
CrashReports::SetAnnotation("new_minmaxwas_minmaxadd", logged.join(";"));
|
||||
CrashReports::SetAnnotation(
|
||||
"new_minmaxwas_minmaxadd",
|
||||
logged.join(";"));
|
||||
|
||||
if (!firstAdded) {
|
||||
newLoaded = true;
|
||||
setLastMessage(lastAvailableMessage());
|
||||
}
|
||||
addToSharedMedia(medias, wasLoadedAtBottom != loadedAtBottom());
|
||||
addToSharedMedia(added);
|
||||
|
||||
CrashReports::ClearAnnotation("new_minmaxwas_minmaxadd");
|
||||
} else {
|
||||
newLoaded = true;
|
||||
setLastMessage(lastAvailableMessage());
|
||||
addEdgesToSharedMedia();
|
||||
}
|
||||
|
||||
if (!wasLoadedAtBottom) {
|
||||
checkAddAllToUnreadMentions();
|
||||
}
|
||||
|
||||
if (firstAdded) {
|
||||
const auto [from, till] = recountGroupingFromTill(firstAdded->mainView());
|
||||
recountGrouping(from, lastAdded->mainView());
|
||||
}
|
||||
|
||||
if (isChannel()) asChannelHistory()->checkJoinedMessage();
|
||||
checkLastMsg();
|
||||
}
|
||||
|
@ -1703,6 +1603,80 @@ void History::checkLastMsg() {
|
|||
}
|
||||
}
|
||||
|
||||
void History::addItemsToLists(
|
||||
const std::vector<not_null<HistoryItem*>> &items) {
|
||||
std::deque<not_null<UserData*>> *lastAuthors = nullptr;
|
||||
base::flat_set<not_null<PeerData*>> *markupSenders = nullptr;
|
||||
if (peer->isChat()) {
|
||||
lastAuthors = &peer->asChat()->lastAuthors;
|
||||
markupSenders = &peer->asChat()->markupSenders;
|
||||
} else if (peer->isMegagroup()) {
|
||||
// We don't add users to mgInfo->lastParticipants here.
|
||||
// We're scrolling back and we see messages from users that
|
||||
// could be gone from the megagroup already. It is fine for
|
||||
// chat->lastAuthors, because they're used only for field
|
||||
// autocomplete, but this is bad for megagroups, because its
|
||||
// lastParticipants are displayed in Profile as members list.
|
||||
markupSenders = &peer->asChannel()->mgInfo->markupSenders;
|
||||
}
|
||||
for (const auto item : base::reversed(items)) {
|
||||
item->addToUnreadMentions(UnreadMentionType::Existing);
|
||||
if (item->from()->id) {
|
||||
if (lastAuthors) { // chats
|
||||
if (auto user = item->from()->asUser()) {
|
||||
if (!base::contains(*lastAuthors, user)) {
|
||||
lastAuthors->push_back(user);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (item->author()->id) {
|
||||
if (markupSenders) { // chats with bots
|
||||
if (!lastKeyboardInited && item->definesReplyKeyboard() && !item->out()) {
|
||||
auto markupFlags = item->replyKeyboardFlags();
|
||||
if (!(markupFlags & MTPDreplyKeyboardMarkup::Flag::f_selective) || item->mentionsMe()) {
|
||||
bool wasKeyboardHide = markupSenders->contains(item->author());
|
||||
if (!wasKeyboardHide) {
|
||||
markupSenders->insert(item->author());
|
||||
}
|
||||
if (!(markupFlags & MTPDreplyKeyboardMarkup_ClientFlag::f_zero)) {
|
||||
if (!lastKeyboardInited) {
|
||||
bool botNotInChat = false;
|
||||
if (peer->isChat()) {
|
||||
botNotInChat = (!peer->canWrite() || !peer->asChat()->participants.empty()) && item->author()->isUser() && !peer->asChat()->participants.contains(item->author()->asUser());
|
||||
} else if (peer->isMegagroup()) {
|
||||
botNotInChat = (!peer->canWrite() || peer->asChannel()->mgInfo->botStatus != 0) && item->author()->isUser() && !peer->asChannel()->mgInfo->bots.contains(item->author()->asUser());
|
||||
}
|
||||
if (wasKeyboardHide || botNotInChat) {
|
||||
clearLastKeyboard();
|
||||
} else {
|
||||
lastKeyboardInited = true;
|
||||
lastKeyboardId = item->id;
|
||||
lastKeyboardFrom = item->author()->id;
|
||||
lastKeyboardUsed = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (!lastKeyboardInited && item->definesReplyKeyboard() && !item->out()) { // conversations with bots
|
||||
MTPDreplyKeyboardMarkup::Flags markupFlags = item->replyKeyboardFlags();
|
||||
if (!(markupFlags & MTPDreplyKeyboardMarkup::Flag::f_selective) || item->mentionsMe()) {
|
||||
if (markupFlags & MTPDreplyKeyboardMarkup_ClientFlag::f_zero) {
|
||||
clearLastKeyboard();
|
||||
} else {
|
||||
lastKeyboardInited = true;
|
||||
lastKeyboardId = item->id;
|
||||
lastKeyboardFrom = item->author()->id;
|
||||
lastKeyboardUsed = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void History::checkAddAllToUnreadMentions() {
|
||||
if (!loadedAtBottom()) {
|
||||
return;
|
||||
|
@ -1716,25 +1690,34 @@ void History::checkAddAllToUnreadMentions() {
|
|||
}
|
||||
}
|
||||
|
||||
void History::addBlockToSharedMedia(HistoryBlock *block) {
|
||||
void History::addToSharedMedia(
|
||||
const std::vector<not_null<HistoryItem*>> &items) {
|
||||
std::vector<MsgId> medias[Storage::kSharedMediaTypeCount];
|
||||
if (block) {
|
||||
for (const auto &message : block->messages) {
|
||||
const auto item = message->data();
|
||||
if (auto types = item->sharedMediaTypes()) {
|
||||
for (auto i = 0; i != Storage::kSharedMediaTypeCount; ++i) {
|
||||
auto type = static_cast<Storage::SharedMediaType>(i);
|
||||
if (types.test(type)) {
|
||||
if (medias[i].empty()) {
|
||||
medias[i].reserve(block->messages.size());
|
||||
}
|
||||
medias[i].push_back(item->id);
|
||||
for (const auto item : items) {
|
||||
if (const auto types = item->sharedMediaTypes()) {
|
||||
for (auto i = 0; i != Storage::kSharedMediaTypeCount; ++i) {
|
||||
const auto type = static_cast<Storage::SharedMediaType>(i);
|
||||
if (types.test(type)) {
|
||||
if (medias[i].empty()) {
|
||||
medias[i].reserve(items.size());
|
||||
}
|
||||
medias[i].push_back(item->id);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
addToSharedMedia(medias, !block);
|
||||
const auto from = loadedAtTop() ? 0 : minMsgId();
|
||||
const auto till = loadedAtBottom() ? ServerMaxMsgId : maxMsgId();
|
||||
for (auto i = 0; i != Storage::kSharedMediaTypeCount; ++i) {
|
||||
if (!medias[i].empty()) {
|
||||
const auto type = static_cast<Storage::SharedMediaType>(i);
|
||||
Auth().storage().add(Storage::SharedMediaAddSlice(
|
||||
peer->id,
|
||||
type,
|
||||
std::move(medias[i]),
|
||||
{ from, till }));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int History::countUnread(MsgId upTo) {
|
||||
|
@ -2026,8 +2009,7 @@ not_null<HistoryItem*> History::addNewInTheMiddle(
|
|||
const auto it = block->messages.insert(
|
||||
block->messages.begin() + itemIndex,
|
||||
item->createView(
|
||||
App::wnd()->controller(),
|
||||
HistoryView::Context::History));
|
||||
HistoryInner::ElementDelegate()));
|
||||
(*it)->attachToBlock(block.get(), itemIndex);
|
||||
(*it)->previousInBlocksChanged();
|
||||
if (itemIndex + 1 < block->messages.size()) {
|
||||
|
@ -2041,70 +2023,65 @@ not_null<HistoryItem*> History::addNewInTheMiddle(
|
|||
(*it)->nextInBlocksChanged();
|
||||
}
|
||||
|
||||
const auto [groupFrom, groupTill] = recountGroupingFromTill(item->mainView());
|
||||
if (groupFrom != groupTill || groupFrom->data()->groupId()) {
|
||||
recountGrouping(groupFrom, groupTill);
|
||||
}
|
||||
|
||||
return item;
|
||||
}
|
||||
|
||||
HistoryView::Element *History::findNextItem(
|
||||
not_null<HistoryView::Element*> view) const {
|
||||
const auto nextBlockIndex = view->block()->indexInHistory() + 1;
|
||||
const auto nextItemIndex = view->indexInBlock() + 1;
|
||||
if (nextItemIndex < int(view->block()->messages.size())) {
|
||||
return view->block()->messages[nextItemIndex].get();
|
||||
} else if (nextBlockIndex < int(blocks.size())) {
|
||||
return blocks[nextBlockIndex]->messages.front().get();
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
HistoryView::Element *History::findPreviousItem(
|
||||
not_null<HistoryView::Element*> view) const {
|
||||
const auto blockIndex = view->block()->indexInHistory();
|
||||
const auto itemIndex = view->indexInBlock();
|
||||
if (itemIndex > 0) {
|
||||
return view->block()->messages[itemIndex - 1].get();
|
||||
} else if (blockIndex > 0) {
|
||||
return blocks[blockIndex - 1]->messages.back().get();
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
not_null<HistoryView::Element*> History::findGroupFirst(
|
||||
not_null<HistoryView::Element*> view) const {
|
||||
const auto group = view->Get<HistoryView::Group>();
|
||||
Assert(group != nullptr);
|
||||
Assert(group->leader != nullptr);
|
||||
|
||||
const auto leaderGroup = (group->leader == view)
|
||||
? group
|
||||
: group->leader->Get<HistoryView::Group>();
|
||||
Assert(leaderGroup != nullptr);
|
||||
|
||||
return leaderGroup->others.empty()
|
||||
? group->leader
|
||||
: leaderGroup->others.front().get();
|
||||
}
|
||||
|
||||
not_null<HistoryView::Element*> History::findGroupLast(
|
||||
not_null<HistoryView::Element*> view) const {
|
||||
const auto group = view->Get<HistoryView::Group>();
|
||||
Assert(group != nullptr);
|
||||
|
||||
return group->leader;
|
||||
}
|
||||
|
||||
void History::recountGroupingAround(not_null<HistoryItem*> item) {
|
||||
Expects(item->history() == this);
|
||||
|
||||
if (item->mainView() && item->groupId()) {
|
||||
const auto [groupFrom, groupTill] = recountGroupingFromTill(item->mainView());
|
||||
recountGrouping(groupFrom, groupTill);
|
||||
}
|
||||
}
|
||||
//
|
||||
//HistoryView::Element *History::findNextItem(
|
||||
// not_null<HistoryView::Element*> view) const {
|
||||
// const auto nextBlockIndex = view->block()->indexInHistory() + 1;
|
||||
// const auto nextItemIndex = view->indexInBlock() + 1;
|
||||
// if (nextItemIndex < int(view->block()->messages.size())) {
|
||||
// return view->block()->messages[nextItemIndex].get();
|
||||
// } else if (nextBlockIndex < int(blocks.size())) {
|
||||
// return blocks[nextBlockIndex]->messages.front().get();
|
||||
// }
|
||||
// return nullptr;
|
||||
//}
|
||||
//
|
||||
//HistoryView::Element *History::findPreviousItem(
|
||||
// not_null<HistoryView::Element*> view) const {
|
||||
// const auto blockIndex = view->block()->indexInHistory();
|
||||
// const auto itemIndex = view->indexInBlock();
|
||||
// if (itemIndex > 0) {
|
||||
// return view->block()->messages[itemIndex - 1].get();
|
||||
// } else if (blockIndex > 0) {
|
||||
// return blocks[blockIndex - 1]->messages.back().get();
|
||||
// }
|
||||
// return nullptr;
|
||||
//}
|
||||
//
|
||||
//not_null<HistoryView::Element*> History::findGroupFirst(
|
||||
// not_null<HistoryView::Element*> view) const {
|
||||
// const auto group = view->Get<HistoryView::Group>();
|
||||
// Assert(group != nullptr);
|
||||
// Assert(group->leader != nullptr);
|
||||
//
|
||||
// const auto leaderGroup = (group->leader == view)
|
||||
// ? group
|
||||
// : group->leader->Get<HistoryView::Group>();
|
||||
// Assert(leaderGroup != nullptr);
|
||||
//
|
||||
// return leaderGroup->others.empty()
|
||||
// ? group->leader
|
||||
// : leaderGroup->others.front().get();
|
||||
//}
|
||||
//
|
||||
//not_null<HistoryView::Element*> History::findGroupLast(
|
||||
// not_null<HistoryView::Element*> view) const {
|
||||
// const auto group = view->Get<HistoryView::Group>();
|
||||
// Assert(group != nullptr);
|
||||
//
|
||||
// return group->leader;
|
||||
//}
|
||||
//
|
||||
//void History::recountGroupingAround(not_null<HistoryItem*> item) {
|
||||
// Expects(item->history() == this);
|
||||
//
|
||||
// if (item->mainView() && item->groupId()) {
|
||||
// const auto [groupFrom, groupTill] = recountGroupingFromTill(item->mainView());
|
||||
// recountGrouping(groupFrom, groupTill);
|
||||
// }
|
||||
//}
|
||||
|
||||
int History::chatListUnreadCount() const {
|
||||
const auto result = unreadCount();
|
||||
|
@ -2135,65 +2112,65 @@ void History::paintUserpic(
|
|||
int size) const {
|
||||
peer->paintUserpic(p, x, y, size);
|
||||
}
|
||||
|
||||
auto History::recountGroupingFromTill(not_null<HistoryView::Element*> view)
|
||||
-> std::pair<not_null<HistoryView::Element*>, not_null<HistoryView::Element*>> {
|
||||
const auto recountFromItem = [&] {
|
||||
if (const auto prev = findPreviousItem(view)) {
|
||||
if (prev->data()->groupId()) {
|
||||
return findGroupFirst(prev);
|
||||
}
|
||||
}
|
||||
return view;
|
||||
}();
|
||||
if (recountFromItem == view && !view->data()->groupId()) {
|
||||
return { view, view };
|
||||
}
|
||||
const auto recountTillItem = [&] {
|
||||
if (const auto next = findNextItem(view)) {
|
||||
if (next->data()->groupId()) {
|
||||
return findGroupLast(next);
|
||||
}
|
||||
}
|
||||
return view;
|
||||
}();
|
||||
return { recountFromItem, recountTillItem };
|
||||
}
|
||||
|
||||
void History::recountGrouping(
|
||||
not_null<HistoryView::Element*> from,
|
||||
not_null<HistoryView::Element*> till) {
|
||||
from->validateGroupId();
|
||||
auto others = std::vector<not_null<HistoryView::Element*>>();
|
||||
auto currentGroupId = from->data()->groupId();
|
||||
auto prev = from;
|
||||
while (prev != till) {
|
||||
auto item = findNextItem(prev);
|
||||
item->validateGroupId();
|
||||
const auto groupId = item->data()->groupId();
|
||||
if (currentGroupId) {
|
||||
if (groupId == currentGroupId) {
|
||||
others.push_back(prev);
|
||||
} else {
|
||||
for (const auto other : others) {
|
||||
other->makeGroupMember(prev);
|
||||
}
|
||||
prev->makeGroupLeader(base::take(others));
|
||||
currentGroupId = groupId;
|
||||
}
|
||||
} else if (groupId) {
|
||||
currentGroupId = groupId;
|
||||
}
|
||||
prev = item;
|
||||
}
|
||||
|
||||
if (currentGroupId) {
|
||||
for (const auto other : others) {
|
||||
other->makeGroupMember(prev);
|
||||
}
|
||||
till->makeGroupLeader(base::take(others));
|
||||
}
|
||||
}
|
||||
//
|
||||
//auto History::recountGroupingFromTill(not_null<HistoryView::Element*> view)
|
||||
//-> std::pair<not_null<HistoryView::Element*>, not_null<HistoryView::Element*>> {
|
||||
// const auto recountFromItem = [&] {
|
||||
// if (const auto prev = findPreviousItem(view)) {
|
||||
// if (prev->data()->groupId()) {
|
||||
// return findGroupFirst(prev);
|
||||
// }
|
||||
// }
|
||||
// return view;
|
||||
// }();
|
||||
// if (recountFromItem == view && !view->data()->groupId()) {
|
||||
// return { view, view };
|
||||
// }
|
||||
// const auto recountTillItem = [&] {
|
||||
// if (const auto next = findNextItem(view)) {
|
||||
// if (next->data()->groupId()) {
|
||||
// return findGroupLast(next);
|
||||
// }
|
||||
// }
|
||||
// return view;
|
||||
// }();
|
||||
// return { recountFromItem, recountTillItem };
|
||||
//}
|
||||
//
|
||||
//void History::recountGrouping(
|
||||
// not_null<HistoryView::Element*> from,
|
||||
// not_null<HistoryView::Element*> till) {
|
||||
// from->validateGroupId();
|
||||
// auto others = std::vector<not_null<HistoryView::Element*>>();
|
||||
// auto currentGroupId = from->data()->groupId();
|
||||
// auto prev = from;
|
||||
// while (prev != till) {
|
||||
// auto item = findNextItem(prev);
|
||||
// item->validateGroupId();
|
||||
// const auto groupId = item->data()->groupId();
|
||||
// if (currentGroupId) {
|
||||
// if (groupId == currentGroupId) {
|
||||
// others.push_back(prev);
|
||||
// } else {
|
||||
// for (const auto other : others) {
|
||||
// other->makeGroupMember(prev);
|
||||
// }
|
||||
// prev->makeGroupLeader(base::take(others));
|
||||
// currentGroupId = groupId;
|
||||
// }
|
||||
// } else if (groupId) {
|
||||
// currentGroupId = groupId;
|
||||
// }
|
||||
// prev = item;
|
||||
// }
|
||||
//
|
||||
// if (currentGroupId) {
|
||||
// for (const auto other : others) {
|
||||
// other->makeGroupMember(prev);
|
||||
// }
|
||||
// till->makeGroupLeader(base::take(others));
|
||||
// }
|
||||
//}
|
||||
|
||||
void History::startBuildingFrontBlock(int expectedItemsCount) {
|
||||
Assert(!isBuildingFrontBlock());
|
||||
|
@ -2451,14 +2428,6 @@ void History::clear(bool leaveItems) {
|
|||
} else {
|
||||
setLastMessage(nullptr);
|
||||
notifies.clear();
|
||||
auto &pending = Global::RefPendingRepaintItems();
|
||||
for (auto i = pending.begin(); i != pending.end();) {
|
||||
if ((*i)->history() == this) {
|
||||
i = pending.erase(i);
|
||||
} else {
|
||||
++i;
|
||||
}
|
||||
}
|
||||
Auth().storage().remove(Storage::SharedMediaRemoveAll(peer->id));
|
||||
Auth().data().markHistoryCleared(this);
|
||||
}
|
||||
|
@ -2621,18 +2590,6 @@ void HistoryBlock::clear(bool leaveItems) {
|
|||
void HistoryBlock::remove(not_null<Element*> view) {
|
||||
Expects(view->block() == this);
|
||||
|
||||
auto [groupFrom, groupTill] = _history->recountGroupingFromTill(view);
|
||||
const auto groupHistory = _history;
|
||||
const auto needGroupRecount = (groupFrom != groupTill);
|
||||
if (needGroupRecount) {
|
||||
if (groupFrom == view) {
|
||||
groupFrom = groupHistory->findNextItem(groupFrom);
|
||||
}
|
||||
if (groupTill == view) {
|
||||
groupTill = groupHistory->findPreviousItem(groupTill);
|
||||
}
|
||||
}
|
||||
|
||||
const auto item = view->data();
|
||||
auto blockIndex = indexInHistory();
|
||||
auto itemIndex = view->indexInBlock();
|
||||
|
@ -2664,10 +2621,6 @@ void HistoryBlock::remove(not_null<Element*> view) {
|
|||
} else if (!_history->blocks.empty() && !_history->blocks.back()->messages.empty()) {
|
||||
_history->blocks.back()->messages.back()->nextInBlocksChanged();
|
||||
}
|
||||
|
||||
if (needGroupRecount) {
|
||||
groupHistory->recountGrouping(groupFrom, groupTill);
|
||||
}
|
||||
}
|
||||
|
||||
HistoryBlock::~HistoryBlock() {
|
||||
|
|
|
@ -167,6 +167,8 @@ public:
|
|||
HistoryItem *createItem(
|
||||
const MTPMessage &message,
|
||||
bool detachExistingItem);
|
||||
std::vector<not_null<HistoryItem*>> createItems(
|
||||
const QVector<MTPMessage> &data);
|
||||
|
||||
void addOlderSlice(const QVector<MTPMessage> &slice);
|
||||
void addNewerSlice(const QVector<MTPMessage> &slice);
|
||||
|
@ -448,8 +450,10 @@ private:
|
|||
|
||||
template <int kSharedMediaTypeCount>
|
||||
void addToSharedMedia(std::vector<MsgId> (&medias)[kSharedMediaTypeCount], bool force);
|
||||
void addBlockToSharedMedia(HistoryBlock *block);
|
||||
void addToSharedMedia(const std::vector<not_null<HistoryItem*>> &items);
|
||||
void addEdgesToSharedMedia();
|
||||
|
||||
void addItemsToLists(const std::vector<not_null<HistoryItem*>> &items);
|
||||
void clearSendAction(not_null<UserData*> from);
|
||||
|
||||
HistoryView::Element *findPreviousItem(
|
||||
|
|
|
@ -14,8 +14,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_message.h"
|
||||
#include "history/history_media_types.h"
|
||||
#include "history/history_item_components.h"
|
||||
#include "history/view/history_view_message.h"
|
||||
#include "history/view/history_view_service_message.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
#include "ui/text_options.h"
|
||||
#include "ui/widgets/popup_menu.h"
|
||||
#include "window/window_controller.h"
|
||||
|
@ -414,31 +414,31 @@ TextSelection HistoryInner::computeRenderSelection(
|
|||
}
|
||||
return TextSelection();
|
||||
};
|
||||
const auto group = view->Get<HistoryView::Group>();
|
||||
if (group) {
|
||||
if (group->leader != view) {
|
||||
return TextSelection();
|
||||
}
|
||||
auto result = TextSelection();
|
||||
auto allFullSelected = true;
|
||||
const auto count = int(group->others.size());
|
||||
for (auto i = 0; i != count; ++i) {
|
||||
if (itemSelection(group->others[i]->data()) == FullSelection) {
|
||||
result = AddGroupItemSelection(result, i);
|
||||
} else {
|
||||
allFullSelected = false;
|
||||
}
|
||||
}
|
||||
const auto leaderSelection = itemSelection(view->data());
|
||||
if (leaderSelection == FullSelection) {
|
||||
return allFullSelected
|
||||
? FullSelection
|
||||
: AddGroupItemSelection(result, count);
|
||||
} else if (leaderSelection != TextSelection()) {
|
||||
return leaderSelection;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
// #TODO group selection
|
||||
//if (const auto group = view->Get<HistoryView::Group>()) {
|
||||
// if (group->leader != view) {
|
||||
// return TextSelection();
|
||||
// }
|
||||
// auto result = TextSelection();
|
||||
// auto allFullSelected = true;
|
||||
// const auto count = int(group->others.size());
|
||||
// for (auto i = 0; i != count; ++i) {
|
||||
// if (itemSelection(group->others[i]->data()) == FullSelection) {
|
||||
// result = AddGroupItemSelection(result, i);
|
||||
// } else {
|
||||
// allFullSelected = false;
|
||||
// }
|
||||
// }
|
||||
// const auto leaderSelection = itemSelection(view->data());
|
||||
// if (leaderSelection == FullSelection) {
|
||||
// return allFullSelected
|
||||
// ? FullSelection
|
||||
// : AddGroupItemSelection(result, count);
|
||||
// } else if (leaderSelection != TextSelection()) {
|
||||
// return leaderSelection;
|
||||
// }
|
||||
// return result;
|
||||
//}
|
||||
return itemSelection(view->data());
|
||||
}
|
||||
|
||||
|
@ -597,7 +597,7 @@ void HistoryInner::paintEvent(QPaintEvent *e) {
|
|||
}
|
||||
|
||||
if (!readMentions.empty() && App::wnd()->doWeReadMentions()) {
|
||||
App::main()->mediaMarkRead(readMentions);
|
||||
Auth().api().markMediaRead(readMentions);
|
||||
}
|
||||
|
||||
if (mtop >= 0 || htop >= 0) {
|
||||
|
@ -1697,9 +1697,8 @@ void HistoryInner::saveContextGif(FullMsgId itemId) {
|
|||
void HistoryInner::copyContextText(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
if (const auto view = item->mainView()) {
|
||||
const auto group = view->getFullGroup();
|
||||
const auto leader = group ? group->leader : view;
|
||||
setToClipboard(leader->selectedText(FullSelection));
|
||||
// #TODO check for a group
|
||||
setToClipboard(view->selectedText(FullSelection));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1764,24 +1763,25 @@ TextWithEntities HistoryInner::getSelectedText() const {
|
|||
continue;
|
||||
}
|
||||
|
||||
if (const auto group = view->getFullGroup()) {
|
||||
if (groupLeadersAdded.contains(group->leader)) {
|
||||
continue;
|
||||
}
|
||||
const auto leaderSelection = computeRenderSelection(
|
||||
&selected,
|
||||
group->leader);
|
||||
if (leaderSelection == FullSelection) {
|
||||
groupLeadersAdded.emplace(group->leader);
|
||||
addItem(group->leader, FullSelection);
|
||||
} else if (view == group->leader) {
|
||||
const auto leaderFullSelection = AddGroupItemSelection(
|
||||
TextSelection(),
|
||||
int(group->others.size()));
|
||||
addItem(view, leaderFullSelection);
|
||||
} else {
|
||||
addItem(view, FullSelection);
|
||||
}
|
||||
if (const auto group = Auth().data().groups().find(item)) {
|
||||
// #TODO group copy
|
||||
//if (groupLeadersAdded.contains(group->leader)) {
|
||||
// continue;
|
||||
//}
|
||||
//const auto leaderSelection = computeRenderSelection(
|
||||
// &selected,
|
||||
// group->leader);
|
||||
//if (leaderSelection == FullSelection) {
|
||||
// groupLeadersAdded.emplace(group->leader);
|
||||
// addItem(group->leader, FullSelection);
|
||||
//} else if (view == group->leader) {
|
||||
// const auto leaderFullSelection = AddGroupItemSelection(
|
||||
// TextSelection(),
|
||||
// int(group->others.size()));
|
||||
// addItem(view, leaderFullSelection);
|
||||
//} else {
|
||||
// addItem(view, FullSelection);
|
||||
//}
|
||||
} else {
|
||||
addItem(view, FullSelection);
|
||||
}
|
||||
|
@ -2901,3 +2901,24 @@ void HistoryInner::onParentGeometryChanged() {
|
|||
mouseActionUpdate(mousePos);
|
||||
}
|
||||
}
|
||||
|
||||
not_null<HistoryView::ElementDelegate*> HistoryInner::ElementDelegate() {
|
||||
class Result : public HistoryView::ElementDelegate {
|
||||
public:
|
||||
std::unique_ptr<HistoryView::Element> elementCreate(
|
||||
not_null<HistoryMessage*> message) override {
|
||||
return std::make_unique<HistoryView::Message>(
|
||||
message,
|
||||
HistoryView::Context::History);
|
||||
}
|
||||
std::unique_ptr<HistoryView::Element> elementCreate(
|
||||
not_null<HistoryService*> message) override {
|
||||
return std::make_unique<HistoryView::Service>(
|
||||
message,
|
||||
HistoryView::Context::History);
|
||||
}
|
||||
|
||||
};
|
||||
static Result result;
|
||||
return &result;
|
||||
}
|
||||
|
|
|
@ -13,6 +13,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/view/history_view_cursor_state.h"
|
||||
#include "history/view/history_view_top_bar_widget.h"
|
||||
|
||||
namespace HistoryView {
|
||||
class ElementDelegate;
|
||||
} // namespace HistoryView
|
||||
|
||||
namespace Window {
|
||||
class Controller;
|
||||
} // namespace Window
|
||||
|
@ -83,10 +87,13 @@ public:
|
|||
// to move scroll position so that mouse points to the same button row.
|
||||
int moveScrollFollowingInlineKeyboard(const HistoryItem *item, int oldKeyboardTop, int newKeyboardTop);
|
||||
|
||||
// AbstractTooltipShower interface
|
||||
// Ui::AbstractTooltipShower interface.
|
||||
QString tooltipText() const override;
|
||||
QPoint tooltipPos() const override;
|
||||
|
||||
// HistoryView::ElementDelegate interface.
|
||||
static not_null<HistoryView::ElementDelegate*> ElementDelegate();
|
||||
|
||||
~HistoryInner();
|
||||
|
||||
protected:
|
||||
|
|
|
@ -110,7 +110,8 @@ void HistoryItem::finishEdition(int oldKeyboardTop) {
|
|||
}
|
||||
|
||||
void HistoryItem::setGroupId(MessageGroupId groupId) {
|
||||
Auth().data().groups().unregisterMessage(this);
|
||||
Expects(!_groupId);
|
||||
|
||||
_groupId = groupId;
|
||||
Auth().data().groups().registerMessage(this);
|
||||
}
|
||||
|
@ -218,7 +219,7 @@ void HistoryItem::addLogEntryOriginal(
|
|||
Expects(isLogEntry());
|
||||
|
||||
AddComponents(HistoryMessageLogEntryOriginal::Bit());
|
||||
Get<HistoryMessageLogEntryOriginal>()->page = App::feedWebPage(
|
||||
Get<HistoryMessageLogEntryOriginal>()->page = Auth().data().webpage(
|
||||
localId,
|
||||
label,
|
||||
content);
|
||||
|
@ -286,7 +287,6 @@ void HistoryItem::destroy() {
|
|||
}
|
||||
}
|
||||
}
|
||||
Global::RefPendingRepaintItems().remove(this);
|
||||
delete this;
|
||||
}
|
||||
|
||||
|
@ -339,7 +339,7 @@ void HistoryItem::setRealId(MsgId newId) {
|
|||
}
|
||||
|
||||
Auth().data().markItemIdChange({ this, oldId });
|
||||
Auth().data().requestItemRepaint(this);
|
||||
Auth().data().requestItemViewRepaint(this);
|
||||
}
|
||||
|
||||
bool HistoryItem::isPinned() const {
|
||||
|
@ -585,21 +585,17 @@ void HistoryItem::setUnreadBarCount(int count) {
|
|||
Expects(!isLogEntry());
|
||||
|
||||
if (count > 0) {
|
||||
HistoryMessageUnreadBar *bar;
|
||||
if (!Has<HistoryMessageUnreadBar>()) {
|
||||
AddComponents(HistoryMessageUnreadBar::Bit());
|
||||
Auth().data().requestItemViewResize(this);
|
||||
// #TODO recount attach to previous
|
||||
|
||||
bar = Get<HistoryMessageUnreadBar>();
|
||||
} else {
|
||||
bar = Get<HistoryMessageUnreadBar>();
|
||||
if (bar->_freezed) {
|
||||
return;
|
||||
}
|
||||
Global::RefPendingRepaintItems().insert(this);
|
||||
}
|
||||
const auto bar = Get<HistoryMessageUnreadBar>();
|
||||
if (bar->_freezed) {
|
||||
return;
|
||||
}
|
||||
bar->init(count);
|
||||
Auth().data().requestItemViewRepaint(this);
|
||||
} else {
|
||||
destroyUnreadBar();
|
||||
}
|
||||
|
@ -668,13 +664,12 @@ void HistoryItem::clipCallback(Media::Clip::Notification notification) {
|
|||
// if (!stopped) {
|
||||
// Auth().data().requestItemViewResize(this);
|
||||
// Auth().data().markItemLayoutChange(this);
|
||||
// Global::RefPendingRepaintItems().insert(this);
|
||||
// }
|
||||
//} break;
|
||||
|
||||
//case NotificationRepaint: {
|
||||
// if (!reader->currentDisplayed()) {
|
||||
// Auth().data().requestItemRepaint(this);
|
||||
// Auth().data().requestItemViewRepaint(this);
|
||||
// }
|
||||
//} break;
|
||||
//}
|
||||
|
|
|
@ -10,7 +10,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "base/runtime_composer.h"
|
||||
#include "base/flags.h"
|
||||
#include "base/value_ordering.h"
|
||||
#include "history/history_media_pointer.h"
|
||||
#include "history/view/history_view_cursor_state.h"
|
||||
|
||||
enum class UnreadMentionType;
|
||||
|
@ -49,6 +48,7 @@ class Controller;
|
|||
|
||||
namespace HistoryView {
|
||||
enum class Context : char;
|
||||
class ElementDelegate;
|
||||
} // namespace HistoryView
|
||||
|
||||
class HistoryItem : public RuntimeComposer<HistoryItem> {
|
||||
|
@ -148,7 +148,7 @@ public:
|
|||
}
|
||||
virtual void applyEdition(const MTPDmessageService &message) {
|
||||
}
|
||||
virtual void updateMedia(const MTPMessageMedia *media) {
|
||||
virtual void updateSentMedia(const MTPMessageMedia *media) {
|
||||
}
|
||||
virtual void updateReplyMarkup(const MTPReplyMarkup *markup) {
|
||||
}
|
||||
|
@ -272,8 +272,7 @@ public:
|
|||
HistoryItem *nextItem() const;
|
||||
|
||||
virtual std::unique_ptr<HistoryView::Element> createView(
|
||||
not_null<Window::Controller*> controller,
|
||||
HistoryView::Context context) = 0;
|
||||
not_null<HistoryView::ElementDelegate*> delegate) = 0;
|
||||
|
||||
virtual ~HistoryItem();
|
||||
|
||||
|
|
|
@ -562,11 +562,13 @@ ReplyKeyboard::ButtonCoords ReplyKeyboard::findButtonCoordsByClickHandler(const
|
|||
return { -1, -1 };
|
||||
}
|
||||
|
||||
void ReplyKeyboard::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) {
|
||||
if (!p) return;
|
||||
void ReplyKeyboard::clickHandlerPressedChanged(
|
||||
const ClickHandlerPtr &handler,
|
||||
bool pressed) {
|
||||
if (!handler) return;
|
||||
|
||||
_savedPressed = pressed ? p : ClickHandlerPtr();
|
||||
auto coords = findButtonCoordsByClickHandler(p);
|
||||
_savedPressed = pressed ? handler : ClickHandlerPtr();
|
||||
auto coords = findButtonCoordsByClickHandler(handler);
|
||||
if (coords.i >= 0) {
|
||||
auto &button = _rows[coords.i][coords.j];
|
||||
if (pressed) {
|
||||
|
@ -584,7 +586,7 @@ void ReplyKeyboard::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pr
|
|||
if (button.ripple) {
|
||||
button.ripple->lastStop();
|
||||
}
|
||||
if (_savedActive != p) {
|
||||
if (_savedActive != handler) {
|
||||
startAnimation(coords.i, coords.j, -1);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -16,7 +16,7 @@ Storage::SharedMediaTypesMask HistoryMedia::sharedMediaTypes() const {
|
|||
}
|
||||
|
||||
bool HistoryMedia::isDisplayed() const {
|
||||
return !_parent->isHiddenByGroup();
|
||||
return true;
|
||||
}
|
||||
|
||||
QSize HistoryMedia::countCurrentSize(int newWidth) {
|
||||
|
|
|
@ -58,11 +58,6 @@ public:
|
|||
}
|
||||
|
||||
virtual HistoryMediaType type() const = 0;
|
||||
virtual std::unique_ptr<HistoryMedia> clone(
|
||||
not_null<Element*> newParent,
|
||||
not_null<Element*> realParent) const {
|
||||
Unexpected("Attempt to clone a media that can't be grouped.");
|
||||
}
|
||||
|
||||
virtual TextWithEntities selectedText(TextSelection selection) const = 0;
|
||||
|
||||
|
@ -82,7 +77,7 @@ public:
|
|||
virtual bool allowsFastShare() const {
|
||||
return false;
|
||||
}
|
||||
virtual void refreshParentId(not_null<Element*> realParent) {
|
||||
virtual void refreshParentId(not_null<HistoryItem*> realParent) {
|
||||
}
|
||||
virtual void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const = 0;
|
||||
virtual HistoryTextState getState(QPoint point, HistoryStateRequest request) const = 0;
|
||||
|
@ -148,14 +143,6 @@ public:
|
|||
return false;
|
||||
}
|
||||
|
||||
virtual void attachToParent() {
|
||||
}
|
||||
virtual void detachFromParent() {
|
||||
}
|
||||
|
||||
virtual bool canBeGrouped() const {
|
||||
return false;
|
||||
}
|
||||
virtual QSize sizeForGrouping() const {
|
||||
Unexpected("Grouping method call.");
|
||||
}
|
||||
|
@ -177,8 +164,9 @@ public:
|
|||
virtual std::unique_ptr<HistoryMedia> takeLastFromGroup() {
|
||||
return nullptr;
|
||||
}
|
||||
virtual bool applyGroup(const std::vector<not_null<Element*>> &others) {
|
||||
return others.empty();
|
||||
virtual bool applyGroup(
|
||||
const std::vector<not_null<HistoryItem*>> &items) {
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual bool animating() const {
|
||||
|
@ -242,6 +230,10 @@ public:
|
|||
return true;
|
||||
}
|
||||
|
||||
// Should be called only by Data::Session.
|
||||
virtual void updateSharedContactUserId(UserId userId) {
|
||||
}
|
||||
|
||||
virtual ~HistoryMedia() = default;
|
||||
|
||||
protected:
|
||||
|
|
|
@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_media_types.h"
|
||||
#include "history/history_message.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
#include "data/data_media_types.h"
|
||||
#include "storage/storage_shared_media.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "ui/grouped_layout.h"
|
||||
|
@ -18,16 +19,16 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "styles/style_history.h"
|
||||
#include "layout.h"
|
||||
|
||||
HistoryGroupedMedia::Part::Part(not_null<Element*> view)
|
||||
: view(view) {
|
||||
HistoryGroupedMedia::Part::Part(not_null<HistoryItem*> item)
|
||||
: item(item) {
|
||||
}
|
||||
|
||||
HistoryGroupedMedia::HistoryGroupedMedia(
|
||||
not_null<Element*> parent,
|
||||
const std::vector<not_null<Element*>> &others)
|
||||
const std::vector<not_null<HistoryItem*>> &items)
|
||||
: HistoryMedia(parent)
|
||||
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
|
||||
const auto result = applyGroup(others);
|
||||
const auto result = applyGroup(items);
|
||||
|
||||
Ensures(result);
|
||||
}
|
||||
|
@ -124,9 +125,10 @@ QSize HistoryGroupedMedia::countCurrentSize(int newWidth) {
|
|||
return { newWidth, newHeight };
|
||||
}
|
||||
|
||||
void HistoryGroupedMedia::refreshParentId(not_null<Element*> realParent) {
|
||||
void HistoryGroupedMedia::refreshParentId(
|
||||
not_null<HistoryItem*> realParent) {
|
||||
for (const auto &part : _parts) {
|
||||
part.content->refreshParentId(part.view);
|
||||
part.content->refreshParentId(part.item);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -193,7 +195,7 @@ HistoryTextState HistoryGroupedMedia::getPartState(
|
|||
part.geometry,
|
||||
point,
|
||||
request);
|
||||
result.itemId = part.view->data()->fullId();
|
||||
result.itemId = part.item->fullId();
|
||||
return result;
|
||||
}
|
||||
}
|
||||
|
@ -284,21 +286,8 @@ void HistoryGroupedMedia::clickHandlerPressedChanged(
|
|||
for (const auto &part : _parts) {
|
||||
part.content->clickHandlerPressedChanged(p, pressed);
|
||||
if (pressed && part.content->dragItemByHandler(p)) {
|
||||
App::pressedLinkItem(part.view);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void HistoryGroupedMedia::attachToParent() {
|
||||
for (const auto &part : _parts) {
|
||||
part.content->attachToParent();
|
||||
}
|
||||
}
|
||||
|
||||
void HistoryGroupedMedia::detachFromParent() {
|
||||
for (const auto &part : _parts) {
|
||||
if (part.content) {
|
||||
part.content->detachFromParent();
|
||||
// #TODO drag by item from album
|
||||
// App::pressedLinkItem(part.view);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -308,42 +297,32 @@ std::unique_ptr<HistoryMedia> HistoryGroupedMedia::takeLastFromGroup() {
|
|||
}
|
||||
|
||||
bool HistoryGroupedMedia::applyGroup(
|
||||
const std::vector<not_null<Element*>> &others) {
|
||||
if (others.empty()) {
|
||||
const std::vector<not_null<HistoryItem*>> &items) {
|
||||
if (items.empty()) {
|
||||
return false;
|
||||
}
|
||||
const auto pushElement = [&](not_null<Element*> view) {
|
||||
const auto media = view->media();
|
||||
Assert(media != nullptr && media->canBeGrouped());
|
||||
|
||||
_parts.push_back(Part(view));
|
||||
_parts.back().content = media->clone(_parent, view);
|
||||
};
|
||||
if (_parts.empty()) {
|
||||
pushElement(_parent);
|
||||
} else if (validateGroupParts(others)) {
|
||||
if (validateGroupParts(items)) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// We're updating other elements, so we just need to preserve the main.
|
||||
auto mainPart = std::move(_parts.back());
|
||||
_parts.erase(_parts.begin(), _parts.end());
|
||||
_parts.reserve(others.size() + 1);
|
||||
for (const auto view : others) {
|
||||
pushElement(view);
|
||||
}
|
||||
_parts.push_back(std::move(mainPart));
|
||||
//_parent->setPendingInitDimensions(); // #TODO group view
|
||||
const auto pushElement = [&](not_null<HistoryItem*> item) {
|
||||
const auto media = item->media();
|
||||
Assert(media != nullptr && media->canBeGrouped());
|
||||
|
||||
_parts.push_back(Part(item));
|
||||
_parts.back().content = media->createView(_parent, item);
|
||||
};
|
||||
ranges::for_each(items, pushElement);
|
||||
return true;
|
||||
}
|
||||
|
||||
bool HistoryGroupedMedia::validateGroupParts(
|
||||
const std::vector<not_null<Element*>> &others) const {
|
||||
if (_parts.size() != others.size() + 1) {
|
||||
const std::vector<not_null<HistoryItem*>> &items) const {
|
||||
if (_parts.size() != items.size()) {
|
||||
return false;
|
||||
}
|
||||
for (auto i = 0, count = int(others.size()); i != count; ++i) {
|
||||
if (_parts[i].view != others[i]) {
|
||||
for (auto i = 0, count = int(items.size()); i != count; ++i) {
|
||||
if (_parts[i].item != items[i]) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
@ -382,15 +361,16 @@ DocumentData *HistoryGroupedMedia::getDocument() const {
|
|||
|
||||
HistoryMessageEdited *HistoryGroupedMedia::displayedEditBadge() const {
|
||||
if (!_caption.isEmpty()) {
|
||||
return _parts.front().view->data()->Get<HistoryMessageEdited>();
|
||||
return _parts.front().item->Get<HistoryMessageEdited>();
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
void HistoryGroupedMedia::updateNeedBubbleState() {
|
||||
const auto getItemCaption = [](const Part &part) {
|
||||
if (const auto media = part.view->media()) {
|
||||
return media->getCaption();
|
||||
if (const auto media = part.item->media()) {
|
||||
return TextWithEntities{ media->caption(), EntitiesInText() };
|
||||
// #TODO group caption
|
||||
}
|
||||
return part.content->getCaption();
|
||||
};
|
||||
|
|
|
@ -15,13 +15,13 @@ class HistoryGroupedMedia : public HistoryMedia {
|
|||
public:
|
||||
HistoryGroupedMedia(
|
||||
not_null<Element*> parent,
|
||||
const std::vector<not_null<Element*>> &others);
|
||||
const std::vector<not_null<HistoryItem*>> &items);
|
||||
|
||||
HistoryMediaType type() const override {
|
||||
return MediaTypeGrouped;
|
||||
}
|
||||
|
||||
void refreshParentId(not_null<Element*> realParent) override;
|
||||
void refreshParentId(not_null<HistoryItem*> realParent) override;
|
||||
|
||||
void draw(
|
||||
Painter &p,
|
||||
|
@ -58,11 +58,9 @@ public:
|
|||
const ClickHandlerPtr &p,
|
||||
bool pressed) override;
|
||||
|
||||
void attachToParent() override;
|
||||
void detachFromParent() override;
|
||||
std::unique_ptr<HistoryMedia> takeLastFromGroup() override;
|
||||
bool applyGroup(
|
||||
const std::vector<not_null<Element*>> &others) override;
|
||||
const std::vector<not_null<HistoryItem*>> &items) override;
|
||||
|
||||
bool hasReplyPreview() const override;
|
||||
ImagePtr replyPreview() override;
|
||||
|
@ -74,10 +72,6 @@ public:
|
|||
}
|
||||
HistoryMessageEdited *displayedEditBadge() const override;
|
||||
|
||||
bool canBeGrouped() const override {
|
||||
return true;
|
||||
}
|
||||
|
||||
bool skipBubbleTail() const override {
|
||||
return isBubbleBottom() && _caption.isEmpty();
|
||||
}
|
||||
|
@ -92,9 +86,9 @@ public:
|
|||
|
||||
private:
|
||||
struct Part {
|
||||
Part(not_null<Element*> view);
|
||||
Part(not_null<HistoryItem*> item);
|
||||
|
||||
not_null<Element*> view;
|
||||
not_null<HistoryItem*> item;
|
||||
std::unique_ptr<HistoryMedia> content;
|
||||
|
||||
RectParts sides = RectPart::None;
|
||||
|
@ -112,7 +106,7 @@ private:
|
|||
bool computeNeedBubble() const;
|
||||
not_null<HistoryMedia*> main() const;
|
||||
bool validateGroupParts(
|
||||
const std::vector<not_null<Element*>> &others) const;
|
||||
const std::vector<not_null<HistoryItem*>> &items) const;
|
||||
HistoryTextState getPartState(
|
||||
QPoint point,
|
||||
HistoryStateRequest request) const;
|
||||
|
|
|
@ -1,38 +0,0 @@
|
|||
/*
|
||||
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 "history/history_media_pointer.h"
|
||||
|
||||
#include "history/history_media.h"
|
||||
|
||||
HistoryMediaPtr::HistoryMediaPtr() = default;
|
||||
|
||||
HistoryMediaPtr::HistoryMediaPtr(std::unique_ptr<HistoryMedia> pointer)
|
||||
: _pointer(std::move(pointer)) {
|
||||
if (_pointer) {
|
||||
_pointer->attachToParent();
|
||||
}
|
||||
}
|
||||
|
||||
void HistoryMediaPtr::reset(std::unique_ptr<HistoryMedia> pointer) {
|
||||
*this = std::move(pointer);
|
||||
}
|
||||
|
||||
HistoryMediaPtr &HistoryMediaPtr::operator=(std::unique_ptr<HistoryMedia> pointer) {
|
||||
if (_pointer) {
|
||||
_pointer->detachFromParent();
|
||||
}
|
||||
_pointer = std::move(pointer);
|
||||
if (_pointer) {
|
||||
_pointer->attachToParent();
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
HistoryMediaPtr::~HistoryMediaPtr() {
|
||||
reset();
|
||||
}
|
|
@ -1,45 +0,0 @@
|
|||
/*
|
||||
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
|
||||
|
||||
class HistoryMedia;
|
||||
|
||||
// HistoryMedia has a special owning smart pointer
|
||||
// which regs/unregs this media to the holding HistoryItem
|
||||
class HistoryMediaPtr {
|
||||
public:
|
||||
HistoryMediaPtr();
|
||||
HistoryMediaPtr(const HistoryMediaPtr &other) = delete;
|
||||
HistoryMediaPtr &operator=(const HistoryMediaPtr &other) = delete;
|
||||
HistoryMediaPtr(std::unique_ptr<HistoryMedia> other);
|
||||
HistoryMediaPtr &operator=(std::unique_ptr<HistoryMedia> other);
|
||||
|
||||
HistoryMedia *get() const {
|
||||
return _pointer.get();
|
||||
}
|
||||
void reset(std::unique_ptr<HistoryMedia> pointer = nullptr);
|
||||
bool isNull() const {
|
||||
return !_pointer;
|
||||
}
|
||||
|
||||
HistoryMedia *operator->() const {
|
||||
return get();
|
||||
}
|
||||
HistoryMedia &operator*() const {
|
||||
Expects(!isNull());
|
||||
return *get();
|
||||
}
|
||||
explicit operator bool() const {
|
||||
return !isNull();
|
||||
}
|
||||
~HistoryMediaPtr();
|
||||
|
||||
private:
|
||||
std::unique_ptr<HistoryMedia> _pointer;
|
||||
|
||||
};
|
|
@ -62,6 +62,39 @@ int gifMaxStatusWidth(DocumentData *document) {
|
|||
return result;
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryMedia> CreateAttach(
|
||||
not_null<HistoryView::Element*> parent,
|
||||
DocumentData *document,
|
||||
PhotoData *photo) {
|
||||
if (document) {
|
||||
if (document->sticker()) {
|
||||
return std::make_unique<HistorySticker>(parent, document);
|
||||
} else if (document->isAnimation()) {
|
||||
return std::make_unique<HistoryGif>(
|
||||
parent,
|
||||
document,
|
||||
QString());
|
||||
} else if (document->isVideoFile()) {
|
||||
return std::make_unique<HistoryVideo>(
|
||||
parent,
|
||||
parent->data(),
|
||||
document,
|
||||
QString());
|
||||
}
|
||||
return std::make_unique<HistoryDocument>(
|
||||
parent,
|
||||
document,
|
||||
QString());
|
||||
} else if (photo) {
|
||||
return std::make_unique<HistoryPhoto>(
|
||||
parent,
|
||||
parent->data(),
|
||||
photo,
|
||||
QString());
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
TextWithEntities WithCaptionSelectedText(
|
||||
|
@ -115,8 +148,8 @@ void HistoryFileMedia::setLinks(
|
|||
_cancell = std::move(cancell);
|
||||
}
|
||||
|
||||
void HistoryFileMedia::refreshParentId(not_null<Element*> realParent) {
|
||||
const auto contextId = realParent->data()->fullId();
|
||||
void HistoryFileMedia::refreshParentId(not_null<HistoryItem*> realParent) {
|
||||
const auto contextId = realParent->fullId();
|
||||
_openl->setMessageId(contextId);
|
||||
_savel->setMessageId(contextId);
|
||||
_cancell->setMessageId(contextId);
|
||||
|
@ -189,13 +222,13 @@ HistoryFileMedia::~HistoryFileMedia() = default;
|
|||
|
||||
HistoryPhoto::HistoryPhoto(
|
||||
not_null<Element*> parent,
|
||||
not_null<HistoryItem*> realParent,
|
||||
not_null<PhotoData*> photo,
|
||||
const QString &caption)
|
||||
: HistoryFileMedia(parent)
|
||||
, _data(photo)
|
||||
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
|
||||
const auto item = parent->data();
|
||||
const auto fullId = item->fullId();
|
||||
const auto fullId = realParent->fullId();
|
||||
setLinks(
|
||||
std::make_shared<PhotoOpenClickHandler>(_data, fullId),
|
||||
std::make_shared<PhotoSaveClickHandler>(_data, fullId),
|
||||
|
@ -204,9 +237,9 @@ HistoryPhoto::HistoryPhoto(
|
|||
_caption.setText(
|
||||
st::messageTextStyle,
|
||||
caption + _parent->skipBlock(),
|
||||
Ui::ItemTextNoMonoOptions(item));
|
||||
Ui::ItemTextNoMonoOptions(_parent->data()));
|
||||
}
|
||||
init();
|
||||
create(realParent->fullId());
|
||||
}
|
||||
|
||||
HistoryPhoto::HistoryPhoto(
|
||||
|
@ -217,41 +250,15 @@ HistoryPhoto::HistoryPhoto(
|
|||
: HistoryFileMedia(parent)
|
||||
, _data(photo)
|
||||
, _serviceWidth(width) {
|
||||
const auto fullId = parent->data()->fullId();
|
||||
create(parent->data()->fullId(), chat);
|
||||
}
|
||||
|
||||
void HistoryPhoto::create(FullMsgId contextId, PeerData *chat) {
|
||||
Auth().data().registerPhotoView(_data, _parent);
|
||||
setLinks(
|
||||
std::make_shared<PhotoOpenClickHandler>(_data, fullId, chat),
|
||||
std::make_shared<PhotoSaveClickHandler>(_data, fullId, chat),
|
||||
std::make_shared<PhotoCancelClickHandler>(_data, fullId, chat));
|
||||
init();
|
||||
}
|
||||
|
||||
HistoryPhoto::HistoryPhoto(
|
||||
not_null<Element*> parent,
|
||||
not_null<PeerData*> chat,
|
||||
const MTPDphoto &photo,
|
||||
int width)
|
||||
: HistoryPhoto(parent, chat, App::feedPhoto(photo), width) {
|
||||
}
|
||||
|
||||
HistoryPhoto::HistoryPhoto(
|
||||
not_null<Element*> parent,
|
||||
not_null<Element*> realParent,
|
||||
const HistoryPhoto &other)
|
||||
: HistoryFileMedia(parent)
|
||||
, _data(other._data)
|
||||
, _pixw(other._pixw)
|
||||
, _pixh(other._pixh)
|
||||
, _caption(other._caption) {
|
||||
const auto fullId = realParent->data()->fullId();
|
||||
setLinks(
|
||||
std::make_shared<PhotoOpenClickHandler>(_data, fullId),
|
||||
std::make_shared<PhotoSaveClickHandler>(_data, fullId),
|
||||
std::make_shared<PhotoCancelClickHandler>(_data, fullId));
|
||||
|
||||
init();
|
||||
}
|
||||
|
||||
void HistoryPhoto::init() {
|
||||
std::make_shared<PhotoOpenClickHandler>(_data, contextId, chat),
|
||||
std::make_shared<PhotoSaveClickHandler>(_data, contextId, chat),
|
||||
std::make_shared<PhotoCancelClickHandler>(_data, contextId, chat));
|
||||
_data->thumb->load();
|
||||
}
|
||||
|
||||
|
@ -707,40 +714,44 @@ ImagePtr HistoryPhoto::replyPreview() {
|
|||
return _data->makeReplyPreview();
|
||||
}
|
||||
|
||||
HistoryVideo::HistoryVideo(
|
||||
not_null<Element*> parent,
|
||||
not_null<DocumentData*> document,
|
||||
const QString &caption)
|
||||
: HistoryFileMedia(parent)
|
||||
, _data(document)
|
||||
, _thumbw(1)
|
||||
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
|
||||
const auto item = parent->data();
|
||||
if (!caption.isEmpty()) {
|
||||
_caption.setText(
|
||||
st::messageTextStyle,
|
||||
caption + _parent->skipBlock(),
|
||||
Ui::ItemTextNoMonoOptions(item));
|
||||
}
|
||||
HistoryPhoto::~HistoryPhoto() {
|
||||
Auth().data().unregisterPhotoView(_data, _parent);
|
||||
}
|
||||
|
||||
setDocumentLinks(_data, item);
|
||||
DocumentViewRegister::DocumentViewRegister(
|
||||
not_null<HistoryView::Element*> parent,
|
||||
not_null<DocumentData*> document)
|
||||
: _savedParent(parent)
|
||||
, _savedDocument(document) {
|
||||
Auth().data().registerDocumentView(_savedDocument, _savedParent);
|
||||
}
|
||||
|
||||
setStatusSize(FileStatusSizeReady);
|
||||
|
||||
_data->thumb->load();
|
||||
DocumentViewRegister::~DocumentViewRegister() {
|
||||
Auth().data().unregisterDocumentView(_savedDocument, _savedParent);
|
||||
}
|
||||
|
||||
HistoryVideo::HistoryVideo(
|
||||
not_null<Element*> parent,
|
||||
not_null<Element*> realParent,
|
||||
const HistoryVideo &other)
|
||||
not_null<HistoryItem*> realParent,
|
||||
not_null<DocumentData*> document,
|
||||
const QString &caption)
|
||||
: HistoryFileMedia(parent)
|
||||
, _data(other._data)
|
||||
, _thumbw(other._thumbw)
|
||||
, _caption(other._caption) {
|
||||
setDocumentLinks(_data, realParent->data());
|
||||
, DocumentViewRegister(parent, document)
|
||||
, _data(document)
|
||||
, _thumbw(1)
|
||||
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
|
||||
if (!caption.isEmpty()) {
|
||||
_caption.setText(
|
||||
st::messageTextStyle,
|
||||
caption + _parent->skipBlock(),
|
||||
Ui::ItemTextNoMonoOptions(_parent->data()));
|
||||
}
|
||||
|
||||
setStatusSize(other._statusSize);
|
||||
setDocumentLinks(_data, realParent);
|
||||
|
||||
setStatusSize(FileStatusSizeReady);
|
||||
|
||||
_data->thumb->load();
|
||||
}
|
||||
|
||||
QSize HistoryVideo::countOptimalSize() {
|
||||
|
@ -1198,6 +1209,7 @@ HistoryDocument::HistoryDocument(
|
|||
not_null<DocumentData*> document,
|
||||
const QString &caption)
|
||||
: HistoryFileMedia(parent)
|
||||
, DocumentViewRegister(parent, document)
|
||||
, _data(document) {
|
||||
const auto item = parent->data();
|
||||
|
||||
|
@ -1900,19 +1912,19 @@ void HistoryDocument::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool
|
|||
HistoryFileMedia::clickHandlerPressedChanged(p, pressed);
|
||||
}
|
||||
|
||||
void HistoryDocument::refreshParentId(not_null<Element*> realParent) {
|
||||
void HistoryDocument::refreshParentId(not_null<HistoryItem*> realParent) {
|
||||
HistoryFileMedia::refreshParentId(realParent);
|
||||
|
||||
const auto contextId = realParent->data()->fullId();
|
||||
const auto fullId = realParent->fullId();
|
||||
if (auto thumbed = Get<HistoryDocumentThumbed>()) {
|
||||
if (thumbed->_linksavel) {
|
||||
thumbed->_linksavel->setMessageId(contextId);
|
||||
thumbed->_linkcancell->setMessageId(contextId);
|
||||
thumbed->_linksavel->setMessageId(fullId);
|
||||
thumbed->_linkcancell->setMessageId(fullId);
|
||||
}
|
||||
}
|
||||
if (auto voice = Get<HistoryDocumentVoice>()) {
|
||||
if (voice->_seekl) {
|
||||
voice->_seekl->setMessageId(contextId);
|
||||
voice->_seekl->setMessageId(fullId);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1944,6 +1956,7 @@ HistoryGif::HistoryGif(
|
|||
not_null<DocumentData*> document,
|
||||
const QString &caption)
|
||||
: HistoryFileMedia(parent)
|
||||
, DocumentViewRegister(parent, document)
|
||||
, _data(document)
|
||||
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
|
||||
const auto item = parent->data();
|
||||
|
@ -2667,7 +2680,7 @@ bool HistoryGif::playInline(bool autoplay) {
|
|||
stopInline();
|
||||
} else if (_data->loaded(DocumentData::FilePathResolveChecked)) {
|
||||
if (!cAutoPlayGif()) {
|
||||
App::stopGifItems();
|
||||
Auth().data().stopAutoplayAnimations();
|
||||
}
|
||||
const auto mode = (!autoplay && _data->isVideoMessage())
|
||||
? Mode::Video
|
||||
|
@ -2682,7 +2695,7 @@ bool HistoryGif::playInline(bool autoplay) {
|
|||
Auth().data().requestViewRepaint(_parent);
|
||||
});
|
||||
if (App::main()) {
|
||||
App::main()->mediaMarkRead(_data);
|
||||
Auth().data().markMediaRead(_data);
|
||||
}
|
||||
App::wnd()->controller()->enableGifPauseReason(Window::GifPauseReason::RoundPlaying);
|
||||
}
|
||||
|
@ -2709,12 +2722,12 @@ void HistoryGif::stopInline() {
|
|||
|
||||
void HistoryGif::setClipReader(Media::Clip::ReaderPointer gif) {
|
||||
if (_gif) {
|
||||
App::unregGifItem(_gif.get());
|
||||
Auth().data().unregisterAutoplayAnimation(_gif.get());
|
||||
}
|
||||
_gif = std::move(gif);
|
||||
//if (_gif) { // #TODO GIFs
|
||||
// App::regGifItem(_gif.get(), _parent);
|
||||
//}
|
||||
if (_gif) {
|
||||
Auth().data().registerAutoplayAnimation(_gif.get(), _parent);
|
||||
}
|
||||
}
|
||||
|
||||
HistoryGif::~HistoryGif() {
|
||||
|
@ -2745,6 +2758,7 @@ HistorySticker::HistorySticker(
|
|||
not_null<Element*> parent,
|
||||
not_null<DocumentData*> document)
|
||||
: HistoryMedia(parent)
|
||||
, DocumentViewRegister(parent, document)
|
||||
, _data(document)
|
||||
, _emoji(_data->sticker()->alt) {
|
||||
_data->thumb->load();
|
||||
|
@ -3033,6 +3047,8 @@ HistoryContact::HistoryContact(
|
|||
, _fname(first)
|
||||
, _lname(last)
|
||||
, _phone(App::formatPhone(phone)) {
|
||||
Auth().data().registerContactView(userId, parent);
|
||||
|
||||
_name.setText(
|
||||
st::semiboldTextStyle,
|
||||
lng_full_name(lt_first_name, first, lt_last_name, last).trimmed(),
|
||||
|
@ -3040,6 +3056,18 @@ HistoryContact::HistoryContact(
|
|||
_phonew = st::normalFont->width(_phone);
|
||||
}
|
||||
|
||||
HistoryContact::~HistoryContact() {
|
||||
Auth().data().unregisterContactView(_userId, _parent);
|
||||
}
|
||||
|
||||
void HistoryContact::updateSharedContactUserId(UserId userId) {
|
||||
if (_userId != userId) {
|
||||
Auth().data().unregisterContactView(_userId, _parent);
|
||||
_userId = userId;
|
||||
Auth().data().registerContactView(_userId, _parent);
|
||||
}
|
||||
}
|
||||
|
||||
QSize HistoryContact::countOptimalSize() {
|
||||
const auto item = _parent->data();
|
||||
auto maxWidth = st::msgFileMinWidth;
|
||||
|
@ -3173,20 +3201,6 @@ TextWithEntities HistoryContact::selectedText(TextSelection selection) const {
|
|||
return { qsl("[ ") + lang(lng_in_dlg_contact) + qsl(" ]\n") + _name.originalText() + '\n' + _phone, EntitiesInText() };
|
||||
}
|
||||
|
||||
void HistoryContact::attachToParent() {
|
||||
//if (_userId) {
|
||||
// App::regSharedContactItem(_userId, _parent);
|
||||
//} // #TODO contacts
|
||||
}
|
||||
|
||||
void HistoryContact::detachFromParent() {
|
||||
//if (_userId) {
|
||||
// App::unregSharedContactItem(_userId, _parent);
|
||||
//} // #TODO contacts
|
||||
}
|
||||
|
||||
HistoryContact::~HistoryContact() = default;
|
||||
|
||||
HistoryCall::HistoryCall(
|
||||
not_null<Element*> parent,
|
||||
not_null<Data::Call*> call)
|
||||
|
@ -3306,6 +3320,7 @@ HistoryWebPage::HistoryWebPage(
|
|||
, _data(data)
|
||||
, _title(st::msgMinWidth - st::webPageLeft)
|
||||
, _description(st::msgMinWidth - st::webPageLeft) {
|
||||
Auth().data().registerWebPageView(_data, _parent);
|
||||
}
|
||||
|
||||
QSize HistoryWebPage::countOptimalSize() {
|
||||
|
@ -3316,10 +3331,7 @@ QSize HistoryWebPage::countOptimalSize() {
|
|||
if (versionChanged) {
|
||||
_dataVersion = _data->version;
|
||||
_openl = nullptr;
|
||||
if (_attach) {
|
||||
_attach->detachFromParent();
|
||||
_attach = nullptr;
|
||||
}
|
||||
_attach = nullptr;
|
||||
_title = Text(st::msgMinWidth - st::webPageLeft);
|
||||
_description = Text(st::msgMinWidth - st::webPageLeft);
|
||||
_siteNameWidth = 0;
|
||||
|
@ -3349,22 +3361,7 @@ QSize HistoryWebPage::countOptimalSize() {
|
|||
|
||||
// init attach
|
||||
if (!_attach && !_asArticle) {
|
||||
if (_data->document) {
|
||||
if (_data->document->sticker()) {
|
||||
_attach = std::make_unique<HistorySticker>(_parent, _data->document);
|
||||
} else if (_data->document->isAnimation()) {
|
||||
_attach = std::make_unique<HistoryGif>(_parent, _data->document, QString());
|
||||
} else if (_data->document->isVideoFile()) {
|
||||
_attach = std::make_unique<HistoryVideo>(_parent, _data->document, QString());
|
||||
} else {
|
||||
_attach = std::make_unique<HistoryDocument>(_parent, _data->document, QString());
|
||||
}
|
||||
} else if (_data->photo) {
|
||||
_attach = std::make_unique<HistoryPhoto>(_parent, _data->photo, QString());
|
||||
}
|
||||
if (_attach) {
|
||||
_attach->attachToParent();
|
||||
}
|
||||
_attach = CreateAttach(_parent, _data->document, _data->photo);
|
||||
}
|
||||
|
||||
auto textFloatsAroundInfo = !_asArticle && !_attach && isBubbleBottom();
|
||||
|
@ -3570,7 +3567,7 @@ TextSelection HistoryWebPage::fromDescriptionSelection(
|
|||
return HistoryView::ShiftItemSelection(selection, _title);
|
||||
}
|
||||
|
||||
void HistoryWebPage::refreshParentId(not_null<Element*> realParent) {
|
||||
void HistoryWebPage::refreshParentId(not_null<HistoryItem*> realParent) {
|
||||
if (_attach) {
|
||||
_attach->refreshParentId(realParent);
|
||||
}
|
||||
|
@ -3829,16 +3826,6 @@ bool HistoryWebPage::isDisplayed() const {
|
|||
&& !item->Has<HistoryMessageLogEntryOriginal>();
|
||||
}
|
||||
|
||||
void HistoryWebPage::attachToParent() {
|
||||
//App::regWebPageItem(_data, _parent); // #TODO webpages
|
||||
if (_attach) _attach->attachToParent();
|
||||
}
|
||||
|
||||
void HistoryWebPage::detachFromParent() {
|
||||
//App::unregWebPageItem(_data, _parent); // #TODO webpages
|
||||
if (_attach) _attach->detachFromParent();
|
||||
}
|
||||
|
||||
TextWithEntities HistoryWebPage::selectedText(TextSelection selection) const {
|
||||
if (selection == FullSelection && !isLogEntryOriginal()) {
|
||||
return TextWithEntities();
|
||||
|
@ -3890,6 +3877,10 @@ int HistoryWebPage::bottomInfoPadding() const {
|
|||
return result;
|
||||
}
|
||||
|
||||
HistoryWebPage::~HistoryWebPage() {
|
||||
Auth().data().unregisterWebPageView(_data, _parent);
|
||||
}
|
||||
|
||||
HistoryGame::HistoryGame(
|
||||
not_null<Element*> parent,
|
||||
not_null<GameData*> data,
|
||||
|
@ -3904,6 +3895,7 @@ HistoryGame::HistoryGame(
|
|||
consumed,
|
||||
Ui::ItemTextOptions(parent->data()));
|
||||
}
|
||||
Auth().data().registerGameView(_data, _parent);
|
||||
}
|
||||
|
||||
QSize HistoryGame::countOptimalSize() {
|
||||
|
@ -3923,19 +3915,7 @@ QSize HistoryGame::countOptimalSize() {
|
|||
|
||||
// init attach
|
||||
if (!_attach) {
|
||||
if (_data->document) {
|
||||
if (_data->document->sticker()) {
|
||||
_attach = std::make_unique<HistorySticker>(_parent, _data->document);
|
||||
} else if (_data->document->isAnimation()) {
|
||||
_attach = std::make_unique<HistoryGif>(_parent, _data->document, QString());
|
||||
} else if (_data->document->isVideoFile()) {
|
||||
_attach = std::make_unique<HistoryVideo>(_parent, _data->document, QString());
|
||||
} else {
|
||||
_attach = std::make_unique<HistoryDocument>(_parent, _data->document, QString());
|
||||
}
|
||||
} else if (_data->photo) {
|
||||
_attach = std::make_unique<HistoryPhoto>(_parent, _data->photo, QString());
|
||||
}
|
||||
_attach = CreateAttach(_parent, _data->document, _data->photo);
|
||||
}
|
||||
|
||||
// init strings
|
||||
|
@ -4003,9 +3983,9 @@ QSize HistoryGame::countOptimalSize() {
|
|||
return { maxWidth, minHeight };
|
||||
}
|
||||
|
||||
void HistoryGame::refreshParentId(not_null<Element*> realParent) {
|
||||
void HistoryGame::refreshParentId(not_null<HistoryItem*> realParent) {
|
||||
if (_openl) {
|
||||
_openl->setMessageId(_parent->data()->fullId());
|
||||
_openl->setMessageId(realParent->fullId());
|
||||
}
|
||||
if (_attach) {
|
||||
_attach->refreshParentId(realParent);
|
||||
|
@ -4242,16 +4222,6 @@ void HistoryGame::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pres
|
|||
}
|
||||
}
|
||||
|
||||
void HistoryGame::attachToParent() {
|
||||
// App::regGameItem(_data, _parent); // #TODO regitems
|
||||
if (_attach) _attach->attachToParent();
|
||||
}
|
||||
|
||||
void HistoryGame::detachFromParent() {
|
||||
// App::unregGameItem(_data, _parent); // #TODO regitems
|
||||
if (_attach) _attach->detachFromParent();
|
||||
}
|
||||
|
||||
TextWithEntities HistoryGame::selectedText(TextSelection selection) const {
|
||||
if (selection == FullSelection) {
|
||||
return TextWithEntities();
|
||||
|
@ -4295,6 +4265,10 @@ int HistoryGame::bottomInfoPadding() const {
|
|||
return result;
|
||||
}
|
||||
|
||||
HistoryGame::~HistoryGame() {
|
||||
Auth().data().unregisterGameView(_data, _parent);
|
||||
}
|
||||
|
||||
HistoryInvoice::HistoryInvoice(
|
||||
not_null<Element*> parent,
|
||||
not_null<Data::Invoice*> invoice)
|
||||
|
@ -4473,7 +4447,7 @@ TextSelection HistoryInvoice::fromDescriptionSelection(
|
|||
return HistoryView::ShiftItemSelection(selection, _title);
|
||||
}
|
||||
|
||||
void HistoryInvoice::refreshParentId(not_null<Element*> realParent) {
|
||||
void HistoryInvoice::refreshParentId(not_null<HistoryItem*> realParent) {
|
||||
if (_attach) {
|
||||
_attach->refreshParentId(realParent);
|
||||
}
|
||||
|
@ -4638,14 +4612,6 @@ void HistoryInvoice::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool p
|
|||
}
|
||||
}
|
||||
|
||||
void HistoryInvoice::attachToParent() {
|
||||
if (_attach) _attach->attachToParent();
|
||||
}
|
||||
|
||||
void HistoryInvoice::detachFromParent() {
|
||||
if (_attach) _attach->detachFromParent();
|
||||
}
|
||||
|
||||
TextWithEntities HistoryInvoice::selectedText(TextSelection selection) const {
|
||||
if (selection == FullSelection) {
|
||||
return TextWithEntities();
|
||||
|
|
|
@ -56,7 +56,7 @@ public:
|
|||
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override;
|
||||
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override;
|
||||
|
||||
void refreshParentId(not_null<Element*> realParent) override;
|
||||
void refreshParentId(not_null<HistoryItem*> realParent) override;
|
||||
|
||||
bool allowsFastShare() const override {
|
||||
return true;
|
||||
|
@ -129,6 +129,7 @@ class HistoryPhoto : public HistoryFileMedia {
|
|||
public:
|
||||
HistoryPhoto(
|
||||
not_null<Element*> parent,
|
||||
not_null<HistoryItem*> realParent,
|
||||
not_null<PhotoData*> photo,
|
||||
const QString &caption);
|
||||
HistoryPhoto(
|
||||
|
@ -136,25 +137,10 @@ public:
|
|||
not_null<PeerData*> chat,
|
||||
not_null<PhotoData*> photo,
|
||||
int width);
|
||||
HistoryPhoto(
|
||||
not_null<Element*> parent,
|
||||
not_null<PeerData*> chat,
|
||||
const MTPDphoto &photo,
|
||||
int width);
|
||||
HistoryPhoto(
|
||||
not_null<Element*> parent,
|
||||
not_null<Element*> realParent,
|
||||
const HistoryPhoto &other);
|
||||
|
||||
void init();
|
||||
HistoryMediaType type() const override {
|
||||
return MediaTypePhoto;
|
||||
}
|
||||
std::unique_ptr<HistoryMedia> clone(
|
||||
not_null<Element*> newParent,
|
||||
not_null<Element*> realParent) const override {
|
||||
return std::make_unique<HistoryPhoto>(newParent, realParent, *this);
|
||||
}
|
||||
|
||||
void draw(Painter &p, const QRect &clip, TextSelection selection, TimeMs ms) const override;
|
||||
HistoryTextState getState(QPoint point, HistoryStateRequest request) const override;
|
||||
|
@ -177,9 +163,6 @@ public:
|
|||
return _data;
|
||||
}
|
||||
|
||||
bool canBeGrouped() const override {
|
||||
return true;
|
||||
}
|
||||
QSize sizeForGrouping() const override;
|
||||
void drawGrouped(
|
||||
Painter &p,
|
||||
|
@ -214,12 +197,16 @@ public:
|
|||
return _data->loaded();
|
||||
}
|
||||
|
||||
~HistoryPhoto();
|
||||
|
||||
protected:
|
||||
float64 dataProgress() const override;
|
||||
bool dataFinished() const override;
|
||||
bool dataLoaded() const override;
|
||||
|
||||
private:
|
||||
void create(FullMsgId contextId, PeerData *chat = nullptr);
|
||||
|
||||
QSize countOptimalSize() override;
|
||||
QSize countCurrentSize(int newWidth) override;
|
||||
|
||||
|
@ -238,25 +225,30 @@ private:
|
|||
|
||||
};
|
||||
|
||||
class HistoryVideo : public HistoryFileMedia {
|
||||
class DocumentViewRegister {
|
||||
public:
|
||||
DocumentViewRegister(
|
||||
not_null<HistoryView::Element*> parent,
|
||||
not_null<DocumentData*> document);
|
||||
~DocumentViewRegister();
|
||||
|
||||
private:
|
||||
not_null<HistoryView::Element*> _savedParent;
|
||||
not_null<DocumentData*> _savedDocument;
|
||||
|
||||
};
|
||||
|
||||
class HistoryVideo : public HistoryFileMedia, public DocumentViewRegister {
|
||||
public:
|
||||
HistoryVideo(
|
||||
not_null<Element*> parent,
|
||||
not_null<HistoryItem*> realParent,
|
||||
not_null<DocumentData*> document,
|
||||
const QString &caption);
|
||||
HistoryVideo(
|
||||
not_null<Element*> parent,
|
||||
not_null<Element*> realParent,
|
||||
const HistoryVideo &other);
|
||||
|
||||
HistoryMediaType type() const override {
|
||||
return MediaTypeVideo;
|
||||
}
|
||||
std::unique_ptr<HistoryMedia> clone(
|
||||
not_null<Element*> newParent,
|
||||
not_null<Element*> realParent) const override {
|
||||
return std::make_unique<HistoryVideo>(newParent, realParent, *this);
|
||||
}
|
||||
|
||||
void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override;
|
||||
HistoryTextState getState(QPoint point, HistoryStateRequest request) const override;
|
||||
|
@ -279,9 +271,6 @@ public:
|
|||
return _data;
|
||||
}
|
||||
|
||||
bool canBeGrouped() const override {
|
||||
return true;
|
||||
}
|
||||
QSize sizeForGrouping() const override;
|
||||
void drawGrouped(
|
||||
Painter &p,
|
||||
|
@ -342,6 +331,7 @@ private:
|
|||
|
||||
class HistoryDocument
|
||||
: public HistoryFileMedia
|
||||
, public DocumentViewRegister
|
||||
, public RuntimeComposer<HistoryDocument> {
|
||||
public:
|
||||
HistoryDocument(
|
||||
|
@ -400,7 +390,7 @@ public:
|
|||
|
||||
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override;
|
||||
|
||||
void refreshParentId(not_null<Element*> realParent) override;
|
||||
void refreshParentId(not_null<HistoryItem*> realParent) override;
|
||||
|
||||
protected:
|
||||
float64 dataProgress() const override;
|
||||
|
@ -426,7 +416,7 @@ private:
|
|||
|
||||
};
|
||||
|
||||
class HistoryGif : public HistoryFileMedia {
|
||||
class HistoryGif : public HistoryFileMedia, public DocumentViewRegister {
|
||||
public:
|
||||
HistoryGif(
|
||||
not_null<Element*> parent,
|
||||
|
@ -529,7 +519,7 @@ private:
|
|||
|
||||
};
|
||||
|
||||
class HistorySticker : public HistoryMedia {
|
||||
class HistorySticker : public HistoryMedia, public DocumentViewRegister {
|
||||
public:
|
||||
HistorySticker(
|
||||
not_null<Element*> parent,
|
||||
|
@ -597,6 +587,7 @@ public:
|
|||
const QString &first,
|
||||
const QString &last,
|
||||
const QString &phone);
|
||||
~HistoryContact();
|
||||
|
||||
HistoryMediaType type() const override {
|
||||
return MediaTypeContact;
|
||||
|
@ -614,9 +605,6 @@ public:
|
|||
|
||||
TextWithEntities selectedText(TextSelection selection) const override;
|
||||
|
||||
void attachToParent() override;
|
||||
void detachFromParent() override;
|
||||
|
||||
bool needsBubble() const override {
|
||||
return true;
|
||||
}
|
||||
|
@ -634,7 +622,8 @@ public:
|
|||
return _phone;
|
||||
}
|
||||
|
||||
~HistoryContact();
|
||||
// Should be called only by Data::Session.
|
||||
void updateSharedContactUserId(UserId userId) override;
|
||||
|
||||
private:
|
||||
QSize countOptimalSize() override;
|
||||
|
@ -709,7 +698,7 @@ public:
|
|||
return MediaTypeWebPage;
|
||||
}
|
||||
|
||||
void refreshParentId(not_null<Element*> realParent) override;
|
||||
void refreshParentId(not_null<HistoryItem*> realParent) override;
|
||||
|
||||
void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override;
|
||||
HistoryTextState getState(QPoint point, HistoryStateRequest request) const override;
|
||||
|
@ -753,9 +742,6 @@ public:
|
|||
if (_attach) _attach->stopInline();
|
||||
}
|
||||
|
||||
void attachToParent() override;
|
||||
void detachFromParent() override;
|
||||
|
||||
bool hasReplyPreview() const override;
|
||||
ImagePtr replyPreview() override;
|
||||
|
||||
|
@ -777,6 +763,8 @@ public:
|
|||
return _attach.get();
|
||||
}
|
||||
|
||||
~HistoryWebPage();
|
||||
|
||||
private:
|
||||
QSize countOptimalSize() override;
|
||||
QSize countCurrentSize(int newWidth) override;
|
||||
|
@ -818,7 +806,7 @@ public:
|
|||
return MediaTypeGame;
|
||||
}
|
||||
|
||||
void refreshParentId(not_null<Element*> realParent) override;
|
||||
void refreshParentId(not_null<HistoryItem*> realParent) override;
|
||||
|
||||
void draw(Painter &p, const QRect &r, TextSelection selection, TimeMs ms) const override;
|
||||
HistoryTextState getState(QPoint point, HistoryStateRequest request) const override;
|
||||
|
@ -864,9 +852,6 @@ public:
|
|||
if (_attach) _attach->stopInline();
|
||||
}
|
||||
|
||||
void attachToParent() override;
|
||||
void detachFromParent() override;
|
||||
|
||||
bool hasReplyPreview() const override {
|
||||
return (_data->photo && !_data->photo->thumb->isNull()) || (_data->document && !_data->document->thumb->isNull());
|
||||
}
|
||||
|
@ -890,6 +875,8 @@ public:
|
|||
return _attach.get();
|
||||
}
|
||||
|
||||
~HistoryGame();
|
||||
|
||||
private:
|
||||
QSize countOptimalSize() override;
|
||||
QSize countCurrentSize(int newWidth) override;
|
||||
|
@ -921,7 +908,7 @@ public:
|
|||
return MediaTypeInvoice;
|
||||
}
|
||||
|
||||
void refreshParentId(not_null<Element*> realParent) override;
|
||||
void refreshParentId(not_null<HistoryItem*> realParent) override;
|
||||
|
||||
MsgId getReceiptMsgId() const {
|
||||
return _receiptMsgId;
|
||||
|
@ -956,9 +943,6 @@ public:
|
|||
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override;
|
||||
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override;
|
||||
|
||||
void attachToParent() override;
|
||||
void detachFromParent() override;
|
||||
|
||||
bool hasReplyPreview() const override {
|
||||
return _attach && _attach->hasReplyPreview();
|
||||
}
|
||||
|
|
|
@ -314,7 +314,9 @@ HistoryMessage::HistoryMessage(
|
|||
|
||||
createComponents(config);
|
||||
|
||||
initMedia(msg.has_media() ? (&msg.vmedia) : nullptr);
|
||||
if (msg.has_media()) {
|
||||
setMedia(msg.vmedia);
|
||||
}
|
||||
|
||||
auto text = TextUtilities::Clean(qs(msg.vmessage));
|
||||
auto entities = msg.has_entities()
|
||||
|
@ -702,84 +704,136 @@ QString FormatViewsCount(int views) {
|
|||
return qsl("1");
|
||||
}
|
||||
|
||||
void HistoryMessage::initMedia(const MTPMessageMedia *media) {
|
||||
void HistoryMessage::refreshMedia(const MTPMessageMedia *media) {
|
||||
const auto wasGrouped = Auth().data().groups().isGrouped(this);
|
||||
_media = nullptr;
|
||||
if (media) {
|
||||
setMedia(*media);
|
||||
}
|
||||
if (wasGrouped) {
|
||||
Auth().data().groups().refreshMessage(this);
|
||||
}
|
||||
}
|
||||
|
||||
switch (media ? media->type() : mtpc_messageMediaEmpty) {
|
||||
void HistoryMessage::setMedia(const MTPMessageMedia &media) {
|
||||
_media = CreateMedia(this, media);
|
||||
if (const auto invoice = _media->invoice()) {
|
||||
if (invoice->receiptMsgId) {
|
||||
replaceBuyWithReceiptInMarkup();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
std::unique_ptr<Data::Media> HistoryMessage::CreateMedia(
|
||||
not_null<HistoryMessage*> item,
|
||||
const MTPMessageMedia &media) {
|
||||
switch (media.type()) {
|
||||
case mtpc_messageMediaContact: {
|
||||
auto &d = media->c_messageMediaContact();
|
||||
_media = std::make_unique<Data::MediaContact>(this, d.vuser_id.v, qs(d.vfirst_name), qs(d.vlast_name), qs(d.vphone_number));
|
||||
const auto &data = media.c_messageMediaContact();
|
||||
return std::make_unique<Data::MediaContact>(
|
||||
item,
|
||||
data.vuser_id.v,
|
||||
qs(data.vfirst_name),
|
||||
qs(data.vlast_name),
|
||||
qs(data.vphone_number));
|
||||
} break;
|
||||
case mtpc_messageMediaGeo: {
|
||||
auto &point = media->c_messageMediaGeo().vgeo;
|
||||
if (point.type() == mtpc_geoPoint) {
|
||||
_media = std::make_unique<Data::MediaLocation>(this, LocationCoords(point.c_geoPoint()));
|
||||
const auto &data = media.c_messageMediaGeo().vgeo;
|
||||
if (data.type() == mtpc_geoPoint) {
|
||||
return std::make_unique<Data::MediaLocation>(
|
||||
item,
|
||||
LocationCoords(data.c_geoPoint()));
|
||||
}
|
||||
} break;
|
||||
case mtpc_messageMediaGeoLive: {
|
||||
auto &point = media->c_messageMediaGeoLive().vgeo;
|
||||
if (point.type() == mtpc_geoPoint) {
|
||||
_media = std::make_unique<Data::MediaLocation>(this, LocationCoords(point.c_geoPoint()));
|
||||
const auto &data = media.c_messageMediaGeoLive().vgeo;
|
||||
if (data.type() == mtpc_geoPoint) {
|
||||
return std::make_unique<Data::MediaLocation>(
|
||||
item,
|
||||
LocationCoords(data.c_geoPoint()));
|
||||
}
|
||||
} break;
|
||||
case mtpc_messageMediaVenue: {
|
||||
auto &d = media->c_messageMediaVenue();
|
||||
if (d.vgeo.type() == mtpc_geoPoint) {
|
||||
_media = std::make_unique<Data::MediaLocation>(this, LocationCoords(d.vgeo.c_geoPoint()), qs(d.vtitle), qs(d.vaddress));
|
||||
const auto &data = media.c_messageMediaVenue();
|
||||
if (data.vgeo.type() == mtpc_geoPoint) {
|
||||
return std::make_unique<Data::MediaLocation>(
|
||||
item,
|
||||
LocationCoords(data.vgeo.c_geoPoint()),
|
||||
qs(data.vtitle),
|
||||
qs(data.vaddress));
|
||||
}
|
||||
} break;
|
||||
case mtpc_messageMediaPhoto: {
|
||||
auto &photo = media->c_messageMediaPhoto();
|
||||
if (photo.has_ttl_seconds()) {
|
||||
LOG(("App Error: Unexpected MTPMessageMediaPhoto with ttl_seconds in HistoryMessage."));
|
||||
} else if (photo.has_photo() && photo.vphoto.type() == mtpc_photo) {
|
||||
_media = std::make_unique<Data::MediaPhoto>(
|
||||
this,
|
||||
App::feedPhoto(photo.vphoto.c_photo()),
|
||||
photo.has_caption() ? qs(photo.vcaption) : QString());
|
||||
const auto &data = media.c_messageMediaPhoto();
|
||||
if (data.has_ttl_seconds()) {
|
||||
LOG(("App Error: "
|
||||
"Unexpected MTPMessageMediaPhoto "
|
||||
"with ttl_seconds in HistoryMessage."));
|
||||
} else if (data.has_photo() && data.vphoto.type() == mtpc_photo) {
|
||||
return std::make_unique<Data::MediaPhoto>(
|
||||
item,
|
||||
Auth().data().photo(data.vphoto.c_photo()),
|
||||
data.has_caption() ? qs(data.vcaption) : QString());
|
||||
} else {
|
||||
LOG(("API Error: Got MTPMessageMediaPhoto without photo and without ttl_seconds."));
|
||||
LOG(("API Error: "
|
||||
"Got MTPMessageMediaPhoto "
|
||||
"without photo and without ttl_seconds."));
|
||||
}
|
||||
} break;
|
||||
case mtpc_messageMediaDocument: {
|
||||
auto &document = media->c_messageMediaDocument();
|
||||
if (document.has_ttl_seconds()) {
|
||||
LOG(("App Error: Unexpected MTPMessageMediaDocument with ttl_seconds in HistoryMessage."));
|
||||
} else if (document.has_document() && document.vdocument.type() == mtpc_document) {
|
||||
_media = std::make_unique<Data::MediaFile>(
|
||||
this,
|
||||
App::feedDocument(document.vdocument.c_document()),
|
||||
document.has_caption() ? qs(document.vcaption) : QString());
|
||||
const auto &data = media.c_messageMediaDocument();
|
||||
if (data.has_ttl_seconds()) {
|
||||
LOG(("App Error: "
|
||||
"Unexpected MTPMessageMediaDocument "
|
||||
"with ttl_seconds in HistoryMessage."));
|
||||
} else if (data.has_document()
|
||||
&& data.vdocument.type() == mtpc_document) {
|
||||
return std::make_unique<Data::MediaFile>(
|
||||
item,
|
||||
Auth().data().document(data.vdocument.c_document()),
|
||||
data.has_caption() ? qs(data.vcaption) : QString());
|
||||
} else {
|
||||
LOG(("API Error: Got MTPMessageMediaDocument without document and without ttl_seconds."));
|
||||
LOG(("API Error: "
|
||||
"Got MTPMessageMediaDocument "
|
||||
"without document and without ttl_seconds."));
|
||||
}
|
||||
} break;
|
||||
case mtpc_messageMediaWebPage: {
|
||||
auto &d = media->c_messageMediaWebPage().vwebpage;
|
||||
switch (d.type()) {
|
||||
const auto &data = media.c_messageMediaWebPage().vwebpage;
|
||||
switch (data.type()) {
|
||||
case mtpc_webPageEmpty: break;
|
||||
case mtpc_webPagePending: {
|
||||
_media = std::make_unique<Data::MediaWebPage>(this, App::feedWebPage(d.c_webPagePending()));
|
||||
} break;
|
||||
case mtpc_webPage: {
|
||||
_media = std::make_unique<Data::MediaWebPage>(this, App::feedWebPage(d.c_webPage()));
|
||||
} break;
|
||||
case mtpc_webPageNotModified: LOG(("API Error: webPageNotModified is unexpected in message media.")); break;
|
||||
case mtpc_webPagePending:
|
||||
return std::make_unique<Data::MediaWebPage>(
|
||||
item,
|
||||
Auth().data().webpage(data.c_webPagePending()));
|
||||
break;
|
||||
case mtpc_webPage:
|
||||
return std::make_unique<Data::MediaWebPage>(
|
||||
item,
|
||||
Auth().data().webpage(data.c_webPage()));
|
||||
break;
|
||||
case mtpc_webPageNotModified:
|
||||
LOG(("API Error: "
|
||||
"webPageNotModified is unexpected in message media."));
|
||||
break;
|
||||
}
|
||||
} break;
|
||||
case mtpc_messageMediaGame: {
|
||||
auto &d = media->c_messageMediaGame().vgame;
|
||||
if (d.type() == mtpc_game) {
|
||||
_media = std::make_unique<Data::MediaGame>(this, App::feedGame(d.c_game()));
|
||||
const auto &data = media.c_messageMediaGame().vgame;
|
||||
if (data.type() == mtpc_game) {
|
||||
return std::make_unique<Data::MediaGame>(
|
||||
item,
|
||||
Auth().data().game(data.c_game()));
|
||||
}
|
||||
} break;
|
||||
case mtpc_messageMediaInvoice: {
|
||||
_media = std::make_unique<Data::MediaInvoice>(this, media->c_messageMediaInvoice());
|
||||
if (_media->invoice()->receiptMsgId) {
|
||||
replaceBuyWithReceiptInMarkup();
|
||||
}
|
||||
return std::make_unique<Data::MediaInvoice>(
|
||||
item,
|
||||
media.c_messageMediaInvoice());
|
||||
} break;
|
||||
};
|
||||
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
void HistoryMessage::replaceBuyWithReceiptInMarkup() {
|
||||
|
@ -822,7 +876,7 @@ void HistoryMessage::applyEdition(const MTPDmessage &message) {
|
|||
}
|
||||
setText(textWithEntities);
|
||||
setReplyMarkup(message.has_reply_markup() ? (&message.vreply_markup) : nullptr);
|
||||
initMedia(message.has_media() ? (&message.vmedia) : nullptr);
|
||||
refreshMedia(message.has_media() ? (&message.vmedia) : nullptr);
|
||||
setViewsCount(message.has_views() ? message.vviews.v : -1);
|
||||
|
||||
finishEdition(keyboardTop);
|
||||
|
@ -836,22 +890,22 @@ void HistoryMessage::applyEdition(const MTPDmessageService &message) {
|
|||
|
||||
void HistoryMessage::applyEditionToEmpty() {
|
||||
setEmptyText();
|
||||
initMedia(nullptr);
|
||||
refreshMedia(nullptr);
|
||||
setReplyMarkup(nullptr);
|
||||
setViewsCount(-1);
|
||||
|
||||
finishEditionToEmpty();
|
||||
}
|
||||
|
||||
void HistoryMessage::updateMedia(const MTPMessageMedia *media) {
|
||||
void HistoryMessage::updateSentMedia(const MTPMessageMedia *media) {
|
||||
if (_flags & MTPDmessage_ClientFlag::f_from_inline_bot) {
|
||||
if (!media || !_media || !_media->updateInlineResultMedia(*media)) {
|
||||
initMedia(media);
|
||||
refreshMedia(media);
|
||||
}
|
||||
_flags &= ~MTPDmessage_ClientFlag::f_from_inline_bot;
|
||||
} else {
|
||||
if (!media || !_media || !_media->updateSentMedia(*media)) {
|
||||
initMedia(media);
|
||||
refreshMedia(media);
|
||||
}
|
||||
}
|
||||
Auth().data().requestItemViewResize(this);
|
||||
|
@ -897,7 +951,7 @@ Storage::SharedMediaTypesMask HistoryMessage::sharedMediaTypes() const {
|
|||
// mediaRemovedSkipBlock = _media->isDisplayed() && _media->isBubbleBottom();
|
||||
// _media.reset();
|
||||
//}
|
||||
//initMedia(media);
|
||||
//refreshMedia(media);
|
||||
//auto mediaDisplayed = _media && _media->isDisplayed();
|
||||
//if (mediaDisplayed && _media->isBubbleBottom() && !mediaRemovedSkipBlock) {
|
||||
// _text.removeSkipBlock();
|
||||
|
@ -996,15 +1050,23 @@ bool HistoryMessage::textHasLinks() const {
|
|||
}
|
||||
|
||||
void HistoryMessage::setViewsCount(int32 count) {
|
||||
auto views = Get<HistoryMessageViews>();
|
||||
if (!views || views->_views == count || (count >= 0 && views->_views > count)) return;
|
||||
const auto views = Get<HistoryMessageViews>();
|
||||
if (!views
|
||||
|| views->_views == count
|
||||
|| (count >= 0 && views->_views > count)) {
|
||||
return;
|
||||
}
|
||||
|
||||
int32 was = views->_viewsWidth;
|
||||
const auto was = views->_viewsWidth;
|
||||
views->_views = count;
|
||||
views->_viewsText = (views->_views >= 0) ? FormatViewsCount(views->_views) : QString();
|
||||
views->_viewsWidth = views->_viewsText.isEmpty() ? 0 : st::msgDateFont->width(views->_viewsText);
|
||||
views->_viewsText = (views->_views >= 0)
|
||||
? FormatViewsCount(views->_views)
|
||||
: QString();
|
||||
views->_viewsWidth = views->_viewsText.isEmpty()
|
||||
? 0
|
||||
: st::msgDateFont->width(views->_viewsText);
|
||||
if (was == views->_viewsWidth) {
|
||||
Auth().data().requestItemRepaint(this);
|
||||
Auth().data().requestItemViewRepaint(this);
|
||||
} else {
|
||||
Auth().data().requestItemViewResize(this);
|
||||
}
|
||||
|
@ -1012,6 +1074,7 @@ void HistoryMessage::setViewsCount(int32 count) {
|
|||
|
||||
void HistoryMessage::setRealId(MsgId newId) {
|
||||
HistoryItem::setRealId(newId);
|
||||
Auth().data().groups().refreshMessage(this);
|
||||
Auth().data().requestItemViewResize(this);
|
||||
}
|
||||
|
||||
|
@ -1026,9 +1089,8 @@ QString HistoryMessage::notificationHeader() const {
|
|||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> HistoryMessage::createView(
|
||||
not_null<Window::Controller*> controller,
|
||||
HistoryView::Context context) {
|
||||
return controller->createMessageView(this, context);
|
||||
not_null<HistoryView::ElementDelegate*> delegate) {
|
||||
return delegate->elementCreate(this);
|
||||
}
|
||||
|
||||
HistoryMessage::~HistoryMessage() {
|
||||
|
|
|
@ -143,7 +143,11 @@ public:
|
|||
markup);
|
||||
}
|
||||
|
||||
void initMedia(const MTPMessageMedia *media);
|
||||
void refreshMedia(const MTPMessageMedia *media);
|
||||
void setMedia(const MTPMessageMedia &media);
|
||||
static std::unique_ptr<Data::Media> CreateMedia(
|
||||
not_null<HistoryMessage*> item,
|
||||
const MTPMessageMedia &media);
|
||||
|
||||
int32 plainMaxWidth() const;
|
||||
|
||||
|
@ -163,7 +167,7 @@ public:
|
|||
|
||||
void applyEdition(const MTPDmessage &message) override;
|
||||
void applyEdition(const MTPDmessageService &message) override;
|
||||
void updateMedia(const MTPMessageMedia *media) override;
|
||||
void updateSentMedia(const MTPMessageMedia *media) override;
|
||||
void updateReplyMarkup(const MTPReplyMarkup *markup) override {
|
||||
setReplyMarkup(markup);
|
||||
}
|
||||
|
@ -191,8 +195,7 @@ public:
|
|||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> createView(
|
||||
not_null<Window::Controller*> controller,
|
||||
HistoryView::Context context) override;
|
||||
not_null<HistoryView::ElementDelegate*> delegate) override;
|
||||
|
||||
~HistoryMessage();
|
||||
|
||||
|
|
|
@ -208,7 +208,7 @@ void HistoryService::setMessageByAction(const MTPmessageAction &action) {
|
|||
_media = std::make_unique<Data::MediaPhoto>(
|
||||
this,
|
||||
history()->peer,
|
||||
App::feedPhoto(photo.c_photo()));
|
||||
Auth().data().photo(photo.c_photo()));
|
||||
}
|
||||
} break;
|
||||
|
||||
|
@ -452,9 +452,8 @@ QString HistoryService::inReplyText() const {
|
|||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> HistoryService::createView(
|
||||
not_null<Window::Controller*> controller,
|
||||
HistoryView::Context context) {
|
||||
return controller->createMessageView(this, context);
|
||||
not_null<HistoryView::ElementDelegate*> delegate) {
|
||||
return delegate->elementCreate(this);
|
||||
}
|
||||
|
||||
void HistoryService::setServiceText(const PreparedText &prepared) {
|
||||
|
|
|
@ -97,8 +97,7 @@ public:
|
|||
QString inReplyText() const override;
|
||||
|
||||
std::unique_ptr<HistoryView::Element> createView(
|
||||
not_null<Window::Controller*> controller,
|
||||
HistoryView::Context context) override;
|
||||
not_null<HistoryView::ElementDelegate*> delegate) override;
|
||||
|
||||
~HistoryService();
|
||||
|
||||
|
|
|
@ -559,7 +559,7 @@ HistoryWidget::HistoryWidget(QWidget *parent, not_null<Window::Controller*> cont
|
|||
) | rpl::start_with_next(
|
||||
[this](auto item) { itemRemoved(item); },
|
||||
lifetime());
|
||||
Auth().data().itemRepaintRequest(
|
||||
Auth().data().itemViewRepaintRequest(
|
||||
) | rpl::start_with_next(
|
||||
[this](auto item) { repaintHistoryItem(item); },
|
||||
lifetime());
|
||||
|
@ -777,8 +777,10 @@ void HistoryWidget::scrollToAnimationCallback(FullMsgId attachToId) {
|
|||
|
||||
void HistoryWidget::enqueueMessageHighlight(
|
||||
not_null<HistoryView::Element*> view) {
|
||||
if (const auto group = view->getFullGroup()) {
|
||||
view = group->leader;
|
||||
if (const auto group = Auth().data().groups().find(view->data())) {
|
||||
if (const auto leader = group->items.back()->mainView()) {
|
||||
view = leader;
|
||||
}
|
||||
}
|
||||
auto enqueueMessageId = [this](MsgId universalId) {
|
||||
if (_highlightQueue.empty() && !_highlightTimer.isActive()) {
|
||||
|
@ -845,8 +847,9 @@ void HistoryWidget::checkNextHighlight() {
|
|||
}
|
||||
|
||||
void HistoryWidget::updateHighlightedMessage() {
|
||||
auto item = getItemFromHistoryOrMigrated(_highlightedMessageId);
|
||||
if (!item || !item->mainView()) {
|
||||
const auto item = getItemFromHistoryOrMigrated(_highlightedMessageId);
|
||||
const auto view = item ? item->mainView() : nullptr;
|
||||
if (!view) {
|
||||
return stopMessageHighlight();
|
||||
}
|
||||
auto duration = st::activeFadeInDuration + st::activeFadeOutDuration;
|
||||
|
@ -854,7 +857,7 @@ void HistoryWidget::updateHighlightedMessage() {
|
|||
return stopMessageHighlight();
|
||||
}
|
||||
|
||||
Auth().data().requestItemRepaint(item);
|
||||
Auth().data().requestViewRepaint(view);
|
||||
}
|
||||
|
||||
TimeMs HistoryWidget::highlightStartTime(not_null<const HistoryItem*> item) const {
|
||||
|
@ -884,8 +887,10 @@ void HistoryWidget::clearHighlightMessages() {
|
|||
|
||||
int HistoryWidget::itemTopForHighlight(
|
||||
not_null<HistoryView::Element*> view) const {
|
||||
if (const auto group = view->getFullGroup()) {
|
||||
view = group->leader;
|
||||
if (const auto group = Auth().data().groups().find(view->data())) {
|
||||
if (const auto leader = group->items.back()->mainView()) {
|
||||
view = leader;
|
||||
}
|
||||
}
|
||||
auto itemTop = _list->itemTop(view);
|
||||
Assert(itemTop >= 0);
|
||||
|
@ -1649,7 +1654,7 @@ void HistoryWidget::showHistory(const PeerId &peerId, MsgId showAtMsgId, bool re
|
|||
}
|
||||
|
||||
if (!cAutoPlayGif()) {
|
||||
App::stopGifItems();
|
||||
Auth().data().stopAutoplayAnimations();
|
||||
}
|
||||
clearReplyReturns();
|
||||
|
||||
|
@ -2187,7 +2192,7 @@ void HistoryWidget::newUnreadMsg(History *history, HistoryItem *item) {
|
|||
}
|
||||
if (App::wnd()->doWeReadServerHistory()) {
|
||||
if (item->mentionsMe() && item->isMediaUnread()) {
|
||||
App::main()->mediaMarkRead(item);
|
||||
Auth().api().markMediaRead(item);
|
||||
}
|
||||
Auth().api().readServerHistoryForce(history);
|
||||
return;
|
||||
|
@ -3367,8 +3372,15 @@ void HistoryWidget::app_sendBotCallback(
|
|||
flags |= MTPmessages_GetBotCallbackAnswer::Flag::f_data;
|
||||
sendData = button->data;
|
||||
}
|
||||
button->requestId = MTP::send(MTPmessages_GetBotCallbackAnswer(MTP_flags(flags), _peer->input, MTP_int(msg->id), MTP_bytes(sendData)), rpcDone(&HistoryWidget::botCallbackDone, info), rpcFail(&HistoryWidget::botCallbackFail, info));
|
||||
Auth().data().requestItemRepaint(msg);
|
||||
button->requestId = MTP::send(
|
||||
MTPmessages_GetBotCallbackAnswer(
|
||||
MTP_flags(flags),
|
||||
_peer->input,
|
||||
MTP_int(msg->id),
|
||||
MTP_bytes(sendData)),
|
||||
rpcDone(&HistoryWidget::botCallbackDone, info),
|
||||
rpcFail(&HistoryWidget::botCallbackFail, info));
|
||||
Auth().data().requestItemViewRepaint(msg);
|
||||
|
||||
if (_replyToId == msg->id) {
|
||||
cancelReply();
|
||||
|
@ -3379,14 +3391,18 @@ void HistoryWidget::app_sendBotCallback(
|
|||
}
|
||||
}
|
||||
|
||||
void HistoryWidget::botCallbackDone(BotCallbackInfo info, const MTPmessages_BotCallbackAnswer &answer, mtpRequestId req) {
|
||||
void HistoryWidget::botCallbackDone(
|
||||
BotCallbackInfo info,
|
||||
const MTPmessages_BotCallbackAnswer &answer,
|
||||
mtpRequestId req) {
|
||||
auto item = App::histItemById(info.msgId);
|
||||
if (item) {
|
||||
if (auto markup = item->Get<HistoryMessageReplyMarkup>()) {
|
||||
if (info.row < markup->rows.size() && info.col < markup->rows.at(info.row).size()) {
|
||||
if (markup->rows.at(info.row).at(info.col).requestId == req) {
|
||||
markup->rows.at(info.row).at(info.col).requestId = 0;
|
||||
Auth().data().requestItemRepaint(item);
|
||||
if (const auto markup = item->Get<HistoryMessageReplyMarkup>()) {
|
||||
if (info.row < markup->rows.size()
|
||||
&& info.col < markup->rows[info.row].size()) {
|
||||
if (markup->rows[info.row][info.col].requestId == req) {
|
||||
markup->rows[info.row][info.col].requestId = 0;
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -3414,14 +3430,18 @@ void HistoryWidget::botCallbackDone(BotCallbackInfo info, const MTPmessages_BotC
|
|||
}
|
||||
}
|
||||
|
||||
bool HistoryWidget::botCallbackFail(BotCallbackInfo info, const RPCError &error, mtpRequestId req) {
|
||||
bool HistoryWidget::botCallbackFail(
|
||||
BotCallbackInfo info,
|
||||
const RPCError &error,
|
||||
mtpRequestId req) {
|
||||
// show error?
|
||||
if (auto item = App::histItemById(info.msgId)) {
|
||||
if (auto markup = item->Get<HistoryMessageReplyMarkup>()) {
|
||||
if (info.row < markup->rows.size() && info.col < markup->rows.at(info.row).size()) {
|
||||
if (markup->rows.at(info.row).at(info.col).requestId == req) {
|
||||
markup->rows.at(info.row).at(info.col).requestId = 0;
|
||||
Auth().data().requestItemRepaint(item);
|
||||
if (const auto item = App::histItemById(info.msgId)) {
|
||||
if (const auto markup = item->Get<HistoryMessageReplyMarkup>()) {
|
||||
if (info.row < markup->rows.size()
|
||||
&& info.col < markup->rows[info.row].size()) {
|
||||
if (markup->rows[info.row][info.col].requestId == req) {
|
||||
markup->rows[info.row][info.col].requestId = 0;
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -4364,7 +4384,7 @@ void HistoryWidget::onPhotoProgress(const FullMsgId &newId) {
|
|||
? item->media()->photo()
|
||||
: nullptr;
|
||||
updateSendAction(item->history(), SendAction::Type::UploadPhoto, 0);
|
||||
Auth().data().requestItemRepaint(item);
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -4382,14 +4402,17 @@ void HistoryWidget::onDocumentProgress(const FullMsgId &newId) {
|
|||
item->history(),
|
||||
sendAction,
|
||||
progress);
|
||||
Auth().data().requestItemRepaint(item);
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
}
|
||||
}
|
||||
|
||||
void HistoryWidget::onPhotoFailed(const FullMsgId &newId) {
|
||||
if (const auto item = App::histItemById(newId)) {
|
||||
updateSendAction(item->history(), SendAction::Type::UploadPhoto, -1);
|
||||
Auth().data().requestItemRepaint(item);
|
||||
updateSendAction(
|
||||
item->history(),
|
||||
SendAction::Type::UploadPhoto,
|
||||
-1);
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -4401,7 +4424,7 @@ void HistoryWidget::onDocumentFailed(const FullMsgId &newId) {
|
|||
? SendAction::Type::UploadVoice
|
||||
: SendAction::Type::UploadFile;
|
||||
updateSendAction(item->history(), sendAction, -1);
|
||||
Auth().data().requestItemRepaint(item);
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -4510,7 +4533,7 @@ void HistoryWidget::repaintHistoryItem(
|
|||
// It is possible that repaintHistoryItem() will be called from
|
||||
// _scroll->setOwnedWidget() because it calls onScroll() that
|
||||
// sendSynteticMouseEvent() and it could lead to some Info layout
|
||||
// calling Auth().data().requestItemRepaint(), while we still are
|
||||
// calling Auth().data().requestItemViewRepaint(), while we still are
|
||||
// in progrss of showing the history. Just ignore them for now :/
|
||||
if (!_list) {
|
||||
return;
|
||||
|
@ -5909,7 +5932,7 @@ void HistoryWidget::onPreviewCheck() {
|
|||
if (i == _previewCache.cend()) {
|
||||
_previewRequest = MTP::send(MTPmessages_GetWebPagePreview(MTP_string(_previewLinks)), rpcDone(&HistoryWidget::gotPreview, _previewLinks));
|
||||
} else if (i.value()) {
|
||||
_previewData = App::webPage(i.value());
|
||||
_previewData = Auth().data().webpage(i.value());
|
||||
updatePreview();
|
||||
} else {
|
||||
if (_previewData && _previewData->pendingTill >= 0) previewCancel();
|
||||
|
@ -5929,16 +5952,19 @@ void HistoryWidget::gotPreview(QString links, const MTPMessageMedia &result, mtp
|
|||
_previewRequest = 0;
|
||||
}
|
||||
if (result.type() == mtpc_messageMediaWebPage) {
|
||||
auto data = App::feedWebPage(result.c_messageMediaWebPage().vwebpage);
|
||||
_previewCache.insert(links, data->id);
|
||||
if (data->pendingTill > 0 && data->pendingTill <= unixtime()) {
|
||||
data->pendingTill = -1;
|
||||
const auto &data = result.c_messageMediaWebPage().vwebpage;
|
||||
const auto page = Auth().data().webpage(data);
|
||||
_previewCache.insert(links, page->id);
|
||||
if (page->pendingTill > 0 && page->pendingTill <= unixtime()) {
|
||||
page->pendingTill = -1;
|
||||
}
|
||||
if (links == _previewLinks && !_previewCancelled) {
|
||||
_previewData = (data->id && data->pendingTill >= 0) ? data : 0;
|
||||
_previewData = (page->id && page->pendingTill >= 0)
|
||||
? page.get()
|
||||
: nullptr;
|
||||
updatePreview();
|
||||
}
|
||||
if (App::main()) App::main()->webPagesOrGamesUpdate();
|
||||
Auth().data().sendWebPageGameNotifications();
|
||||
} else if (result.type() == mtpc_messageMediaEmpty) {
|
||||
_previewCache.insert(links, 0);
|
||||
if (links == _previewLinks && !_previewCancelled) {
|
||||
|
|
|
@ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_media_grouped.h"
|
||||
#include "history/history.h"
|
||||
#include "data/data_session.h"
|
||||
#include "data/data_groups.h"
|
||||
#include "data/data_media_types.h"
|
||||
#include "auth_session.h"
|
||||
#include "layout.h"
|
||||
|
@ -55,10 +56,9 @@ TextSelection ShiftItemSelection(
|
|||
|
||||
Element::Element(not_null<HistoryItem*> data, Context context)
|
||||
: _data(data)
|
||||
, _media(_data->media() ? _data->media()->createView(this) : nullptr)
|
||||
, _context(context) {
|
||||
Auth().data().registerItemView(this);
|
||||
initGroup();
|
||||
refreshMedia();
|
||||
}
|
||||
|
||||
not_null<HistoryItem*> Element::data() const {
|
||||
|
@ -145,93 +145,110 @@ int Element::infoWidth() const {
|
|||
bool Element::isHiddenByGroup() const {
|
||||
return _flags & Flag::HiddenByGroup;
|
||||
}
|
||||
//
|
||||
//void Element::makeGroupMember(not_null<Element*> leader) {
|
||||
// Expects(leader != this);
|
||||
//
|
||||
// const auto group = Get<Group>();
|
||||
// Assert(group != nullptr);
|
||||
// if (group->leader == this) {
|
||||
// if (auto single = _media ? _media->takeLastFromGroup() : nullptr) {
|
||||
// _media = std::move(single);
|
||||
// }
|
||||
// _flags |= Flag::HiddenByGroup;
|
||||
// Auth().data().requestViewResize(this);
|
||||
//
|
||||
// group->leader = leader;
|
||||
// base::take(group->others);
|
||||
// } else if (group->leader != leader) {
|
||||
// group->leader = leader;
|
||||
// }
|
||||
//
|
||||
// Ensures(isHiddenByGroup());
|
||||
// Ensures(group->others.empty());
|
||||
//}
|
||||
//
|
||||
//void Element::makeGroupLeader(std::vector<not_null<HistoryItem*>> &&others) {
|
||||
// const auto group = Get<Group>();
|
||||
// Assert(group != nullptr);
|
||||
//
|
||||
// const auto leaderChanged = (group->leader != this);
|
||||
// if (leaderChanged) {
|
||||
// group->leader = this;
|
||||
// _flags &= ~Flag::HiddenByGroup;
|
||||
// Auth().data().requestViewResize(this);
|
||||
// }
|
||||
// group->others = std::move(others);
|
||||
// if (!_media || !_media->applyGroup(group->others)) {
|
||||
// resetGroupMedia(group->others);
|
||||
// data()->invalidateChatsListEntry();
|
||||
// }
|
||||
//
|
||||
// Ensures(!isHiddenByGroup());
|
||||
//}
|
||||
//
|
||||
//bool Element::groupIdValidityChanged() {
|
||||
// if (Has<Group>()) {
|
||||
// if (_media && _media->canBeGrouped()) {
|
||||
// return false;
|
||||
// }
|
||||
// RemoveComponents(Group::Bit());
|
||||
// Auth().data().requestViewResize(this);
|
||||
// return true;
|
||||
// }
|
||||
// return false;
|
||||
//}
|
||||
//
|
||||
//void Element::validateGroupId() {
|
||||
// // Just ignore the result.
|
||||
// groupIdValidityChanged();
|
||||
//}
|
||||
//
|
||||
//Group *Element::getFullGroup() {
|
||||
// if (const auto group = Get<Group>()) {
|
||||
// if (group->leader == this) {
|
||||
// return group;
|
||||
// }
|
||||
// return group->leader->Get<Group>();
|
||||
// }
|
||||
// return nullptr;
|
||||
//}
|
||||
|
||||
void Element::makeGroupMember(not_null<Element*> leader) {
|
||||
Expects(leader != this);
|
||||
void Element::refreshMedia() {
|
||||
_flags &= ~Flag::HiddenByGroup;
|
||||
|
||||
const auto group = Get<Group>();
|
||||
Assert(group != nullptr);
|
||||
if (group->leader == this) {
|
||||
if (auto single = _media ? _media->takeLastFromGroup() : nullptr) {
|
||||
_media = std::move(single);
|
||||
const auto item = data();
|
||||
const auto media = item->media();
|
||||
if (media && media->canBeGrouped()) {
|
||||
if (const auto group = Auth().data().groups().find(item)) {
|
||||
if (group->items.back() != item) {
|
||||
_media = nullptr;
|
||||
_flags |= Flag::HiddenByGroup;
|
||||
} else {
|
||||
_media = std::make_unique<HistoryGroupedMedia>(
|
||||
this,
|
||||
group->items);
|
||||
Auth().data().requestViewResize(this);
|
||||
}
|
||||
return;
|
||||
}
|
||||
_flags |= Flag::HiddenByGroup;
|
||||
Auth().data().requestViewResize(this);
|
||||
|
||||
group->leader = leader;
|
||||
base::take(group->others);
|
||||
} else if (group->leader != leader) {
|
||||
group->leader = leader;
|
||||
}
|
||||
|
||||
Ensures(isHiddenByGroup());
|
||||
Ensures(group->others.empty());
|
||||
}
|
||||
|
||||
void Element::makeGroupLeader(std::vector<not_null<Element*>> &&others) {
|
||||
const auto group = Get<Group>();
|
||||
Assert(group != nullptr);
|
||||
|
||||
const auto leaderChanged = (group->leader != this);
|
||||
if (leaderChanged) {
|
||||
group->leader = this;
|
||||
_flags &= ~Flag::HiddenByGroup;
|
||||
Auth().data().requestViewResize(this);
|
||||
}
|
||||
group->others = std::move(others);
|
||||
if (!_media || !_media->applyGroup(group->others)) {
|
||||
resetGroupMedia(group->others);
|
||||
data()->invalidateChatsListEntry();
|
||||
}
|
||||
|
||||
Ensures(!isHiddenByGroup());
|
||||
}
|
||||
|
||||
bool Element::groupIdValidityChanged() {
|
||||
if (Has<Group>()) {
|
||||
if (_media && _media->canBeGrouped()) {
|
||||
return false;
|
||||
}
|
||||
RemoveComponents(Group::Bit());
|
||||
Auth().data().requestViewResize(this);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void Element::validateGroupId() {
|
||||
// Just ignore the result.
|
||||
groupIdValidityChanged();
|
||||
}
|
||||
|
||||
Group *Element::getFullGroup() {
|
||||
if (const auto group = Get<Group>()) {
|
||||
if (group->leader == this) {
|
||||
return group;
|
||||
}
|
||||
return group->leader->Get<Group>();
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
void Element::initGroup() {
|
||||
if (const auto groupId = _data->groupId()) {
|
||||
AddComponents(Group::Bit());
|
||||
const auto group = Get<Group>();
|
||||
group->groupId = groupId;
|
||||
group->leader = this;
|
||||
if (_data->media()) {
|
||||
_media = _data->media()->createView(this);
|
||||
} else {
|
||||
_media = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
void Element::resetGroupMedia(
|
||||
const std::vector<not_null<Element*>> &others) {
|
||||
if (!others.empty()) {
|
||||
_media = std::make_unique<HistoryGroupedMedia>(this, others);
|
||||
} else if (_media) {
|
||||
_media = _media->takeLastFromGroup();
|
||||
}
|
||||
Auth().data().requestViewResize(this);
|
||||
}
|
||||
//void Element::resetGroupMedia(
|
||||
// const std::vector<not_null<Element*>> &others) {
|
||||
// if (!others.empty()) {
|
||||
// _media = std::make_unique<HistoryGroupedMedia>(this, others);
|
||||
// } else if (_media) {
|
||||
// _media = _media->takeLastFromGroup();
|
||||
// }
|
||||
// Auth().data().requestViewResize(this);
|
||||
//}
|
||||
|
||||
void Element::previousInBlocksChanged() {
|
||||
recountDisplayDateInBlocks();
|
||||
|
@ -245,12 +262,13 @@ void Element::nextInBlocksChanged() {
|
|||
|
||||
void Element::refreshDataId() {
|
||||
if (const auto media = this->media()) {
|
||||
media->refreshParentId(this);
|
||||
if (const auto group = Get<Group>()) {
|
||||
if (group->leader != this) {
|
||||
group->leader->refreshDataId();
|
||||
}
|
||||
}
|
||||
media->refreshParentId(data());
|
||||
// #TODO refresh sent album items
|
||||
//if (const auto group = Get<Group>()) {
|
||||
// if (group->leader != this) {
|
||||
// group->leader->refreshDataId();
|
||||
// }
|
||||
//}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -325,20 +343,20 @@ void Element::setDisplayDate(bool displayDate) {
|
|||
void Element::setAttachToNext(bool attachToNext) {
|
||||
if (attachToNext && !(_flags & Flag::AttachedToNext)) {
|
||||
_flags |= Flag::AttachedToNext;
|
||||
Auth().data().requestItemRepaint(data());
|
||||
Auth().data().requestViewResize(this);
|
||||
} else if (!attachToNext && (_flags & Flag::AttachedToNext)) {
|
||||
_flags &= ~Flag::AttachedToNext;
|
||||
Auth().data().requestItemRepaint(data());
|
||||
Auth().data().requestViewResize(this);
|
||||
}
|
||||
}
|
||||
|
||||
void Element::setAttachToPrevious(bool attachToPrevious) {
|
||||
if (attachToPrevious && !(_flags & Flag::AttachedToPrevious)) {
|
||||
_flags |= Flag::AttachedToPrevious;
|
||||
setPendingResize();
|
||||
Auth().data().requestViewResize(this);
|
||||
} else if (!attachToPrevious && (_flags & Flag::AttachedToPrevious)) {
|
||||
_flags &= ~Flag::AttachedToPrevious;
|
||||
setPendingResize();
|
||||
Auth().data().requestViewResize(this);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -502,7 +520,7 @@ void Element::clickHandlerActiveChanged(
|
|||
}
|
||||
}
|
||||
App::hoveredLinkItem(active ? this : nullptr);
|
||||
Auth().data().requestItemRepaint(_data);
|
||||
Auth().data().requestViewRepaint(this);
|
||||
if (const auto media = this->media()) {
|
||||
media->clickHandlerActiveChanged(handler, active);
|
||||
}
|
||||
|
@ -517,7 +535,7 @@ void Element::clickHandlerPressedChanged(
|
|||
}
|
||||
}
|
||||
App::pressedLinkItem(pressed ? this : nullptr);
|
||||
Auth().data().requestItemRepaint(_data);
|
||||
Auth().data().requestViewRepaint(this);
|
||||
if (const auto media = this->media()) {
|
||||
media->clickHandlerPressedChanged(handler, pressed);
|
||||
}
|
||||
|
|
|
@ -13,6 +13,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
|
||||
class HistoryBlock;
|
||||
class HistoryItem;
|
||||
class HistoryMessage;
|
||||
class HistoryService;
|
||||
class HistoryMedia;
|
||||
class HistoryWebPage;
|
||||
struct HistoryTextState;
|
||||
|
@ -21,6 +23,16 @@ enum InfoDisplayType : char;
|
|||
|
||||
namespace HistoryView {
|
||||
|
||||
class Element;
|
||||
class ElementDelegate {
|
||||
public:
|
||||
virtual std::unique_ptr<Element> elementCreate(
|
||||
not_null<HistoryMessage*> message) = 0;
|
||||
virtual std::unique_ptr<Element> elementCreate(
|
||||
not_null<HistoryService*> message) = 0;
|
||||
|
||||
};
|
||||
|
||||
TextSelection UnshiftItemSelection(
|
||||
TextSelection selection,
|
||||
uint16 byLength);
|
||||
|
@ -35,11 +47,11 @@ TextSelection ShiftItemSelection(
|
|||
const Text &byText);
|
||||
|
||||
class Element;
|
||||
struct Group : public RuntimeComponent<Group, Element> {
|
||||
MessageGroupId groupId = MessageGroupId::None;
|
||||
Element *leader = nullptr;
|
||||
std::vector<not_null<Element*>> others;
|
||||
};
|
||||
//struct Group : public RuntimeComponent<Group, Element> {
|
||||
// MessageGroupId groupId = MessageGroupId::None;
|
||||
// Element *leader = nullptr;
|
||||
// std::vector<not_null<HistoryItem*>> others;
|
||||
//};
|
||||
|
||||
enum class Context : char {
|
||||
History,
|
||||
|
@ -86,11 +98,11 @@ public:
|
|||
virtual int infoWidth() const;
|
||||
|
||||
bool isHiddenByGroup() const;
|
||||
void makeGroupMember(not_null<Element*> leader);
|
||||
void makeGroupLeader(std::vector<not_null<Element*>> &&others);
|
||||
bool groupIdValidityChanged();
|
||||
void validateGroupId();
|
||||
Group *getFullGroup();
|
||||
//void makeGroupMember(not_null<Element*> leader);
|
||||
//void makeGroupLeader(std::vector<not_null<HistoryItem*>> &&others);
|
||||
//bool groupIdValidityChanged();
|
||||
//void validateGroupId();
|
||||
//Group *getFullGroup();
|
||||
|
||||
// For blocks context this should be called only from recountAttachToPreviousInBlocks().
|
||||
void setAttachToPrevious(bool attachToNext);
|
||||
|
@ -197,8 +209,8 @@ private:
|
|||
virtual QSize performCountOptimalSize() = 0;
|
||||
virtual QSize performCountCurrentSize(int newWidth) = 0;
|
||||
|
||||
void initGroup();
|
||||
void resetGroupMedia(const std::vector<not_null<Element*>> &others);
|
||||
void refreshMedia();
|
||||
//void resetGroupMedia(const std::vector<not_null<HistoryItem*>> &others);
|
||||
|
||||
const not_null<HistoryItem*> _data;
|
||||
std::unique_ptr<HistoryMedia> _media;
|
||||
|
|
|
@ -221,7 +221,7 @@ ListWidget::ListWidget(
|
|||
, _scrollDateCheck([this] { scrollDateCheck(); }) {
|
||||
setMouseTracking(true);
|
||||
_scrollDateHideTimer.setCallback([this] { scrollDateHideByTimer(); });
|
||||
Auth().data().itemRepaintRequest(
|
||||
Auth().data().itemViewRepaintRequest(
|
||||
) | rpl::start_with_next([this](auto item) {
|
||||
if (const auto view = viewForItem(item)) {
|
||||
repaintItem(view);
|
||||
|
@ -306,7 +306,7 @@ not_null<Element*> ListWidget::enforceViewForItem(
|
|||
}
|
||||
const auto [i, ok] = _views.emplace(
|
||||
item,
|
||||
item->createView(_controller, _context));
|
||||
item->createView(_delegate));
|
||||
return i->second.get();
|
||||
}
|
||||
|
||||
|
|
|
@ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "base/timer.h"
|
||||
#include "data/data_messages.h"
|
||||
#include "history/view/history_view_cursor_state.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
|
||||
namespace Ui {
|
||||
class PopupMenu;
|
||||
|
@ -25,9 +26,8 @@ class Controller;
|
|||
namespace HistoryView {
|
||||
|
||||
enum class Context : char;
|
||||
class Element;
|
||||
|
||||
class ListDelegate {
|
||||
class ListDelegate : public ElementDelegate {
|
||||
public:
|
||||
virtual Context listContext() = 0;
|
||||
virtual void listScrollTo(int top) = 0;
|
||||
|
@ -36,6 +36,7 @@ public:
|
|||
Data::MessagePosition aroundId,
|
||||
int limitBefore,
|
||||
int limitAfter) = 0;
|
||||
|
||||
};
|
||||
|
||||
class ListMemento {
|
||||
|
|
|
@ -57,7 +57,7 @@ const style::TextStyle &KeyboardStyle::textStyle() const {
|
|||
}
|
||||
|
||||
void KeyboardStyle::repaint(not_null<const HistoryItem*> item) const {
|
||||
Auth().data().requestItemRepaint(item);
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
}
|
||||
|
||||
int KeyboardStyle::buttonRadius() const {
|
||||
|
|
|
@ -569,7 +569,7 @@ void ListWidget::start() {
|
|||
) | rpl::start_with_next([this](auto item) {
|
||||
itemRemoved(item);
|
||||
}, lifetime());
|
||||
Auth().data().itemRepaintRequest(
|
||||
Auth().data().itemViewRepaintRequest(
|
||||
) | rpl::start_with_next([this](auto item) {
|
||||
repaintItem(item);
|
||||
}, lifetime());
|
||||
|
|
|
@ -210,19 +210,26 @@ const DocumentItems *documentItems() {
|
|||
|
||||
namespace internal {
|
||||
|
||||
void regDocumentItem(DocumentData *document, ItemBase *item) {
|
||||
void regDocumentItem(
|
||||
not_null<const DocumentData*> document,
|
||||
not_null<ItemBase*> item) {
|
||||
documentItemsMap.createIfNull();
|
||||
(*documentItemsMap)[document].insert(item);
|
||||
}
|
||||
|
||||
void unregDocumentItem(DocumentData *document, ItemBase *item) {
|
||||
void unregDocumentItem(
|
||||
not_null<const DocumentData*> document,
|
||||
not_null<ItemBase*> item) {
|
||||
if (documentItemsMap) {
|
||||
auto i = documentItemsMap->find(document);
|
||||
if (i != documentItemsMap->cend()) {
|
||||
if (i->remove(item) && i->isEmpty()) {
|
||||
if (i->second.remove(item) && i->second.empty()) {
|
||||
documentItemsMap->erase(i);
|
||||
}
|
||||
}
|
||||
if (documentItemsMap->empty()) {
|
||||
documentItemsMap.clear();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -115,13 +115,19 @@ private:
|
|||
|
||||
};
|
||||
|
||||
using DocumentItems = QMap<DocumentData*, OrderedSet<ItemBase*>>;
|
||||
using DocumentItems = std::map<
|
||||
not_null<const DocumentData*>,
|
||||
base::flat_set<not_null<ItemBase*>>>;
|
||||
const DocumentItems *documentItems();
|
||||
|
||||
namespace internal {
|
||||
|
||||
void regDocumentItem(DocumentData *document, ItemBase *item);
|
||||
void unregDocumentItem(DocumentData *document, ItemBase *item);
|
||||
void regDocumentItem(
|
||||
not_null<const DocumentData*> document,
|
||||
not_null<ItemBase*> item);
|
||||
void unregDocumentItem(
|
||||
not_null<const DocumentData*> document,
|
||||
not_null<ItemBase*> item);
|
||||
|
||||
} // namespace internal
|
||||
} // namespace Layout
|
||||
|
|
|
@ -9,11 +9,13 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
|
||||
#include "data/data_photo.h"
|
||||
#include "data/data_document.h"
|
||||
#include "data/data_session.h"
|
||||
#include "inline_bots/inline_bot_layout_item.h"
|
||||
#include "inline_bots/inline_bot_send_data.h"
|
||||
#include "storage/file_download.h"
|
||||
#include "core/file_utilities.h"
|
||||
#include "mainwidget.h"
|
||||
#include "auth_session.h"
|
||||
|
||||
namespace InlineBots {
|
||||
|
||||
|
@ -78,10 +80,10 @@ std::unique_ptr<Result> Result::create(uint64 queryId, const MTPBotInlineResult
|
|||
if (r.has_title()) result->_title = qs(r.vtitle);
|
||||
if (r.has_description()) result->_description = qs(r.vdescription);
|
||||
if (r.has_photo()) {
|
||||
result->_photo = App::feedPhoto(r.vphoto);
|
||||
result->_photo = Auth().data().photo(r.vphoto);
|
||||
}
|
||||
if (r.has_document()) {
|
||||
result->_document = App::feedDocument(r.vdocument);
|
||||
result->_document = Auth().data().document(r.vdocument);
|
||||
}
|
||||
message = &r.vsend_message;
|
||||
} break;
|
||||
|
@ -315,7 +317,13 @@ void Result::createPhoto() {
|
|||
|
||||
ImagePtr full = ImagePtr(_content_url, _width, _height);
|
||||
auto photoId = rand_value<PhotoId>();
|
||||
_photo = App::photoSet(photoId, 0, 0, unixtime(), _thumb, medium, full);
|
||||
_photo = Auth().data().photo(
|
||||
photoId,
|
||||
uint64(0),
|
||||
unixtime(),
|
||||
_thumb,
|
||||
medium,
|
||||
full);
|
||||
_photo->thumb = _thumb;
|
||||
}
|
||||
|
||||
|
@ -354,15 +362,32 @@ void Result::createDocument() {
|
|||
}
|
||||
|
||||
auto documentId = rand_value<DocumentId>();
|
||||
_document = App::documentSet(documentId, nullptr, 0, 0, unixtime(), attributes, mime, _thumb, MTP::maindc(), 0, StorageImageLocation());
|
||||
_document = Auth().data().document(
|
||||
documentId,
|
||||
uint64(0),
|
||||
int32(0),
|
||||
unixtime(),
|
||||
attributes,
|
||||
mime,
|
||||
_thumb,
|
||||
MTP::maindc(),
|
||||
int32(0),
|
||||
StorageImageLocation());
|
||||
_document->setContentUrl(_content_url);
|
||||
}
|
||||
|
||||
void Result::createGame() {
|
||||
if (_game) return;
|
||||
|
||||
auto gameId = rand_value<GameId>();
|
||||
_game = App::gameSet(gameId, nullptr, 0, QString(), _title, _description, _photo, _document);
|
||||
const auto gameId = rand_value<GameId>();
|
||||
_game = Auth().data().game(
|
||||
gameId,
|
||||
0,
|
||||
QString(),
|
||||
_title,
|
||||
_description,
|
||||
_photo,
|
||||
_document);
|
||||
}
|
||||
|
||||
} // namespace InlineBots
|
||||
|
|
|
@ -256,9 +256,6 @@ MainWidget::MainWidget(
|
|||
connect(&_updateMutedTimer, SIGNAL(timeout()), this, SLOT(onUpdateMuted()));
|
||||
connect(&_viewsIncrementTimer, SIGNAL(timeout()), this, SLOT(onViewsIncrement()));
|
||||
|
||||
_webPageOrGameUpdater.setSingleShot(true);
|
||||
connect(&_webPageOrGameUpdater, SIGNAL(timeout()), this, SLOT(webPagesOrGamesUpdate()));
|
||||
|
||||
using Update = Window::Theme::BackgroundUpdate;
|
||||
subscribe(Window::Theme::Background(), [this](const Update &update) {
|
||||
if (update.type == Update::Type::New || update.type == Update::Type::Changed) {
|
||||
|
@ -689,44 +686,6 @@ void MainWidget::finishForwarding(not_null<History*> history) {
|
|||
_history->peerMessagesUpdated(history->peer->id);
|
||||
}
|
||||
|
||||
void MainWidget::webPageUpdated(WebPageData *data) {
|
||||
_webPagesUpdated.insert(data->id);
|
||||
_webPageOrGameUpdater.start(0);
|
||||
}
|
||||
|
||||
void MainWidget::gameUpdated(GameData *data) {
|
||||
_gamesUpdated.insert(data->id);
|
||||
_webPageOrGameUpdater.start(0);
|
||||
}
|
||||
|
||||
void MainWidget::webPagesOrGamesUpdate() {
|
||||
_webPageOrGameUpdater.stop();
|
||||
if (!_webPagesUpdated.isEmpty()) {
|
||||
auto &items = App::webPageItems();
|
||||
for_const (auto webPageId, _webPagesUpdated) {
|
||||
auto j = items.constFind(App::webPage(webPageId));
|
||||
if (j != items.cend()) {
|
||||
for_const (auto item, j.value()) {
|
||||
Auth().data().requestItemViewResize(item);
|
||||
}
|
||||
}
|
||||
}
|
||||
_webPagesUpdated.clear();
|
||||
}
|
||||
if (!_gamesUpdated.isEmpty()) {
|
||||
auto &items = App::gameItems();
|
||||
for_const (auto gameId, _gamesUpdated) {
|
||||
auto j = items.constFind(App::game(gameId));
|
||||
if (j != items.cend()) {
|
||||
for_const (auto item, j.value()) {
|
||||
Auth().data().requestItemViewResize(item);
|
||||
}
|
||||
}
|
||||
}
|
||||
_gamesUpdated.clear();
|
||||
}
|
||||
}
|
||||
|
||||
void MainWidget::updateMutedIn(TimeMs delay) {
|
||||
accumulate_max(delay, 24 * 3600 * 1000LL);
|
||||
if (!_updateMutedTimer.isActive()
|
||||
|
@ -1435,8 +1394,11 @@ void MainWidget::sendMessage(const MessageToSend &message) {
|
|||
if (message.webPageId == CancelledWebPageId) {
|
||||
sendFlags |= MTPmessages_SendMessage::Flag::f_no_webpage;
|
||||
} else if (message.webPageId) {
|
||||
auto page = App::webPage(message.webPageId);
|
||||
media = MTP_messageMediaWebPage(MTP_webPagePending(MTP_long(page->id), MTP_int(page->pendingTill)));
|
||||
auto page = Auth().data().webpage(message.webPageId);
|
||||
media = MTP_messageMediaWebPage(
|
||||
MTP_webPagePending(
|
||||
MTP_long(page->id),
|
||||
MTP_int(page->pendingTill)));
|
||||
flags |= MTPDmessage::Flag::f_media;
|
||||
}
|
||||
bool channelPost = peer->isChannel() && !peer->isMegagroup();
|
||||
|
@ -1600,20 +1562,14 @@ void MainWidget::messagesAffected(
|
|||
}
|
||||
}
|
||||
|
||||
void MainWidget::messagesContentsRead(
|
||||
const MTPmessages_AffectedMessages &result) {
|
||||
const auto &data = result.c_messages_affectedMessages();
|
||||
ptsUpdateAndApply(data.vpts.v, data.vpts_count.v);
|
||||
}
|
||||
|
||||
void MainWidget::handleAudioUpdate(const AudioMsgId &audioId) {
|
||||
using State = Media::Player::State;
|
||||
const auto document = audioId.audio();
|
||||
auto state = Media::Player::mixer()->currentState(audioId.type());
|
||||
if (state.id == audioId && state.state == State::StoppedAtStart) {
|
||||
state.state = State::Stopped;
|
||||
Media::Player::mixer()->clearStoppedAtStart(audioId);
|
||||
|
||||
auto document = audioId.audio();
|
||||
auto filepath = document->filepath(DocumentData::FilePathResolveSaveFromData);
|
||||
if (!filepath.isEmpty()) {
|
||||
if (documentIsValidMediaFile(filepath)) {
|
||||
|
@ -1628,13 +1584,15 @@ void MainWidget::handleAudioUpdate(const AudioMsgId &audioId) {
|
|||
}
|
||||
}
|
||||
|
||||
if (auto item = App::histItemById(audioId.contextId())) {
|
||||
Auth().data().requestItemRepaint(item);
|
||||
if (const auto item = App::histItemById(audioId.contextId())) {
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
item->audioTrackUpdated();
|
||||
}
|
||||
if (auto items = InlineBots::Layout::documentItems()) {
|
||||
for (auto item : items->value(audioId.audio())) {
|
||||
item->update();
|
||||
if (const auto items = InlineBots::Layout::documentItems()) {
|
||||
if (const auto i = items->find(document); i != items->end()) {
|
||||
for (const auto item : i->second) {
|
||||
item->update();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1795,7 +1753,7 @@ void MainWidget::callTopBarHeightUpdated(int callTopBarHeight) {
|
|||
|
||||
void MainWidget::documentLoadProgress(FileLoader *loader) {
|
||||
if (auto documentId = loader ? loader->objId() : 0) {
|
||||
documentLoadProgress(App::document(documentId));
|
||||
documentLoadProgress(Auth().data().document(documentId));
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1804,13 +1762,7 @@ void MainWidget::documentLoadProgress(DocumentData *document) {
|
|||
document->performActionOnLoad();
|
||||
}
|
||||
|
||||
auto &items = App::documentItems();
|
||||
auto i = items.constFind(document);
|
||||
if (i != items.cend()) {
|
||||
for_const (auto item, i.value()) {
|
||||
Auth().data().requestItemRepaint(item);
|
||||
}
|
||||
}
|
||||
Auth().data().requestDocumentViewRepaint(document);
|
||||
Auth().documentUpdated.notify(document, true);
|
||||
|
||||
if (!document->loaded() && document->isAudioFile()) {
|
||||
|
@ -1822,7 +1774,7 @@ void MainWidget::documentLoadFailed(FileLoader *loader, bool started) {
|
|||
auto documentId = loader ? loader->objId() : 0;
|
||||
if (!documentId) return;
|
||||
|
||||
auto document = App::document(documentId);
|
||||
auto document = Auth().data().document(documentId);
|
||||
if (started) {
|
||||
auto failedFileName = loader->fileName();
|
||||
Ui::show(Box<ConfirmBox>(lang(lng_download_finish_failed), base::lambda_guarded(this, [this, document, failedFileName] {
|
||||
|
@ -1862,63 +1814,6 @@ void MainWidget::inlineResultLoadFailed(FileLoader *loader, bool started) {
|
|||
//Ui::repaintInlineItem();
|
||||
}
|
||||
|
||||
void MainWidget::mediaMarkRead(not_null<DocumentData*> data) {
|
||||
auto &items = App::documentItems();
|
||||
auto i = items.constFind(data);
|
||||
if (i != items.cend()) {
|
||||
mediaMarkRead({ i.value().begin(), i.value().end() });
|
||||
}
|
||||
}
|
||||
|
||||
void MainWidget::mediaMarkRead(
|
||||
const base::flat_set<not_null<HistoryItem*>> &items) {
|
||||
QVector<MTPint> markedIds;
|
||||
base::flat_map<not_null<ChannelData*>, QVector<MTPint>> channelMarkedIds;
|
||||
markedIds.reserve(items.size());
|
||||
for (const auto item : items) {
|
||||
if (!item->isMediaUnread() || (item->out() && !item->mentionsMe())) {
|
||||
continue;
|
||||
}
|
||||
item->markMediaRead();
|
||||
if (item->id > 0) {
|
||||
if (const auto channel = item->history()->peer->asChannel()) {
|
||||
channelMarkedIds[channel].push_back(MTP_int(item->id));
|
||||
} else {
|
||||
markedIds.push_back(MTP_int(item->id));
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!markedIds.isEmpty()) {
|
||||
MTP::send(
|
||||
MTPmessages_ReadMessageContents(MTP_vector<MTPint>(markedIds)),
|
||||
rpcDone(&MainWidget::messagesContentsRead));
|
||||
}
|
||||
for (const auto &channelIds : channelMarkedIds) {
|
||||
MTP::send(MTPchannels_ReadMessageContents(
|
||||
channelIds.first->inputChannel,
|
||||
MTP_vector<MTPint>(channelIds.second)));
|
||||
}
|
||||
}
|
||||
|
||||
void MainWidget::mediaMarkRead(not_null<HistoryItem*> item) {
|
||||
if ((!item->out() || item->mentionsMe()) && item->isMediaUnread()) {
|
||||
item->markMediaRead();
|
||||
if (item->id > 0) {
|
||||
const auto ids = MTP_vector<MTPint>(1, MTP_int(item->id));
|
||||
if (const auto channel = item->history()->peer->asChannel()) {
|
||||
MTP::send(
|
||||
MTPchannels_ReadMessageContents(
|
||||
channel->inputChannel,
|
||||
ids));
|
||||
} else {
|
||||
MTP::send(
|
||||
MTPmessages_ReadMessageContents(ids),
|
||||
rpcDone(&MainWidget::messagesContentsRead));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void MainWidget::onSendFileConfirm(
|
||||
const std::shared_ptr<FileLoadResult> &file) {
|
||||
_history->sendFileConfirmed(file);
|
||||
|
@ -4808,8 +4703,9 @@ void MainWidget::feedUpdates(const MTPUpdates &updates, uint64 randomId) {
|
|||
const auto entities = d.has_entities()
|
||||
? TextUtilities::EntitiesFromMTP(d.ventities.v)
|
||||
: EntitiesInText();
|
||||
const auto media = d.has_media() ? &d.vmedia : nullptr;
|
||||
item->setText({ text, entities });
|
||||
item->updateMedia(d.has_media() ? (&d.vmedia) : nullptr);
|
||||
item->updateSentMedia(media);
|
||||
if (!wasAlready) {
|
||||
item->indexAsNewItem();
|
||||
}
|
||||
|
@ -4927,7 +4823,8 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
|
|||
auto &d = update.c_updateChannelReadMessagesContents();
|
||||
auto channel = App::channelLoaded(d.vchannel_id.v);
|
||||
if (!channel) {
|
||||
if (!_byMinChannelTimer.isActive()) { // getDifference after timeout
|
||||
if (!_byMinChannelTimer.isActive()) {
|
||||
// getDifference after timeout.
|
||||
_byMinChannelTimer.start(WaitForSkippedTimeout);
|
||||
}
|
||||
return;
|
||||
|
@ -4937,7 +4834,7 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
|
|||
if (auto item = App::histItemById(channel, msgId.v)) {
|
||||
if (item->isMediaUnread()) {
|
||||
item->markMediaRead();
|
||||
Auth().data().requestItemRepaint(item);
|
||||
Auth().data().requestItemViewRepaint(item);
|
||||
}
|
||||
} else {
|
||||
// Perhaps it was an unread mention!
|
||||
|
@ -5040,9 +4937,9 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
|
|||
auto &d = update.c_updateWebPage();
|
||||
|
||||
// Update web page anyway.
|
||||
App::feedWebPage(d.vwebpage);
|
||||
Auth().data().webpage(d.vwebpage);
|
||||
_history->updatePreview();
|
||||
webPagesOrGamesUpdate();
|
||||
Auth().data().sendWebPageGameNotifications();
|
||||
|
||||
ptsUpdateAndApply(d.vpts.v, d.vpts_count.v, update);
|
||||
} break;
|
||||
|
@ -5051,9 +4948,9 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
|
|||
auto &d = update.c_updateChannelWebPage();
|
||||
|
||||
// Update web page anyway.
|
||||
App::feedWebPage(d.vwebpage);
|
||||
Auth().data().webpage(d.vwebpage);
|
||||
_history->updatePreview();
|
||||
webPagesOrGamesUpdate();
|
||||
Auth().data().sendWebPageGameNotifications();
|
||||
|
||||
auto channel = App::channelLoaded(d.vchannel_id.v);
|
||||
if (channel && !_handlingChannelDifference) {
|
||||
|
@ -5419,8 +5316,8 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
|
|||
it->stickers.clear();
|
||||
it->stickers.reserve(v.size());
|
||||
for (int i = 0, l = v.size(); i < l; ++i) {
|
||||
auto doc = App::feedDocument(v.at(i));
|
||||
if (!doc || !doc->sticker()) continue;
|
||||
const auto doc = Auth().data().document(v.at(i));
|
||||
if (!doc->sticker()) continue;
|
||||
|
||||
it->stickers.push_back(doc);
|
||||
if (doc->sticker()->set.type() != mtpc_inputStickerSetID) {
|
||||
|
@ -5439,8 +5336,8 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
|
|||
Stickers::Pack p;
|
||||
p.reserve(stickers.size());
|
||||
for (auto j = 0, c = stickers.size(); j != c; ++j) {
|
||||
auto doc = App::document(stickers[j].v);
|
||||
if (!doc || !doc->sticker()) continue;
|
||||
auto doc = Auth().data().document(stickers[j].v);
|
||||
if (!doc->sticker()) continue;
|
||||
|
||||
p.push_back(doc);
|
||||
}
|
||||
|
|
|
@ -279,12 +279,6 @@ public:
|
|||
void cancelForwarding(not_null<History*> history);
|
||||
void finishForwarding(not_null<History*> history);
|
||||
|
||||
void mediaMarkRead(not_null<DocumentData*> data);
|
||||
void mediaMarkRead(const base::flat_set<not_null<HistoryItem*>> &items);
|
||||
void mediaMarkRead(not_null<HistoryItem*> item);
|
||||
|
||||
void webPageUpdated(WebPageData *page);
|
||||
void gameUpdated(GameData *game);
|
||||
void updateMutedIn(TimeMs delay);
|
||||
|
||||
// Does offerPeer or showPeerHistory.
|
||||
|
@ -349,8 +343,6 @@ signals:
|
|||
void dialogsUpdated();
|
||||
|
||||
public slots:
|
||||
void webPagesOrGamesUpdate();
|
||||
|
||||
void documentLoadProgress(FileLoader *loader);
|
||||
void documentLoadFailed(FileLoader *loader, bool started);
|
||||
void inlineResultLoadProgress(FileLoader *loader);
|
||||
|
@ -456,7 +448,6 @@ private:
|
|||
void messagesAffected(
|
||||
not_null<PeerData*> peer,
|
||||
const MTPmessages_AffectedMessages &result);
|
||||
void messagesContentsRead(const MTPmessages_AffectedMessages &result);
|
||||
|
||||
Window::SectionSlideParams prepareShowAnimation(
|
||||
bool willHaveTopBarShadow);
|
||||
|
@ -552,10 +543,6 @@ private:
|
|||
not_null<Window::Controller*> _controller;
|
||||
bool _started = false;
|
||||
|
||||
OrderedSet<WebPageId> _webPagesUpdated;
|
||||
OrderedSet<GameId> _gamesUpdated;
|
||||
QTimer _webPageOrGameUpdater;
|
||||
|
||||
SingleTimer _updateMutedTimer;
|
||||
|
||||
QString _inviteHash;
|
||||
|
|
|
@ -47,7 +47,7 @@ Float::Float(
|
|||
|
||||
rpl::merge(
|
||||
Auth().data().itemLayoutChanged(),
|
||||
Auth().data().itemRepaintRequest()
|
||||
Auth().data().itemViewRepaintRequest()
|
||||
) | rpl::start_with_next([this](auto item) {
|
||||
if (_item == item) {
|
||||
repaintItem();
|
||||
|
|
|
@ -12,8 +12,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "data/data_photo.h"
|
||||
#include "data/data_document.h"
|
||||
#include "data/data_media_types.h"
|
||||
#include "data/data_session.h"
|
||||
#include "history/history.h"
|
||||
#include "history/history_media.h"
|
||||
#include "auth_session.h"
|
||||
#include "styles/style_mediaview.h"
|
||||
|
||||
namespace Media {
|
||||
|
@ -470,7 +472,7 @@ void GroupThumbs::animatePreviouslyAlive(
|
|||
|
||||
auto GroupThumbs::createThumb(Key key) -> std::unique_ptr<Thumb> {
|
||||
if (const auto photoId = base::get_if<PhotoId>(&key)) {
|
||||
const auto photo = App::photo(*photoId);
|
||||
const auto photo = Auth().data().photo(*photoId);
|
||||
return createThumb(key, photo->date ? photo->thumb : ImagePtr());
|
||||
} else if (const auto msgId = base::get_if<FullMsgId>(&key)) {
|
||||
if (const auto item = App::histItemById(*msgId)) {
|
||||
|
|
|
@ -23,6 +23,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_message.h"
|
||||
#include "history/history_media_types.h"
|
||||
#include "data/data_media_types.h"
|
||||
#include "data/data_session.h"
|
||||
#include "window/themes/window_theme_preview.h"
|
||||
#include "window/window_peer_menu.h"
|
||||
#include "observer_peer.h"
|
||||
|
@ -1299,7 +1300,7 @@ void MediaView::initGroupThumbs() {
|
|||
_groupThumbs->activateRequests(
|
||||
) | rpl::start_with_next([this](Media::View::GroupThumbs::Key key) {
|
||||
if (const auto photoId = base::get_if<PhotoId>(&key)) {
|
||||
const auto photo = App::photo(*photoId);
|
||||
const auto photo = Auth().data().photo(*photoId);
|
||||
moveToEntity({ photo, nullptr });
|
||||
} else if (const auto itemId = base::get_if<FullMsgId>(&key)) {
|
||||
moveToEntity(entityForItemId(*itemId));
|
||||
|
@ -2425,7 +2426,7 @@ MediaView::Entity MediaView::entityForUserPhotos(int index) const {
|
|||
if (index < 0 || index >= _userPhotosData->size()) {
|
||||
return { base::none, nullptr };
|
||||
}
|
||||
if (auto photo = App::photo((*_userPhotosData)[index])) {
|
||||
if (auto photo = Auth().data().photo((*_userPhotosData)[index])) {
|
||||
return { photo, nullptr };
|
||||
}
|
||||
return { base::none, nullptr };
|
||||
|
|
|
@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include <rpl/complete.h>
|
||||
#include "data/data_photo.h"
|
||||
#include "data/data_document.h"
|
||||
#include "data/data_session.h"
|
||||
#include "base/timer.h"
|
||||
#include "storage/localstorage.h"
|
||||
#include "platform/platform_specific.h"
|
||||
|
@ -530,8 +531,8 @@ void Messenger::chatPhotoCleared(PeerId peer, const MTPUpdates &updates) {
|
|||
|
||||
void Messenger::selfPhotoDone(const MTPphotos_Photo &result) {
|
||||
if (!App::self()) return;
|
||||
const auto &photo(result.c_photos_photo());
|
||||
App::feedPhoto(photo.vphoto);
|
||||
const auto &photo = result.c_photos_photo();
|
||||
Auth().data().photo(photo.vphoto);
|
||||
App::feedUsers(photo.vusers);
|
||||
cancelPhotoUpdate(App::self()->id);
|
||||
emit peerPhotoDone(App::self()->id);
|
||||
|
|
|
@ -127,7 +127,7 @@ ItemBase::ItemBase(not_null<HistoryItem*> parent) : _parent(parent) {
|
|||
void ItemBase::clickHandlerActiveChanged(
|
||||
const ClickHandlerPtr &action,
|
||||
bool active) {
|
||||
Auth().data().requestItemRepaint(_parent);
|
||||
Auth().data().requestItemViewRepaint(_parent);
|
||||
if (_check) {
|
||||
_check->setActive(active);
|
||||
}
|
||||
|
@ -136,7 +136,7 @@ void ItemBase::clickHandlerActiveChanged(
|
|||
void ItemBase::clickHandlerPressedChanged(
|
||||
const ClickHandlerPtr &action,
|
||||
bool pressed) {
|
||||
Auth().data().requestItemRepaint(_parent);
|
||||
Auth().data().requestItemViewRepaint(_parent);
|
||||
if (_check) {
|
||||
_check->setPressed(pressed);
|
||||
}
|
||||
|
@ -168,7 +168,7 @@ const style::RoundCheckbox &ItemBase::checkboxStyle() const {
|
|||
void ItemBase::ensureCheckboxCreated() {
|
||||
if (!_check) {
|
||||
_check = std::make_unique<Checkbox>(
|
||||
[this] { Auth().data().requestItemRepaint(_parent); },
|
||||
[=] { Auth().data().requestItemViewRepaint(_parent); },
|
||||
checkboxStyle());
|
||||
}
|
||||
}
|
||||
|
@ -203,7 +203,7 @@ void RadialProgressItem::clickHandlerActiveChanged(const ClickHandlerPtr &action
|
|||
if (action == _openl || action == _savel || action == _cancell) {
|
||||
if (iconAnimated()) {
|
||||
_a_iconOver.start(
|
||||
[this] { Auth().data().requestItemRepaint(parent()); },
|
||||
[=] { Auth().data().requestItemViewRepaint(parent()); },
|
||||
active ? 0. : 1.,
|
||||
active ? 1. : 0.,
|
||||
st::msgFileOverDuration);
|
||||
|
@ -219,7 +219,7 @@ void RadialProgressItem::setLinks(ClickHandlerPtr &&openl, ClickHandlerPtr &&sav
|
|||
|
||||
void RadialProgressItem::step_radial(TimeMs ms, bool timer) {
|
||||
if (timer) {
|
||||
Auth().data().requestItemRepaint(parent());
|
||||
Auth().data().requestItemViewRepaint(parent());
|
||||
} else {
|
||||
_radial->update(dataProgress(), dataFinished(), ms);
|
||||
if (!_radial->animating()) {
|
||||
|
|
|
@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "settings/settings_cover.h"
|
||||
|
||||
#include "data/data_photo.h"
|
||||
#include "data/data_session.h"
|
||||
#include "ui/widgets/labels.h"
|
||||
#include "ui/widgets/buttons.h"
|
||||
#include "ui/special_buttons.h"
|
||||
|
@ -84,7 +85,7 @@ PhotoData *CoverWidget::validatePhoto() const {
|
|||
Expects(_self != nullptr);
|
||||
|
||||
const auto photo = _self->userpicPhotoId()
|
||||
? App::photo(_self->userpicPhotoId())
|
||||
? Auth().data().photo(_self->userpicPhotoId()).get()
|
||||
: nullptr;
|
||||
_userpicButton->setPointerCursor(photo != nullptr && photo->date != 0);
|
||||
if (_self->userpicPhotoUnknown() || (photo && !photo->date)) {
|
||||
|
|
|
@ -14,8 +14,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "boxes/username_box.h"
|
||||
#include "boxes/add_contact_box.h"
|
||||
#include "boxes/change_phone_box.h"
|
||||
#include "data/data_session.h"
|
||||
#include "observer_peer.h"
|
||||
#include "messenger.h"
|
||||
#include "auth_session.h"
|
||||
|
||||
namespace Settings {
|
||||
|
||||
|
@ -47,12 +49,8 @@ void InfoWidget::refreshControls() {
|
|||
|
||||
void InfoWidget::refreshMobileNumber() {
|
||||
TextWithEntities phoneText;
|
||||
if (auto user = self()->asUser()) {
|
||||
if (!user->phone().isEmpty()) {
|
||||
phoneText.text = App::formatPhone(user->phone());
|
||||
} else {
|
||||
phoneText.text = App::phoneFromSharedContact(peerToUser(user->id));
|
||||
}
|
||||
if (const auto user = self()->asUser()) {
|
||||
phoneText.text = Auth().data().findContactPhone(user);
|
||||
}
|
||||
setLabeledText(
|
||||
_mobileNumber,
|
||||
|
|
|
@ -10,6 +10,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "storage/localimageloader.h"
|
||||
#include "data/data_document.h"
|
||||
#include "data/data_photo.h"
|
||||
#include "data/data_session.h"
|
||||
#include "auth_session.h"
|
||||
|
||||
namespace Storage {
|
||||
namespace {
|
||||
|
@ -116,14 +118,11 @@ Uploader::Uploader() {
|
|||
|
||||
void Uploader::uploadMedia(const FullMsgId &msgId, const SendMediaReady &media) {
|
||||
if (media.type == SendMediaType::Photo) {
|
||||
App::feedPhoto(media.photo, media.photoThumbs);
|
||||
Auth().data().photo(media.photo, media.photoThumbs);
|
||||
} else if (media.type == SendMediaType::File || media.type == SendMediaType::Audio) {
|
||||
DocumentData *document;
|
||||
if (media.photoThumbs.isEmpty()) {
|
||||
document = App::feedDocument(media.document);
|
||||
} else {
|
||||
document = App::feedDocument(media.document, media.photoThumbs.begin().value());
|
||||
}
|
||||
const auto document = media.photoThumbs.isEmpty()
|
||||
? Auth().data().document(media.document)
|
||||
: Auth().data().document(media.document, media.photoThumbs.begin().value());
|
||||
if (!media.data.isEmpty()) {
|
||||
document->setData(media.data);
|
||||
}
|
||||
|
@ -139,10 +138,12 @@ void Uploader::upload(
|
|||
const FullMsgId &msgId,
|
||||
const std::shared_ptr<FileLoadResult> &file) {
|
||||
if (file->type == SendMediaType::Photo) {
|
||||
auto photo = App::feedPhoto(file->photo, file->photoThumbs);
|
||||
auto photo = Auth().data().photo(file->photo, file->photoThumbs);
|
||||
photo->uploadingData = std::make_unique<Data::UploadState>(file->partssize);
|
||||
} else if (file->type == SendMediaType::File || file->type == SendMediaType::Audio) {
|
||||
auto document = file->thumb.isNull() ? App::feedDocument(file->document) : App::feedDocument(file->document, file->thumb);
|
||||
auto document = file->thumb.isNull()
|
||||
? Auth().data().document(file->document)
|
||||
: Auth().data().document(file->document, file->thumb);
|
||||
document->uploadingData = std::make_unique<Data::UploadState>(document->size);
|
||||
if (!file->content.isEmpty()) {
|
||||
document->setData(file->content);
|
||||
|
@ -161,7 +162,7 @@ void Uploader::currentFailed() {
|
|||
if (j->second.type() == SendMediaType::Photo) {
|
||||
emit photoFailed(j->first);
|
||||
} else if (j->second.type() == SendMediaType::File) {
|
||||
const auto document = App::document(j->second.id());
|
||||
const auto document = Auth().data().document(j->second.id());
|
||||
if (document->uploading()) {
|
||||
document->status = FileUploadFailed;
|
||||
}
|
||||
|
@ -451,7 +452,7 @@ void Uploader::partLoaded(const MTPBool &result, mtpRequestId requestId) {
|
|||
sentSizes[dc] -= sentPartSize;
|
||||
if (file.type() == SendMediaType::Photo) {
|
||||
file.fileSentSize += sentPartSize;
|
||||
const auto photo = App::photo(file.id());
|
||||
const auto photo = Auth().data().photo(file.id());
|
||||
if (photo->uploading() && file.file) {
|
||||
photo->uploadingData->size = file.file->partssize;
|
||||
photo->uploadingData->offset = file.fileSentSize;
|
||||
|
@ -459,7 +460,7 @@ void Uploader::partLoaded(const MTPBool &result, mtpRequestId requestId) {
|
|||
emit photoProgress(fullId);
|
||||
} else if (file.type() == SendMediaType::File
|
||||
|| file.type() == SendMediaType::Audio) {
|
||||
const auto document = App::document(file.id());
|
||||
const auto document = Auth().data().document(file.id());
|
||||
if (document->uploading()) {
|
||||
const auto doneParts = file.docSentParts
|
||||
- int(docRequestsSent.size());
|
||||
|
|
|
@ -3146,13 +3146,7 @@ public:
|
|||
voice->waveform[0] = -2;
|
||||
voice->wavemax = 0;
|
||||
}
|
||||
auto &items = App::documentItems();
|
||||
auto i = items.constFind(_doc);
|
||||
if (i != items.cend()) {
|
||||
for_const (auto item, i.value()) {
|
||||
Auth().data().requestItemRepaint(item);
|
||||
}
|
||||
}
|
||||
Auth().data().requestDocumentViewRepaint(_doc);
|
||||
}
|
||||
}
|
||||
virtual ~CountWaveformTask() {
|
||||
|
@ -3411,8 +3405,8 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
|
|||
for (int32 k = 0; k < stickersCount; ++k) {
|
||||
quint64 id;
|
||||
stickers.stream >> id;
|
||||
DocumentData *doc = App::document(id);
|
||||
if (!doc || !doc->sticker()) continue;
|
||||
const auto doc = Auth().data().document(id);
|
||||
if (!doc->sticker()) continue;
|
||||
|
||||
pack.push_back(doc);
|
||||
}
|
||||
|
@ -3562,7 +3556,17 @@ void importOldRecentStickers() {
|
|||
attributes.push_back(MTP_documentAttributeImageSize(MTP_int(width), MTP_int(height)));
|
||||
}
|
||||
|
||||
DocumentData *doc = App::documentSet(id, 0, access, 0, date, attributes, mime, ImagePtr(), dc, size, StorageImageLocation());
|
||||
const auto doc = Auth().data().document(
|
||||
id,
|
||||
access,
|
||||
int32(0),
|
||||
date,
|
||||
attributes,
|
||||
mime,
|
||||
ImagePtr(),
|
||||
dc,
|
||||
size,
|
||||
StorageImageLocation());
|
||||
if (!doc->sticker()) continue;
|
||||
|
||||
if (value > 0) {
|
||||
|
|
|
@ -9,6 +9,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
|
||||
#include "storage/serialize_common.h"
|
||||
#include "chat_helpers/stickers.h"
|
||||
#include "data/data_session.h"
|
||||
#include "auth_session.h"
|
||||
|
||||
namespace {
|
||||
|
||||
|
@ -119,7 +121,17 @@ DocumentData *Document::readFromStreamHelper(int streamAppVersion, QDataStream &
|
|||
if (!dc && !access) {
|
||||
return nullptr;
|
||||
}
|
||||
return App::documentSet(id, nullptr, access, version, date, attributes, mime, thumb.isNull() ? ImagePtr() : ImagePtr(thumb), dc, size, thumb);
|
||||
return Auth().data().document(
|
||||
id,
|
||||
access,
|
||||
version,
|
||||
date,
|
||||
attributes,
|
||||
mime,
|
||||
thumb.isNull() ? ImagePtr() : ImagePtr(thumb),
|
||||
dc,
|
||||
size,
|
||||
thumb);
|
||||
}
|
||||
|
||||
DocumentData *Document::readStickerFromStream(int streamAppVersion, QDataStream &stream, const StickerSetInfo &info) {
|
||||
|
|
|
@ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "ui/effects/ripple_animation.h"
|
||||
#include "ui/empty_userpic.h"
|
||||
#include "data/data_photo.h"
|
||||
#include "data/data_session.h"
|
||||
#include "core/file_utilities.h"
|
||||
#include "boxes/photo_crop_box.h"
|
||||
#include "boxes/confirm_box.h"
|
||||
|
@ -479,7 +480,7 @@ void UserpicButton::openPeerPhoto() {
|
|||
if (!id) {
|
||||
return;
|
||||
}
|
||||
const auto photo = App::photo(id);
|
||||
const auto photo = Auth().data().photo(id);
|
||||
if (photo->date) {
|
||||
Messenger::Instance().showPhoto(photo, _peer);
|
||||
}
|
||||
|
|
|
@ -9,11 +9,9 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
|
||||
#include "window/main_window.h"
|
||||
#include "info/info_memento.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
#include "history/view/history_view_message.h"
|
||||
#include "history/view/history_view_service_message.h"
|
||||
#include "history/history.h"
|
||||
#include "history/history_item.h"
|
||||
#include "history/view/history_view_element.h"
|
||||
#include "mainwidget.h"
|
||||
#include "mainwindow.h"
|
||||
#include "styles/style_window.h"
|
||||
|
@ -407,16 +405,4 @@ not_null<MainWidget*> Controller::chats() const {
|
|||
return App::wnd()->chatsWidget();
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> Controller::createMessageView(
|
||||
not_null<HistoryMessage*> message,
|
||||
HistoryView::Context context) {
|
||||
return std::make_unique<HistoryView::Message>(message, context);
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> Controller::createMessageView(
|
||||
not_null<HistoryService*> message,
|
||||
HistoryView::Context context) {
|
||||
return std::make_unique<HistoryView::Service>(message, context);
|
||||
}
|
||||
|
||||
} // namespace Window
|
||||
|
|
|
@ -14,10 +14,6 @@ class MainWidget;
|
|||
class HistoryMessage;
|
||||
class HistoryService;
|
||||
|
||||
namespace HistoryView {
|
||||
enum class Context : char;
|
||||
} // namespace HistoryView
|
||||
|
||||
namespace Window {
|
||||
|
||||
class LayerWidget;
|
||||
|
@ -201,13 +197,6 @@ public:
|
|||
return this;
|
||||
}
|
||||
|
||||
std::unique_ptr<HistoryView::Element> createMessageView(
|
||||
not_null<HistoryMessage*> message,
|
||||
HistoryView::Context context);
|
||||
std::unique_ptr<HistoryView::Element> createMessageView(
|
||||
not_null<HistoryService*> message,
|
||||
HistoryView::Context context);
|
||||
|
||||
private:
|
||||
int minimalThreeColumnWidth() const;
|
||||
not_null<MainWidget*> chats() const;
|
||||
|
|
|
@ -469,12 +469,10 @@ void PeerMenuDeleteContact(not_null<UserData*> user) {
|
|||
}
|
||||
|
||||
void PeerMenuAddContact(not_null<UserData*> user) {
|
||||
auto firstName = user->firstName;
|
||||
auto lastName = user->lastName;
|
||||
auto phone = user->phone().isEmpty()
|
||||
? App::phoneFromSharedContact(user->bareId())
|
||||
: user->phone();
|
||||
Ui::show(Box<AddContactBox>(firstName, lastName, phone));
|
||||
Ui::show(Box<AddContactBox>(
|
||||
user->firstName,
|
||||
user->lastName,
|
||||
Auth().data().findContactPhone(user)));
|
||||
}
|
||||
|
||||
void PeerMenuShareContactBox(not_null<UserData*> user) {
|
||||
|
|
|
@ -258,8 +258,6 @@
|
|||
<(src_loc)/history/history_media.cpp
|
||||
<(src_loc)/history/history_media_grouped.h
|
||||
<(src_loc)/history/history_media_grouped.cpp
|
||||
<(src_loc)/history/history_media_pointer.h
|
||||
<(src_loc)/history/history_media_pointer.cpp
|
||||
<(src_loc)/history/history_media_types.cpp
|
||||
<(src_loc)/history/history_media_types.h
|
||||
<(src_loc)/history/history_message.cpp
|
||||
|
|
Loading…
Reference in New Issue