mirror of https://github.com/procxx/kepka.git
Support new ('modern') API file locations.
This commit is contained in:
parent
eba2a98703
commit
aa8f62da9d
|
@ -89,9 +89,8 @@ constexpr auto kFeedReadTimeout = crl::time(1000);
|
||||||
constexpr auto kStickersByEmojiInvalidateTimeout = crl::time(60 * 60 * 1000);
|
constexpr auto kStickersByEmojiInvalidateTimeout = crl::time(60 * 60 * 1000);
|
||||||
constexpr auto kNotifySettingSaveTimeout = crl::time(1000);
|
constexpr auto kNotifySettingSaveTimeout = crl::time(1000);
|
||||||
|
|
||||||
using SimpleFileLocationId = Data::SimpleFileLocationId;
|
using PhotoFileLocationId = Data::PhotoFileLocationId;
|
||||||
using DocumentFileLocationId = Data::DocumentFileLocationId;
|
using DocumentFileLocationId = Data::DocumentFileLocationId;
|
||||||
using FileLocationId = Data::FileLocationId;
|
|
||||||
using UpdatedFileReferences = Data::UpdatedFileReferences;
|
using UpdatedFileReferences = Data::UpdatedFileReferences;
|
||||||
|
|
||||||
bool IsSilentPost(not_null<HistoryItem*> item, bool silent) {
|
bool IsSilentPost(not_null<HistoryItem*> item, bool silent) {
|
||||||
|
@ -2840,7 +2839,13 @@ void ApiWrap::requestFileReference(
|
||||||
&origin);
|
&origin);
|
||||||
if (documentId) {
|
if (documentId) {
|
||||||
_session->data().document(
|
_session->data().document(
|
||||||
*documentId
|
documentId->id
|
||||||
|
)->refreshFileReference(reference);
|
||||||
|
}
|
||||||
|
const auto photoId = base::get_if<PhotoFileLocationId>(&origin);
|
||||||
|
if (photoId) {
|
||||||
|
_session->data().photo(
|
||||||
|
photoId->id
|
||||||
)->refreshFileReference(reference);
|
)->refreshFileReference(reference);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2857,7 +2862,7 @@ void ApiWrap::requestFileReference(
|
||||||
auto handlers = std::move(i->second);
|
auto handlers = std::move(i->second);
|
||||||
_fileReferenceHandlers.erase(i);
|
_fileReferenceHandlers.erase(i);
|
||||||
for (auto &handler : handlers) {
|
for (auto &handler : handlers) {
|
||||||
handler(Data::UpdatedFileReferences());
|
handler(UpdatedFileReferences());
|
||||||
}
|
}
|
||||||
}).send();
|
}).send();
|
||||||
}
|
}
|
||||||
|
@ -2868,7 +2873,7 @@ void ApiWrap::refreshFileReference(
|
||||||
int requestId,
|
int requestId,
|
||||||
const QByteArray ¤t) {
|
const QByteArray ¤t) {
|
||||||
return refreshFileReference(origin, crl::guard(loader, [=](
|
return refreshFileReference(origin, crl::guard(loader, [=](
|
||||||
const Data::UpdatedFileReferences &data) {
|
const UpdatedFileReferences &data) {
|
||||||
loader->refreshFileReferenceFrom(data, requestId, current);
|
loader->refreshFileReferenceFrom(data, requestId, current);
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
@ -2894,7 +2899,7 @@ void ApiWrap::refreshFileReference(
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
const auto fail = [&] {
|
const auto fail = [&] {
|
||||||
handler(Data::UpdatedFileReferences());
|
handler(UpdatedFileReferences());
|
||||||
};
|
};
|
||||||
origin.data.match([&](Data::FileOriginMessage data) {
|
origin.data.match([&](Data::FileOriginMessage data) {
|
||||||
if (const auto item = App::histItemById(data)) {
|
if (const auto item = App::histItemById(data)) {
|
||||||
|
@ -2924,22 +2929,7 @@ void ApiWrap::refreshFileReference(
|
||||||
fail();
|
fail();
|
||||||
}
|
}
|
||||||
}, [&](Data::FileOriginPeerPhoto data) {
|
}, [&](Data::FileOriginPeerPhoto data) {
|
||||||
if (const auto peer = _session->data().peer(data.peerId)) {
|
fail();
|
||||||
if (const auto user = peer->asUser()) {
|
|
||||||
request(MTPusers_GetUsers(
|
|
||||||
MTP_vector<MTPInputUser>(1, user->inputUser)));
|
|
||||||
} else if (const auto chat = peer->asChat()) {
|
|
||||||
request(MTPmessages_GetChats(
|
|
||||||
MTP_vector<MTPint>(1, chat->inputChat)));
|
|
||||||
} else if (const auto channel = peer->asChannel()) {
|
|
||||||
request(MTPchannels_GetChannels(
|
|
||||||
MTP_vector<MTPInputChannel>(1, channel->inputChannel)));
|
|
||||||
} else {
|
|
||||||
fail();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
fail();
|
|
||||||
}
|
|
||||||
}, [&](Data::FileOriginStickerSet data) {
|
}, [&](Data::FileOriginStickerSet data) {
|
||||||
if (data.setId == Stickers::CloudRecentSetId
|
if (data.setId == Stickers::CloudRecentSetId
|
||||||
|| data.setId == Stickers::RecentSetId) {
|
|| data.setId == Stickers::RecentSetId) {
|
||||||
|
@ -3152,7 +3142,7 @@ void ApiWrap::toggleFavedSticker(
|
||||||
if (error.code() == 400
|
if (error.code() == 400
|
||||||
&& error.type().startsWith(qstr("FILE_REFERENCE_"))) {
|
&& error.type().startsWith(qstr("FILE_REFERENCE_"))) {
|
||||||
const auto current = document->fileReference();
|
const auto current = document->fileReference();
|
||||||
auto refreshed = [=](const Data::UpdatedFileReferences &data) {
|
auto refreshed = [=](const UpdatedFileReferences &data) {
|
||||||
if (document->fileReference() != current) {
|
if (document->fileReference() != current) {
|
||||||
performRequest();
|
performRequest();
|
||||||
}
|
}
|
||||||
|
@ -3190,7 +3180,7 @@ void ApiWrap::toggleSavedGif(
|
||||||
if (error.code() == 400
|
if (error.code() == 400
|
||||||
&& error.type().startsWith(qstr("FILE_REFERENCE_"))) {
|
&& error.type().startsWith(qstr("FILE_REFERENCE_"))) {
|
||||||
const auto current = document->fileReference();
|
const auto current = document->fileReference();
|
||||||
auto refreshed = [=](const Data::UpdatedFileReferences &data) {
|
auto refreshed = [=](const UpdatedFileReferences &data) {
|
||||||
if (document->fileReference() != current) {
|
if (document->fileReference() != current) {
|
||||||
performRequest();
|
performRequest();
|
||||||
}
|
}
|
||||||
|
@ -4937,7 +4927,7 @@ void ApiWrap::sendExistingDocument(
|
||||||
if (error.code() == 400
|
if (error.code() == 400
|
||||||
&& error.type().startsWith(qstr("FILE_REFERENCE_"))) {
|
&& error.type().startsWith(qstr("FILE_REFERENCE_"))) {
|
||||||
const auto current = document->fileReference();
|
const auto current = document->fileReference();
|
||||||
auto refreshed = [=](const Data::UpdatedFileReferences &data) {
|
auto refreshed = [=](const UpdatedFileReferences &data) {
|
||||||
if (document->fileReference() != current) {
|
if (document->fileReference() != current) {
|
||||||
performRequest();
|
performRequest();
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -228,100 +228,6 @@ namespace App {
|
||||||
return feedMsgs(msgs.v, type);
|
return feedMsgs(msgs.v, type);
|
||||||
}
|
}
|
||||||
|
|
||||||
ImagePtr image(const MTPPhotoSize &size) {
|
|
||||||
switch (size.type()) {
|
|
||||||
case mtpc_photoSize: {
|
|
||||||
auto &d = size.c_photoSize();
|
|
||||||
if (d.vlocation.type() == mtpc_fileLocation) {
|
|
||||||
auto &l = d.vlocation.c_fileLocation();
|
|
||||||
return Images::Create(
|
|
||||||
StorageImageLocation(
|
|
||||||
d.vw.v,
|
|
||||||
d.vh.v,
|
|
||||||
l.vdc_id.v,
|
|
||||||
l.vvolume_id.v,
|
|
||||||
l.vlocal_id.v,
|
|
||||||
l.vsecret.v,
|
|
||||||
l.vfile_reference.v),
|
|
||||||
d.vsize.v);
|
|
||||||
}
|
|
||||||
} break;
|
|
||||||
case mtpc_photoCachedSize: {
|
|
||||||
auto &d = size.c_photoCachedSize();
|
|
||||||
if (d.vlocation.type() == mtpc_fileLocation) {
|
|
||||||
auto &l = d.vlocation.c_fileLocation();
|
|
||||||
auto bytes = qba(d.vbytes);
|
|
||||||
return Images::Create(
|
|
||||||
StorageImageLocation(
|
|
||||||
d.vw.v,
|
|
||||||
d.vh.v,
|
|
||||||
l.vdc_id.v,
|
|
||||||
l.vvolume_id.v,
|
|
||||||
l.vlocal_id.v,
|
|
||||||
l.vsecret.v,
|
|
||||||
l.vfile_reference.v),
|
|
||||||
bytes);
|
|
||||||
} else if (d.vlocation.type() == mtpc_fileLocationUnavailable) {
|
|
||||||
const auto bytes = qba(d.vbytes);
|
|
||||||
if (auto image = App::readImage(bytes); !image.isNull()) {
|
|
||||||
return Images::Create(std::move(image), "JPG");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} break;
|
|
||||||
case mtpc_photoStrippedSize: {
|
|
||||||
const auto &d = size.c_photoStrippedSize();
|
|
||||||
auto bytes = qba(d.vbytes);
|
|
||||||
if (bytes.size() >= 3 && bytes[0] == '\x01') {
|
|
||||||
const char header[] = "\xff\xd8\xff\xe0\x00\x10\x4a\x46\x49"
|
|
||||||
"\x46\x00\x01\x01\x00\x00\x01\x00\x01\x00\x00\xff\xdb\x00\x43\x00\x28\x1c"
|
|
||||||
"\x1e\x23\x1e\x19\x28\x23\x21\x23\x2d\x2b\x28\x30\x3c\x64\x41\x3c\x37\x37"
|
|
||||||
"\x3c\x7b\x58\x5d\x49\x64\x91\x80\x99\x96\x8f\x80\x8c\x8a\xa0\xb4\xe6\xc3"
|
|
||||||
"\xa0\xaa\xda\xad\x8a\x8c\xc8\xff\xcb\xda\xee\xf5\xff\xff\xff\x9b\xc1\xff"
|
|
||||||
"\xff\xff\xfa\xff\xe6\xfd\xff\xf8\xff\xdb\x00\x43\x01\x2b\x2d\x2d\x3c\x35"
|
|
||||||
"\x3c\x76\x41\x41\x76\xf8\xa5\x8c\xa5\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8"
|
|
||||||
"\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8"
|
|
||||||
"\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8"
|
|
||||||
"\xf8\xf8\xf8\xf8\xf8\xff\xc0\x00\x11\x08\x00\x00\x00\x00\x03\x01\x22\x00"
|
|
||||||
"\x02\x11\x01\x03\x11\x01\xff\xc4\x00\x1f\x00\x00\x01\x05\x01\x01\x01\x01"
|
|
||||||
"\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x01\x02\x03\x04\x05\x06\x07\x08"
|
|
||||||
"\x09\x0a\x0b\xff\xc4\x00\xb5\x10\x00\x02\x01\x03\x03\x02\x04\x03\x05\x05"
|
|
||||||
"\x04\x04\x00\x00\x01\x7d\x01\x02\x03\x00\x04\x11\x05\x12\x21\x31\x41\x06"
|
|
||||||
"\x13\x51\x61\x07\x22\x71\x14\x32\x81\x91\xa1\x08\x23\x42\xb1\xc1\x15\x52"
|
|
||||||
"\xd1\xf0\x24\x33\x62\x72\x82\x09\x0a\x16\x17\x18\x19\x1a\x25\x26\x27\x28"
|
|
||||||
"\x29\x2a\x34\x35\x36\x37\x38\x39\x3a\x43\x44\x45\x46\x47\x48\x49\x4a\x53"
|
|
||||||
"\x54\x55\x56\x57\x58\x59\x5a\x63\x64\x65\x66\x67\x68\x69\x6a\x73\x74\x75"
|
|
||||||
"\x76\x77\x78\x79\x7a\x83\x84\x85\x86\x87\x88\x89\x8a\x92\x93\x94\x95\x96"
|
|
||||||
"\x97\x98\x99\x9a\xa2\xa3\xa4\xa5\xa6\xa7\xa8\xa9\xaa\xb2\xb3\xb4\xb5\xb6"
|
|
||||||
"\xb7\xb8\xb9\xba\xc2\xc3\xc4\xc5\xc6\xc7\xc8\xc9\xca\xd2\xd3\xd4\xd5\xd6"
|
|
||||||
"\xd7\xd8\xd9\xda\xe1\xe2\xe3\xe4\xe5\xe6\xe7\xe8\xe9\xea\xf1\xf2\xf3\xf4"
|
|
||||||
"\xf5\xf6\xf7\xf8\xf9\xfa\xff\xc4\x00\x1f\x01\x00\x03\x01\x01\x01\x01\x01"
|
|
||||||
"\x01\x01\x01\x01\x00\x00\x00\x00\x00\x00\x01\x02\x03\x04\x05\x06\x07\x08"
|
|
||||||
"\x09\x0a\x0b\xff\xc4\x00\xb5\x11\x00\x02\x01\x02\x04\x04\x03\x04\x07\x05"
|
|
||||||
"\x04\x04\x00\x01\x02\x77\x00\x01\x02\x03\x11\x04\x05\x21\x31\x06\x12\x41"
|
|
||||||
"\x51\x07\x61\x71\x13\x22\x32\x81\x08\x14\x42\x91\xa1\xb1\xc1\x09\x23\x33"
|
|
||||||
"\x52\xf0\x15\x62\x72\xd1\x0a\x16\x24\x34\xe1\x25\xf1\x17\x18\x19\x1a\x26"
|
|
||||||
"\x27\x28\x29\x2a\x35\x36\x37\x38\x39\x3a\x43\x44\x45\x46\x47\x48\x49\x4a"
|
|
||||||
"\x53\x54\x55\x56\x57\x58\x59\x5a\x63\x64\x65\x66\x67\x68\x69\x6a\x73\x74"
|
|
||||||
"\x75\x76\x77\x78\x79\x7a\x82\x83\x84\x85\x86\x87\x88\x89\x8a\x92\x93\x94"
|
|
||||||
"\x95\x96\x97\x98\x99\x9a\xa2\xa3\xa4\xa5\xa6\xa7\xa8\xa9\xaa\xb2\xb3\xb4"
|
|
||||||
"\xb5\xb6\xb7\xb8\xb9\xba\xc2\xc3\xc4\xc5\xc6\xc7\xc8\xc9\xca\xd2\xd3\xd4"
|
|
||||||
"\xd5\xd6\xd7\xd8\xd9\xda\xe2\xe3\xe4\xe5\xe6\xe7\xe8\xe9\xea\xf2\xf3\xf4"
|
|
||||||
"\xf5\xf6\xf7\xf8\xf9\xfa\xff\xda\x00\x0c\x03\x01\x00\x02\x11\x03\x11\x00"
|
|
||||||
"\x3f\x00";
|
|
||||||
const char footer[] = "\xff\xd9";
|
|
||||||
auto real = QByteArray(header, sizeof(header) - 1);
|
|
||||||
real[164] = bytes[1];
|
|
||||||
real[166] = bytes[2];
|
|
||||||
bytes = real + bytes.mid(3) + QByteArray::fromRawData(footer, sizeof(footer) - 1);
|
|
||||||
if (auto image = App::readImage(bytes); !image.isNull()) {
|
|
||||||
return Images::Create(std::move(image), "JPG");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} break;
|
|
||||||
}
|
|
||||||
return ImagePtr();
|
|
||||||
}
|
|
||||||
|
|
||||||
void feedInboxRead(const PeerId &peer, MsgId upTo) {
|
void feedInboxRead(const PeerId &peer, MsgId upTo) {
|
||||||
if (const auto history = Auth().data().historyLoaded(peer)) {
|
if (const auto history = Auth().data().historyLoaded(peer)) {
|
||||||
history->inboxRead(upTo);
|
history->inboxRead(upTo);
|
||||||
|
|
|
@ -77,8 +77,6 @@ namespace App {
|
||||||
void feedWereDeleted(ChannelId channelId, const QVector<MTPint> &msgsIds);
|
void feedWereDeleted(ChannelId channelId, const QVector<MTPint> &msgsIds);
|
||||||
void feedUserLink(MTPint userId, const MTPContactLink &myLink, const MTPContactLink &foreignLink);
|
void feedUserLink(MTPint userId, const MTPContactLink &myLink, const MTPContactLink &foreignLink);
|
||||||
|
|
||||||
ImagePtr image(const MTPPhotoSize &size);
|
|
||||||
|
|
||||||
[[nodiscard]] QString peerName(const PeerData *peer, bool forDialogs = false);
|
[[nodiscard]] QString peerName(const PeerData *peer, bool forDialogs = false);
|
||||||
|
|
||||||
[[nodiscard]] HistoryItem *histItemById(ChannelId channelId, MsgId itemId);
|
[[nodiscard]] HistoryItem *histItemById(ChannelId channelId, MsgId itemId);
|
||||||
|
|
|
@ -25,6 +25,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "window/window_controller.h"
|
#include "window/window_controller.h"
|
||||||
#include "storage/localstorage.h"
|
#include "storage/localstorage.h"
|
||||||
#include "data/data_session.h"
|
#include "data/data_session.h"
|
||||||
|
#include "data/data_photo.h"
|
||||||
#include "data/data_channel.h"
|
#include "data/data_channel.h"
|
||||||
#include "data/data_chat.h"
|
#include "data/data_chat.h"
|
||||||
#include "data/data_user.h"
|
#include "data/data_user.h"
|
||||||
|
@ -802,24 +803,17 @@ ConfirmInviteBox::ConfirmInviteBox(
|
||||||
}();
|
}();
|
||||||
_title->setText(title);
|
_title->setText(title);
|
||||||
_status->setText(status);
|
_status->setText(status);
|
||||||
if (data.vphoto.type() == mtpc_chatPhoto) {
|
|
||||||
const auto &photo = data.vphoto.c_chatPhoto();
|
const auto photo = Auth().data().processPhoto(data.vphoto);
|
||||||
const auto size = 160;
|
if (!photo->isNull()) {
|
||||||
const auto location = StorageImageLocation::FromMTP(
|
_photo = photo->thumbnail();
|
||||||
size,
|
if (!_photo->loaded()) {
|
||||||
size,
|
subscribe(Auth().downloaderTaskFinished(), [=] {
|
||||||
photo.vphoto_small);
|
update();
|
||||||
if (!location.isNull()) {
|
});
|
||||||
_photo = Images::Create(location);
|
_photo->load(Data::FileOrigin());
|
||||||
if (!_photo->loaded()) {
|
|
||||||
subscribe(Auth().downloaderTaskFinished(), [=] {
|
|
||||||
update();
|
|
||||||
});
|
|
||||||
_photo->load(Data::FileOrigin());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
if (!_photo) {
|
|
||||||
_photoEmpty = std::make_unique<Ui::EmptyUserpic>(
|
_photoEmpty = std::make_unique<Ui::EmptyUserpic>(
|
||||||
Data::PeerUserpicColor(0),
|
Data::PeerUserpicColor(0),
|
||||||
title);
|
title);
|
||||||
|
|
|
@ -211,7 +211,7 @@ private:
|
||||||
Fn<void()> _submit;
|
Fn<void()> _submit;
|
||||||
object_ptr<Ui::FlatLabel> _title;
|
object_ptr<Ui::FlatLabel> _title;
|
||||||
object_ptr<Ui::FlatLabel> _status;
|
object_ptr<Ui::FlatLabel> _status;
|
||||||
ImagePtr _photo;
|
Image *_photo = nullptr;
|
||||||
std::unique_ptr<Ui::EmptyUserpic> _photoEmpty;
|
std::unique_ptr<Ui::EmptyUserpic> _photoEmpty;
|
||||||
std::vector<not_null<UserData*>> _participants;
|
std::vector<not_null<UserData*>> _participants;
|
||||||
bool _isChannel = false;
|
bool _isChannel = false;
|
||||||
|
|
|
@ -225,7 +225,7 @@ void StickerSetBox::Inner::gotSet(const MTPmessages_StickerSet &set) {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
data.vset.match([&](const MTPDstickerSet & set) {
|
data.vset.match([&](const MTPDstickerSet &set) {
|
||||||
_setTitle = Stickers::GetSetTitle(set);
|
_setTitle = Stickers::GetSetTitle(set);
|
||||||
_setShortName = qs(set.vshort_name);
|
_setShortName = qs(set.vshort_name);
|
||||||
_setId = set.vid.v;
|
_setId = set.vid.v;
|
||||||
|
@ -237,7 +237,7 @@ void StickerSetBox::Inner::gotSet(const MTPmessages_StickerSet &set) {
|
||||||
? set.vinstalled_date.v
|
? set.vinstalled_date.v
|
||||||
: TimeId(0);
|
: TimeId(0);
|
||||||
_setThumbnail = set.has_thumb()
|
_setThumbnail = set.has_thumb()
|
||||||
? App::image(set.vthumb)
|
? Images::Create(set, set.vthumb)
|
||||||
: ImagePtr();
|
: ImagePtr();
|
||||||
auto &sets = Auth().data().stickerSetsRef();
|
auto &sets = Auth().data().stickerSetsRef();
|
||||||
const auto it = sets.find(_setId);
|
const auto it = sets.find(_setId);
|
||||||
|
|
|
@ -90,7 +90,8 @@ bool ApplyArchivedResultFake() {
|
||||||
MTP_long(set.access),
|
MTP_long(set.access),
|
||||||
MTP_string(set.title),
|
MTP_string(set.title),
|
||||||
MTP_string(set.shortName),
|
MTP_string(set.shortName),
|
||||||
MTP_photoSizeEmpty(MTP_string("a")),
|
MTP_photoSizeEmpty(MTP_string(QString())),
|
||||||
|
MTP_int(0),
|
||||||
MTP_int(set.count),
|
MTP_int(set.count),
|
||||||
MTP_int(set.hash));
|
MTP_int(set.hash));
|
||||||
sets.push_back(MTP_stickerSetCovered(
|
sets.push_back(MTP_stickerSetCovered(
|
||||||
|
@ -595,7 +596,7 @@ void FeaturedSetsReceived(
|
||||||
? set->vinstalled_date.v
|
? set->vinstalled_date.v
|
||||||
: TimeId(0);
|
: TimeId(0);
|
||||||
const auto thumbnail = set->has_thumb()
|
const auto thumbnail = set->has_thumb()
|
||||||
? App::image(set->vthumb)
|
? Images::Create(*set, set->vthumb)
|
||||||
: ImagePtr();
|
: ImagePtr();
|
||||||
if (it == sets.cend()) {
|
if (it == sets.cend()) {
|
||||||
auto setClientFlags = MTPDstickerSet_ClientFlag::f_featured
|
auto setClientFlags = MTPDstickerSet_ClientFlag::f_featured
|
||||||
|
@ -885,7 +886,7 @@ Set *FeedSet(const MTPDstickerSet &set) {
|
||||||
set.vhash.v,
|
set.vhash.v,
|
||||||
set.vflags.v | MTPDstickerSet_ClientFlag::f_not_loaded,
|
set.vflags.v | MTPDstickerSet_ClientFlag::f_not_loaded,
|
||||||
set.has_installed_date() ? set.vinstalled_date.v : TimeId(0),
|
set.has_installed_date() ? set.vinstalled_date.v : TimeId(0),
|
||||||
set.has_thumb() ? App::image(set.vthumb) : ImagePtr()));
|
set.has_thumb() ? Images::Create(set, set.vthumb) : ImagePtr()));
|
||||||
} else {
|
} else {
|
||||||
it->access = set.vaccess_hash.v;
|
it->access = set.vaccess_hash.v;
|
||||||
it->title = title;
|
it->title = title;
|
||||||
|
@ -901,7 +902,7 @@ Set *FeedSet(const MTPDstickerSet &set) {
|
||||||
? (set.vinstalled_date.v ? set.vinstalled_date.v : unixtime())
|
? (set.vinstalled_date.v ? set.vinstalled_date.v : unixtime())
|
||||||
: TimeId(0);
|
: TimeId(0);
|
||||||
it->thumbnail = set.has_thumb()
|
it->thumbnail = set.has_thumb()
|
||||||
? App::image(set.vthumb)
|
? Images::Create(set, set.vthumb)
|
||||||
: ImagePtr();
|
: ImagePtr();
|
||||||
if (it->count != set.vcount.v
|
if (it->count != set.vcount.v
|
||||||
|| it->hash != set.vhash.v
|
|| it->hash != set.vhash.v
|
||||||
|
|
|
@ -64,12 +64,11 @@ void ChannelData::setPhoto(const MTPChatPhoto &photo) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChannelData::setPhoto(PhotoId photoId, const MTPChatPhoto &photo) {
|
void ChannelData::setPhoto(PhotoId photoId, const MTPChatPhoto &photo) {
|
||||||
if (photo.type() == mtpc_chatPhoto) {
|
photo.match([&](const MTPDchatPhoto & data) {
|
||||||
const auto &data = photo.c_chatPhoto();
|
updateUserpic(photoId, data.vdc_id.v, data.vphoto_small);
|
||||||
updateUserpic(photoId, data.vphoto_small);
|
}, [&](const MTPDchatPhotoEmpty &) {
|
||||||
} else {
|
|
||||||
clearUserpic();
|
clearUserpic();
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChannelData::setName(const QString &newName, const QString &newUsername) {
|
void ChannelData::setName(const QString &newName, const QString &newUsername) {
|
||||||
|
|
|
@ -31,12 +31,11 @@ void ChatData::setPhoto(const MTPChatPhoto &photo) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChatData::setPhoto(PhotoId photoId, const MTPChatPhoto &photo) {
|
void ChatData::setPhoto(PhotoId photoId, const MTPChatPhoto &photo) {
|
||||||
if (photo.type() == mtpc_chatPhoto) {
|
photo.match([&](const MTPDchatPhoto &data) {
|
||||||
const auto &data = photo.c_chatPhoto();
|
updateUserpic(photoId, data.vdc_id.v, data.vphoto_small);
|
||||||
updateUserpic(photoId, data.vphoto_small);
|
}, [&](const MTPDchatPhotoEmpty &) {
|
||||||
} else {
|
|
||||||
clearUserpic();
|
clearUserpic();
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
auto ChatData::DefaultAdminRights() -> AdminRights {
|
auto ChatData::DefaultAdminRights() -> AdminRights {
|
||||||
|
|
|
@ -1199,7 +1199,8 @@ auto DocumentData::createStreamingLoader(Data::FileOrigin origin) const
|
||||||
MTP_inputDocumentFileLocation(
|
MTP_inputDocumentFileLocation(
|
||||||
MTP_long(id),
|
MTP_long(id),
|
||||||
MTP_long(_access),
|
MTP_long(_access),
|
||||||
MTP_bytes(_fileReference)),
|
MTP_bytes(_fileReference),
|
||||||
|
MTP_string(QString())),
|
||||||
size,
|
size,
|
||||||
origin)
|
origin)
|
||||||
: nullptr;
|
: nullptr;
|
||||||
|
@ -1229,6 +1230,10 @@ QByteArray DocumentData::fileReference() const {
|
||||||
|
|
||||||
void DocumentData::refreshFileReference(const QByteArray &value) {
|
void DocumentData::refreshFileReference(const QByteArray &value) {
|
||||||
_fileReference = value;
|
_fileReference = value;
|
||||||
|
_thumbnail->refreshFileReference(value);
|
||||||
|
if (const auto data = sticker()) {
|
||||||
|
data->loc.refreshFileReference(value);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void DocumentData::refreshStickerThumbFileReference() {
|
void DocumentData::refreshStickerThumbFileReference() {
|
||||||
|
|
|
@ -212,7 +212,7 @@ int GoodThumbSource::loadOffset() {
|
||||||
}
|
}
|
||||||
|
|
||||||
const StorageImageLocation &GoodThumbSource::location() {
|
const StorageImageLocation &GoodThumbSource::location() {
|
||||||
return StorageImageLocation::Null;
|
return StorageImageLocation::Invalid();
|
||||||
}
|
}
|
||||||
|
|
||||||
void GoodThumbSource::refreshFileReference(const QByteArray &data) {
|
void GoodThumbSource::refreshFileReference(const QByteArray &data) {
|
||||||
|
|
|
@ -17,37 +17,18 @@ struct FileReferenceAccumulator {
|
||||||
push(item);
|
push(item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void push(const MTPFileLocation &data) {
|
|
||||||
data.match([&](const MTPDfileLocation &data) {
|
|
||||||
result.data.emplace(SimpleFileLocationId(
|
|
||||||
data.vvolume_id.v,
|
|
||||||
data.vdc_id.v,
|
|
||||||
data.vlocal_id.v), data.vfile_reference.v);
|
|
||||||
}, [](const MTPDfileLocationUnavailable &data) {
|
|
||||||
});
|
|
||||||
}
|
|
||||||
void push(const MTPPhotoSize &data) {
|
|
||||||
data.match([](const MTPDphotoSizeEmpty &) {
|
|
||||||
}, [](const MTPDphotoStrippedSize &) {
|
|
||||||
}, [&](const auto &data) {
|
|
||||||
push(data.vlocation);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
void push(const MTPPhoto &data) {
|
void push(const MTPPhoto &data) {
|
||||||
data.match([&](const MTPDphoto &data) {
|
data.match([&](const MTPDphoto &data) {
|
||||||
for (const auto &size : data.vsizes.v) {
|
result.data.emplace(
|
||||||
push(size);
|
PhotoFileLocationId{ data.vid.v },
|
||||||
}
|
data.vfile_reference.v);
|
||||||
}, [](const MTPDphotoEmpty &data) {
|
}, [](const MTPDphotoEmpty &data) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
void push(const MTPDocument &data) {
|
void push(const MTPDocument &data) {
|
||||||
data.match([&](const MTPDdocument &data) {
|
data.match([&](const MTPDdocument &data) {
|
||||||
for (const auto &thumb : data.vthumbs.v) {
|
|
||||||
push(thumb);
|
|
||||||
}
|
|
||||||
result.data.emplace(
|
result.data.emplace(
|
||||||
DocumentFileLocationId(data.vid.v),
|
DocumentFileLocationId{ data.vid.v },
|
||||||
data.vfile_reference.v);
|
data.vfile_reference.v);
|
||||||
}, [](const MTPDdocumentEmpty &data) {
|
}, [](const MTPDdocumentEmpty &data) {
|
||||||
});
|
});
|
||||||
|
@ -57,36 +38,6 @@ struct FileReferenceAccumulator {
|
||||||
push(data.vdocument);
|
push(data.vdocument);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
void push(const MTPUserProfilePhoto &data) {
|
|
||||||
data.match([&](const MTPDuserProfilePhoto &data) {
|
|
||||||
push(data.vphoto_small);
|
|
||||||
push(data.vphoto_big);
|
|
||||||
}, [](const MTPDuserProfilePhotoEmpty &data) {
|
|
||||||
});
|
|
||||||
}
|
|
||||||
void push(const MTPChatPhoto &data) {
|
|
||||||
data.match([&](const MTPDchatPhoto &data) {
|
|
||||||
push(data.vphoto_small);
|
|
||||||
push(data.vphoto_big);
|
|
||||||
}, [](const MTPDchatPhotoEmpty &data) {
|
|
||||||
});
|
|
||||||
}
|
|
||||||
void push(const MTPUser &data) {
|
|
||||||
data.match([&](const MTPDuser &data) {
|
|
||||||
if (data.has_photo()) {
|
|
||||||
push(data.vphoto);
|
|
||||||
}
|
|
||||||
}, [](const MTPDuserEmpty &data) {
|
|
||||||
});
|
|
||||||
}
|
|
||||||
void push(const MTPChat &data) {
|
|
||||||
data.match([](const MTPDchatEmpty &data) {
|
|
||||||
}, [](const MTPDchannelForbidden &data) {
|
|
||||||
}, [](const MTPDchatForbidden &data) {
|
|
||||||
}, [&](const auto &data) {
|
|
||||||
push(data.vphoto);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
void push(const MTPWebPage &data) {
|
void push(const MTPWebPage &data) {
|
||||||
data.match([&](const MTPDwebPage &data) {
|
data.match([&](const MTPDwebPage &data) {
|
||||||
if (data.has_document()) {
|
if (data.has_document()) {
|
||||||
|
@ -139,22 +90,14 @@ struct FileReferenceAccumulator {
|
||||||
void push(const MTPmessages_Messages &data) {
|
void push(const MTPmessages_Messages &data) {
|
||||||
data.match([](const MTPDmessages_messagesNotModified &) {
|
data.match([](const MTPDmessages_messagesNotModified &) {
|
||||||
}, [&](const auto &data) {
|
}, [&](const auto &data) {
|
||||||
push(data.vusers);
|
|
||||||
push(data.vchats);
|
|
||||||
push(data.vmessages);
|
push(data.vmessages);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
void push(const MTPphotos_Photos &data) {
|
void push(const MTPphotos_Photos &data) {
|
||||||
data.match([&](const auto &data) {
|
data.match([&](const auto &data) {
|
||||||
push(data.vusers);
|
|
||||||
push(data.vphotos);
|
push(data.vphotos);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
void push(const MTPmessages_Chats &data) {
|
|
||||||
data.match([&](const auto &data) {
|
|
||||||
push(data.vchats);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
void push(const MTPmessages_RecentStickers &data) {
|
void push(const MTPmessages_RecentStickers &data) {
|
||||||
data.match([&](const MTPDmessages_recentStickers &data) {
|
data.match([&](const MTPDmessages_recentStickers &data) {
|
||||||
push(data.vstickers);
|
push(data.vstickers);
|
||||||
|
@ -191,22 +134,6 @@ UpdatedFileReferences GetFileReferencesHelper(const Type &data) {
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
SimpleFileLocationId::SimpleFileLocationId(
|
|
||||||
uint64 volumeId,
|
|
||||||
int32 dcId,
|
|
||||||
int32 localId)
|
|
||||||
: volumeId(volumeId)
|
|
||||||
, dcId(dcId)
|
|
||||||
, localId(localId) {
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator<(
|
|
||||||
const SimpleFileLocationId &a,
|
|
||||||
const SimpleFileLocationId &b) {
|
|
||||||
return std::tie(a.volumeId, a.dcId, a.localId)
|
|
||||||
< std::tie(b.volumeId, b.dcId, b.localId);
|
|
||||||
}
|
|
||||||
|
|
||||||
UpdatedFileReferences GetFileReferences(const MTPmessages_Messages &data) {
|
UpdatedFileReferences GetFileReferences(const MTPmessages_Messages &data) {
|
||||||
return GetFileReferencesHelper(data);
|
return GetFileReferencesHelper(data);
|
||||||
}
|
}
|
||||||
|
@ -215,14 +142,6 @@ UpdatedFileReferences GetFileReferences(const MTPphotos_Photos &data) {
|
||||||
return GetFileReferencesHelper(data);
|
return GetFileReferencesHelper(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
UpdatedFileReferences GetFileReferences(const MTPVector<MTPUser> &data) {
|
|
||||||
return GetFileReferencesHelper(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
UpdatedFileReferences GetFileReferences(const MTPmessages_Chats &data) {
|
|
||||||
return GetFileReferencesHelper(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
UpdatedFileReferences GetFileReferences(
|
UpdatedFileReferences GetFileReferences(
|
||||||
const MTPmessages_RecentStickers &data) {
|
const MTPmessages_RecentStickers &data) {
|
||||||
return GetFileReferencesHelper(data);
|
return GetFileReferencesHelper(data);
|
||||||
|
|
|
@ -107,31 +107,32 @@ struct FileOrigin {
|
||||||
Variant data;
|
Variant data;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Volume_id, dc_id, local_id.
|
struct DocumentFileLocationId {
|
||||||
struct SimpleFileLocationId {
|
uint64 id = 0;
|
||||||
SimpleFileLocationId(uint64 volumeId, int32 dcId, int32 localId);
|
|
||||||
|
|
||||||
uint64 volumeId = 0;
|
|
||||||
int32 dcId = 0;
|
|
||||||
int32 localId = 0;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
bool operator<(
|
inline bool operator<(DocumentFileLocationId a, DocumentFileLocationId b) {
|
||||||
const SimpleFileLocationId &a,
|
return a.id < b.id;
|
||||||
const SimpleFileLocationId &b);
|
}
|
||||||
|
|
||||||
|
struct PhotoFileLocationId {
|
||||||
|
uint64 id = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
inline bool operator<(PhotoFileLocationId a, PhotoFileLocationId b) {
|
||||||
|
return a.id < b.id;
|
||||||
|
}
|
||||||
|
|
||||||
using DocumentFileLocationId = uint64;
|
|
||||||
using FileLocationId = base::variant<
|
using FileLocationId = base::variant<
|
||||||
SimpleFileLocationId,
|
DocumentFileLocationId,
|
||||||
DocumentFileLocationId>;
|
PhotoFileLocationId>;
|
||||||
|
|
||||||
struct UpdatedFileReferences {
|
struct UpdatedFileReferences {
|
||||||
std::map<FileLocationId, QByteArray> data;
|
std::map<FileLocationId, QByteArray> data;
|
||||||
};
|
};
|
||||||
|
|
||||||
UpdatedFileReferences GetFileReferences(const MTPmessages_Messages &data);
|
UpdatedFileReferences GetFileReferences(const MTPmessages_Messages &data);
|
||||||
UpdatedFileReferences GetFileReferences(const MTPphotos_Photos &data);
|
UpdatedFileReferences GetFileReferences(const MTPphotos_Photos &data);
|
||||||
UpdatedFileReferences GetFileReferences(const MTPVector<MTPUser> &data);
|
|
||||||
UpdatedFileReferences GetFileReferences(const MTPmessages_Chats &data);
|
|
||||||
UpdatedFileReferences GetFileReferences(
|
UpdatedFileReferences GetFileReferences(
|
||||||
const MTPmessages_RecentStickers &data);
|
const MTPmessages_RecentStickers &data);
|
||||||
UpdatedFileReferences GetFileReferences(
|
UpdatedFileReferences GetFileReferences(
|
||||||
|
|
|
@ -362,29 +362,36 @@ bool MediaPhoto::updateSentMedia(const MTPMessageMedia &media) {
|
||||||
"or with ttl_seconds in updateSentMedia()"));
|
"or with ttl_seconds in updateSentMedia()"));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
const auto &photo = mediaPhoto.vphoto;
|
parent()->history()->owner().photoConvert(_photo, mediaPhoto.vphoto);
|
||||||
parent()->history()->owner().photoConvert(_photo, photo);
|
|
||||||
|
|
||||||
if (photo.type() != mtpc_photo) {
|
if (mediaPhoto.vphoto.type() != mtpc_photo) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
const auto &photo = mediaPhoto.vphoto.c_photo();
|
||||||
|
|
||||||
struct SizeData {
|
struct SizeData {
|
||||||
char letter = 0;
|
MTPstring type = MTP_string(QString());
|
||||||
int width = 0;
|
int width = 0;
|
||||||
int height = 0;
|
int height = 0;
|
||||||
const MTPFileLocation *location = nullptr;
|
|
||||||
QByteArray bytes;
|
QByteArray bytes;
|
||||||
};
|
};
|
||||||
const auto saveImageToCache = [&](
|
const auto saveImageToCache = [&](
|
||||||
not_null<Image*> image,
|
not_null<Image*> image,
|
||||||
SizeData size) {
|
SizeData size) {
|
||||||
Expects(size.location != nullptr);
|
Expects(!size.type.v.isEmpty());
|
||||||
|
|
||||||
const auto key = StorageImageLocation(
|
const auto key = StorageImageLocation(
|
||||||
|
StorageFileLocation(
|
||||||
|
photo.vdc_id.v,
|
||||||
|
_photo->session().userId(),
|
||||||
|
MTP_inputPhotoFileLocation(
|
||||||
|
photo.vid,
|
||||||
|
photo.vaccess_hash,
|
||||||
|
photo.vfile_reference,
|
||||||
|
size.type)),
|
||||||
size.width,
|
size.width,
|
||||||
size.height,
|
size.height);
|
||||||
size.location->c_fileLocation());
|
if (!key.valid() || image->isNull() || !image->loaded()) {
|
||||||
if (key.isNull() || image->isNull() || !image->loaded()) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (size.bytes.isEmpty()) {
|
if (size.bytes.isEmpty()) {
|
||||||
|
@ -403,24 +410,22 @@ bool MediaPhoto::updateSentMedia(const MTPMessageMedia &media) {
|
||||||
image->replaceSource(
|
image->replaceSource(
|
||||||
std::make_unique<Images::StorageSource>(key, length));
|
std::make_unique<Images::StorageSource>(key, length));
|
||||||
};
|
};
|
||||||
auto &sizes = photo.c_photo().vsizes.v;
|
auto &sizes = photo.vsizes.v;
|
||||||
auto max = 0;
|
auto max = 0;
|
||||||
auto maxSize = SizeData();
|
auto maxSize = SizeData();
|
||||||
for (const auto &data : sizes) {
|
for (const auto &data : sizes) {
|
||||||
const auto size = data.match([](const MTPDphotoSize &data) {
|
const auto size = data.match([](const MTPDphotoSize &data) {
|
||||||
return SizeData{
|
return SizeData{
|
||||||
data.vtype.v.isEmpty() ? char(0) : data.vtype.v[0],
|
data.vtype,
|
||||||
data.vw.v,
|
data.vw.v,
|
||||||
data.vh.v,
|
data.vh.v,
|
||||||
&data.vlocation,
|
|
||||||
QByteArray()
|
QByteArray()
|
||||||
};
|
};
|
||||||
}, [](const MTPDphotoCachedSize &data) {
|
}, [](const MTPDphotoCachedSize &data) {
|
||||||
return SizeData{
|
return SizeData{
|
||||||
data.vtype.v.isEmpty() ? char(0) : data.vtype.v[0],
|
data.vtype,
|
||||||
data.vw.v,
|
data.vw.v,
|
||||||
data.vh.v,
|
data.vh.v,
|
||||||
&data.vlocation,
|
|
||||||
qba(data.vbytes)
|
qba(data.vbytes)
|
||||||
};
|
};
|
||||||
}, [](const MTPDphotoSizeEmpty &) {
|
}, [](const MTPDphotoSizeEmpty &) {
|
||||||
|
@ -429,25 +434,26 @@ bool MediaPhoto::updateSentMedia(const MTPMessageMedia &media) {
|
||||||
// No need to save stripped images to local cache.
|
// No need to save stripped images to local cache.
|
||||||
return SizeData();
|
return SizeData();
|
||||||
});
|
});
|
||||||
if (!size.location || size.location->type() != mtpc_fileLocation) {
|
const auto letter = size.type.v.isEmpty() ? char(0) : size.type.v[0];
|
||||||
|
if (!letter) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (size.letter == 's') {
|
if (letter == 's') {
|
||||||
saveImageToCache(_photo->thumbnailSmall(), size);
|
saveImageToCache(_photo->thumbnailSmall(), size);
|
||||||
} else if (size.letter == 'm') {
|
} else if (letter == 'm') {
|
||||||
saveImageToCache(_photo->thumbnail(), size);
|
saveImageToCache(_photo->thumbnail(), size);
|
||||||
} else if (size.letter == 'x' && max < 1) {
|
} else if (letter == 'x' && max < 1) {
|
||||||
max = 1;
|
max = 1;
|
||||||
maxSize = size;
|
maxSize = size;
|
||||||
} else if (size.letter == 'y' && max < 2) {
|
} else if (letter == 'y' && max < 2) {
|
||||||
max = 2;
|
max = 2;
|
||||||
maxSize = size;
|
maxSize = size;
|
||||||
//} else if (size.letter == 'w' && max < 3) {
|
//} else if (letter == 'w' && max < 3) {
|
||||||
// max = 3;
|
// max = 3;
|
||||||
// maxSize = size;
|
// maxSize = size;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (maxSize.location) {
|
if (!maxSize.type.v.isEmpty()) {
|
||||||
saveImageToCache(_photo->large(), maxSize);
|
saveImageToCache(_photo->large(), maxSize);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -310,11 +310,24 @@ Data::FileOrigin PeerData::userpicPhotoOrigin() const {
|
||||||
|
|
||||||
void PeerData::updateUserpic(
|
void PeerData::updateUserpic(
|
||||||
PhotoId photoId,
|
PhotoId photoId,
|
||||||
|
MTP::DcId dcId,
|
||||||
const MTPFileLocation &location) {
|
const MTPFileLocation &location) {
|
||||||
const auto size = kUserpicSize;
|
const auto size = kUserpicSize;
|
||||||
const auto loc = StorageImageLocation::FromMTP(size, size, location);
|
const auto loc = location.match([&](
|
||||||
const auto photo = loc.isNull() ? ImagePtr() : Images::Create(loc);
|
const MTPDfileLocationToBeDeprecated &deprecated) {
|
||||||
setUserpicChecked(photoId, loc, photo);
|
return StorageImageLocation(
|
||||||
|
StorageFileLocation(
|
||||||
|
dcId,
|
||||||
|
session().userId(),
|
||||||
|
MTP_inputPeerPhotoFileLocation(
|
||||||
|
MTP_flags(0),
|
||||||
|
input,
|
||||||
|
deprecated.vvolume_id,
|
||||||
|
deprecated.vlocal_id)),
|
||||||
|
size,
|
||||||
|
size);
|
||||||
|
});
|
||||||
|
setUserpicChecked(photoId, loc, Images::Create(loc));
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerData::clearUserpic() {
|
void PeerData::clearUserpic() {
|
||||||
|
|
|
@ -303,7 +303,10 @@ protected:
|
||||||
const QString &newName,
|
const QString &newName,
|
||||||
const QString &newNameOrPhone,
|
const QString &newNameOrPhone,
|
||||||
const QString &newUsername);
|
const QString &newUsername);
|
||||||
void updateUserpic(PhotoId photoId, const MTPFileLocation &location);
|
void updateUserpic(
|
||||||
|
PhotoId photoId,
|
||||||
|
MTP::DcId dcId,
|
||||||
|
const MTPFileLocation &location);
|
||||||
void clearUserpic();
|
void clearUserpic();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -135,11 +135,33 @@ Image *PhotoData::getReplyPreview(Data::FileOrigin origin) {
|
||||||
return _replyPreview.image();
|
return _replyPreview.image();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void PhotoData::setRemoteLocation(
|
||||||
|
int32 dc,
|
||||||
|
uint64 access,
|
||||||
|
const QByteArray &fileReference) {
|
||||||
|
_fileReference = fileReference;
|
||||||
|
if (_dc != dc || _access != access) {
|
||||||
|
_dc = dc;
|
||||||
|
_access = access;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
MTPInputPhoto PhotoData::mtpInput() const {
|
MTPInputPhoto PhotoData::mtpInput() const {
|
||||||
return MTP_inputPhoto(
|
return MTP_inputPhoto(
|
||||||
MTP_long(id),
|
MTP_long(id),
|
||||||
MTP_long(access),
|
MTP_long(_access),
|
||||||
MTP_bytes(fileReference));
|
MTP_bytes(_fileReference));
|
||||||
|
}
|
||||||
|
|
||||||
|
QByteArray PhotoData::fileReference() const {
|
||||||
|
return _fileReference;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PhotoData::refreshFileReference(const QByteArray &value) {
|
||||||
|
_fileReference = value;
|
||||||
|
_thumbnailSmall->refreshFileReference(value);
|
||||||
|
_thumbnail->refreshFileReference(value);
|
||||||
|
_large->refreshFileReference(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
void PhotoData::collectLocalData(not_null<PhotoData*> local) {
|
void PhotoData::collectLocalData(not_null<PhotoData*> local) {
|
||||||
|
@ -206,7 +228,7 @@ void PhotoData::updateImages(
|
||||||
if (!was) {
|
if (!was) {
|
||||||
was = now;
|
was = now;
|
||||||
} else if (was->isDelayedStorageImage()) {
|
} else if (was->isDelayedStorageImage()) {
|
||||||
if (const auto location = now->location(); !location.isNull()) {
|
if (const auto location = now->location(); location.valid()) {
|
||||||
was->setDelayedStorageLocation(
|
was->setDelayedStorageLocation(
|
||||||
Data::FileOrigin(),
|
Data::FileOrigin(),
|
||||||
location);
|
location);
|
||||||
|
|
|
@ -42,7 +42,13 @@ public:
|
||||||
void unload();
|
void unload();
|
||||||
[[nodiscard]] Image *getReplyPreview(Data::FileOrigin origin);
|
[[nodiscard]] Image *getReplyPreview(Data::FileOrigin origin);
|
||||||
|
|
||||||
|
void setRemoteLocation(
|
||||||
|
int32 dc,
|
||||||
|
uint64 access,
|
||||||
|
const QByteArray &fileReference);
|
||||||
[[nodiscard]] MTPInputPhoto mtpInput() const;
|
[[nodiscard]] MTPInputPhoto mtpInput() const;
|
||||||
|
[[nodiscard]] QByteArray fileReference() const;
|
||||||
|
void refreshFileReference(const QByteArray &value);
|
||||||
|
|
||||||
// When we have some client-side generated photo
|
// When we have some client-side generated photo
|
||||||
// (for example for displaying an external inline bot result)
|
// (for example for displaying an external inline bot result)
|
||||||
|
@ -72,8 +78,6 @@ public:
|
||||||
ImagePtr large);
|
ImagePtr large);
|
||||||
|
|
||||||
PhotoId id = 0;
|
PhotoId id = 0;
|
||||||
uint64 access = 0;
|
|
||||||
QByteArray fileReference;
|
|
||||||
TimeId date = 0;
|
TimeId date = 0;
|
||||||
bool hasSticker = false;
|
bool hasSticker = false;
|
||||||
|
|
||||||
|
@ -88,6 +92,9 @@ private:
|
||||||
ImagePtr _thumbnail;
|
ImagePtr _thumbnail;
|
||||||
ImagePtr _large;
|
ImagePtr _large;
|
||||||
|
|
||||||
|
int32 _dc = 0;
|
||||||
|
uint64 _access = 0;
|
||||||
|
QByteArray _fileReference;
|
||||||
Data::ReplyPreview _replyPreview;
|
Data::ReplyPreview _replyPreview;
|
||||||
|
|
||||||
not_null<Data::Session*> _owner;
|
not_null<Data::Session*> _owner;
|
||||||
|
|
|
@ -1700,6 +1700,7 @@ not_null<PhotoData*> Session::processPhoto(
|
||||||
data.vaccess_hash.v,
|
data.vaccess_hash.v,
|
||||||
data.vfile_reference.v,
|
data.vfile_reference.v,
|
||||||
data.vdate.v,
|
data.vdate.v,
|
||||||
|
data.vdc_id.v,
|
||||||
data.is_has_stickers(),
|
data.is_has_stickers(),
|
||||||
thumbnailInline,
|
thumbnailInline,
|
||||||
thumbnailSmall,
|
thumbnailSmall,
|
||||||
|
@ -1715,6 +1716,7 @@ not_null<PhotoData*> Session::photo(
|
||||||
const uint64 &access,
|
const uint64 &access,
|
||||||
const QByteArray &fileReference,
|
const QByteArray &fileReference,
|
||||||
TimeId date,
|
TimeId date,
|
||||||
|
int32 dc,
|
||||||
bool hasSticker,
|
bool hasSticker,
|
||||||
const ImagePtr &thumbnailInline,
|
const ImagePtr &thumbnailInline,
|
||||||
const ImagePtr &thumbnailSmall,
|
const ImagePtr &thumbnailSmall,
|
||||||
|
@ -1726,6 +1728,7 @@ not_null<PhotoData*> Session::photo(
|
||||||
access,
|
access,
|
||||||
fileReference,
|
fileReference,
|
||||||
date,
|
date,
|
||||||
|
dc,
|
||||||
hasSticker,
|
hasSticker,
|
||||||
thumbnailInline,
|
thumbnailInline,
|
||||||
thumbnailSmall,
|
thumbnailSmall,
|
||||||
|
@ -1790,6 +1793,7 @@ PhotoData *Session::photoFromWeb(
|
||||||
uint64(0),
|
uint64(0),
|
||||||
QByteArray(),
|
QByteArray(),
|
||||||
unixtime(),
|
unixtime(),
|
||||||
|
0,
|
||||||
false,
|
false,
|
||||||
thumbnailInline,
|
thumbnailInline,
|
||||||
thumbnailSmall,
|
thumbnailSmall,
|
||||||
|
@ -1828,7 +1832,7 @@ void Session::photoApplyFields(
|
||||||
};
|
};
|
||||||
const auto image = [&](const QByteArray &levels) {
|
const auto image = [&](const QByteArray &levels) {
|
||||||
const auto i = find(levels);
|
const auto i = find(levels);
|
||||||
return (i == sizes.end()) ? ImagePtr() : App::image(*i);
|
return (i == sizes.end()) ? ImagePtr() : Images::Create(data, *i);
|
||||||
};
|
};
|
||||||
const auto thumbnailInline = image(InlineLevels);
|
const auto thumbnailInline = image(InlineLevels);
|
||||||
const auto thumbnailSmall = image(SmallLevels);
|
const auto thumbnailSmall = image(SmallLevels);
|
||||||
|
@ -1840,6 +1844,7 @@ void Session::photoApplyFields(
|
||||||
data.vaccess_hash.v,
|
data.vaccess_hash.v,
|
||||||
data.vfile_reference.v,
|
data.vfile_reference.v,
|
||||||
data.vdate.v,
|
data.vdate.v,
|
||||||
|
data.vdc_id.v,
|
||||||
data.is_has_stickers(),
|
data.is_has_stickers(),
|
||||||
thumbnailInline,
|
thumbnailInline,
|
||||||
thumbnailSmall,
|
thumbnailSmall,
|
||||||
|
@ -1853,6 +1858,7 @@ void Session::photoApplyFields(
|
||||||
const uint64 &access,
|
const uint64 &access,
|
||||||
const QByteArray &fileReference,
|
const QByteArray &fileReference,
|
||||||
TimeId date,
|
TimeId date,
|
||||||
|
int32 dc,
|
||||||
bool hasSticker,
|
bool hasSticker,
|
||||||
const ImagePtr &thumbnailInline,
|
const ImagePtr &thumbnailInline,
|
||||||
const ImagePtr &thumbnailSmall,
|
const ImagePtr &thumbnailSmall,
|
||||||
|
@ -1861,8 +1867,7 @@ void Session::photoApplyFields(
|
||||||
if (!date) {
|
if (!date) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
photo->access = access;
|
photo->setRemoteLocation(dc, access, fileReference);
|
||||||
photo->fileReference = fileReference;
|
|
||||||
photo->date = date;
|
photo->date = date;
|
||||||
photo->hasSticker = hasSticker;
|
photo->hasSticker = hasSticker;
|
||||||
photo->updateImages(
|
photo->updateImages(
|
||||||
|
@ -2061,7 +2066,8 @@ void Session::documentApplyFields(
|
||||||
not_null<DocumentData*> document,
|
not_null<DocumentData*> document,
|
||||||
const MTPDdocument &data) {
|
const MTPDdocument &data) {
|
||||||
const auto thumbnailInline = FindDocumentInlineThumbnail(data);
|
const auto thumbnailInline = FindDocumentInlineThumbnail(data);
|
||||||
const auto thumbnail = FindDocumentThumbnail(data);
|
const auto thumbnailSize = FindDocumentThumbnail(data);
|
||||||
|
const auto thumbnail = Images::Create(data, thumbnailSize);
|
||||||
documentApplyFields(
|
documentApplyFields(
|
||||||
document,
|
document,
|
||||||
data.vaccess_hash.v,
|
data.vaccess_hash.v,
|
||||||
|
@ -2069,11 +2075,11 @@ void Session::documentApplyFields(
|
||||||
data.vdate.v,
|
data.vdate.v,
|
||||||
data.vattributes.v,
|
data.vattributes.v,
|
||||||
qs(data.vmime_type),
|
qs(data.vmime_type),
|
||||||
App::image(thumbnailInline),
|
Images::Create(data, thumbnailInline),
|
||||||
App::image(thumbnail),
|
thumbnail,
|
||||||
data.vdc_id.v,
|
data.vdc_id.v,
|
||||||
data.vsize.v,
|
data.vsize.v,
|
||||||
StorageImageLocation::FromMTP(thumbnail));
|
thumbnail->location());
|
||||||
}
|
}
|
||||||
|
|
||||||
void Session::documentApplyFields(
|
void Session::documentApplyFields(
|
||||||
|
@ -2101,8 +2107,8 @@ void Session::documentApplyFields(
|
||||||
document->size = size;
|
document->size = size;
|
||||||
document->recountIsImage();
|
document->recountIsImage();
|
||||||
if (document->sticker()
|
if (document->sticker()
|
||||||
&& document->sticker()->loc.isNull()
|
&& !document->sticker()->loc.valid()
|
||||||
&& !thumbLocation.isNull()) {
|
&& thumbLocation.valid()) {
|
||||||
document->sticker()->loc = thumbLocation;
|
document->sticker()->loc = thumbLocation;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -347,6 +347,7 @@ public:
|
||||||
const uint64 &access,
|
const uint64 &access,
|
||||||
const QByteArray &fileReference,
|
const QByteArray &fileReference,
|
||||||
TimeId date,
|
TimeId date,
|
||||||
|
int32 dc,
|
||||||
bool hasSticker,
|
bool hasSticker,
|
||||||
const ImagePtr &thumbnailInline,
|
const ImagePtr &thumbnailInline,
|
||||||
const ImagePtr &thumbnailSmall,
|
const ImagePtr &thumbnailSmall,
|
||||||
|
@ -580,6 +581,7 @@ private:
|
||||||
const uint64 &access,
|
const uint64 &access,
|
||||||
const QByteArray &fileReference,
|
const QByteArray &fileReference,
|
||||||
TimeId date,
|
TimeId date,
|
||||||
|
int32 dc,
|
||||||
bool hasSticker,
|
bool hasSticker,
|
||||||
const ImagePtr &thumbnailInline,
|
const ImagePtr &thumbnailInline,
|
||||||
const ImagePtr &thumbnailSmall,
|
const ImagePtr &thumbnailSmall,
|
||||||
|
|
|
@ -62,11 +62,7 @@ Storage::Cache::Key DocumentThumbCacheKey(int32 dcId, uint64 id) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Storage::Cache::Key StorageCacheKey(const StorageImageLocation &location) {
|
Storage::Cache::Key StorageCacheKey(const StorageImageLocation &location) {
|
||||||
const auto dcId = uint64(location.dc()) & 0xFFULL;
|
return location.file().cacheKey();
|
||||||
return Storage::Cache::Key{
|
|
||||||
Data::kStorageCacheTag | (dcId << 32) | uint32(location.local()),
|
|
||||||
location.volume()
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Storage::Cache::Key WebDocumentCacheKey(const WebFileLocation &location) {
|
Storage::Cache::Key WebDocumentCacheKey(const WebFileLocation &location) {
|
||||||
|
|
|
@ -79,7 +79,7 @@ void UserData::setContactStatus(ContactStatus status) {
|
||||||
void UserData::setPhoto(const MTPUserProfilePhoto &photo) {
|
void UserData::setPhoto(const MTPUserProfilePhoto &photo) {
|
||||||
if (photo.type() == mtpc_userProfilePhoto) {
|
if (photo.type() == mtpc_userProfilePhoto) {
|
||||||
const auto &data = photo.c_userProfilePhoto();
|
const auto &data = photo.c_userProfilePhoto();
|
||||||
updateUserpic(data.vphoto_id.v, data.vphoto_small);
|
updateUserpic(data.vphoto_id.v, data.vdc_id.v, data.vphoto_small);
|
||||||
} else {
|
} else {
|
||||||
clearUserpic();
|
clearUserpic();
|
||||||
}
|
}
|
||||||
|
|
|
@ -238,7 +238,7 @@ void WebPageData::replaceDocumentGoodThumbnail() {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const auto &location = photo->large()->location();
|
const auto &location = photo->large()->location();
|
||||||
if (!location.isNull()) {
|
if (location.valid()) {
|
||||||
document->replaceGoodThumbnail(
|
document->replaceGoodThumbnail(
|
||||||
std::make_unique<Images::StorageSource>(
|
std::make_unique<Images::StorageSource>(
|
||||||
location,
|
location,
|
||||||
|
|
|
@ -203,36 +203,14 @@ Utf8String FillLeft(const Utf8String &data, int length, char filler) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
FileLocation ParseLocation(const MTPFileLocation &data) {
|
|
||||||
return data.match([](const MTPDfileLocation &data) {
|
|
||||||
return FileLocation{
|
|
||||||
data.vdc_id.v,
|
|
||||||
MTP_inputFileLocation(
|
|
||||||
data.vvolume_id,
|
|
||||||
data.vlocal_id,
|
|
||||||
data.vsecret,
|
|
||||||
data.vfile_reference)
|
|
||||||
};
|
|
||||||
}, [](const MTPDfileLocationUnavailable &data) {
|
|
||||||
return FileLocation{
|
|
||||||
0,
|
|
||||||
MTP_inputFileLocation(
|
|
||||||
data.vvolume_id,
|
|
||||||
data.vlocal_id,
|
|
||||||
data.vsecret,
|
|
||||||
MTP_bytes(QByteArray()))
|
|
||||||
};
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
Image ParseMaxImage(
|
Image ParseMaxImage(
|
||||||
const MTPVector<MTPPhotoSize> &data,
|
const MTPDphoto &photo,
|
||||||
const QString &suggestedPath) {
|
const QString &suggestedPath) {
|
||||||
auto result = Image();
|
auto result = Image();
|
||||||
result.file.suggestedPath = suggestedPath;
|
result.file.suggestedPath = suggestedPath;
|
||||||
|
|
||||||
auto maxArea = int64(0);
|
auto maxArea = int64(0);
|
||||||
for (const auto &size : data.v) {
|
for (const auto &size : photo.vsizes.v) {
|
||||||
size.match([](const MTPDphotoSizeEmpty &) {
|
size.match([](const MTPDphotoSizeEmpty &) {
|
||||||
}, [](const MTPDphotoStrippedSize &) {
|
}, [](const MTPDphotoStrippedSize &) {
|
||||||
// Max image size should not be a stripped image.
|
// Max image size should not be a stripped image.
|
||||||
|
@ -241,7 +219,13 @@ Image ParseMaxImage(
|
||||||
if (area > maxArea) {
|
if (area > maxArea) {
|
||||||
result.width = data.vw.v;
|
result.width = data.vw.v;
|
||||||
result.height = data.vh.v;
|
result.height = data.vh.v;
|
||||||
result.file.location = ParseLocation(data.vlocation);
|
result.file.location = FileLocation{
|
||||||
|
photo.vdc_id.v,
|
||||||
|
MTP_inputPhotoFileLocation(
|
||||||
|
photo.vid,
|
||||||
|
photo.vaccess_hash,
|
||||||
|
photo.vfile_reference,
|
||||||
|
data.vtype) };
|
||||||
if constexpr (MTPDphotoCachedSize::Is<decltype(data)>()) {
|
if constexpr (MTPDphotoCachedSize::Is<decltype(data)>()) {
|
||||||
result.file.content = data.vbytes.v;
|
result.file.content = data.vbytes.v;
|
||||||
result.file.size = result.file.content.size();
|
result.file.size = result.file.content.size();
|
||||||
|
@ -261,7 +245,7 @@ Photo ParsePhoto(const MTPPhoto &data, const QString &suggestedPath) {
|
||||||
data.match([&](const MTPDphoto &data) {
|
data.match([&](const MTPDphoto &data) {
|
||||||
result.id = data.vid.v;
|
result.id = data.vid.v;
|
||||||
result.date = data.vdate.v;
|
result.date = data.vdate.v;
|
||||||
result.image = ParseMaxImage(data.vsizes, suggestedPath);
|
result.image = ParseMaxImage(data, suggestedPath);
|
||||||
}, [&](const MTPDphotoEmpty &data) {
|
}, [&](const MTPDphotoEmpty &data) {
|
||||||
result.id = data.vid.v;
|
result.id = data.vid.v;
|
||||||
});
|
});
|
||||||
|
@ -407,7 +391,7 @@ QString DocumentFolder(const Document &data) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Image ParseDocumentThumb(
|
Image ParseDocumentThumb(
|
||||||
const QVector<MTPPhotoSize> &thumbs,
|
const MTPDdocument &document,
|
||||||
const QString &documentPath) {
|
const QString &documentPath) {
|
||||||
const auto area = [](const MTPPhotoSize &size) {
|
const auto area = [](const MTPPhotoSize &size) {
|
||||||
return size.match([](const MTPDphotoSizeEmpty &) {
|
return size.match([](const MTPDphotoSizeEmpty &) {
|
||||||
|
@ -418,6 +402,7 @@ Image ParseDocumentThumb(
|
||||||
return data.vw.v * data.vh.v;
|
return data.vw.v * data.vh.v;
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
const auto &thumbs = document.vthumbs.v;
|
||||||
const auto i = ranges::max_element(thumbs, ranges::less(), area);
|
const auto i = ranges::max_element(thumbs, ranges::less(), area);
|
||||||
if (i == thumbs.end()) {
|
if (i == thumbs.end()) {
|
||||||
return Image();
|
return Image();
|
||||||
|
@ -430,7 +415,13 @@ Image ParseDocumentThumb(
|
||||||
auto result = Image();
|
auto result = Image();
|
||||||
result.width = data.vw.v;
|
result.width = data.vw.v;
|
||||||
result.height = data.vh.v;
|
result.height = data.vh.v;
|
||||||
result.file.location = ParseLocation(data.vlocation);
|
result.file.location = FileLocation{
|
||||||
|
document.vdc_id.v,
|
||||||
|
MTP_inputDocumentFileLocation(
|
||||||
|
document.vid,
|
||||||
|
document.vaccess_hash,
|
||||||
|
document.vfile_reference,
|
||||||
|
data.vtype) };
|
||||||
if constexpr (MTPDphotoCachedSize::Is<decltype(data)>()) {
|
if constexpr (MTPDphotoCachedSize::Is<decltype(data)>()) {
|
||||||
result.file.content = data.vbytes.v;
|
result.file.content = data.vbytes.v;
|
||||||
result.file.size = result.file.content.size();
|
result.file.size = result.file.content.size();
|
||||||
|
@ -460,13 +451,14 @@ Document ParseDocument(
|
||||||
result.file.location.data = MTP_inputDocumentFileLocation(
|
result.file.location.data = MTP_inputDocumentFileLocation(
|
||||||
data.vid,
|
data.vid,
|
||||||
data.vaccess_hash,
|
data.vaccess_hash,
|
||||||
data.vfile_reference);
|
data.vfile_reference,
|
||||||
|
MTP_string(QString()));
|
||||||
result.file.suggestedPath = suggestedFolder
|
result.file.suggestedPath = suggestedFolder
|
||||||
+ DocumentFolder(result) + '/'
|
+ DocumentFolder(result) + '/'
|
||||||
+ CleanDocumentName(ComputeDocumentName(context, result, date));
|
+ CleanDocumentName(ComputeDocumentName(context, result, date));
|
||||||
|
|
||||||
result.thumb = ParseDocumentThumb(
|
result.thumb = ParseDocumentThumb(
|
||||||
data.vthumbs.v,
|
data,
|
||||||
result.file.suggestedPath);
|
result.file.suggestedPath);
|
||||||
}, [&](const MTPDdocumentEmpty &data) {
|
}, [&](const MTPDdocumentEmpty &data) {
|
||||||
result.id = data.vid.v;
|
result.id = data.vid.v;
|
||||||
|
|
|
@ -103,40 +103,6 @@ TextWithEntities ExtractEditedText(const MTPMessage &message) {
|
||||||
return { text, entities };
|
return { text, entities };
|
||||||
}
|
}
|
||||||
|
|
||||||
PhotoData *GenerateChatPhoto(
|
|
||||||
ChannelId channelId,
|
|
||||||
uint64 logEntryId,
|
|
||||||
TimeId date,
|
|
||||||
const MTPDchatPhoto &photo) {
|
|
||||||
// We try to make a unique photoId that will stay the same for each pair (channelId, logEntryId).
|
|
||||||
static const auto RandomIdPart = rand_value<uint64>();
|
|
||||||
auto mixinIdPart = (static_cast<uint64>(static_cast<uint32>(channelId)) << 32) ^ logEntryId;
|
|
||||||
auto photoId = RandomIdPart ^ mixinIdPart;
|
|
||||||
|
|
||||||
const auto fileReference = [&]() -> const MTPbytes * {
|
|
||||||
const auto takeFrom = [](const MTPFileLocation &location) {
|
|
||||||
return (location.type() == mtpc_fileLocation)
|
|
||||||
? &location.c_fileLocation().vfile_reference
|
|
||||||
: nullptr;
|
|
||||||
};
|
|
||||||
if (const auto result = takeFrom(photo.vphoto_big)) {
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
return takeFrom(photo.vphoto_small);
|
|
||||||
}();
|
|
||||||
auto photoSizes = QVector<MTPPhotoSize>();
|
|
||||||
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 Auth().data().processPhoto(MTP_photo(
|
|
||||||
MTP_flags(0),
|
|
||||||
MTP_long(photoId),
|
|
||||||
MTP_long(0),
|
|
||||||
fileReference ? (*fileReference) : MTP_bytes(QByteArray()),
|
|
||||||
MTP_int(date),
|
|
||||||
MTP_vector<MTPPhotoSize>(photoSizes)));
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto CollectChanges = [](auto &phraseMap, auto plusFlags, auto minusFlags) {
|
const auto CollectChanges = [](auto &phraseMap, auto plusFlags, auto minusFlags) {
|
||||||
auto withPrefix = [&phraseMap](auto flags, QChar prefix) {
|
auto withPrefix = [&phraseMap](auto flags, QChar prefix) {
|
||||||
auto result = QString();
|
auto result = QString();
|
||||||
|
@ -444,18 +410,14 @@ void GenerateItems(
|
||||||
};
|
};
|
||||||
|
|
||||||
auto createChangePhoto = [&](const MTPDchannelAdminLogEventActionChangePhoto &action) {
|
auto createChangePhoto = [&](const MTPDchannelAdminLogEventActionChangePhoto &action) {
|
||||||
switch (action.vnew_photo.type()) {
|
action.vnew_photo.match([&](const MTPDphoto &data) {
|
||||||
case mtpc_chatPhoto: {
|
auto photo = Auth().data().processPhoto(data);
|
||||||
auto photo = GenerateChatPhoto(channel->bareId(), id, date, action.vnew_photo.c_chatPhoto());
|
|
||||||
auto text = (channel->isMegagroup() ? lng_admin_log_changed_photo_group : lng_admin_log_changed_photo_channel)(lt_from, fromLinkText);
|
auto text = (channel->isMegagroup() ? lng_admin_log_changed_photo_group : lng_admin_log_changed_photo_channel)(lt_from, fromLinkText);
|
||||||
addSimpleServiceMessage(text, photo);
|
addSimpleServiceMessage(text, photo);
|
||||||
} break;
|
}, [&](const MTPDphotoEmpty &data) {
|
||||||
case mtpc_chatPhotoEmpty: {
|
|
||||||
auto text = (channel->isMegagroup() ? lng_admin_log_removed_photo_group : lng_admin_log_removed_photo_channel)(lt_from, fromLinkText);
|
auto text = (channel->isMegagroup() ? lng_admin_log_removed_photo_group : lng_admin_log_removed_photo_channel)(lt_from, fromLinkText);
|
||||||
addSimpleServiceMessage(text);
|
addSimpleServiceMessage(text);
|
||||||
} break;
|
});
|
||||||
default: Unexpected("ChatPhoto type in createChangePhoto()");
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
auto createToggleInvites = [&](const MTPDchannelAdminLogEventActionToggleInvites &action) {
|
auto createToggleInvites = [&](const MTPDchannelAdminLogEventActionToggleInvites &action) {
|
||||||
|
|
|
@ -1057,11 +1057,11 @@ void History::applyServiceChanges(
|
||||||
|
|
||||||
case mtpc_messageActionChatEditPhoto: {
|
case mtpc_messageActionChatEditPhoto: {
|
||||||
auto &d = action.c_messageActionChatEditPhoto();
|
auto &d = action.c_messageActionChatEditPhoto();
|
||||||
if (d.vphoto.type() == mtpc_photo) {
|
d.vphoto.match([&](const MTPDphoto &data) {
|
||||||
auto &sizes = d.vphoto.c_photo().vsizes.v;
|
const auto &sizes = data.vsizes.v;
|
||||||
if (!sizes.isEmpty()) {
|
if (!sizes.isEmpty()) {
|
||||||
auto photo = _owner->processPhoto(d.vphoto.c_photo());
|
auto photo = _owner->processPhoto(data);
|
||||||
if (photo) photo->peer = peer;
|
photo->peer = peer;
|
||||||
auto &smallSize = sizes.front();
|
auto &smallSize = sizes.front();
|
||||||
auto &bigSize = sizes.back();
|
auto &bigSize = sizes.back();
|
||||||
const MTPFileLocation *smallLoc = 0, *bigLoc = 0;
|
const MTPFileLocation *smallLoc = 0, *bigLoc = 0;
|
||||||
|
@ -1074,16 +1074,21 @@ void History::applyServiceChanges(
|
||||||
case mtpc_photoCachedSize: bigLoc = &bigSize.c_photoCachedSize().vlocation; break;
|
case mtpc_photoCachedSize: bigLoc = &bigSize.c_photoCachedSize().vlocation; break;
|
||||||
}
|
}
|
||||||
if (smallLoc && bigLoc) {
|
if (smallLoc && bigLoc) {
|
||||||
const auto newPhotoId = photo ? photo->id : PhotoId();
|
|
||||||
if (const auto chat = peer->asChat()) {
|
if (const auto chat = peer->asChat()) {
|
||||||
chat->setPhoto(newPhotoId, MTP_chatPhoto(*smallLoc, *bigLoc));
|
chat->setPhoto(photo->id, MTP_chatPhoto(*smallLoc, *bigLoc, data.vdc_id));
|
||||||
} else if (const auto channel = peer->asChannel()) {
|
} else if (const auto channel = peer->asChannel()) {
|
||||||
channel->setPhoto(newPhotoId, MTP_chatPhoto(*smallLoc, *bigLoc));
|
channel->setPhoto(photo->id, MTP_chatPhoto(*smallLoc, *bigLoc, data.vdc_id));
|
||||||
}
|
}
|
||||||
peer->loadUserpic();
|
peer->loadUserpic();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}, [&](const MTPDphotoEmpty &data) {
|
||||||
|
if (const auto chat = peer->asChat()) {
|
||||||
|
chat->setPhoto(MTP_chatPhotoEmpty());
|
||||||
|
} else if (const auto channel = peer->asChannel()) {
|
||||||
|
channel->setPhoto(MTP_chatPhotoEmpty());
|
||||||
|
}
|
||||||
|
});
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case mtpc_messageActionChatEditTitle: {
|
case mtpc_messageActionChatEditTitle: {
|
||||||
|
|
|
@ -240,16 +240,16 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, crl
|
||||||
}
|
}
|
||||||
|
|
||||||
p.setOpacity(radialOpacity);
|
p.setOpacity(radialOpacity);
|
||||||
auto icon = ([radial, this, selected]() -> const style::icon* {
|
auto icon = [&]() -> const style::icon* {
|
||||||
if (radial || _data->loading()) {
|
if (radial || _data->loading()) {
|
||||||
if (_data->uploading()
|
if (_data->uploading()
|
||||||
|| !_data->large()->location().isNull()) {
|
|| _data->large()->location().valid()) {
|
||||||
return &(selected ? st::historyFileThumbCancelSelected : st::historyFileThumbCancel);
|
return &(selected ? st::historyFileThumbCancelSelected : st::historyFileThumbCancel);
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return &(selected ? st::historyFileThumbDownloadSelected : st::historyFileThumbDownload);
|
return &(selected ? st::historyFileThumbDownloadSelected : st::historyFileThumbDownload);
|
||||||
})();
|
}();
|
||||||
if (icon) {
|
if (icon) {
|
||||||
icon->paintInCenter(p, inner);
|
icon->paintInCenter(p, inner);
|
||||||
}
|
}
|
||||||
|
@ -311,7 +311,7 @@ TextState HistoryPhoto::textState(QPoint point, StateRequest request) const {
|
||||||
} else if (_data->loaded()) {
|
} else if (_data->loaded()) {
|
||||||
result.link = _openl;
|
result.link = _openl;
|
||||||
} else if (_data->loading()) {
|
} else if (_data->loading()) {
|
||||||
if (!_data->large()->location().isNull()) {
|
if (_data->large()->location().valid()) {
|
||||||
result.link = _cancell;
|
result.link = _cancell;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -414,7 +414,8 @@ void HistoryPhoto::drawGrouped(
|
||||||
if (_data->waitingForAlbum()) {
|
if (_data->waitingForAlbum()) {
|
||||||
return &(selected ? st::historyFileThumbWaitingSelected : st::historyFileThumbWaiting);
|
return &(selected ? st::historyFileThumbWaitingSelected : st::historyFileThumbWaiting);
|
||||||
} else if (radial || _data->loading()) {
|
} else if (radial || _data->loading()) {
|
||||||
if (_data->uploading() || !_data->large()->location().isNull()) {
|
if (_data->uploading()
|
||||||
|
|| _data->large()->location().valid()) {
|
||||||
return &(selected ? st::historyFileThumbCancelSelected : st::historyFileThumbCancel);
|
return &(selected ? st::historyFileThumbCancelSelected : st::historyFileThumbCancel);
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
@ -459,9 +460,9 @@ TextState HistoryPhoto::getStateGrouped(
|
||||||
: _data->loaded()
|
: _data->loaded()
|
||||||
? _openl
|
? _openl
|
||||||
: _data->loading()
|
: _data->loading()
|
||||||
? (_data->large()->location().isNull()
|
? (_data->large()->location().valid()
|
||||||
? ClickHandlerPtr()
|
? _cancell
|
||||||
: _cancell)
|
: nullptr)
|
||||||
: _savel);
|
: _savel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -156,7 +156,8 @@ void LoaderMtproto::requestFailed(
|
||||||
}
|
}
|
||||||
const auto callback = [=](const Data::UpdatedFileReferences &updated) {
|
const auto callback = [=](const Data::UpdatedFileReferences &updated) {
|
||||||
_location.match([&](const MTPDinputDocumentFileLocation &location) {
|
_location.match([&](const MTPDinputDocumentFileLocation &location) {
|
||||||
const auto i = updated.data.find(location.vid.v);
|
const auto i = updated.data.find(
|
||||||
|
Data::DocumentFileLocationId{ location.vid.v });
|
||||||
if (i == end(updated.data)) {
|
if (i == end(updated.data)) {
|
||||||
return fail();
|
return fail();
|
||||||
}
|
}
|
||||||
|
@ -167,7 +168,8 @@ void LoaderMtproto::requestFailed(
|
||||||
_location = MTP_inputDocumentFileLocation(
|
_location = MTP_inputDocumentFileLocation(
|
||||||
MTP_long(location.vid.v),
|
MTP_long(location.vid.v),
|
||||||
MTP_long(location.vaccess_hash.v),
|
MTP_long(location.vaccess_hash.v),
|
||||||
MTP_bytes(reference));
|
MTP_bytes(reference),
|
||||||
|
MTP_string(QString()));
|
||||||
}
|
}
|
||||||
if (!_requests.take(offset)) {
|
if (!_requests.take(offset)) {
|
||||||
// Request with such offset was already cancelled.
|
// Request with such offset was already cancelled.
|
||||||
|
|
|
@ -72,7 +72,8 @@ std::optional<DedicatedLoader::File> ParseFile(
|
||||||
const auto location = MTP_inputDocumentFileLocation(
|
const auto location = MTP_inputDocumentFileLocation(
|
||||||
fields.vid,
|
fields.vid,
|
||||||
fields.vaccess_hash,
|
fields.vaccess_hash,
|
||||||
fields.vfile_reference);
|
fields.vfile_reference,
|
||||||
|
MTP_string(QString()));
|
||||||
return DedicatedLoader::File{ name, size, fields.vdc_id.v, location };
|
return DedicatedLoader::File{ name, size, fields.vdc_id.v, location };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
|
|
||||||
#include "data/data_document.h"
|
#include "data/data_document.h"
|
||||||
#include "data/data_session.h"
|
#include "data/data_session.h"
|
||||||
|
#include "data/data_file_origin.h"
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
#include "mainwindow.h"
|
#include "mainwindow.h"
|
||||||
#include "core/application.h"
|
#include "core/application.h"
|
||||||
|
@ -540,7 +541,7 @@ mtpFileLoader::mtpFileLoader(
|
||||||
fromCloud,
|
fromCloud,
|
||||||
autoLoading,
|
autoLoading,
|
||||||
cacheTag)
|
cacheTag)
|
||||||
, _dcId(location->dc())
|
, _dcId(location->file().dcId())
|
||||||
, _location(location)
|
, _location(location)
|
||||||
, _origin(origin) {
|
, _origin(origin) {
|
||||||
auto shiftedDcId = MTP::downloadDcId(_dcId, 0);
|
auto shiftedDcId = MTP::downloadDcId(_dcId, 0);
|
||||||
|
@ -645,25 +646,22 @@ void mtpFileLoader::refreshFileReferenceFrom(
|
||||||
const Data::UpdatedFileReferences &updates,
|
const Data::UpdatedFileReferences &updates,
|
||||||
int requestId,
|
int requestId,
|
||||||
const QByteArray ¤t) {
|
const QByteArray ¤t) {
|
||||||
const auto updated = [&] {
|
|
||||||
if (_location) {
|
|
||||||
const auto i = updates.data.find(Data::SimpleFileLocationId(
|
|
||||||
_location->volume(),
|
|
||||||
_location->dc(),
|
|
||||||
_location->local()));
|
|
||||||
return (i == end(updates.data)) ? QByteArray() : i->second;
|
|
||||||
}
|
|
||||||
const auto i = updates.data.find(_id);
|
|
||||||
return (i == end(updates.data)) ? QByteArray() : i->second;
|
|
||||||
}();
|
|
||||||
if (updated.isEmpty() || updated == current) {
|
|
||||||
cancel(true);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (_location) {
|
if (_location) {
|
||||||
_location->refreshFileReference(updated);
|
_location->refreshFileReference(updates);
|
||||||
|
if (_location->fileReference() == current) {
|
||||||
|
cancel(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
_fileReference = updated;
|
const auto i = updates.data.find(
|
||||||
|
Data::DocumentFileLocationId{ _id });
|
||||||
|
if (i != end(updates.data) && !i->second.isEmpty()) {
|
||||||
|
_fileReference = i->second;
|
||||||
|
}
|
||||||
|
if (_fileReference == current) {
|
||||||
|
cancel(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
const auto offset = finishSentRequestGetOffset(requestId);
|
const auto offset = finishSentRequestGetOffset(requestId);
|
||||||
makeRequest(offset);
|
makeRequest(offset);
|
||||||
|
@ -774,11 +772,7 @@ void mtpFileLoader::makeRequest(int offset) {
|
||||||
|
|
||||||
MTPInputFileLocation mtpFileLoader::computeLocation() const {
|
MTPInputFileLocation mtpFileLoader::computeLocation() const {
|
||||||
if (_location) {
|
if (_location) {
|
||||||
return MTP_inputFileLocation(
|
return _location->file().tl(Auth().userId());
|
||||||
MTP_long(_location->volume()),
|
|
||||||
MTP_int(_location->local()),
|
|
||||||
MTP_long(_location->secret()),
|
|
||||||
MTP_bytes(_location->fileReference()));
|
|
||||||
} else if (_locationType == SecureFileLocation) {
|
} else if (_locationType == SecureFileLocation) {
|
||||||
return MTP_inputSecureFileLocation(
|
return MTP_inputSecureFileLocation(
|
||||||
MTP_long(_id),
|
MTP_long(_id),
|
||||||
|
@ -787,7 +781,8 @@ MTPInputFileLocation mtpFileLoader::computeLocation() const {
|
||||||
return MTP_inputDocumentFileLocation(
|
return MTP_inputDocumentFileLocation(
|
||||||
MTP_long(_id),
|
MTP_long(_id),
|
||||||
MTP_long(_accessHash),
|
MTP_long(_accessHash),
|
||||||
MTP_bytes(_fileReference));
|
MTP_bytes(_fileReference),
|
||||||
|
MTP_string(QString()));
|
||||||
}
|
}
|
||||||
|
|
||||||
void mtpFileLoader::requestMoreCdnFileHashes() {
|
void mtpFileLoader::requestMoreCdnFileHashes() {
|
||||||
|
|
|
@ -65,7 +65,7 @@ PreparedFileThumbnail PrepareFileThumbnail(QImage &&original) {
|
||||||
: std::move(original);
|
: std::move(original);
|
||||||
result.mtpSize = MTP_photoSize(
|
result.mtpSize = MTP_photoSize(
|
||||||
MTP_string(""),
|
MTP_string(""),
|
||||||
MTP_fileLocationUnavailable(MTP_long(0), MTP_int(0), MTP_long(0)),
|
MTP_fileLocationToBeDeprecated(MTP_long(0), MTP_int(0)),
|
||||||
MTP_int(result.image.width()),
|
MTP_int(result.image.width()),
|
||||||
MTP_int(result.image.height()),
|
MTP_int(result.image.height()),
|
||||||
MTP_int(0));
|
MTP_int(0));
|
||||||
|
@ -192,10 +192,7 @@ SendMediaReady PreparePeerPhoto(PeerId peerId, QImage &&image) {
|
||||||
const auto push = [&](const char *type, QImage &&image) {
|
const auto push = [&](const char *type, QImage &&image) {
|
||||||
photoSizes.push_back(MTP_photoSize(
|
photoSizes.push_back(MTP_photoSize(
|
||||||
MTP_string(type),
|
MTP_string(type),
|
||||||
MTP_fileLocationUnavailable(
|
MTP_fileLocationToBeDeprecated(MTP_long(0), MTP_int(0)),
|
||||||
MTP_long(0),
|
|
||||||
MTP_int(0),
|
|
||||||
MTP_long(0)),
|
|
||||||
MTP_int(image.width()),
|
MTP_int(image.width()),
|
||||||
MTP_int(image.height()), MTP_int(0)));
|
MTP_int(image.height()), MTP_int(0)));
|
||||||
photoThumbs.emplace(type[0], std::move(image));
|
photoThumbs.emplace(type[0], std::move(image));
|
||||||
|
@ -211,7 +208,8 @@ SendMediaReady PreparePeerPhoto(PeerId peerId, QImage &&image) {
|
||||||
MTP_long(0),
|
MTP_long(0),
|
||||||
MTP_bytes(QByteArray()),
|
MTP_bytes(QByteArray()),
|
||||||
MTP_int(unixtime()),
|
MTP_int(unixtime()),
|
||||||
MTP_vector<MTPPhotoSize>(photoSizes));
|
MTP_vector<MTPPhotoSize>(photoSizes),
|
||||||
|
MTP_int(MTP::maindc()));
|
||||||
|
|
||||||
QString file, filename;
|
QString file, filename;
|
||||||
int32 filesize = 0;
|
int32 filesize = 0;
|
||||||
|
@ -252,10 +250,7 @@ SendMediaReady PrepareWallPaper(const QImage &image) {
|
||||||
const auto push = [&](const char *type, QImage &&image) {
|
const auto push = [&](const char *type, QImage &&image) {
|
||||||
sizes.push_back(MTP_photoSize(
|
sizes.push_back(MTP_photoSize(
|
||||||
MTP_string(type),
|
MTP_string(type),
|
||||||
MTP_fileLocationUnavailable(
|
MTP_fileLocationToBeDeprecated(MTP_long(0), MTP_int(0)),
|
||||||
MTP_long(0),
|
|
||||||
MTP_int(0),
|
|
||||||
MTP_long(0)),
|
|
||||||
MTP_int(image.width()),
|
MTP_int(image.width()),
|
||||||
MTP_int(image.height()), MTP_int(0)));
|
MTP_int(image.height()), MTP_int(0)));
|
||||||
thumbnails.emplace(type[0], std::move(image));
|
thumbnails.emplace(type[0], std::move(image));
|
||||||
|
@ -861,15 +856,15 @@ void FileLoadTask::process() {
|
||||||
} else if (_type != SendMediaType::File) {
|
} else if (_type != SendMediaType::File) {
|
||||||
auto thumb = (w > 100 || h > 100) ? fullimage.scaled(100, 100, Qt::KeepAspectRatio, Qt::SmoothTransformation) : fullimage;
|
auto thumb = (w > 100 || h > 100) ? fullimage.scaled(100, 100, Qt::KeepAspectRatio, Qt::SmoothTransformation) : fullimage;
|
||||||
photoThumbs.emplace('s', thumb);
|
photoThumbs.emplace('s', thumb);
|
||||||
photoSizes.push_back(MTP_photoSize(MTP_string("s"), MTP_fileLocationUnavailable(MTP_long(0), MTP_int(0), MTP_long(0)), MTP_int(thumb.width()), MTP_int(thumb.height()), MTP_int(0)));
|
photoSizes.push_back(MTP_photoSize(MTP_string("s"), MTP_fileLocationToBeDeprecated(MTP_long(0), MTP_int(0)), MTP_int(thumb.width()), MTP_int(thumb.height()), MTP_int(0)));
|
||||||
|
|
||||||
auto medium = (w > 320 || h > 320) ? fullimage.scaled(320, 320, Qt::KeepAspectRatio, Qt::SmoothTransformation) : fullimage;
|
auto medium = (w > 320 || h > 320) ? fullimage.scaled(320, 320, Qt::KeepAspectRatio, Qt::SmoothTransformation) : fullimage;
|
||||||
photoThumbs.emplace('m', medium);
|
photoThumbs.emplace('m', medium);
|
||||||
photoSizes.push_back(MTP_photoSize(MTP_string("m"), MTP_fileLocationUnavailable(MTP_long(0), MTP_int(0), MTP_long(0)), MTP_int(medium.width()), MTP_int(medium.height()), MTP_int(0)));
|
photoSizes.push_back(MTP_photoSize(MTP_string("m"), MTP_fileLocationToBeDeprecated(MTP_long(0), MTP_int(0)), MTP_int(medium.width()), MTP_int(medium.height()), MTP_int(0)));
|
||||||
|
|
||||||
auto full = (w > 1280 || h > 1280) ? fullimage.scaled(1280, 1280, Qt::KeepAspectRatio, Qt::SmoothTransformation) : fullimage;
|
auto full = (w > 1280 || h > 1280) ? fullimage.scaled(1280, 1280, Qt::KeepAspectRatio, Qt::SmoothTransformation) : fullimage;
|
||||||
photoThumbs.emplace('y', full);
|
photoThumbs.emplace('y', full);
|
||||||
photoSizes.push_back(MTP_photoSize(MTP_string("y"), MTP_fileLocationUnavailable(MTP_long(0), MTP_int(0), MTP_long(0)), MTP_int(full.width()), MTP_int(full.height()), MTP_int(0)));
|
photoSizes.push_back(MTP_photoSize(MTP_string("y"), MTP_fileLocationToBeDeprecated(MTP_long(0), MTP_int(0)), MTP_int(full.width()), MTP_int(full.height()), MTP_int(0)));
|
||||||
|
|
||||||
{
|
{
|
||||||
QBuffer buffer(&filedata);
|
QBuffer buffer(&filedata);
|
||||||
|
@ -882,7 +877,8 @@ void FileLoadTask::process() {
|
||||||
MTP_long(0),
|
MTP_long(0),
|
||||||
MTP_bytes(QByteArray()),
|
MTP_bytes(QByteArray()),
|
||||||
MTP_int(unixtime()),
|
MTP_int(unixtime()),
|
||||||
MTP_vector<MTPPhotoSize>(photoSizes));
|
MTP_vector<MTPPhotoSize>(photoSizes),
|
||||||
|
MTP_int(MTP::maindc()));
|
||||||
|
|
||||||
if (filesize < 0) {
|
if (filesize < 0) {
|
||||||
filesize = _result->filesize = filedata.size();
|
filesize = _result->filesize = filedata.size();
|
||||||
|
|
|
@ -3483,8 +3483,7 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
|
||||||
qint32 version = 0;
|
qint32 version = 0;
|
||||||
stickers.stream >> versionTag >> version;
|
stickers.stream >> versionTag >> version;
|
||||||
if (versionTag != kStickersVersionTag
|
if (versionTag != kStickersVersionTag
|
||||||
|| version <= 0
|
|| version != kStickersSerializeVersion) {
|
||||||
|| version > kStickersSerializeVersion) {
|
|
||||||
// Old data, without sticker set thumbnails.
|
// Old data, without sticker set thumbnails.
|
||||||
return failed();
|
return failed();
|
||||||
}
|
}
|
||||||
|
@ -3496,6 +3495,8 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
|
||||||
return failed();
|
return failed();
|
||||||
}
|
}
|
||||||
for (auto i = 0; i != count; ++i) {
|
for (auto i = 0; i != count; ++i) {
|
||||||
|
using LocationType = StorageFileLocation::Type;
|
||||||
|
|
||||||
quint64 setId = 0, setAccess = 0;
|
quint64 setId = 0, setAccess = 0;
|
||||||
QString setTitle, setShortName;
|
QString setTitle, setShortName;
|
||||||
qint32 scnt = 0;
|
qint32 scnt = 0;
|
||||||
|
@ -3514,11 +3515,19 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
|
||||||
>> setHash
|
>> setHash
|
||||||
>> setFlagsValue
|
>> setFlagsValue
|
||||||
>> setInstallDate;
|
>> setInstallDate;
|
||||||
setThumbnail = Serialize::readStorageImageLocation(
|
const auto thumbnail = Serialize::readStorageImageLocation(
|
||||||
stickers.version,
|
stickers.version,
|
||||||
stickers.stream);
|
stickers.stream);
|
||||||
if (!_checkStreamStatus(stickers.stream)) {
|
if (!thumbnail || !_checkStreamStatus(stickers.stream)) {
|
||||||
return failed();
|
return failed();
|
||||||
|
} else if (thumbnail->valid()
|
||||||
|
&& thumbnail->type() == LocationType::Legacy) {
|
||||||
|
setThumbnail = thumbnail->convertToModern(
|
||||||
|
LocationType::StickerSetThumb,
|
||||||
|
setId,
|
||||||
|
setAccess);
|
||||||
|
} else {
|
||||||
|
setThumbnail = *thumbnail;
|
||||||
}
|
}
|
||||||
|
|
||||||
setFlags = MTPDstickerSet::Flags::from_raw(setFlagsValue);
|
setFlags = MTPDstickerSet::Flags::from_raw(setFlagsValue);
|
||||||
|
@ -3551,7 +3560,7 @@ void _readStickerSets(FileKey &stickersKey, Stickers::Order *outOrder = nullptr,
|
||||||
setHash,
|
setHash,
|
||||||
MTPDstickerSet::Flags(setFlags),
|
MTPDstickerSet::Flags(setFlags),
|
||||||
setInstallDate,
|
setInstallDate,
|
||||||
setThumbnail.isNull() ? ImagePtr() : Images::Create(setThumbnail)));
|
Images::Create(setThumbnail)));
|
||||||
}
|
}
|
||||||
auto &set = it.value();
|
auto &set = it.value();
|
||||||
auto inputSet = MTP_inputStickerSetID(MTP_long(set.id), MTP_long(set.access));
|
auto inputSet = MTP_inputStickerSetID(MTP_long(set.id), MTP_long(set.access));
|
||||||
|
|
|
@ -15,49 +15,67 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "ui/image/image.h"
|
#include "ui/image/image.h"
|
||||||
|
|
||||||
namespace Serialize {
|
namespace Serialize {
|
||||||
|
namespace {
|
||||||
|
|
||||||
void writeStorageImageLocation(
|
constexpr auto kModernImageLocationTag = std::numeric_limits<qint32>::min();
|
||||||
QDataStream &stream,
|
|
||||||
const StorageImageLocation &location) {
|
|
||||||
stream
|
|
||||||
<< qint32(location.width())
|
|
||||||
<< qint32(location.height())
|
|
||||||
<< qint32(location.dc())
|
|
||||||
<< quint64(location.volume())
|
|
||||||
<< qint32(location.local())
|
|
||||||
<< quint64(location.secret());
|
|
||||||
stream << location.fileReference();
|
|
||||||
}
|
|
||||||
|
|
||||||
StorageImageLocation readStorageImageLocation(
|
} // namespace
|
||||||
|
|
||||||
|
std::optional<StorageImageLocation> readLegacyStorageImageLocationOrTag(
|
||||||
int streamAppVersion,
|
int streamAppVersion,
|
||||||
QDataStream &stream) {
|
QDataStream &stream) {
|
||||||
qint32 width, height, dc, local;
|
qint32 width, height, dc, local;
|
||||||
quint64 volume, secret;
|
quint64 volume, secret;
|
||||||
QByteArray fileReference;
|
QByteArray fileReference;
|
||||||
stream >> width >> height >> dc >> volume >> local >> secret;
|
stream >> width;
|
||||||
|
if (width == kModernImageLocationTag) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
stream >> height >> dc >> volume >> local >> secret;
|
||||||
if (streamAppVersion >= 1003013) {
|
if (streamAppVersion >= 1003013) {
|
||||||
stream >> fileReference;
|
stream >> fileReference;
|
||||||
}
|
}
|
||||||
|
if (stream.status() != QDataStream::Ok) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
return StorageImageLocation(
|
return StorageImageLocation(
|
||||||
|
StorageFileLocation(
|
||||||
|
dc,
|
||||||
|
UserId(0),
|
||||||
|
MTP_inputFileLocation(
|
||||||
|
MTP_long(volume),
|
||||||
|
MTP_int(local),
|
||||||
|
MTP_long(secret),
|
||||||
|
MTP_bytes(fileReference))),
|
||||||
width,
|
width,
|
||||||
height,
|
height);
|
||||||
dc,
|
|
||||||
volume,
|
|
||||||
local,
|
|
||||||
secret,
|
|
||||||
fileReference);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int storageImageLocationSize(const StorageImageLocation &location) {
|
int storageImageLocationSize(const StorageImageLocation &location) {
|
||||||
// width + height + dc + volume + local + secret + fileReference
|
// Modern image location tag + (size + content) of the serialization.
|
||||||
return sizeof(qint32)
|
return sizeof(qint32) * 2 + location.serializeSize();
|
||||||
+ sizeof(qint32)
|
}
|
||||||
+ sizeof(qint32)
|
|
||||||
+ sizeof(quint64)
|
void writeStorageImageLocation(
|
||||||
+ sizeof(qint32)
|
QDataStream &stream,
|
||||||
+ sizeof(quint64)
|
const StorageImageLocation &location) {
|
||||||
+ bytearraySize(location.fileReference());
|
stream << kModernImageLocationTag << location.serialize();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<StorageImageLocation> readStorageImageLocation(
|
||||||
|
int streamAppVersion,
|
||||||
|
QDataStream &stream) {
|
||||||
|
const auto legacy = readLegacyStorageImageLocationOrTag(
|
||||||
|
streamAppVersion,
|
||||||
|
stream);
|
||||||
|
if (legacy) {
|
||||||
|
return legacy;
|
||||||
|
}
|
||||||
|
auto serialized = QByteArray();
|
||||||
|
stream >> serialized;
|
||||||
|
return (stream.status() == QDataStream::Ok)
|
||||||
|
? StorageImageLocation::FromSerialized(serialized)
|
||||||
|
: std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32 peerSize(not_null<PeerData*> peer) {
|
uint32 peerSize(not_null<PeerData*> peer) {
|
||||||
|
@ -151,14 +169,15 @@ PeerData *readPeer(int streamAppVersion, QDataStream &stream) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto photoLoc = readStorageImageLocation(
|
const auto userpic = readStorageImageLocation(streamAppVersion, stream);
|
||||||
streamAppVersion,
|
auto userpicAccessHash = uint64(0);
|
||||||
stream);
|
if (!userpic) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
PeerData *result = Auth().data().peerLoaded(peerId);
|
const auto loaded = Auth().data().peerLoaded(peerId);
|
||||||
bool wasLoaded = (result != nullptr);
|
const auto result = loaded ? loaded : Auth().data().peer(peerId).get();
|
||||||
if (!wasLoaded) {
|
if (!loaded) {
|
||||||
result = Auth().data().peer(peerId);
|
|
||||||
result->loadedStatus = PeerData::FullLoaded;
|
result->loadedStatus = PeerData::FullLoaded;
|
||||||
}
|
}
|
||||||
if (const auto user = result->asUser()) {
|
if (const auto user = result->asUser()) {
|
||||||
|
@ -174,6 +193,8 @@ PeerData *readPeer(int streamAppVersion, QDataStream &stream) {
|
||||||
}
|
}
|
||||||
stream >> onlineTill >> contact >> botInfoVersion;
|
stream >> onlineTill >> contact >> botInfoVersion;
|
||||||
|
|
||||||
|
userpicAccessHash = access;
|
||||||
|
|
||||||
const auto showPhone = !user->isServiceUser()
|
const auto showPhone = !user->isServiceUser()
|
||||||
&& (user->id != Auth().userPeerId())
|
&& (user->id != Auth().userPeerId())
|
||||||
&& (contact <= 0);
|
&& (contact <= 0);
|
||||||
|
@ -181,7 +202,7 @@ PeerData *readPeer(int streamAppVersion, QDataStream &stream) {
|
||||||
? App::formatPhone(phone)
|
? App::formatPhone(phone)
|
||||||
: QString();
|
: QString();
|
||||||
|
|
||||||
if (!wasLoaded) {
|
if (!loaded) {
|
||||||
user->setPhone(phone);
|
user->setPhone(phone);
|
||||||
user->setName(first, last, pname, username);
|
user->setName(first, last, pname, username);
|
||||||
|
|
||||||
|
@ -223,7 +244,7 @@ PeerData *readPeer(int streamAppVersion, QDataStream &stream) {
|
||||||
if (oldForbidden) {
|
if (oldForbidden) {
|
||||||
flags |= quint32(MTPDchat_ClientFlag::f_forbidden);
|
flags |= quint32(MTPDchat_ClientFlag::f_forbidden);
|
||||||
}
|
}
|
||||||
if (!wasLoaded) {
|
if (!loaded) {
|
||||||
chat->setName(name);
|
chat->setName(name);
|
||||||
chat->count = count;
|
chat->count = count;
|
||||||
chat->date = date;
|
chat->date = date;
|
||||||
|
@ -245,10 +266,13 @@ PeerData *readPeer(int streamAppVersion, QDataStream &stream) {
|
||||||
qint32 date, version, oldForbidden;
|
qint32 date, version, oldForbidden;
|
||||||
quint32 flags;
|
quint32 flags;
|
||||||
stream >> name >> access >> date >> version >> oldForbidden >> flags >> inviteLink;
|
stream >> name >> access >> date >> version >> oldForbidden >> flags >> inviteLink;
|
||||||
|
|
||||||
|
userpicAccessHash = access;
|
||||||
|
|
||||||
if (oldForbidden) {
|
if (oldForbidden) {
|
||||||
flags |= quint32(MTPDchannel_ClientFlag::f_forbidden);
|
flags |= quint32(MTPDchannel_ClientFlag::f_forbidden);
|
||||||
}
|
}
|
||||||
if (!wasLoaded) {
|
if (!loaded) {
|
||||||
channel->setName(name, QString());
|
channel->setName(name, QString());
|
||||||
channel->access = access;
|
channel->access = access;
|
||||||
channel->date = date;
|
channel->date = date;
|
||||||
|
@ -264,11 +288,16 @@ PeerData *readPeer(int streamAppVersion, QDataStream &stream) {
|
||||||
channel->inputChannel = MTP_inputChannel(MTP_int(peerToChannel(channel->id)), MTP_long(access));
|
channel->inputChannel = MTP_inputChannel(MTP_int(peerToChannel(channel->id)), MTP_long(access));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!wasLoaded) {
|
if (!loaded) {
|
||||||
result->setUserpic(
|
using LocationType = StorageFileLocation::Type;
|
||||||
photoId,
|
const auto location = (userpic->valid()
|
||||||
photoLoc,
|
&& userpic->type() == LocationType::Legacy)
|
||||||
photoLoc.isNull() ? ImagePtr() : Images::Create(photoLoc));
|
? userpic->convertToModern(
|
||||||
|
LocationType::PeerPhoto,
|
||||||
|
result->id,
|
||||||
|
userpicAccessHash)
|
||||||
|
: *userpic;
|
||||||
|
result->setUserpic(photoId, location, Images::Create(location));
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -277,13 +306,12 @@ QString peekUserPhone(int streamAppVersion, QDataStream &stream) {
|
||||||
quint64 peerId = 0, photoId = 0;
|
quint64 peerId = 0, photoId = 0;
|
||||||
stream >> peerId >> photoId;
|
stream >> peerId >> photoId;
|
||||||
DEBUG_LOG(("peekUserPhone.id: %1").arg(peerId));
|
DEBUG_LOG(("peekUserPhone.id: %1").arg(peerId));
|
||||||
if (!peerId || !peerIsUser(peerId)) {
|
if (!peerId
|
||||||
|
|| !peerIsUser(peerId)
|
||||||
|
|| !readStorageImageLocation(streamAppVersion, stream)) {
|
||||||
return QString();
|
return QString();
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto photoLoc = readStorageImageLocation(
|
|
||||||
streamAppVersion,
|
|
||||||
stream);
|
|
||||||
QString first, last, phone;
|
QString first, last, phone;
|
||||||
stream >> first >> last >> phone;
|
stream >> first >> last >> phone;
|
||||||
DEBUG_LOG(("peekUserPhone.data: %1 %2 %3"
|
DEBUG_LOG(("peekUserPhone.data: %1 %2 %3"
|
||||||
|
|
|
@ -84,13 +84,16 @@ inline int dateTimeSize() {
|
||||||
return (sizeof(qint64) + sizeof(quint32) + sizeof(qint8));
|
return (sizeof(qint64) + sizeof(quint32) + sizeof(qint8));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int storageImageLocationSize(const StorageImageLocation &location);
|
||||||
void writeStorageImageLocation(
|
void writeStorageImageLocation(
|
||||||
QDataStream &stream,
|
QDataStream &stream,
|
||||||
const StorageImageLocation &location);
|
const StorageImageLocation &location);
|
||||||
StorageImageLocation readStorageImageLocation(
|
|
||||||
|
// NB! This method can return StorageFileLocation with Type::Generic!
|
||||||
|
// The reader should discard it or convert to one of the valid modern types.
|
||||||
|
std::optional<StorageImageLocation> readStorageImageLocation(
|
||||||
int streamAppVersion,
|
int streamAppVersion,
|
||||||
QDataStream &stream);
|
QDataStream &stream);
|
||||||
int storageImageLocationSize(const StorageImageLocation &location);
|
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
inline T read(QDataStream &stream) {
|
inline T read(QDataStream &stream) {
|
||||||
|
|
|
@ -81,7 +81,7 @@ DocumentData *Document::readFromStreamHelper(int streamAppVersion, QDataStream &
|
||||||
}
|
}
|
||||||
|
|
||||||
qint32 duration = -1;
|
qint32 duration = -1;
|
||||||
StorageImageLocation thumb;
|
std::optional<StorageImageLocation> thumb;
|
||||||
if (type == StickerDocument) {
|
if (type == StickerDocument) {
|
||||||
QString alt;
|
QString alt;
|
||||||
qint32 typeOfSet;
|
qint32 typeOfSet;
|
||||||
|
@ -131,9 +131,14 @@ DocumentData *Document::readFromStreamHelper(int streamAppVersion, QDataStream &
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!dc && !access) {
|
if ((!dc && !access) || !thumb) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
using LocationType = StorageFileLocation::Type;
|
||||||
|
const auto location = (thumb->valid()
|
||||||
|
&& thumb->type() == LocationType::Legacy)
|
||||||
|
? thumb->convertToModern(LocationType::Document, id, access)
|
||||||
|
: *thumb;
|
||||||
return Auth().data().document(
|
return Auth().data().document(
|
||||||
id,
|
id,
|
||||||
access,
|
access,
|
||||||
|
@ -142,10 +147,10 @@ DocumentData *Document::readFromStreamHelper(int streamAppVersion, QDataStream &
|
||||||
attributes,
|
attributes,
|
||||||
mime,
|
mime,
|
||||||
ImagePtr(),
|
ImagePtr(),
|
||||||
thumb.isNull() ? ImagePtr() : Images::Create(thumb),
|
Images::Create(location),
|
||||||
dc,
|
dc,
|
||||||
size,
|
size,
|
||||||
thumb);
|
location);
|
||||||
}
|
}
|
||||||
|
|
||||||
DocumentData *Document::readStickerFromStream(int streamAppVersion, QDataStream &stream, const StickerSetInfo &info) {
|
DocumentData *Document::readStickerFromStream(int streamAppVersion, QDataStream &stream, const StickerSetInfo &info) {
|
||||||
|
|
|
@ -162,6 +162,9 @@ ImagePtr Create(int width, int height) {
|
||||||
}
|
}
|
||||||
|
|
||||||
ImagePtr Create(const StorageImageLocation &location, int size) {
|
ImagePtr Create(const StorageImageLocation &location, int size) {
|
||||||
|
if (!location.valid()) {
|
||||||
|
return ImagePtr();
|
||||||
|
}
|
||||||
const auto key = inMemoryKey(location);
|
const auto key = inMemoryKey(location);
|
||||||
const auto i = StorageImages.find(key);
|
const auto i = StorageImages.find(key);
|
||||||
const auto found = (i != end(StorageImages));
|
const auto found = (i != end(StorageImages));
|
||||||
|
@ -198,6 +201,131 @@ ImagePtr Create(
|
||||||
return ImagePtr(image);
|
return ImagePtr(image);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename CreateLocation>
|
||||||
|
ImagePtr CreateFromPhotoSize(
|
||||||
|
CreateLocation &&createLocation,
|
||||||
|
const MTPPhotoSize &size) {
|
||||||
|
return size.match([&](const MTPDphotoSize &data) {
|
||||||
|
const auto &location = data.vlocation.c_fileLocationToBeDeprecated();
|
||||||
|
return Create(
|
||||||
|
StorageImageLocation(
|
||||||
|
createLocation(data.vtype, location),
|
||||||
|
data.vw.v,
|
||||||
|
data.vh.v),
|
||||||
|
data.vsize.v);
|
||||||
|
}, [&](const MTPDphotoCachedSize &data) {
|
||||||
|
const auto bytes = qba(data.vbytes);
|
||||||
|
const auto &location = data.vlocation.c_fileLocationToBeDeprecated();
|
||||||
|
return Create(
|
||||||
|
StorageImageLocation(
|
||||||
|
createLocation(data.vtype, location),
|
||||||
|
data.vw.v,
|
||||||
|
data.vh.v),
|
||||||
|
bytes);
|
||||||
|
}, [&](const MTPDphotoStrippedSize &data) {
|
||||||
|
const auto bytes = qba(data.vbytes);
|
||||||
|
if (bytes.size() < 3 || bytes[0] != '\x01') {
|
||||||
|
return ImagePtr();
|
||||||
|
}
|
||||||
|
const char header[] = "\xff\xd8\xff\xe0\x00\x10\x4a\x46\x49"
|
||||||
|
"\x46\x00\x01\x01\x00\x00\x01\x00\x01\x00\x00\xff\xdb\x00\x43\x00\x28\x1c"
|
||||||
|
"\x1e\x23\x1e\x19\x28\x23\x21\x23\x2d\x2b\x28\x30\x3c\x64\x41\x3c\x37\x37"
|
||||||
|
"\x3c\x7b\x58\x5d\x49\x64\x91\x80\x99\x96\x8f\x80\x8c\x8a\xa0\xb4\xe6\xc3"
|
||||||
|
"\xa0\xaa\xda\xad\x8a\x8c\xc8\xff\xcb\xda\xee\xf5\xff\xff\xff\x9b\xc1\xff"
|
||||||
|
"\xff\xff\xfa\xff\xe6\xfd\xff\xf8\xff\xdb\x00\x43\x01\x2b\x2d\x2d\x3c\x35"
|
||||||
|
"\x3c\x76\x41\x41\x76\xf8\xa5\x8c\xa5\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8"
|
||||||
|
"\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8"
|
||||||
|
"\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8\xf8"
|
||||||
|
"\xf8\xf8\xf8\xf8\xf8\xff\xc0\x00\x11\x08\x00\x00\x00\x00\x03\x01\x22\x00"
|
||||||
|
"\x02\x11\x01\x03\x11\x01\xff\xc4\x00\x1f\x00\x00\x01\x05\x01\x01\x01\x01"
|
||||||
|
"\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x01\x02\x03\x04\x05\x06\x07\x08"
|
||||||
|
"\x09\x0a\x0b\xff\xc4\x00\xb5\x10\x00\x02\x01\x03\x03\x02\x04\x03\x05\x05"
|
||||||
|
"\x04\x04\x00\x00\x01\x7d\x01\x02\x03\x00\x04\x11\x05\x12\x21\x31\x41\x06"
|
||||||
|
"\x13\x51\x61\x07\x22\x71\x14\x32\x81\x91\xa1\x08\x23\x42\xb1\xc1\x15\x52"
|
||||||
|
"\xd1\xf0\x24\x33\x62\x72\x82\x09\x0a\x16\x17\x18\x19\x1a\x25\x26\x27\x28"
|
||||||
|
"\x29\x2a\x34\x35\x36\x37\x38\x39\x3a\x43\x44\x45\x46\x47\x48\x49\x4a\x53"
|
||||||
|
"\x54\x55\x56\x57\x58\x59\x5a\x63\x64\x65\x66\x67\x68\x69\x6a\x73\x74\x75"
|
||||||
|
"\x76\x77\x78\x79\x7a\x83\x84\x85\x86\x87\x88\x89\x8a\x92\x93\x94\x95\x96"
|
||||||
|
"\x97\x98\x99\x9a\xa2\xa3\xa4\xa5\xa6\xa7\xa8\xa9\xaa\xb2\xb3\xb4\xb5\xb6"
|
||||||
|
"\xb7\xb8\xb9\xba\xc2\xc3\xc4\xc5\xc6\xc7\xc8\xc9\xca\xd2\xd3\xd4\xd5\xd6"
|
||||||
|
"\xd7\xd8\xd9\xda\xe1\xe2\xe3\xe4\xe5\xe6\xe7\xe8\xe9\xea\xf1\xf2\xf3\xf4"
|
||||||
|
"\xf5\xf6\xf7\xf8\xf9\xfa\xff\xc4\x00\x1f\x01\x00\x03\x01\x01\x01\x01\x01"
|
||||||
|
"\x01\x01\x01\x01\x00\x00\x00\x00\x00\x00\x01\x02\x03\x04\x05\x06\x07\x08"
|
||||||
|
"\x09\x0a\x0b\xff\xc4\x00\xb5\x11\x00\x02\x01\x02\x04\x04\x03\x04\x07\x05"
|
||||||
|
"\x04\x04\x00\x01\x02\x77\x00\x01\x02\x03\x11\x04\x05\x21\x31\x06\x12\x41"
|
||||||
|
"\x51\x07\x61\x71\x13\x22\x32\x81\x08\x14\x42\x91\xa1\xb1\xc1\x09\x23\x33"
|
||||||
|
"\x52\xf0\x15\x62\x72\xd1\x0a\x16\x24\x34\xe1\x25\xf1\x17\x18\x19\x1a\x26"
|
||||||
|
"\x27\x28\x29\x2a\x35\x36\x37\x38\x39\x3a\x43\x44\x45\x46\x47\x48\x49\x4a"
|
||||||
|
"\x53\x54\x55\x56\x57\x58\x59\x5a\x63\x64\x65\x66\x67\x68\x69\x6a\x73\x74"
|
||||||
|
"\x75\x76\x77\x78\x79\x7a\x82\x83\x84\x85\x86\x87\x88\x89\x8a\x92\x93\x94"
|
||||||
|
"\x95\x96\x97\x98\x99\x9a\xa2\xa3\xa4\xa5\xa6\xa7\xa8\xa9\xaa\xb2\xb3\xb4"
|
||||||
|
"\xb5\xb6\xb7\xb8\xb9\xba\xc2\xc3\xc4\xc5\xc6\xc7\xc8\xc9\xca\xd2\xd3\xd4"
|
||||||
|
"\xd5\xd6\xd7\xd8\xd9\xda\xe2\xe3\xe4\xe5\xe6\xe7\xe8\xe9\xea\xf2\xf3\xf4"
|
||||||
|
"\xf5\xf6\xf7\xf8\xf9\xfa\xff\xda\x00\x0c\x03\x01\x00\x02\x11\x03\x11\x00"
|
||||||
|
"\x3f\x00";
|
||||||
|
const char footer[] = "\xff\xd9";
|
||||||
|
auto real = QByteArray(header, sizeof(header) - 1);
|
||||||
|
real[164] = bytes[1];
|
||||||
|
real[166] = bytes[2];
|
||||||
|
const auto ready = real
|
||||||
|
+ bytes.mid(3)
|
||||||
|
+ QByteArray::fromRawData(footer, sizeof(footer) - 1);
|
||||||
|
auto image = App::readImage(ready);
|
||||||
|
return !image.isNull()
|
||||||
|
? Images::Create(std::move(image), "JPG")
|
||||||
|
: ImagePtr();
|
||||||
|
}, [&](const MTPDphotoSizeEmpty &) {
|
||||||
|
return ImagePtr();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
ImagePtr Create(const MTPDstickerSet &set, const MTPPhotoSize &size) {
|
||||||
|
const auto create = [&](
|
||||||
|
const MTPstring &thumbSize,
|
||||||
|
const MTPDfileLocationToBeDeprecated &location) {
|
||||||
|
return StorageFileLocation(
|
||||||
|
set.vthumb_dc_id.v,
|
||||||
|
Auth().userId(),
|
||||||
|
MTP_inputStickerSetThumb(
|
||||||
|
MTP_inputStickerSetID(set.vid, set.vaccess_hash),
|
||||||
|
location.vvolume_id,
|
||||||
|
location.vlocal_id));
|
||||||
|
};
|
||||||
|
return CreateFromPhotoSize(create, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
ImagePtr Create(const MTPDphoto &photo, const MTPPhotoSize &size) {
|
||||||
|
const auto create = [&](
|
||||||
|
const MTPstring &thumbSize,
|
||||||
|
const MTPDfileLocationToBeDeprecated &location) {
|
||||||
|
return StorageFileLocation(
|
||||||
|
photo.vdc_id.v,
|
||||||
|
Auth().userId(),
|
||||||
|
MTP_inputPhotoFileLocation(
|
||||||
|
photo.vid,
|
||||||
|
photo.vaccess_hash,
|
||||||
|
photo.vfile_reference,
|
||||||
|
thumbSize));
|
||||||
|
};
|
||||||
|
return CreateFromPhotoSize(create, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
ImagePtr Create(const MTPDdocument &document, const MTPPhotoSize &size) {
|
||||||
|
const auto create = [&](
|
||||||
|
const MTPstring &thumbSize,
|
||||||
|
const MTPDfileLocationToBeDeprecated &location) {
|
||||||
|
return StorageFileLocation(
|
||||||
|
document.vdc_id.v,
|
||||||
|
Auth().userId(),
|
||||||
|
MTP_inputDocumentFileLocation(
|
||||||
|
document.vid,
|
||||||
|
document.vaccess_hash,
|
||||||
|
document.vfile_reference,
|
||||||
|
thumbSize));
|
||||||
|
};
|
||||||
|
return CreateFromPhotoSize(create, size);
|
||||||
|
}
|
||||||
|
|
||||||
QSize getImageSize(const QVector<MTPDocumentAttribute> &attributes) {
|
QSize getImageSize(const QVector<MTPDocumentAttribute> &attributes) {
|
||||||
for (const auto &attribute : attributes) {
|
for (const auto &attribute : attributes) {
|
||||||
if (attribute.type() == mtpc_documentAttributeImageSize) {
|
if (attribute.type() == mtpc_documentAttributeImageSize) {
|
||||||
|
|
|
@ -30,6 +30,9 @@ ImagePtr Create(const StorageImageLocation &location, int size = 0);
|
||||||
ImagePtr Create( // photoCachedSize
|
ImagePtr Create( // photoCachedSize
|
||||||
const StorageImageLocation &location,
|
const StorageImageLocation &location,
|
||||||
const QByteArray &bytes);
|
const QByteArray &bytes);
|
||||||
|
ImagePtr Create(const MTPDstickerSet &set, const MTPPhotoSize &size);
|
||||||
|
ImagePtr Create(const MTPDphoto &photo, const MTPPhotoSize &size);
|
||||||
|
ImagePtr Create(const MTPDdocument &document, const MTPPhotoSize &size);
|
||||||
ImagePtr Create(const MTPWebDocument &location);
|
ImagePtr Create(const MTPWebDocument &location);
|
||||||
ImagePtr Create(const MTPWebDocument &location, QSize box);
|
ImagePtr Create(const MTPWebDocument &location, QSize box);
|
||||||
ImagePtr Create(
|
ImagePtr Create(
|
||||||
|
|
|
@ -9,6 +9,33 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
|
|
||||||
#include "ui/image/image.h"
|
#include "ui/image/image.h"
|
||||||
#include "platform/platform_specific.h"
|
#include "platform/platform_specific.h"
|
||||||
|
#include "storage/cache/storage_cache_types.h"
|
||||||
|
#include "storage/serialize_common.h"
|
||||||
|
#include "data/data_file_origin.h"
|
||||||
|
#include "auth_session.h"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
MTPInputPeer GenerateInputPeer(uint64 id, uint64 accessHash, int32 self) {
|
||||||
|
const auto bareId = [&] {
|
||||||
|
return peerToBareMTPInt(id);
|
||||||
|
};
|
||||||
|
if (!id) {
|
||||||
|
return MTP_inputPeerEmpty();
|
||||||
|
} else if (id == peerFromUser(self)) {
|
||||||
|
return MTP_inputPeerSelf();
|
||||||
|
} else if (peerIsUser(id)) {
|
||||||
|
return MTP_inputPeerUser(bareId(), MTP_long(accessHash));
|
||||||
|
} else if (peerIsChat(id)) {
|
||||||
|
return MTP_inputPeerChat(bareId());
|
||||||
|
} else if (peerIsChannel(id)) {
|
||||||
|
return MTP_inputPeerChannel(bareId(), MTP_long(accessHash));
|
||||||
|
} else {
|
||||||
|
return MTP_inputPeerEmpty();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
ImagePtr::ImagePtr() : _data(Image::Empty()) {
|
ImagePtr::ImagePtr() : _data(Image::Empty()) {
|
||||||
}
|
}
|
||||||
|
@ -29,9 +56,228 @@ ImagePtr::operator bool() const {
|
||||||
|
|
||||||
WebFileLocation WebFileLocation::Null;
|
WebFileLocation WebFileLocation::Null;
|
||||||
|
|
||||||
|
StorageFileLocation::StorageFileLocation(
|
||||||
|
int32 dcId,
|
||||||
|
int32 self,
|
||||||
|
const MTPInputFileLocation &tl)
|
||||||
|
: _dcId(dcId) {
|
||||||
|
tl.match([&](const MTPDinputFileLocation &data) {
|
||||||
|
_type = Type::Legacy;
|
||||||
|
_volumeId = data.vvolume_id.v;
|
||||||
|
_localId = data.vlocal_id.v;
|
||||||
|
_accessHash = data.vsecret.v;
|
||||||
|
_fileReference = data.vfile_reference.v;
|
||||||
|
}, [&](const MTPDinputEncryptedFileLocation &data) {
|
||||||
|
_type = Type::Encrypted;
|
||||||
|
_id = data.vid.v;
|
||||||
|
_accessHash = data.vaccess_hash.v;
|
||||||
|
}, [&](const MTPDinputDocumentFileLocation &data) {
|
||||||
|
_type = Type::Document;
|
||||||
|
_id = data.vid.v;
|
||||||
|
_accessHash = data.vaccess_hash.v;
|
||||||
|
_fileReference = data.vfile_reference.v;
|
||||||
|
_sizeLetter = data.vthumb_size.v.isEmpty()
|
||||||
|
? uint8(0)
|
||||||
|
: uint8(data.vthumb_size.v[0]);
|
||||||
|
}, [&](const MTPDinputSecureFileLocation &data) {
|
||||||
|
_type = Type::Secure;
|
||||||
|
_id = data.vid.v;
|
||||||
|
_accessHash = data.vaccess_hash.v;
|
||||||
|
}, [&](const MTPDinputTakeoutFileLocation &data) {
|
||||||
|
_type = Type::Takeout;
|
||||||
|
}, [&](const MTPDinputPhotoFileLocation &data) {
|
||||||
|
_type = Type::Photo;
|
||||||
|
_id = data.vid.v;
|
||||||
|
_accessHash = data.vaccess_hash.v;
|
||||||
|
_fileReference = data.vfile_reference.v;
|
||||||
|
_sizeLetter = data.vthumb_size.v.isEmpty()
|
||||||
|
? char(0)
|
||||||
|
: data.vthumb_size.v[0];
|
||||||
|
}, [&](const MTPDinputPeerPhotoFileLocation &data) {
|
||||||
|
_type = Type::PeerPhoto;
|
||||||
|
data.vpeer.match([&](const MTPDinputPeerEmpty &data) {
|
||||||
|
_id = 0;
|
||||||
|
}, [&](const MTPDinputPeerSelf &data) {
|
||||||
|
_id = peerFromUser(self);
|
||||||
|
}, [&](const MTPDinputPeerChat &data) {
|
||||||
|
_id = peerFromChat(data.vchat_id);
|
||||||
|
}, [&](const MTPDinputPeerUser &data) {
|
||||||
|
_id = peerFromUser(data.vuser_id);
|
||||||
|
_accessHash = data.vaccess_hash.v;
|
||||||
|
}, [&](const MTPDinputPeerChannel &data) {
|
||||||
|
_id = peerFromChannel(data.vchannel_id);
|
||||||
|
_accessHash = data.vaccess_hash.v;
|
||||||
|
});
|
||||||
|
_volumeId = data.vvolume_id.v;
|
||||||
|
_localId = data.vlocal_id.v;
|
||||||
|
_sizeLetter = data.is_big() ? 'c' : 'a';
|
||||||
|
}, [&](const MTPDinputStickerSetThumb &data) {
|
||||||
|
_type = Type::StickerSetThumb;
|
||||||
|
data.vstickerset.match([&](const MTPDinputStickerSetEmpty &data) {
|
||||||
|
_id = 0;
|
||||||
|
}, [&](const MTPDinputStickerSetID &data) {
|
||||||
|
_id = data.vid.v;
|
||||||
|
_accessHash = data.vaccess_hash.v;
|
||||||
|
}, [&](const MTPDinputStickerSetShortName &data) {
|
||||||
|
Unexpected("inputStickerSetShortName in StorageFileLocation().");
|
||||||
|
});
|
||||||
|
_volumeId = data.vvolume_id.v;
|
||||||
|
_localId = data.vlocal_id.v;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
StorageFileLocation StorageFileLocation::convertToModern(
|
||||||
|
Type type,
|
||||||
|
uint64 id,
|
||||||
|
uint64 accessHash) const {
|
||||||
|
Expects(_type == Type::Legacy);
|
||||||
|
Expects(type == Type::Document
|
||||||
|
|| type == Type::PeerPhoto
|
||||||
|
|| type == Type::StickerSetThumb);
|
||||||
|
|
||||||
|
auto result = *this;
|
||||||
|
result._type = type;
|
||||||
|
result._id = id;
|
||||||
|
result._accessHash = accessHash;
|
||||||
|
result._sizeLetter = (type == Type::PeerPhoto) ? uint8('a') : uint8(0);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32 StorageFileLocation::dcId() const {
|
||||||
|
return _dcId;
|
||||||
|
}
|
||||||
|
|
||||||
|
MTPInputFileLocation StorageFileLocation::tl(int32 self) const {
|
||||||
|
switch (_type) {
|
||||||
|
case Type::Legacy:
|
||||||
|
return MTP_inputFileLocation(
|
||||||
|
MTP_long(_volumeId),
|
||||||
|
MTP_int(_localId),
|
||||||
|
MTP_long(_accessHash),
|
||||||
|
MTP_bytes(_fileReference));
|
||||||
|
|
||||||
|
case Type::Encrypted:
|
||||||
|
return MTP_inputSecureFileLocation(
|
||||||
|
MTP_long(_id),
|
||||||
|
MTP_long(_accessHash));
|
||||||
|
|
||||||
|
case Type::Document:
|
||||||
|
return MTP_inputDocumentFileLocation(
|
||||||
|
MTP_long(_id),
|
||||||
|
MTP_long(_accessHash),
|
||||||
|
MTP_bytes(_fileReference),
|
||||||
|
MTP_string(_sizeLetter
|
||||||
|
? std::string(1, char(_sizeLetter))
|
||||||
|
: std::string()));
|
||||||
|
|
||||||
|
case Type::Secure:
|
||||||
|
return MTP_inputSecureFileLocation(
|
||||||
|
MTP_long(_id),
|
||||||
|
MTP_long(_accessHash));
|
||||||
|
|
||||||
|
case Type::Takeout:
|
||||||
|
return MTP_inputTakeoutFileLocation();
|
||||||
|
|
||||||
|
case Type::Photo:
|
||||||
|
return MTP_inputPhotoFileLocation(
|
||||||
|
MTP_long(_id),
|
||||||
|
MTP_long(_accessHash),
|
||||||
|
MTP_bytes(_fileReference),
|
||||||
|
MTP_string(std::string(1, char(_sizeLetter))));
|
||||||
|
|
||||||
|
case Type::PeerPhoto:
|
||||||
|
return MTP_inputPeerPhotoFileLocation(
|
||||||
|
MTP_flags((_sizeLetter == 'c')
|
||||||
|
? MTPDinputPeerPhotoFileLocation::Flag::f_big
|
||||||
|
: MTPDinputPeerPhotoFileLocation::Flag(0)),
|
||||||
|
GenerateInputPeer(_id, _accessHash, self),
|
||||||
|
MTP_long(_volumeId),
|
||||||
|
MTP_int(_localId));
|
||||||
|
|
||||||
|
case Type::StickerSetThumb:
|
||||||
|
return MTP_inputStickerSetThumb(
|
||||||
|
MTP_inputStickerSetID(MTP_long(_id), MTP_long(_accessHash)),
|
||||||
|
MTP_long(_volumeId),
|
||||||
|
MTP_int(_localId));
|
||||||
|
|
||||||
|
}
|
||||||
|
Unexpected("Type in StorageFileLocation::tl.");
|
||||||
|
}
|
||||||
|
|
||||||
|
QByteArray StorageFileLocation::serialize() const {
|
||||||
|
auto result = QByteArray();
|
||||||
|
result.reserve(serializeSize());
|
||||||
|
if (valid()) {
|
||||||
|
auto buffer = QBuffer(&result);
|
||||||
|
buffer.open(QIODevice::WriteOnly);
|
||||||
|
auto stream = QDataStream(&buffer);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
stream
|
||||||
|
<< quint16(_dcId)
|
||||||
|
<< quint8(_type)
|
||||||
|
<< quint8(_sizeLetter)
|
||||||
|
<< qint32(_localId)
|
||||||
|
<< quint64(_id)
|
||||||
|
<< quint64(_accessHash)
|
||||||
|
<< quint64(_volumeId)
|
||||||
|
<< _fileReference;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
int StorageFileLocation::serializeSize() const {
|
||||||
|
return valid()
|
||||||
|
? int(sizeof(uint64) * 4 + Serialize::bytearraySize(_fileReference))
|
||||||
|
: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<StorageFileLocation> StorageFileLocation::FromSerialized(
|
||||||
|
const QByteArray &serialized) {
|
||||||
|
if (serialized.isEmpty()) {
|
||||||
|
return StorageFileLocation();
|
||||||
|
}
|
||||||
|
|
||||||
|
quint16 dcId = 0;
|
||||||
|
quint8 type = 0;
|
||||||
|
quint8 sizeLetter = 0;
|
||||||
|
qint32 localId = 0;
|
||||||
|
quint64 id = 0;
|
||||||
|
quint64 accessHash = 0;
|
||||||
|
quint64 volumeId = 0;
|
||||||
|
QByteArray fileReference;
|
||||||
|
auto stream = QDataStream(serialized);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
stream
|
||||||
|
>> dcId
|
||||||
|
>> type
|
||||||
|
>> sizeLetter
|
||||||
|
>> localId
|
||||||
|
>> id
|
||||||
|
>> accessHash
|
||||||
|
>> volumeId
|
||||||
|
>> fileReference;
|
||||||
|
|
||||||
|
auto result = StorageFileLocation();
|
||||||
|
result._dcId = dcId;
|
||||||
|
result._type = Type(type);
|
||||||
|
result._sizeLetter = sizeLetter;
|
||||||
|
result._localId = localId;
|
||||||
|
result._id = id;
|
||||||
|
result._accessHash = accessHash;
|
||||||
|
result._volumeId = volumeId;
|
||||||
|
result._fileReference = fileReference;
|
||||||
|
return (stream.status() == QDataStream::Ok && result.valid())
|
||||||
|
? std::make_optional(result)
|
||||||
|
: std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
StorageFileLocation::Type StorageFileLocation::type() const {
|
||||||
|
return _type;
|
||||||
|
}
|
||||||
|
|
||||||
bool StorageFileLocation::valid() const {
|
bool StorageFileLocation::valid() const {
|
||||||
switch (_type) {
|
switch (_type) {
|
||||||
case Type::General:
|
case Type::Legacy:
|
||||||
return (_dcId != 0) && (_volumeId != 0) && (_localId != 0);
|
return (_dcId != 0) && (_volumeId != 0) && (_localId != 0);
|
||||||
|
|
||||||
case Type::Encrypted:
|
case Type::Encrypted:
|
||||||
|
@ -52,33 +298,70 @@ bool StorageFileLocation::valid() const {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
InMemoryKey StorageFileLocation::inMemoryKey() const {
|
Storage::Cache::Key StorageFileLocation::cacheKey() const {
|
||||||
|
using Key = Storage::Cache::Key;
|
||||||
|
|
||||||
|
// Skip '1' and '2' for legacy document cache keys.
|
||||||
|
const auto shifted = ((uint64(_type) + 3) << 8);
|
||||||
|
const auto sliced = uint64(_dcId) & 0xFFULL;
|
||||||
switch (_type) {
|
switch (_type) {
|
||||||
case Type::General:
|
case Type::Legacy:
|
||||||
case Type::PeerPhoto:
|
case Type::PeerPhoto:
|
||||||
case Type::StickerSetThumb:
|
case Type::StickerSetThumb:
|
||||||
return InMemoryKey(
|
return Key{
|
||||||
(uint64(_type) << 56) | (uint64(_dcId) << 40) | uint32(_localId),
|
shifted | sliced | (uint64(uint32(_localId)) << 16),
|
||||||
_volumeId);
|
_volumeId };
|
||||||
|
|
||||||
case Type::Encrypted:
|
case Type::Encrypted:
|
||||||
case Type::Secure:
|
case Type::Secure:
|
||||||
return InMemoryKey(
|
return Key{ shifted | sliced, _id };
|
||||||
(uint64(_type) << 56) | (uint64(_dcId) << 40),
|
|
||||||
_id);
|
|
||||||
|
|
||||||
case Type::Document:
|
case Type::Document:
|
||||||
|
// Keep old cache keys for documents and document 'm' thumbnails.
|
||||||
|
if (_sizeLetter == 0) {
|
||||||
|
return Data::DocumentCacheKey(_dcId, _id);
|
||||||
|
//return Key{ 0x100ULL | sliced, _id };
|
||||||
|
} else if (_sizeLetter == uint8('m')) {
|
||||||
|
return Data::DocumentThumbCacheKey(_dcId, _id);
|
||||||
|
//return Key{ 0x200ULL | sliced, _id };
|
||||||
|
}
|
||||||
|
[[fallthrough]];
|
||||||
case Type::Photo:
|
case Type::Photo:
|
||||||
return InMemoryKey(
|
return Key{ shifted | sliced | (uint64(_sizeLetter) << 16), _id };
|
||||||
(uint64(_type) << 56) | (uint64(_dcId) << 40) | _sizeLetter,
|
|
||||||
_id);
|
|
||||||
|
|
||||||
case Type::Takeout:
|
case Type::Takeout:
|
||||||
return InMemoryKey(
|
return Key{ shifted, 0 };
|
||||||
(uint64(_type) << 56),
|
|
||||||
0);
|
|
||||||
}
|
}
|
||||||
return InMemoryKey();
|
return Key();
|
||||||
|
}
|
||||||
|
|
||||||
|
QByteArray StorageFileLocation::fileReference() const {
|
||||||
|
return _fileReference;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool StorageFileLocation::refreshFileReference(
|
||||||
|
const Data::UpdatedFileReferences &updates) {
|
||||||
|
const auto i = (_type == Type::Document)
|
||||||
|
? updates.data.find(Data::DocumentFileLocationId{ _id })
|
||||||
|
: (_type == Type::Photo)
|
||||||
|
? updates.data.find(Data::PhotoFileLocationId{ _id })
|
||||||
|
: end(updates.data);
|
||||||
|
return (i != end(updates.data))
|
||||||
|
? refreshFileReference(i->second)
|
||||||
|
: false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool StorageFileLocation::refreshFileReference(const QByteArray &data) {
|
||||||
|
if (data.isEmpty() || _fileReference == data) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
_fileReference = data;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
const StorageFileLocation &StorageFileLocation::Invalid() {
|
||||||
|
static auto result = StorageFileLocation();
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool operator==(const StorageFileLocation &a, const StorageFileLocation &b) {
|
bool operator==(const StorageFileLocation &a, const StorageFileLocation &b) {
|
||||||
|
@ -95,7 +378,7 @@ bool operator==(const StorageFileLocation &a, const StorageFileLocation &b) {
|
||||||
|
|
||||||
using Type = StorageFileLocation::Type;
|
using Type = StorageFileLocation::Type;
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case Type::General:
|
case Type::Legacy:
|
||||||
return (a._dcId == b._dcId)
|
return (a._dcId == b._dcId)
|
||||||
&& (a._volumeId == b._volumeId)
|
&& (a._volumeId == b._volumeId)
|
||||||
&& (a._localId == b._localId);
|
&& (a._localId == b._localId);
|
||||||
|
@ -129,6 +412,11 @@ bool operator==(const StorageFileLocation &a, const StorageFileLocation &b) {
|
||||||
Unexpected("Type in StorageFileLocation::operator==.");
|
Unexpected("Type in StorageFileLocation::operator==.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
InMemoryKey inMemoryKey(const StorageFileLocation &location) {
|
||||||
|
const auto key = location.cacheKey();
|
||||||
|
return { key.high, key.low };
|
||||||
|
}
|
||||||
|
|
||||||
StorageImageLocation::StorageImageLocation(
|
StorageImageLocation::StorageImageLocation(
|
||||||
const StorageFileLocation &file,
|
const StorageFileLocation &file,
|
||||||
int width,
|
int width,
|
||||||
|
@ -138,6 +426,51 @@ StorageImageLocation::StorageImageLocation(
|
||||||
, _height(height) {
|
, _height(height) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QByteArray StorageImageLocation::serialize() const {
|
||||||
|
auto result = _file.serialize();
|
||||||
|
if (!result.isEmpty() || (_width > 0) || (_height > 0)) {
|
||||||
|
result.reserve(result.size() + 2 * sizeof(qint32));
|
||||||
|
auto buffer = QBuffer(&result);
|
||||||
|
buffer.open(QIODevice::Append);
|
||||||
|
auto stream = QDataStream(&buffer);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
stream << qint32(_width) << qint32(_height);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
int StorageImageLocation::serializeSize() const {
|
||||||
|
const auto partial = _file.serializeSize();
|
||||||
|
return (partial > 0 || _width > 0 || _height > 0)
|
||||||
|
? (partial + 2 * sizeof(qint32))
|
||||||
|
: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<StorageImageLocation> StorageImageLocation::FromSerialized(
|
||||||
|
const QByteArray &serialized) {
|
||||||
|
if (const auto file = StorageFileLocation::FromSerialized(serialized)) {
|
||||||
|
const auto my = 2 * sizeof(qint32);
|
||||||
|
const auto full = serialized.size();
|
||||||
|
if (!full) {
|
||||||
|
return StorageImageLocation(*file, 0, 0);
|
||||||
|
} else if (full >= my) {
|
||||||
|
qint32 width = 0;
|
||||||
|
qint32 height = 0;
|
||||||
|
|
||||||
|
const auto dimensions = QByteArray::fromRawData(
|
||||||
|
serialized.data() + full - my, my);
|
||||||
|
auto stream = QDataStream(dimensions);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
stream >> width >> height;
|
||||||
|
|
||||||
|
return (stream.status() == QDataStream::Ok)
|
||||||
|
? StorageImageLocation(*file, width, height)
|
||||||
|
: std::optional<StorageImageLocation>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
ReadAccessEnabler::ReadAccessEnabler(const PsFileBookmark *bookmark)
|
ReadAccessEnabler::ReadAccessEnabler(const PsFileBookmark *bookmark)
|
||||||
: _bookmark(bookmark)
|
: _bookmark(bookmark)
|
||||||
, _failed(_bookmark ? !_bookmark->enable() : false) {
|
, _failed(_bookmark ? !_bookmark->enable() : false) {
|
||||||
|
|
|
@ -9,6 +9,16 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
|
|
||||||
class FileLoader;
|
class FileLoader;
|
||||||
|
|
||||||
|
namespace Storage {
|
||||||
|
namespace Cache {
|
||||||
|
struct Key;
|
||||||
|
} // namespace Cache
|
||||||
|
} // namespace Storage
|
||||||
|
|
||||||
|
namespace Data {
|
||||||
|
struct UpdatedFileReferences;
|
||||||
|
} // namespace Data
|
||||||
|
|
||||||
enum LoadFromCloudSetting {
|
enum LoadFromCloudSetting {
|
||||||
LoadFromCloudOrLocal,
|
LoadFromCloudOrLocal,
|
||||||
LoadFromLocalOnly,
|
LoadFromLocalOnly,
|
||||||
|
@ -21,11 +31,27 @@ enum LoadToCacheSetting {
|
||||||
|
|
||||||
using InMemoryKey = std::pair<uint64, uint64>;
|
using InMemoryKey = std::pair<uint64, uint64>;
|
||||||
|
|
||||||
|
namespace std {
|
||||||
|
|
||||||
|
template<>
|
||||||
|
struct hash<InMemoryKey> {
|
||||||
|
size_t operator()(InMemoryKey value) const {
|
||||||
|
auto seed = hash<uint64>()(value.first);
|
||||||
|
seed ^= hash<uint64>()(value.second)
|
||||||
|
+ std::size_t(0x9e3779b9)
|
||||||
|
+ (seed << 6) + (seed >> 2);
|
||||||
|
return seed;
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace std
|
||||||
|
|
||||||
class StorageFileLocation {
|
class StorageFileLocation {
|
||||||
public:
|
public:
|
||||||
// Those are used in serialization, don't change.
|
// Those are used in serialization, don't change.
|
||||||
enum class Type : uchar {
|
enum class Type : uint8 {
|
||||||
General = 0x00,
|
Legacy = 0x00,
|
||||||
Encrypted = 0x01,
|
Encrypted = 0x01,
|
||||||
Document = 0x02,
|
Document = 0x02,
|
||||||
Secure = 0x03,
|
Secure = 0x03,
|
||||||
|
@ -36,26 +62,33 @@ public:
|
||||||
};
|
};
|
||||||
|
|
||||||
StorageFileLocation() = default;
|
StorageFileLocation() = default;
|
||||||
StorageFileLocation(MTP::DcId dcId, const MTPInputFileLocation &tl);
|
StorageFileLocation(
|
||||||
|
int32 dcId,
|
||||||
|
int32 self,
|
||||||
|
const MTPInputFileLocation &tl);
|
||||||
|
|
||||||
[[nodiscard]] MTP::DcId dcId() const;
|
[[nodiscard]] StorageFileLocation convertToModern(
|
||||||
[[nodiscard]] MTPInputFileLocation tl() const;
|
Type type,
|
||||||
|
uint64 id,
|
||||||
|
uint64 accessHash) const;
|
||||||
|
|
||||||
|
[[nodiscard]] int32 dcId() const;
|
||||||
|
[[nodiscard]] MTPInputFileLocation tl(int32 self) const;
|
||||||
|
|
||||||
[[nodiscard]] QByteArray serialize() const;
|
[[nodiscard]] QByteArray serialize() const;
|
||||||
|
[[nodiscard]] int serializeSize() const;
|
||||||
[[nodiscard]] static std::optional<StorageFileLocation> FromSerialized(
|
[[nodiscard]] static std::optional<StorageFileLocation> FromSerialized(
|
||||||
const QByteArray &serialized);
|
const QByteArray &serialized);
|
||||||
|
|
||||||
|
[[nodiscard]] Type type() const;
|
||||||
[[nodiscard]] bool valid() const;
|
[[nodiscard]] bool valid() const;
|
||||||
[[nodiscard]] InMemoryKey inMemoryKey() const;
|
[[nodiscard]] Storage::Cache::Key cacheKey() const;
|
||||||
|
|
||||||
[[nodiscard]] QByteArray fileReference() const;
|
[[nodiscard]] QByteArray fileReference() const;
|
||||||
bool refreshFileReference(const QByteArray &data) {
|
bool refreshFileReference(const Data::UpdatedFileReferences &updates);
|
||||||
if (data.isEmpty() || _fileReference == data) {
|
bool refreshFileReference(const QByteArray &data);
|
||||||
return false;
|
|
||||||
}
|
[[nodiscard]] static const StorageFileLocation &Invalid();
|
||||||
_fileReference = data;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend bool operator==(
|
friend bool operator==(
|
||||||
|
@ -63,7 +96,7 @@ private:
|
||||||
const StorageFileLocation &b);
|
const StorageFileLocation &b);
|
||||||
|
|
||||||
uint16 _dcId = 0;
|
uint16 _dcId = 0;
|
||||||
Type _type = Type::General;
|
Type _type = Type::Legacy;
|
||||||
uint8 _sizeLetter = 0;
|
uint8 _sizeLetter = 0;
|
||||||
int32 _localId = 0;
|
int32 _localId = 0;
|
||||||
uint64 _id = 0;
|
uint64 _id = 0;
|
||||||
|
@ -88,9 +121,20 @@ public:
|
||||||
int height);
|
int height);
|
||||||
|
|
||||||
[[nodiscard]] QByteArray serialize() const;
|
[[nodiscard]] QByteArray serialize() const;
|
||||||
|
[[nodiscard]] int serializeSize() const;
|
||||||
[[nodiscard]] static std::optional<StorageImageLocation> FromSerialized(
|
[[nodiscard]] static std::optional<StorageImageLocation> FromSerialized(
|
||||||
const QByteArray &serialized);
|
const QByteArray &serialized);
|
||||||
|
|
||||||
|
[[nodiscard]] StorageImageLocation convertToModern(
|
||||||
|
StorageFileLocation::Type type,
|
||||||
|
uint64 id,
|
||||||
|
uint64 accessHash) const {
|
||||||
|
return StorageImageLocation(
|
||||||
|
_file.convertToModern(type, id, accessHash),
|
||||||
|
_width,
|
||||||
|
_height);
|
||||||
|
}
|
||||||
|
|
||||||
[[nodiscard]] const StorageFileLocation &file() const {
|
[[nodiscard]] const StorageFileLocation &file() const {
|
||||||
return _file;
|
return _file;
|
||||||
}
|
}
|
||||||
|
@ -106,18 +150,26 @@ public:
|
||||||
_height = height;
|
_height = height;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] StorageFileLocation::Type type() const {
|
||||||
|
return _file.type();
|
||||||
|
}
|
||||||
[[nodiscard]] bool valid() const {
|
[[nodiscard]] bool valid() const {
|
||||||
return _file.valid();
|
return _file.valid();
|
||||||
}
|
}
|
||||||
[[nodiscard]] InMemoryKey inMemoryKey() const {
|
|
||||||
return _file.inMemoryKey();
|
|
||||||
}
|
|
||||||
[[nodiscard]] QByteArray fileReference() const {
|
[[nodiscard]] QByteArray fileReference() const {
|
||||||
return _file.fileReference();
|
return _file.fileReference();
|
||||||
}
|
}
|
||||||
bool refreshFileReference(const QByteArray &data) {
|
bool refreshFileReference(const QByteArray &data) {
|
||||||
return _file.refreshFileReference(data);
|
return _file.refreshFileReference(data);
|
||||||
}
|
}
|
||||||
|
bool refreshFileReference(const Data::UpdatedFileReferences &updates) {
|
||||||
|
return _file.refreshFileReference(updates);
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] static const StorageImageLocation &Invalid() {
|
||||||
|
static auto result = StorageImageLocation();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend inline bool operator==(
|
friend inline bool operator==(
|
||||||
|
@ -224,12 +276,10 @@ private:
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
inline InMemoryKey inMemoryKey(const StorageFileLocation &location) {
|
InMemoryKey inMemoryKey(const StorageFileLocation &location);
|
||||||
return location.inMemoryKey();
|
|
||||||
}
|
|
||||||
|
|
||||||
inline InMemoryKey inMemoryKey(const StorageImageLocation &location) {
|
inline InMemoryKey inMemoryKey(const StorageImageLocation &location) {
|
||||||
return location.inMemoryKey();
|
return inMemoryKey(location.file());
|
||||||
}
|
}
|
||||||
|
|
||||||
inline InMemoryKey inMemoryKey(const WebFileLocation &location) {
|
inline InMemoryKey inMemoryKey(const WebFileLocation &location) {
|
||||||
|
|
|
@ -87,7 +87,7 @@ int ImageSource::loadOffset() {
|
||||||
}
|
}
|
||||||
|
|
||||||
const StorageImageLocation &ImageSource::location() {
|
const StorageImageLocation &ImageSource::location() {
|
||||||
return StorageImageLocation::Null;
|
return StorageImageLocation::Invalid();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ImageSource::refreshFileReference(const QByteArray &data) {
|
void ImageSource::refreshFileReference(const QByteArray &data) {
|
||||||
|
@ -222,7 +222,7 @@ int LocalFileSource::loadOffset() {
|
||||||
}
|
}
|
||||||
|
|
||||||
const StorageImageLocation &LocalFileSource::location() {
|
const StorageImageLocation &LocalFileSource::location() {
|
||||||
return StorageImageLocation::Null;
|
return StorageImageLocation::Invalid();
|
||||||
}
|
}
|
||||||
|
|
||||||
void LocalFileSource::refreshFileReference(const QByteArray &data) {
|
void LocalFileSource::refreshFileReference(const QByteArray &data) {
|
||||||
|
@ -334,7 +334,7 @@ void RemoteSource::setImageBytes(const QByteArray &bytes) {
|
||||||
_loader->finishWithBytes(bytes);
|
_loader->finishWithBytes(bytes);
|
||||||
|
|
||||||
const auto location = this->location();
|
const auto location = this->location();
|
||||||
if (!location.isNull()
|
if (location.valid()
|
||||||
&& !bytes.isEmpty()
|
&& !bytes.isEmpty()
|
||||||
&& bytes.size() <= Storage::kMaxFileInMemory) {
|
&& bytes.size() <= Storage::kMaxFileInMemory) {
|
||||||
Auth().data().cache().putIfEmpty(
|
Auth().data().cache().putIfEmpty(
|
||||||
|
@ -437,7 +437,7 @@ RemoteSource::~RemoteSource() {
|
||||||
}
|
}
|
||||||
|
|
||||||
const StorageImageLocation &RemoteSource::location() {
|
const StorageImageLocation &RemoteSource::location() {
|
||||||
return StorageImageLocation::Null;
|
return StorageImageLocation::Invalid();
|
||||||
}
|
}
|
||||||
|
|
||||||
void RemoteSource::refreshFileReference(const QByteArray &data) {
|
void RemoteSource::refreshFileReference(const QByteArray &data) {
|
||||||
|
@ -472,9 +472,9 @@ const StorageImageLocation &StorageSource::location() {
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<Storage::Cache::Key> StorageSource::cacheKey() {
|
std::optional<Storage::Cache::Key> StorageSource::cacheKey() {
|
||||||
return _location.isNull()
|
return _location.valid()
|
||||||
? std::nullopt
|
? base::make_optional(Data::StorageCacheKey(_location))
|
||||||
: base::make_optional(Data::StorageCacheKey(_location));
|
: std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
int StorageSource::width() {
|
int StorageSource::width() {
|
||||||
|
@ -506,16 +506,15 @@ FileLoader *StorageSource::createLoader(
|
||||||
Data::FileOrigin origin,
|
Data::FileOrigin origin,
|
||||||
LoadFromCloudSetting fromCloud,
|
LoadFromCloudSetting fromCloud,
|
||||||
bool autoLoading) {
|
bool autoLoading) {
|
||||||
if (_location.isNull()) {
|
return _location.valid()
|
||||||
return nullptr;
|
? new mtpFileLoader(
|
||||||
}
|
&_location,
|
||||||
return new mtpFileLoader(
|
origin,
|
||||||
&_location,
|
_size,
|
||||||
origin,
|
fromCloud,
|
||||||
_size,
|
autoLoading,
|
||||||
fromCloud,
|
Data::kImageCacheTag)
|
||||||
autoLoading,
|
: nullptr;
|
||||||
Data::kImageCacheTag);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
WebCachedSource::WebCachedSource(
|
WebCachedSource::WebCachedSource(
|
||||||
|
@ -637,7 +636,7 @@ DelayedStorageSource::DelayedStorageSource()
|
||||||
}
|
}
|
||||||
|
|
||||||
DelayedStorageSource::DelayedStorageSource(int w, int h)
|
DelayedStorageSource::DelayedStorageSource(int w, int h)
|
||||||
: StorageSource(StorageImageLocation(w, h, 0, 0, 0, 0, {}), 0) {
|
: StorageSource(StorageImageLocation(StorageFileLocation(), w, h), 0) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void DelayedStorageSource::setDelayedStorageLocation(
|
void DelayedStorageSource::setDelayedStorageLocation(
|
||||||
|
@ -663,22 +662,22 @@ void DelayedStorageSource::performDelayedLoad(Data::FileOrigin origin) {
|
||||||
void DelayedStorageSource::automaticLoad(
|
void DelayedStorageSource::automaticLoad(
|
||||||
Data::FileOrigin origin,
|
Data::FileOrigin origin,
|
||||||
const HistoryItem *item) {
|
const HistoryItem *item) {
|
||||||
if (_location.isNull()) {
|
if (_location.valid()) {
|
||||||
if (!_loadCancelled && item) {
|
|
||||||
const auto loadFromCloud = Data::AutoDownload::Should(
|
|
||||||
Auth().settings().autoDownload(),
|
|
||||||
item->history()->peer,
|
|
||||||
this);
|
|
||||||
|
|
||||||
if (_loadRequested) {
|
|
||||||
if (loadFromCloud) _loadFromCloud = loadFromCloud;
|
|
||||||
} else {
|
|
||||||
_loadFromCloud = loadFromCloud;
|
|
||||||
_loadRequested = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
StorageSource::automaticLoad(origin, item);
|
StorageSource::automaticLoad(origin, item);
|
||||||
|
return;
|
||||||
|
} else if (_loadCancelled || !item) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto loadFromCloud = Data::AutoDownload::Should(
|
||||||
|
Auth().settings().autoDownload(),
|
||||||
|
item->history()->peer,
|
||||||
|
this);
|
||||||
|
|
||||||
|
if (_loadRequested) {
|
||||||
|
if (loadFromCloud) _loadFromCloud = loadFromCloud;
|
||||||
|
} else {
|
||||||
|
_loadFromCloud = loadFromCloud;
|
||||||
|
_loadRequested = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -691,10 +690,10 @@ void DelayedStorageSource::load(
|
||||||
Data::FileOrigin origin,
|
Data::FileOrigin origin,
|
||||||
bool loadFirst,
|
bool loadFirst,
|
||||||
bool prior) {
|
bool prior) {
|
||||||
if (_location.isNull()) {
|
if (_location.valid()) {
|
||||||
_loadRequested = _loadFromCloud = true;
|
|
||||||
} else {
|
|
||||||
StorageSource::load(origin, loadFirst, prior);
|
StorageSource::load(origin, loadFirst, prior);
|
||||||
|
} else {
|
||||||
|
_loadRequested = _loadFromCloud = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -707,7 +706,7 @@ void DelayedStorageSource::loadEvenCancelled(
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DelayedStorageSource::displayLoading() {
|
bool DelayedStorageSource::displayLoading() {
|
||||||
return _location.isNull() ? true : StorageSource::displayLoading();
|
return _location.valid() ? StorageSource::displayLoading() : true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DelayedStorageSource::cancel() {
|
void DelayedStorageSource::cancel() {
|
||||||
|
|
|
@ -283,7 +283,9 @@ public:
|
||||||
void automaticLoadSettingsChanged() override;
|
void automaticLoadSettingsChanged() override;
|
||||||
|
|
||||||
bool loading() override {
|
bool loading() override {
|
||||||
return _location.isNull() ? _loadRequested : StorageSource::loading();
|
return _location.valid()
|
||||||
|
? StorageSource::loading()
|
||||||
|
: _loadRequested;
|
||||||
}
|
}
|
||||||
bool displayLoading() override;
|
bool displayLoading() override;
|
||||||
void cancel() override;
|
void cancel() override;
|
||||||
|
|
Loading…
Reference in New Issue