init
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Close stale issues and PRs / stale (push) Successful in 13s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Close stale issues and PRs / stale (push) Successful in 13s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
This commit is contained in:
386
Telegram/SourceFiles/ui/image/image_location_factory.cpp
Normal file
386
Telegram/SourceFiles/ui/image/image_location_factory.cpp
Normal file
@@ -0,0 +1,386 @@
|
||||
/*
|
||||
This file is part of Telegram Desktop,
|
||||
the official desktop application for the Telegram messaging service.
|
||||
|
||||
For license and copyright information please follow this link:
|
||||
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||
*/
|
||||
#include "ui/image/image_location_factory.h"
|
||||
|
||||
#include "ui/image/image.h"
|
||||
#include "main/main_session.h"
|
||||
|
||||
#include <QtCore/QBuffer>
|
||||
|
||||
namespace Images {
|
||||
namespace {
|
||||
|
||||
QSize GetSizeForDocument(const QVector<MTPDocumentAttribute> &attributes) {
|
||||
for (const auto &attribute : attributes) {
|
||||
if (attribute.type() == mtpc_documentAttributeImageSize) {
|
||||
auto &size = attribute.c_documentAttributeImageSize();
|
||||
return QSize(size.vw().v, size.vh().v);
|
||||
}
|
||||
}
|
||||
return QSize();
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
ImageWithLocation FromPhotoSize(
|
||||
not_null<Main::Session*> session,
|
||||
const MTPDphoto &photo,
|
||||
const MTPPhotoSize &size) {
|
||||
if (!photo.vaccess_hash().v && photo.vfile_reference().v.isEmpty()) {
|
||||
// Locally created fake photo.
|
||||
return ImageWithLocation();
|
||||
}
|
||||
return size.match([&](const MTPDphotoSize &data) {
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
photo.vdc_id().v,
|
||||
session->userId(),
|
||||
MTP_inputPhotoFileLocation(
|
||||
photo.vid(),
|
||||
photo.vaccess_hash(),
|
||||
photo.vfile_reference(),
|
||||
data.vtype())) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytesCount = data.vsize().v
|
||||
};
|
||||
}, [&](const MTPDphotoCachedSize &data) {
|
||||
const auto bytes = qba(data.vbytes());
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
photo.vdc_id().v,
|
||||
session->userId(),
|
||||
MTP_inputPhotoFileLocation(
|
||||
photo.vid(),
|
||||
photo.vaccess_hash(),
|
||||
photo.vfile_reference(),
|
||||
data.vtype())) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytes = bytes,
|
||||
.bytesCount = int(bytes.size()),
|
||||
};
|
||||
}, [&](const MTPDphotoSizeProgressive &data) {
|
||||
// #TODO layer118
|
||||
if (data.vsizes().v.isEmpty()) {
|
||||
return ImageWithLocation();
|
||||
}
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
photo.vdc_id().v,
|
||||
session->userId(),
|
||||
MTP_inputPhotoFileLocation(
|
||||
photo.vid(),
|
||||
photo.vaccess_hash(),
|
||||
photo.vfile_reference(),
|
||||
data.vtype())) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytesCount = data.vsizes().v.back().v
|
||||
};
|
||||
}, [&](const MTPDphotoStrippedSize &data) {
|
||||
return ImageWithLocation();
|
||||
//const auto bytes = ExpandInlineBytes(qba(data.vbytes()));
|
||||
//return ImageWithLocation{
|
||||
// .location = ImageLocation(
|
||||
// DownloadLocation{ StorageFileLocation(
|
||||
// photo.vdc_id().v,
|
||||
// session->userId(),
|
||||
// MTP_inputPhotoFileLocation(
|
||||
// photo.vid(),
|
||||
// photo.vaccess_hash(),
|
||||
// photo.vfile_reference(),
|
||||
// data.vtype())) },
|
||||
// width, // ???
|
||||
// height), // ???
|
||||
// .bytes = bytes,
|
||||
// .bytesCount = bytes.size(),
|
||||
//};
|
||||
}, [&](const MTPDphotoPathSize &) {
|
||||
return ImageWithLocation();
|
||||
}, [&](const MTPDphotoSizeEmpty &) {
|
||||
return ImageWithLocation();
|
||||
});
|
||||
}
|
||||
|
||||
ImageWithLocation FromProgressiveSize(
|
||||
not_null<Main::Session*> session,
|
||||
const MTPPhotoSize &size,
|
||||
int index) {
|
||||
Expects(size.type() == mtpc_photoSizeProgressive);
|
||||
|
||||
const auto &data = size.c_photoSizeProgressive();
|
||||
if (data.vsizes().v.size() <= index) {
|
||||
return ImageWithLocation();
|
||||
}
|
||||
return ImageWithLocation{
|
||||
.progressivePartSize = data.vsizes().v[index].v,
|
||||
};
|
||||
}
|
||||
|
||||
ImageWithLocation FromPhotoSize(
|
||||
not_null<Main::Session*> session,
|
||||
const MTPDdocument &document,
|
||||
const MTPPhotoSize &size) {
|
||||
return size.match([&](const MTPDphotoSize &data) {
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
document.vdc_id().v,
|
||||
session->userId(),
|
||||
MTP_inputDocumentFileLocation(
|
||||
document.vid(),
|
||||
document.vaccess_hash(),
|
||||
document.vfile_reference(),
|
||||
data.vtype())) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytesCount = data.vsize().v
|
||||
};
|
||||
}, [&](const MTPDphotoCachedSize &data) {
|
||||
const auto bytes = qba(data.vbytes());
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
document.vdc_id().v,
|
||||
session->userId(),
|
||||
MTP_inputDocumentFileLocation(
|
||||
document.vid(),
|
||||
document.vaccess_hash(),
|
||||
document.vfile_reference(),
|
||||
data.vtype())) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytes = bytes,
|
||||
.bytesCount = int(bytes.size()),
|
||||
};
|
||||
}, [&](const MTPDphotoSizeProgressive &data) {
|
||||
if (data.vsizes().v.isEmpty()) {
|
||||
return ImageWithLocation();
|
||||
}
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
document.vdc_id().v,
|
||||
session->userId(),
|
||||
MTP_inputDocumentFileLocation(
|
||||
document.vid(),
|
||||
document.vaccess_hash(),
|
||||
document.vfile_reference(),
|
||||
data.vtype())) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytesCount = data.vsizes().v.back().v
|
||||
};
|
||||
}, [&](const MTPDphotoStrippedSize &data) {
|
||||
return ImageWithLocation();
|
||||
//const auto bytes = ExpandInlineBytes(qba(data.vbytes()));
|
||||
//return ImageWithLocation{
|
||||
// .location = ImageLocation(
|
||||
// DownloadLocation{ StorageFileLocation(
|
||||
// document.vdc_id().v,
|
||||
// session->userId(),
|
||||
// MTP_inputDocumentFileLocation(
|
||||
// document.vid(),
|
||||
// document.vaccess_hash(),
|
||||
// document.vfile_reference(),
|
||||
// data.vtype())) },
|
||||
// width, // ???
|
||||
// height), // ???
|
||||
// .bytes = bytes,
|
||||
// .bytesCount = bytes.size(),
|
||||
//};
|
||||
}, [&](const MTPDphotoPathSize &data) {
|
||||
return ImageWithLocation();
|
||||
}, [&](const MTPDphotoSizeEmpty &) {
|
||||
return ImageWithLocation();
|
||||
});
|
||||
}
|
||||
|
||||
ImageWithLocation FromPhotoSize(
|
||||
not_null<Main::Session*> session,
|
||||
const MTPDstickerSet &set,
|
||||
const MTPPhotoSize &size) {
|
||||
if (!set.vthumb_dc_id() || !set.vthumb_version()) {
|
||||
return ImageWithLocation();
|
||||
}
|
||||
return size.match([&](const MTPDphotoSize &data) {
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
set.vthumb_dc_id()->v,
|
||||
session->userId(),
|
||||
MTP_inputStickerSetThumb(
|
||||
MTP_inputStickerSetID(set.vid(), set.vaccess_hash()),
|
||||
MTP_int(set.vthumb_version()->v))) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytesCount = data.vsize().v
|
||||
};
|
||||
}, [&](const MTPDphotoCachedSize &data) {
|
||||
const auto bytes = qba(data.vbytes());
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
set.vthumb_dc_id()->v,
|
||||
session->userId(),
|
||||
MTP_inputStickerSetThumb(
|
||||
MTP_inputStickerSetID(set.vid(), set.vaccess_hash()),
|
||||
MTP_int(set.vthumb_version()->v))) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytes = bytes,
|
||||
.bytesCount = int(bytes.size()),
|
||||
};
|
||||
}, [&](const MTPDphotoSizeProgressive &data) {
|
||||
if (data.vsizes().v.isEmpty()) {
|
||||
return ImageWithLocation();
|
||||
}
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
set.vthumb_dc_id()->v,
|
||||
session->userId(),
|
||||
MTP_inputStickerSetThumb(
|
||||
MTP_inputStickerSetID(set.vid(), set.vaccess_hash()),
|
||||
MTP_int(set.vthumb_version()->v))) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytesCount = data.vsizes().v.back().v
|
||||
};
|
||||
}, [&](const MTPDphotoStrippedSize &data) {
|
||||
return ImageWithLocation();
|
||||
//const auto bytes = ExpandInlineBytes(qba(data.vbytes()));
|
||||
//return ImageWithLocation{
|
||||
// .location = ImageLocation(
|
||||
// DownloadLocation{ StorageFileLocation(
|
||||
// document.vdc_id().v,
|
||||
// session->userId(),
|
||||
// MTP_inputDocumentFileLocation(
|
||||
// document.vid(),
|
||||
// document.vaccess_hash(),
|
||||
// document.vfile_reference(),
|
||||
// data.vtype())) },
|
||||
// width, // ???
|
||||
// height), // ???
|
||||
// .bytes = bytes,
|
||||
// .bytesCount = bytes.size(),
|
||||
//};
|
||||
}, [&](const MTPDphotoPathSize &data) {
|
||||
return ImageWithLocation();
|
||||
}, [&](const MTPDphotoSizeEmpty &) {
|
||||
return ImageWithLocation();
|
||||
});
|
||||
}
|
||||
|
||||
ImageWithLocation FromImageInMemory(
|
||||
const QImage &image,
|
||||
const char *format,
|
||||
QByteArray bytes) {
|
||||
if (image.isNull()) {
|
||||
return ImageWithLocation();
|
||||
}
|
||||
if (bytes.isEmpty()) {
|
||||
auto buffer = QBuffer(&bytes);
|
||||
image.save(&buffer, format);
|
||||
}
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ InMemoryLocation{ bytes } },
|
||||
image.width(),
|
||||
image.height()),
|
||||
.bytes = bytes,
|
||||
.preloaded = image,
|
||||
.bytesCount = int(bytes.size()),
|
||||
};
|
||||
}
|
||||
|
||||
ImageLocation FromWebDocument(const MTPWebDocument &document) {
|
||||
return document.match([](const MTPDwebDocument &data) {
|
||||
const auto size = GetSizeForDocument(data.vattributes().v);
|
||||
|
||||
// We don't use size from WebDocument, because it is not reliable.
|
||||
// It can be > 0 and different from the real size
|
||||
// that we get in upload.WebFile result.
|
||||
//auto filesize = int64(); // data.vsize().v;
|
||||
return ImageLocation(
|
||||
DownloadLocation{ WebFileLocation(
|
||||
data.vurl().v,
|
||||
data.vaccess_hash().v) },
|
||||
size.width(),
|
||||
size.height());
|
||||
}, [](const MTPDwebDocumentNoProxy &data) {
|
||||
const auto size = GetSizeForDocument(data.vattributes().v);
|
||||
|
||||
// We don't use size from WebDocument, because it is not reliable.
|
||||
// It can be > 0 and different from the real size
|
||||
// that we get in upload.WebFile result.
|
||||
//auto filesize = int64(); // data.vsize().v;
|
||||
return ImageLocation(
|
||||
DownloadLocation{ PlainUrlLocation{ qs(data.vurl()) } },
|
||||
size.width(),
|
||||
size.height());
|
||||
});
|
||||
}
|
||||
|
||||
ImageWithLocation FromVideoSize(
|
||||
not_null<Main::Session*> session,
|
||||
const MTPDdocument &document,
|
||||
const MTPVideoSize &size) {
|
||||
return size.match([&](const MTPDvideoSize &data) {
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
document.vdc_id().v,
|
||||
session->userId(),
|
||||
MTP_inputDocumentFileLocation(
|
||||
document.vid(),
|
||||
document.vaccess_hash(),
|
||||
document.vfile_reference(),
|
||||
data.vtype())) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytesCount = data.vsize().v,
|
||||
};
|
||||
}, [](const MTPDvideoSizeEmojiMarkup &) {
|
||||
return ImageWithLocation();
|
||||
}, [](const MTPDvideoSizeStickerMarkup &) {
|
||||
return ImageWithLocation();
|
||||
});
|
||||
}
|
||||
|
||||
ImageWithLocation FromVideoSize(
|
||||
not_null<Main::Session*> session,
|
||||
const MTPDphoto &photo,
|
||||
const MTPVideoSize &size) {
|
||||
return size.match([&](const MTPDvideoSize &data) {
|
||||
return ImageWithLocation{
|
||||
.location = ImageLocation(
|
||||
DownloadLocation{ StorageFileLocation(
|
||||
photo.vdc_id().v,
|
||||
session->userId(),
|
||||
MTP_inputPhotoFileLocation(
|
||||
photo.vid(),
|
||||
photo.vaccess_hash(),
|
||||
photo.vfile_reference(),
|
||||
data.vtype())) },
|
||||
data.vw().v,
|
||||
data.vh().v),
|
||||
.bytesCount = data.vsize().v,
|
||||
};
|
||||
}, [](const MTPDvideoSizeEmojiMarkup &) {
|
||||
return ImageWithLocation();
|
||||
}, [](const MTPDvideoSizeStickerMarkup &) {
|
||||
return ImageWithLocation();
|
||||
});
|
||||
}
|
||||
|
||||
} // namespace Images
|
||||
Reference in New Issue
Block a user