Start using document bytes from DocumentMedia.

This commit is contained in:
John Preston 2020-04-09 16:27:53 +04:00
parent 888e42df34
commit 33f4946242
31 changed files with 190 additions and 169 deletions

View File

@ -11,23 +11,11 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/rect_part.h" #include "ui/rect_part.h"
enum class ImageRoundRadius; enum class ImageRoundRadius;
class MainWindow;
class MainWidget;
class HistoryItem;
class History;
namespace HistoryView { namespace HistoryView {
class Element; class Element;
} // namespace HistoryView } // namespace HistoryView
namespace Media {
namespace Clip {
class Reader;
} // namespace Clip
} // namespace Media
using HistoryItemsMap = base::flat_set<not_null<HistoryItem*>>;
using GifItems = QHash<Media::Clip::Reader*, HistoryItem*>;
enum RoundCorners : int { enum RoundCorners : int {
SmallMaskCorners = 0x00, // for images SmallMaskCorners = 0x00, // for images
LargeMaskCorners, LargeMaskCorners,

View File

@ -315,9 +315,10 @@ void EditCaptionBox::prepareGifPreview(DocumentData* document) {
const auto callback = [=](Media::Clip::Notification notification) { const auto callback = [=](Media::Clip::Notification notification) {
clipCallback(notification); clipCallback(notification);
}; };
_gifMedia = document ? document->createMediaView() : nullptr;
if (document && document->isAnimation() && document->loaded()) { if (document && document->isAnimation() && document->loaded()) {
_gifPreview = Media::Clip::MakeReader( _gifPreview = Media::Clip::MakeReader(
document, _gifMedia.get(),
_msgId, _msgId,
callback); callback);
} else if (!isListEmpty) { } else if (!isListEmpty) {

View File

@ -23,6 +23,7 @@ class SessionController;
namespace Data { namespace Data {
class Media; class Media;
class DocumentMedia;
} // namespace Data } // namespace Data
namespace Ui { namespace Ui {
@ -94,6 +95,7 @@ private:
bool _doc = false; bool _doc = false;
QPixmap _thumb; QPixmap _thumb;
std::shared_ptr<Data::DocumentMedia> _gifMedia;
Media::Clip::ReaderPointer _gifPreview; Media::Clip::ReaderPointer _gifPreview;
object_ptr<Ui::InputField> _field = { nullptr }; object_ptr<Ui::InputField> _field = { nullptr };

View File

@ -604,7 +604,7 @@ void StickerSetBox::Inner::setupLottie(int index) {
element.animated = Stickers::LottieAnimationFromDocument( element.animated = Stickers::LottieAnimationFromDocument(
getLottiePlayer(), getLottiePlayer(),
document, element.documentMedia.get(),
Stickers::LottieSize::StickerSet, Stickers::LottieSize::StickerSet,
boundingBoxSize() * cIntRetinaFactor()); boundingBoxSize() * cIntRetinaFactor());
} }

View File

@ -642,6 +642,7 @@ StickersBox::Inner::Row::Row(
, accessHash(accessHash) , accessHash(accessHash)
, thumbnail(thumbnail) , thumbnail(thumbnail)
, sticker(sticker) , sticker(sticker)
, stickerMedia(sticker ? sticker->createMediaView() : nullptr)
, count(count) , count(count)
, title(title) , title(title)
, titleWidth(titleWidth) , titleWidth(titleWidth)
@ -947,7 +948,7 @@ void StickersBox::Inner::validateLottieAnimation(not_null<Row*> set) {
} }
auto player = Stickers::LottieThumbnail( auto player = Stickers::LottieThumbnail(
set->thumbnail, set->thumbnail,
set->sticker, set->stickerMedia.get(),
Stickers::LottieSize::SetsListThumbnail, Stickers::LottieSize::SetsListThumbnail,
QSize( QSize(
st::contactsPhotoSize, st::contactsPhotoSize,
@ -1645,6 +1646,7 @@ void StickersBox::Inner::updateRows() {
} }
row->thumbnail = thumbnail; row->thumbnail = thumbnail;
row->sticker = sticker; row->sticker = sticker;
row->stickerMedia = sticker->createMediaView();
row->pixw = pixw; row->pixw = pixw;
row->pixh = pixh; row->pixh = pixh;
} }

View File

@ -233,15 +233,17 @@ private:
bool removed, bool removed,
int32 pixw, int32 pixw,
int32 pixh); int32 pixh);
~Row();
bool isRecentSet() const { bool isRecentSet() const {
return (id == Stickers::CloudRecentSetId); return (id == Stickers::CloudRecentSetId);
} }
~Row();
uint64 id = 0; uint64 id = 0;
uint64 accessHash = 0; uint64 accessHash = 0;
ImagePtr thumbnail; ImagePtr thumbnail;
DocumentData *sticker = nullptr; DocumentData *sticker = nullptr;
std::shared_ptr<Data::DocumentMedia> stickerMedia;
int32 count = 0; int32 count = 0;
QString title; QString title;
int titleWidth = 0; int titleWidth = 0;

View File

@ -1008,7 +1008,7 @@ auto FieldAutocompleteInner::getLottieRenderer()
void FieldAutocompleteInner::setupLottie(StickerSuggestion &suggestion) { void FieldAutocompleteInner::setupLottie(StickerSuggestion &suggestion) {
const auto document = suggestion.document; const auto document = suggestion.document;
suggestion.animated = Stickers::LottiePlayerFromDocument( suggestion.animated = Stickers::LottiePlayerFromDocument(
document, suggestion.documentMedia.get(),
Stickers::LottieSize::InlineResults, Stickers::LottieSize::InlineResults,
stickerBoundingBox() * cIntRetinaFactor(), stickerBoundingBox() * cIntRetinaFactor(),
Lottie::Quality::Default, Lottie::Quality::Default,

View File

@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "stickers.h" #include "stickers.h"
#include "data/data_document.h" #include "data/data_document.h"
#include "data/data_document_media.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "data/data_file_origin.h" #include "data/data_file_origin.h"
#include "boxes/stickers_box.h" #include "boxes/stickers_box.h"
@ -1146,10 +1147,11 @@ auto LottieCachedFromContent(
template <typename Method> template <typename Method>
auto LottieFromDocument( auto LottieFromDocument(
Method &&method, Method &&method,
not_null<DocumentData*> document, not_null<Data::DocumentMedia*> media,
uint8 keyShift, uint8 keyShift,
QSize box) { QSize box) {
const auto data = document->data(); const auto document = media->owner();
const auto data = media->bytes();
const auto filepath = document->filepath(); const auto filepath = document->filepath();
if (box.width() * box.height() > kDontCacheLottieAfterArea) { if (box.width() * box.height() > kDontCacheLottieAfterArea) {
// Don't use frame caching for large stickers. // Don't use frame caching for large stickers.
@ -1172,13 +1174,13 @@ auto LottieFromDocument(
} }
std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument( std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument(
not_null<DocumentData*> document, not_null<Data::DocumentMedia*> media,
LottieSize sizeTag, LottieSize sizeTag,
QSize box, QSize box,
Lottie::Quality quality, Lottie::Quality quality,
std::shared_ptr<Lottie::FrameRenderer> renderer) { std::shared_ptr<Lottie::FrameRenderer> renderer) {
return LottiePlayerFromDocument( return LottiePlayerFromDocument(
document, media,
nullptr, nullptr,
sizeTag, sizeTag,
box, box,
@ -1187,7 +1189,7 @@ std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument(
} }
std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument( std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument(
not_null<DocumentData*> document, not_null<Data::DocumentMedia*> media,
const Lottie::ColorReplacements *replacements, const Lottie::ColorReplacements *replacements,
LottieSize sizeTag, LottieSize sizeTag,
QSize box, QSize box,
@ -1202,18 +1204,18 @@ std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument(
}; };
const auto tag = replacements ? replacements->tag : uint8(0); const auto tag = replacements ? replacements->tag : uint8(0);
const auto keyShift = ((tag << 4) & 0xF0) | (uint8(sizeTag) & 0x0F); const auto keyShift = ((tag << 4) & 0xF0) | (uint8(sizeTag) & 0x0F);
return LottieFromDocument(method, document, uint8(keyShift), box); return LottieFromDocument(method, media, uint8(keyShift), box);
} }
not_null<Lottie::Animation*> LottieAnimationFromDocument( not_null<Lottie::Animation*> LottieAnimationFromDocument(
not_null<Lottie::MultiPlayer*> player, not_null<Lottie::MultiPlayer*> player,
not_null<DocumentData*> document, not_null<Data::DocumentMedia*> media,
LottieSize sizeTag, LottieSize sizeTag,
QSize box) { QSize box) {
const auto method = [&](auto &&...args) { const auto method = [&](auto &&...args) {
return player->append(std::forward<decltype(args)>(args)...); return player->append(std::forward<decltype(args)>(args)...);
}; };
return LottieFromDocument(method, document, uint8(sizeTag), box); return LottieFromDocument(method, media, uint8(sizeTag), box);
} }
bool HasLottieThumbnail( bool HasLottieThumbnail(
@ -1243,19 +1245,20 @@ bool HasLottieThumbnail(
std::unique_ptr<Lottie::SinglePlayer> LottieThumbnail( std::unique_ptr<Lottie::SinglePlayer> LottieThumbnail(
ImagePtr thumbnail, ImagePtr thumbnail,
not_null<DocumentData*> sticker, not_null<Data::DocumentMedia*> media,
LottieSize sizeTag, LottieSize sizeTag,
QSize box, QSize box,
std::shared_ptr<Lottie::FrameRenderer> renderer) { std::shared_ptr<Lottie::FrameRenderer> renderer) {
const auto document = media->owner();
const auto baseKey = thumbnail const auto baseKey = thumbnail
? thumbnail->location().file().bigFileBaseCacheKey() ? thumbnail->location().file().bigFileBaseCacheKey()
: sticker->bigFileBaseCacheKey(); : document->bigFileBaseCacheKey();
if (!baseKey) { if (!baseKey) {
return nullptr; return nullptr;
} }
const auto content = (thumbnail const auto content = (thumbnail
? thumbnail->bytesForCache() ? thumbnail->bytesForCache()
: Lottie::ReadContent(sticker->data(), sticker->filepath())); : Lottie::ReadContent(media->bytes(), document->filepath()));
if (content.isEmpty()) { if (content.isEmpty()) {
return nullptr; return nullptr;
} }
@ -1267,7 +1270,7 @@ std::unique_ptr<Lottie::SinglePlayer> LottieThumbnail(
method, method,
*baseKey, *baseKey,
uint8(sizeTag), uint8(sizeTag),
&sticker->session(), &document->session(),
content, content,
box); box);
} }

View File

@ -12,6 +12,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
class DocumentData; class DocumentData;
namespace Data {
class DocumentMedia;
} // namespace Data
namespace Main { namespace Main {
class Session; class Session;
} // namespace Main } // namespace Main
@ -136,13 +140,13 @@ enum class LottieSize : uchar {
}; };
[[nodiscard]] std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument( [[nodiscard]] std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument(
not_null<DocumentData*> document, not_null<Data::DocumentMedia*> media,
LottieSize sizeTag, LottieSize sizeTag,
QSize box, QSize box,
Lottie::Quality quality = Lottie::Quality(), Lottie::Quality quality = Lottie::Quality(),
std::shared_ptr<Lottie::FrameRenderer> renderer = nullptr); std::shared_ptr<Lottie::FrameRenderer> renderer = nullptr);
[[nodiscard]] std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument( [[nodiscard]] std::unique_ptr<Lottie::SinglePlayer> LottiePlayerFromDocument(
not_null<DocumentData*> document, not_null<Data::DocumentMedia*> media,
const Lottie::ColorReplacements *replacements, const Lottie::ColorReplacements *replacements,
LottieSize sizeTag, LottieSize sizeTag,
QSize box, QSize box,
@ -150,7 +154,7 @@ enum class LottieSize : uchar {
std::shared_ptr<Lottie::FrameRenderer> renderer = nullptr); std::shared_ptr<Lottie::FrameRenderer> renderer = nullptr);
[[nodiscard]] not_null<Lottie::Animation*> LottieAnimationFromDocument( [[nodiscard]] not_null<Lottie::Animation*> LottieAnimationFromDocument(
not_null<Lottie::MultiPlayer*> player, not_null<Lottie::MultiPlayer*> player,
not_null<DocumentData*> document, not_null<Data::DocumentMedia*> media,
LottieSize sizeTag, LottieSize sizeTag,
QSize box); QSize box);
@ -159,7 +163,7 @@ enum class LottieSize : uchar {
not_null<DocumentData*> sticker); not_null<DocumentData*> sticker);
[[nodiscard]] std::unique_ptr<Lottie::SinglePlayer> LottieThumbnail( [[nodiscard]] std::unique_ptr<Lottie::SinglePlayer> LottieThumbnail(
ImagePtr thumbnail, ImagePtr thumbnail,
not_null<DocumentData*> sticker, not_null<Data::DocumentMedia*> media,
LottieSize sizeTag, LottieSize sizeTag,
QSize box, QSize box,
std::shared_ptr<Lottie::FrameRenderer> renderer = nullptr); std::shared_ptr<Lottie::FrameRenderer> renderer = nullptr);

View File

@ -67,6 +67,7 @@ struct StickerIcon {
: setId(setId) : setId(setId)
, thumbnail(thumbnail) , thumbnail(thumbnail)
, sticker(sticker) , sticker(sticker)
, stickerMedia(sticker ? sticker->createMediaView() : nullptr)
, pixw(pixw) , pixw(pixw)
, pixh(pixh) { , pixh(pixh) {
} }
@ -74,6 +75,7 @@ struct StickerIcon {
ImagePtr thumbnail; ImagePtr thumbnail;
mutable Lottie::SinglePlayer *lottie = nullptr; mutable Lottie::SinglePlayer *lottie = nullptr;
DocumentData *sticker = nullptr; DocumentData *sticker = nullptr;
std::shared_ptr<Data::DocumentMedia> stickerMedia;
ChannelData *megagroup = nullptr; ChannelData *megagroup = nullptr;
int pixw = 0; int pixw = 0;
int pixh = 0; int pixh = 0;
@ -704,7 +706,7 @@ void StickersListWidget::Footer::validateIconLottieAnimation(
} }
auto player = Stickers::LottieThumbnail( auto player = Stickers::LottieThumbnail(
icon.thumbnail, icon.thumbnail,
icon.sticker, icon.stickerMedia.get(),
Stickers::LottieSize::StickersFooter, Stickers::LottieSize::StickersFooter,
QSize( QSize(
st::stickerIconWidth - 2 * st::stickerIconPadding, st::stickerIconWidth - 2 * st::stickerIconPadding,
@ -1692,7 +1694,7 @@ void StickersListWidget::setupLottie(Set &set, int section, int index) {
ensureLottiePlayer(set); ensureLottiePlayer(set);
sticker.animated = Stickers::LottieAnimationFromDocument( sticker.animated = Stickers::LottieAnimationFromDocument(
set.lottiePlayer, set.lottiePlayer,
document, sticker.documentMedia.get(),
Stickers::LottieSize::StickersPanel, Stickers::LottieSize::StickersPanel,
boundingBoxSize() * cIntRetinaFactor()); boundingBoxSize() * cIntRetinaFactor());
_lottieData[set.id].items.emplace( _lottieData[set.id].items.emplace(

View File

@ -14,6 +14,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_session.h" #include "data/data_session.h"
#include "data/data_document.h" #include "data/data_document.h"
#include "data/data_file_origin.h" #include "data/data_file_origin.h"
#include "data/data_document_media.h"
#include "main/main_session.h" #include "main/main_session.h"
#include "boxes/confirm_box.h" #include "boxes/confirm_box.h"
#include "core/application.h" // Core::App().showTheme. #include "core/application.h" // Core::App().showTheme.
@ -178,9 +179,11 @@ void CloudThemes::showPreview(const CloudTheme &cloud) {
void CloudThemes::applyFromDocument(const CloudTheme &cloud) { void CloudThemes::applyFromDocument(const CloudTheme &cloud) {
const auto document = _session->data().document(cloud.documentId); const auto document = _session->data().document(cloud.documentId);
loadDocumentAndInvoke(_updatingFrom, cloud, document, [=] { loadDocumentAndInvoke(_updatingFrom, cloud, document, [=](
std::shared_ptr<Data::DocumentMedia> media) {
const auto document = media->owner();
auto preview = Window::Theme::PreviewFromFile( auto preview = Window::Theme::PreviewFromFile(
document->data(), media->bytes(),
document->location().name(), document->location().name(),
cloud); cloud);
if (preview) { if (preview) {
@ -192,7 +195,9 @@ void CloudThemes::applyFromDocument(const CloudTheme &cloud) {
void CloudThemes::previewFromDocument(const CloudTheme &cloud) { void CloudThemes::previewFromDocument(const CloudTheme &cloud) {
const auto document = _session->data().document(cloud.documentId); const auto document = _session->data().document(cloud.documentId);
loadDocumentAndInvoke(_previewFrom, cloud, document, [=] { loadDocumentAndInvoke(_previewFrom, cloud, document, [=](
std::shared_ptr<Data::DocumentMedia> media) {
const auto document = media->owner();
Core::App().showTheme(document, cloud); Core::App().showTheme(document, cloud);
}); });
} }
@ -201,12 +206,13 @@ void CloudThemes::loadDocumentAndInvoke(
LoadingDocument &value, LoadingDocument &value,
const CloudTheme &cloud, const CloudTheme &cloud,
not_null<DocumentData*> document, not_null<DocumentData*> document,
Fn<void()> callback) { Fn<void(std::shared_ptr<Data::DocumentMedia>)> callback) {
const auto alreadyWaiting = (value.document != nullptr); const auto alreadyWaiting = (value.document != nullptr);
if (alreadyWaiting) { if (alreadyWaiting) {
value.document->cancel(); value.document->cancel();
} }
value.document = document; value.document = document;
value.documentMedia = document->createMediaView();
value.document->save( value.document->save(
Data::FileOriginTheme(cloud.id, cloud.accessHash), Data::FileOriginTheme(cloud.id, cloud.accessHash),
QString()); QString());
@ -228,8 +234,9 @@ void CloudThemes::loadDocumentAndInvoke(
void CloudThemes::invokeForLoaded(LoadingDocument &value) { void CloudThemes::invokeForLoaded(LoadingDocument &value) {
const auto onstack = std::move(value.callback); const auto onstack = std::move(value.callback);
auto media = std::move(value.documentMedia);
value = LoadingDocument(); value = LoadingDocument();
onstack(); onstack(std::move(media));
} }
void CloudThemes::scheduleReload() { void CloudThemes::scheduleReload() {

View File

@ -17,6 +17,8 @@ class Session;
namespace Data { namespace Data {
class DocumentMedia;
struct CloudTheme { struct CloudTheme {
uint64 id = 0; uint64 id = 0;
uint64 accessHash = 0; uint64 accessHash = 0;
@ -54,8 +56,9 @@ private:
struct LoadingDocument { struct LoadingDocument {
CloudTheme theme; CloudTheme theme;
DocumentData *document = nullptr; DocumentData *document = nullptr;
std::shared_ptr<Data::DocumentMedia> documentMedia;
rpl::lifetime subscription; rpl::lifetime subscription;
Fn<void()> callback; Fn<void(std::shared_ptr<Data::DocumentMedia>)> callback;
}; };
void parseThemes(const QVector<MTPTheme> &list); void parseThemes(const QVector<MTPTheme> &list);
@ -71,7 +74,7 @@ private:
LoadingDocument &value, LoadingDocument &value,
const CloudTheme &cloud, const CloudTheme &cloud,
not_null<DocumentData*> document, not_null<DocumentData*> document,
Fn<void()> callback); Fn<void(std::shared_ptr<Data::DocumentMedia>)> callback);
void invokeForLoaded(LoadingDocument &value); void invokeForLoaded(LoadingDocument &value);
const not_null<Main::Session*> _session; const not_null<Main::Session*> _session;

View File

@ -587,7 +587,10 @@ void DocumentData::validateLottieSticker() {
} }
void DocumentData::setDataAndCache(const QByteArray &data) { void DocumentData::setDataAndCache(const QByteArray &data) {
setData(data); _data = data;
if (const auto media = activeMediaView()) {
media->setBytes(data);
}
if (saveToCache() && data.size() <= Storage::kMaxFileInMemory) { if (saveToCache() && data.size() <= Storage::kMaxFileInMemory) {
owner().cache().put( owner().cache().put(
cacheKey(), cacheKey(),
@ -734,11 +737,12 @@ auto DocumentData::bigFileBaseCacheKey() const
} }
bool DocumentData::saveToCache() const { bool DocumentData::saveToCache() const {
return (type == StickerDocument && size < Storage::kMaxStickerInMemory) return (size < Storage::kMaxFileInMemory)
|| (isAnimation() && size < Storage::kMaxAnimationInMemory) && ((type == StickerDocument)
|| (isVoiceMessage() && size < Storage::kMaxVoiceInMemory) || isAnimation()
|| (type == WallPaperDocument) || isVoiceMessage()
|| (isTheme() && size < Storage::kMaxFileInMemory); || (type == WallPaperDocument)
|| isTheme());
} }
void DocumentData::unload() { void DocumentData::unload() {
@ -814,6 +818,7 @@ bool DocumentData::loaded(FilePathResolve resolve) const {
that->setGoodThumbnailDataReady(); that->setGoodThumbnailDataReady();
if (const auto media = activeMediaView()) { if (const auto media = activeMediaView()) {
media->setBytes(_loader->bytes());
media->checkStickerLarge(_loader.get()); media->checkStickerLarge(_loader.get());
} }
destroyLoader(); destroyLoader();
@ -824,7 +829,7 @@ bool DocumentData::loaded(FilePathResolve resolve) const {
} }
_owner->notifyDocumentLayoutChanged(this); _owner->notifyDocumentLayoutChanged(this);
} }
return !data().isEmpty() || !filepath(resolve).isEmpty(); return !rawBytes().isEmpty() || !filepath(resolve).isEmpty();
} }
void DocumentData::destroyLoader() const { void DocumentData::destroyLoader() const {
@ -918,10 +923,10 @@ void DocumentData::save(
if (loaded(FilePathResolve::Checked)) { if (loaded(FilePathResolve::Checked)) {
auto &l = location(true); auto &l = location(true);
if (!toFile.isEmpty()) { if (!toFile.isEmpty()) {
if (!_data.isEmpty()) { if (!rawBytes().isEmpty()) {
QFile f(toFile); QFile f(toFile);
f.open(QIODevice::WriteOnly); f.open(QIODevice::WriteOnly);
f.write(_data); f.write(rawBytes());
f.close(); f.close();
setLocation(FileLocation(toFile)); setLocation(FileLocation(toFile));
@ -1085,7 +1090,7 @@ QByteArray documentWaveformEncode5bit(const VoiceWaveform &waveform) {
return result; return result;
} }
QByteArray DocumentData::data() const { QByteArray DocumentData::rawBytes() const {
if (!_data.isEmpty()) { if (!_data.isEmpty()) {
ActiveCache().up(const_cast<DocumentData*>(this)); ActiveCache().up(const_cast<DocumentData*>(this));
} }
@ -1115,8 +1120,10 @@ void DocumentData::setLocation(const FileLocation &loc) {
QString DocumentData::filepath(FilePathResolve resolve) const { QString DocumentData::filepath(FilePathResolve resolve) const {
bool check = (resolve != FilePathResolve::Cached); bool check = (resolve != FilePathResolve::Cached);
QString result = (check && _location.name().isEmpty()) ? QString() : location(check).name(); QString result = (check && _location.name().isEmpty())
bool saveFromData = result.isEmpty() && !data().isEmpty(); ? QString()
: location(check).name();
bool saveFromData = result.isEmpty() && !rawBytes().isEmpty();
if (saveFromData) { if (saveFromData) {
if (resolve != FilePathResolve::SaveFromData if (resolve != FilePathResolve::SaveFromData
&& resolve != FilePathResolve::SaveFromDataSilent) { && resolve != FilePathResolve::SaveFromDataSilent) {
@ -1131,7 +1138,7 @@ QString DocumentData::filepath(FilePathResolve resolve) const {
if (!filename.isEmpty()) { if (!filename.isEmpty()) {
QFile f(filename); QFile f(filename);
if (f.open(QIODevice::WriteOnly)) { if (f.open(QIODevice::WriteOnly)) {
if (f.write(data()) == data().size()) { if (f.write(rawBytes()) == rawBytes().size()) {
f.close(); f.close();
const_cast<DocumentData*>(this)->_location = FileLocation(filename); const_cast<DocumentData*>(this)->_location = FileLocation(filename);
Local::writeFileLocation(mediaKey(), _location); Local::writeFileLocation(mediaKey(), _location);
@ -1263,8 +1270,8 @@ auto DocumentData::createStreamingLoader(
} }
if (!forceRemoteLoader) { if (!forceRemoteLoader) {
const auto &location = this->location(true); const auto &location = this->location(true);
if (!data().isEmpty()) { if (!rawBytes().isEmpty()) {
return Media::Streaming::MakeBytesLoader(data()); return Media::Streaming::MakeBytesLoader(rawBytes());
} else if (!location.isEmpty() && location.accessEnable()) { } else if (!location.isEmpty() && location.accessEnable()) {
auto result = Media::Streaming::MakeFileLoader(location.name()); auto result = Media::Streaming::MakeFileLoader(location.name());
location.accessDisable(); location.accessDisable();
@ -1545,6 +1552,9 @@ void DocumentData::collectLocalData(not_null<DocumentData*> local) {
if (!local->_data.isEmpty()) { if (!local->_data.isEmpty()) {
ActiveCache().decrement(_data.size()); ActiveCache().decrement(_data.size());
_data = local->_data; _data = local->_data;
if (const auto media = activeMediaView()) {
media->setBytes(local->_data);
}
ActiveCache().increment(_data.size()); ActiveCache().increment(_data.size());
if (!_data.isEmpty()) { if (!_data.isEmpty()) {
ActiveCache().up(this); ActiveCache().up(this);
@ -1642,7 +1652,7 @@ base::binary_guard ReadImageAsync(
FnMut<void(QImage&&)> done) { FnMut<void(QImage&&)> done) {
auto result = base::binary_guard(); auto result = base::binary_guard();
crl::async([ crl::async([
bytes = document->data(), bytes = document->rawBytes(),
path = document->filepath(), path = document->filepath(),
postprocess = std::move(postprocess), postprocess = std::move(postprocess),
guard = result.make_guard(), guard = result.make_guard(),
@ -1672,30 +1682,4 @@ base::binary_guard ReadImageAsync(
return result; return result;
} }
//void HandleUnsupportedMedia(
// not_null<DocumentData*> document,
// FullMsgId contextId) {
// using Error = ::Media::Streaming::Error;
//
// document->setInappPlaybackFailed();
// const auto filepath = document->filepath(
// DocumentData::FilePathResolve::SaveFromData);
// if (filepath.isEmpty()) {
// const auto save = [=] {
// Ui::hideLayer();
// DocumentSaveClickHandler::Save(
// (contextId ? contextId : Data::FileOrigin()),
// document,
// document->owner().message(contextId));
// };
// Ui::show(Box<ConfirmBox>(
// tr::lng_player_cant_stream(tr::now),
// tr::lng_player_download(tr::now),
// tr::lng_cancel(tr::now),
// save));
// } else if (IsValidMediaFile(filepath)) {
// File::Launch(filepath);
// }
//}
} // namespace Data } // namespace Data

View File

@ -127,7 +127,7 @@ public:
void setWaitingForAlbum(); void setWaitingForAlbum();
[[nodiscard]] bool waitingForAlbum() const; [[nodiscard]] bool waitingForAlbum() const;
[[nodiscard]] QByteArray data() const; [[nodiscard]] QByteArray rawBytes() const;
[[nodiscard]] const FileLocation &location(bool check = false) const; [[nodiscard]] const FileLocation &location(bool check = false) const;
void setLocation(const FileLocation &loc); void setLocation(const FileLocation &loc);
@ -163,9 +163,6 @@ public:
void recountIsImage(); void recountIsImage();
[[nodiscard]] bool supportsStreaming() const; [[nodiscard]] bool supportsStreaming() const;
void setNotSupportsStreaming(); void setNotSupportsStreaming();
void setData(const QByteArray &data) {
_data = data;
}
void setDataAndCache(const QByteArray &data); void setDataAndCache(const QByteArray &data);
bool checkWallPaperProperties(); bool checkWallPaperProperties();
[[nodiscard]] bool isWallPaper() const; [[nodiscard]] bool isWallPaper() const;

View File

@ -82,11 +82,16 @@ enum class FileType {
} // namespace } // namespace
DocumentMedia::DocumentMedia(not_null<DocumentData*> owner) DocumentMedia::DocumentMedia(not_null<DocumentData*> owner)
: _owner(owner) { : _owner(owner)
, _bytes(owner->rawBytes()) {
} }
DocumentMedia::~DocumentMedia() = default; DocumentMedia::~DocumentMedia() = default;
not_null<DocumentData*> DocumentMedia::owner() const {
return _owner;
}
void DocumentMedia::goodThumbnailWanted() { void DocumentMedia::goodThumbnailWanted() {
_flags |= Flag::GoodThumbnailWanted; _flags |= Flag::GoodThumbnailWanted;
} }
@ -135,7 +140,7 @@ void DocumentMedia::checkStickerLarge() {
if (data->animated || !_owner->loaded()) { if (data->animated || !_owner->loaded()) {
return; return;
} }
const auto bytes = _owner->data(); const auto bytes = _owner->rawBytes();
if (bytes.isEmpty()) { if (bytes.isEmpty()) {
const auto &loc = _owner->location(true); const auto &loc = _owner->location(true);
if (loc.accessEnable()) { if (loc.accessEnable()) {
@ -155,6 +160,16 @@ void DocumentMedia::checkStickerLarge() {
} }
} }
void DocumentMedia::setBytes(const QByteArray &bytes) {
if (!bytes.isEmpty()) {
_bytes = bytes;
}
}
QByteArray DocumentMedia::bytes() const {
return _bytes;
}
void DocumentMedia::checkStickerSmall() { void DocumentMedia::checkStickerSmall() {
const auto data = _owner->sticker(); const auto data = _owner->sticker();
if ((data && data->animated) || _owner->thumbnailEnoughForSticker()) { if ((data && data->animated) || _owner->thumbnailEnoughForSticker()) {
@ -184,18 +199,18 @@ void DocumentMedia::checkStickerLarge(not_null<FileLoader*> loader) {
if (_owner->sticker() if (_owner->sticker()
&& !_sticker && !_sticker
&& !loader->imageData().isNull() && !loader->imageData().isNull()
&& !_owner->data().isEmpty()) { && !_owner->rawBytes().isEmpty()) {
_sticker = std::make_unique<Image>( _sticker = std::make_unique<Image>(
std::make_unique<Images::LocalFileSource>( std::make_unique<Images::LocalFileSource>(
QString(), QString(),
_owner->data(), _owner->rawBytes(),
loader->imageFormat(), loader->imageFormat(),
loader->imageData())); loader->imageData()));
} }
} }
void DocumentMedia::GenerateGoodThumbnail(not_null<DocumentData*> document) { void DocumentMedia::GenerateGoodThumbnail(not_null<DocumentData*> document) {
const auto data = document->data(); const auto data = document->rawBytes();
const auto type = document->isWallPaper() const auto type = document->isWallPaper()
? FileType::WallPaper ? FileType::WallPaper
: document->isTheme() : document->isTheme()

View File

@ -18,6 +18,8 @@ public:
explicit DocumentMedia(not_null<DocumentData*> owner); explicit DocumentMedia(not_null<DocumentData*> owner);
~DocumentMedia(); ~DocumentMedia();
[[nodiscard]] not_null<DocumentData*> owner() const;
void goodThumbnailWanted(); void goodThumbnailWanted();
[[nodiscard]] Image *goodThumbnail() const; [[nodiscard]] Image *goodThumbnail() const;
void setGoodThumbnail(QImage thumbnail); void setGoodThumbnail(QImage thumbnail);
@ -30,6 +32,9 @@ public:
[[nodiscard]] Image *getStickerLarge(); [[nodiscard]] Image *getStickerLarge();
void checkStickerLarge(not_null<FileLoader*> loader); void checkStickerLarge(not_null<FileLoader*> loader);
void setBytes(const QByteArray &bytes);
[[nodiscard]] QByteArray bytes() const;
// For DocumentData. // For DocumentData.
static void CheckGoodThumbnail(not_null<DocumentData*> document); static void CheckGoodThumbnail(not_null<DocumentData*> document);
@ -47,6 +52,7 @@ private:
std::unique_ptr<Image> _goodThumbnail; std::unique_ptr<Image> _goodThumbnail;
mutable std::unique_ptr<Image> _inlineThumbnail; mutable std::unique_ptr<Image> _inlineThumbnail;
std::unique_ptr<Image> _sticker; std::unique_ptr<Image> _sticker;
QByteArray _bytes;
Flags _flags; Flags _flags;
}; };

View File

@ -386,6 +386,8 @@ void Document::draw(Painter &p, const QRect &r, TextSelection selection, crl::ti
auto voiceStatusOverride = QString(); auto voiceStatusOverride = QString();
if (const auto voice = Get<HistoryDocumentVoice>()) { if (const auto voice = Get<HistoryDocumentVoice>()) {
ensureDataMediaCreated();
const VoiceWaveform *wf = nullptr; const VoiceWaveform *wf = nullptr;
uchar norm_value = 0; uchar norm_value = 0;
if (const auto voiceData = _data->voice()) { if (const auto voiceData = _data->voice()) {
@ -393,7 +395,7 @@ void Document::draw(Painter &p, const QRect &r, TextSelection selection, crl::ti
if (wf->isEmpty()) { if (wf->isEmpty()) {
wf = nullptr; wf = nullptr;
if (loaded) { if (loaded) {
Local::countVoiceWaveform(_data); Local::countVoiceWaveform(_dataMedia.get());
} }
} else if (wf->at(0) < 0) { } else if (wf->at(0) < 0) {
wf = nullptr; wf = nullptr;

View File

@ -271,8 +271,10 @@ void Sticker::setDiceIndex(const QString &emoji, int index) {
} }
void Sticker::setupLottie() { void Sticker::setupLottie() {
Expects(_dataMedia != nullptr);
_lottie = Stickers::LottiePlayerFromDocument( _lottie = Stickers::LottiePlayerFromDocument(
_data, _dataMedia.get(),
_replacements, _replacements,
Stickers::LottieSize::MessageHistory, Stickers::LottieSize::MessageHistory,
_size * cIntRetinaFactor(), _size * cIntRetinaFactor(),

View File

@ -146,6 +146,9 @@ int Gif::resizeGetHeight(int width) {
void Gif::paint(Painter &p, const QRect &clip, const PaintContext *context) const { void Gif::paint(Painter &p, const QRect &clip, const PaintContext *context) const {
const auto document = getShownDocument(); const auto document = getShownDocument();
if (document) {
ensureDataMediaCreated(document);
}
document->automaticLoad(fileOrigin(), nullptr); document->automaticLoad(fileOrigin(), nullptr);
bool loaded = document->loaded(), loading = document->loading(), displayLoading = document->displayLoading(); bool loaded = document->loaded(), loading = document->loading(), displayLoading = document->displayLoading();
@ -154,7 +157,7 @@ void Gif::paint(Painter &p, const QRect &clip, const PaintContext *context) cons
&& !_gif.isBad() && !_gif.isBad()
&& CanPlayInline(document)) { && CanPlayInline(document)) {
auto that = const_cast<Gif*>(this); auto that = const_cast<Gif*>(this);
that->_gif = Media::Clip::MakeReader(document, FullMsgId(), [that](Media::Clip::Notification notification) { that->_gif = Media::Clip::MakeReader(_dataMedia.get(), FullMsgId(), [that](Media::Clip::Notification notification) {
that->clipCallback(notification); that->clipCallback(notification);
}); });
if (_gif) _gif->setAutoplay(); if (_gif) _gif->setAutoplay();
@ -508,9 +511,11 @@ QSize Sticker::getThumbSize() const {
return QSize(qMax(w, 1), qMax(h, 1)); return QSize(qMax(w, 1), qMax(h, 1));
} }
void Sticker::setupLottie(not_null<DocumentData*> document) const { void Sticker::setupLottie() const {
Expects(_dataMedia != nullptr);
_lottie = Stickers::LottiePlayerFromDocument( _lottie = Stickers::LottiePlayerFromDocument(
document, _dataMedia.get(),
Stickers::LottieSize::InlineResults, Stickers::LottieSize::InlineResults,
QSize( QSize(
st::stickerPanSize.width() - st::buttonRadius * 2, st::stickerPanSize.width() - st::buttonRadius * 2,
@ -530,7 +535,7 @@ void Sticker::prepareThumbnail() const {
&& document->sticker() && document->sticker()
&& document->sticker()->animated && document->sticker()->animated
&& document->loaded()) { && document->loaded()) {
setupLottie(document); setupLottie();
} }
_dataMedia->checkStickerSmall(); _dataMedia->checkStickerSmall();
if (const auto sticker = _dataMedia->getStickerSmall()) { if (const auto sticker = _dataMedia->getStickerSmall()) {
@ -1318,7 +1323,10 @@ void Game::paint(Painter &p, const QRect &clip, const PaintContext *context) con
// Gif thumb // Gif thumb
auto thumbDisplayed = false, radial = false; auto thumbDisplayed = false, radial = false;
auto document = getResultDocument(); const auto document = getResultDocument();
if (document) {
ensureDataMediaCreated(document);
}
auto animatedThumb = document && document->isAnimation(); auto animatedThumb = document && document->isAnimation();
if (animatedThumb) { if (animatedThumb) {
document->automaticLoad(fileOrigin(), nullptr); document->automaticLoad(fileOrigin(), nullptr);
@ -1326,7 +1334,7 @@ void Game::paint(Painter &p, const QRect &clip, const PaintContext *context) con
bool loaded = document->loaded(), loading = document->loading(), displayLoading = document->displayLoading(); bool loaded = document->loaded(), loading = document->loading(), displayLoading = document->displayLoading();
if (loaded && !_gif && !_gif.isBad()) { if (loaded && !_gif && !_gif.isBad()) {
auto that = const_cast<Game*>(this); auto that = const_cast<Game*>(this);
that->_gif = Media::Clip::MakeReader(document, FullMsgId(), [that](Media::Clip::Notification notification) { that->_gif = Media::Clip::MakeReader(_dataMedia.get(), FullMsgId(), [that](Media::Clip::Notification notification) {
that->clipCallback(notification); that->clipCallback(notification);
}); });
if (_gif) _gif->setAutoplay(); if (_gif) _gif->setAutoplay();
@ -1403,7 +1411,7 @@ void Game::prepareThumbnail(QSize size) const {
validateThumbnail(photo->thumbnail(), size, true); validateThumbnail(photo->thumbnail(), size, true);
validateThumbnail(photo->thumbnailInline(), size, false); validateThumbnail(photo->thumbnailInline(), size, false);
} else if (const auto document = getResultDocument()) { } else if (const auto document = getResultDocument()) {
ensureDataMediaCreated(document); Assert(_dataMedia != nullptr);
validateThumbnail(document->thumbnail(), size, true); validateThumbnail(document->thumbnail(), size, true);
validateThumbnail(_dataMedia->thumbnailInline(), size, false); validateThumbnail(_dataMedia->thumbnailInline(), size, false);
} }

View File

@ -193,7 +193,7 @@ public:
private: private:
void ensureDataMediaCreated(not_null<DocumentData*> document) const; void ensureDataMediaCreated(not_null<DocumentData*> document) const;
void setupLottie(not_null<DocumentData*> document) const; void setupLottie() const;
QSize getThumbSize() const; QSize getThumbSize() const;
void prepareThumbnail() const; void prepareThumbnail() const;

View File

@ -761,7 +761,6 @@ void Mixer::play(
auto type = audio.type(); auto type = audio.type();
AudioMsgId stopped; AudioMsgId stopped;
auto notLoadedYet = false;
{ {
QMutexLocker lock(&AudioMutex); QMutexLocker lock(&AudioMutex);
Audio::AttachToDevice(); Audio::AttachToDevice();
@ -799,42 +798,18 @@ void Mixer::play(
current->state.id = audio; current->state.id = audio;
current->lastUpdateWhen = 0; current->lastUpdateWhen = 0;
current->lastUpdatePosition = 0; current->lastUpdatePosition = 0;
if (externalData) { current->setExternalData(std::move(externalData));
current->setExternalData(std::move(externalData)); current->state.position = (positionMs * current->state.frequency)
} else { / 1000LL;
current->setExternalData(nullptr); current->state.state = current->externalData
current->file = audio.audio()->location(true); ? State::Paused
current->data = audio.audio()->data(); : fadedStart
notLoadedYet = (current->file.isEmpty() && current->data.isEmpty()); ? State::Starting
} : State::Playing;
if (notLoadedYet) { current->loading = true;
auto newState = (type == AudioMsgId::Type::Song) emit loaderOnStart(current->state.id, positionMs);
? State::Stopped if (type == AudioMsgId::Type::Voice) {
: State::StoppedAtError; emit suppressSong();
setStoppedState(current, newState);
} else {
current->state.position = (positionMs * current->state.frequency)
/ 1000LL;
current->state.state = current->externalData
? State::Paused
: fadedStart
? State::Starting
: State::Playing;
current->loading = true;
emit loaderOnStart(current->state.id, positionMs);
if (type == AudioMsgId::Type::Voice) {
emit suppressSong();
}
}
}
if (notLoadedYet) {
if (type == AudioMsgId::Type::Song || type == AudioMsgId::Type::Video) {
DocumentOpenClickHandler::Open(
audio.contextId(),
audio.audio(),
Auth().data().message(audio.contextId()));
} else {
onError(audio);
} }
} }
if (stopped) { if (stopped) {

View File

@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "media/clip/media_clip_reader.h" #include "media/clip/media_clip_reader.h"
#include "data/data_document.h" #include "data/data_document.h"
#include "data/data_document_media.h"
#include "storage/file_download.h" #include "storage/file_download.h"
#include "media/clip/media_clip_ffmpeg.h" #include "media/clip/media_clip_ffmpeg.h"
#include "media/clip/media_clip_check_streaming.h" #include "media/clip/media_clip_check_streaming.h"
@ -91,15 +92,20 @@ Reader::Reader(const QString &filepath, Callback &&callback, Mode mode, crl::tim
init(FileLocation(filepath), QByteArray()); init(FileLocation(filepath), QByteArray());
} }
Reader::Reader(not_null<DocumentData*> document, FullMsgId msgId, Callback &&callback, Mode mode, crl::time seekMs) Reader::Reader(
not_null<Data::DocumentMedia*> media,
FullMsgId msgId,
Callback &&callback,
Mode mode,
crl::time seekMs)
: _callback(std::move(callback)) : _callback(std::move(callback))
, _mode(mode) , _mode(mode)
, _audioMsgId( , _audioMsgId(
document, media->owner(),
msgId, msgId,
(mode == Mode::Video) ? AudioMsgId::CreateExternalPlayId() : 0) (mode == Mode::Video) ? AudioMsgId::CreateExternalPlayId() : 0)
, _seekPositionMs(seekMs) { , _seekPositionMs(seekMs) {
init(document->location(), document->data()); init(media->owner()->location(), media->bytes());
} }
Reader::Reader(const QByteArray &data, Callback &&callback, Mode mode, crl::time seekMs) Reader::Reader(const QByteArray &data, Callback &&callback, Mode mode, crl::time seekMs)

View File

@ -14,6 +14,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
class FileLocation; class FileLocation;
namespace Data {
class DocumentMedia;
} // namespace Data
namespace Media { namespace Media {
namespace Clip { namespace Clip {
@ -56,8 +60,8 @@ public:
Video, Video,
}; };
Reader(not_null<Data::DocumentMedia*> media, FullMsgId msgId, Callback &&callback, Mode mode = Mode::Gif, crl::time seekMs = 0);
Reader(const QString &filepath, Callback &&callback, Mode mode = Mode::Gif, crl::time seekMs = 0); Reader(const QString &filepath, Callback &&callback, Mode mode = Mode::Gif, crl::time seekMs = 0);
Reader(not_null<DocumentData*> document, FullMsgId msgId, Callback &&callback, Mode mode = Mode::Gif, crl::time seekMs = 0);
Reader(const QByteArray &data, Callback &&callback, Mode mode = Mode::Gif, crl::time seekMs = 0); Reader(const QByteArray &data, Callback &&callback, Mode mode = Mode::Gif, crl::time seekMs = 0);
// Reader can be already deleted. // Reader can be already deleted.

View File

@ -1220,8 +1220,9 @@ void OverlayWidget::notifyFileDialogShown(bool shown) {
void OverlayWidget::onSaveAs() { void OverlayWidget::onSaveAs() {
QString file; QString file;
if (_doc) { if (_doc) {
const FileLocation &location(_doc->location(true)); const auto &location = _doc->location(true);
if (!_doc->data().isEmpty() || location.accessEnable()) { const auto bytes = _docMedia->bytes();
if (!bytes.isEmpty() || location.accessEnable()) {
QFileInfo alreadyInfo(location.name()); QFileInfo alreadyInfo(location.name());
QDir alreadyDir(alreadyInfo.dir()); QDir alreadyDir(alreadyInfo.dir());
QString name = alreadyInfo.fileName(), filter; QString name = alreadyInfo.fileName(), filter;
@ -1240,17 +1241,19 @@ void OverlayWidget::onSaveAs() {
file = FileNameForSave(tr::lng_save_file(tr::now), filter, qsl("doc"), name, true, alreadyDir); file = FileNameForSave(tr::lng_save_file(tr::now), filter, qsl("doc"), name, true, alreadyDir);
if (!file.isEmpty() && file != location.name()) { if (!file.isEmpty() && file != location.name()) {
if (_doc->data().isEmpty()) { if (bytes.isEmpty()) {
QFile(file).remove(); QFile(file).remove();
QFile(location.name()).copy(file); QFile(location.name()).copy(file);
} else { } else {
QFile f(file); QFile f(file);
f.open(QIODevice::WriteOnly); f.open(QIODevice::WriteOnly);
f.write(_doc->data()); f.write(bytes);
} }
} }
if (_doc->data().isEmpty()) location.accessDisable(); if (bytes.isEmpty()) {
location.accessDisable();
}
} else { } else {
DocumentSaveClickHandler::Save( DocumentSaveClickHandler::Save(
fileOrigin(), fileOrigin(),
@ -2330,7 +2333,7 @@ void OverlayWidget::initThemePreview() {
Assert(_doc && _doc->isTheme()); Assert(_doc && _doc->isTheme());
const auto bytes = _doc->data(); const auto bytes = _docMedia->bytes();
auto &location = _doc->location(); auto &location = _doc->location();
if (bytes.isEmpty() if (bytes.isEmpty()
&& (location.isEmpty() || !location.accessEnable())) { && (location.isEmpty() || !location.accessEnable())) {

View File

@ -30,11 +30,8 @@ struct Key;
// This value is used in local cache database settings! // This value is used in local cache database settings!
constexpr auto kMaxFileInMemory = 10 * 1024 * 1024; constexpr auto kMaxFileInMemory = 10 * 1024 * 1024;
// 2 MB audio is hold in memory and auto loaded
constexpr auto kMaxVoiceInMemory = 2 * 1024 * 1024;
// 2 MB stickers hold in memory, auto loaded and displayed inline // 2 MB stickers hold in memory, auto loaded and displayed inline
constexpr auto kMaxStickerInMemory = 2 * 1024 * 1024; constexpr auto kMaxStickerBytesSize = 2 * 1024 * 1024;
// 10 MB GIF and mp4 animations held in memory while playing // 10 MB GIF and mp4 animations held in memory while playing
constexpr auto kMaxWallPaperInMemory = kMaxFileInMemory; constexpr auto kMaxWallPaperInMemory = kMaxFileInMemory;

View File

@ -836,7 +836,7 @@ void FileLoadTask::process() {
&& (h > 0) && (h > 0)
&& (w <= StickerMaxSize) && (w <= StickerMaxSize)
&& (h <= StickerMaxSize) && (h <= StickerMaxSize)
&& (filesize < Storage::kMaxStickerInMemory); && (filesize < Storage::kMaxStickerBytesSize);
if (isSticker) { if (isSticker) {
attributes.push_back(MTP_documentAttributeSticker( attributes.push_back(MTP_documentAttributeSticker(
MTP_flags(0), MTP_flags(0),

View File

@ -14,6 +14,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "chat_helpers/stickers.h" #include "chat_helpers/stickers.h"
#include "data/data_drafts.h" #include "data/data_drafts.h"
#include "data/data_user.h" #include "data/data_user.h"
#include "data/data_session.h"
#include "data/data_document_media.h"
#include "boxes/send_files_box.h" #include "boxes/send_files_box.h"
#include "base/flags.h" #include "base/flags.h"
#include "base/platform/base_platform_file_utilities.h" #include "base/platform/base_platform_file_utilities.h"
@ -37,7 +39,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "main/main_session.h" #include "main/main_session.h"
#include "window/themes/window_theme.h" #include "window/themes/window_theme.h"
#include "window/window_session_controller.h" #include "window/window_session_controller.h"
#include "data/data_session.h"
#include "history/history.h" #include "history/history.h"
#include "facades.h" #include "facades.h"
@ -3358,11 +3359,11 @@ Storage::Cache::Database::Settings cacheBigFileSettings() {
class CountWaveformTask : public Task { class CountWaveformTask : public Task {
public: public:
CountWaveformTask(DocumentData *doc) CountWaveformTask(not_null<Data::DocumentMedia*> media)
: _doc(doc) : _doc(media->owner())
, _loc(doc->location(true)) , _loc(_doc->location(true))
, _data(doc->data()) , _data(media->bytes())
, _wavemax(0) { , _wavemax(0) {
if (_data.isEmpty() && !_loc.accessEnable()) { if (_data.isEmpty() && !_loc.accessEnable()) {
_doc = nullptr; _doc = nullptr;
} }
@ -3399,7 +3400,7 @@ public:
} }
protected: protected:
DocumentData *_doc; DocumentData *_doc = nullptr;
FileLocation _loc; FileLocation _loc;
QByteArray _data; QByteArray _data;
VoiceWaveform _waveform; VoiceWaveform _waveform;
@ -3407,13 +3408,14 @@ protected:
}; };
void countVoiceWaveform(DocumentData *document) { void countVoiceWaveform(not_null<Data::DocumentMedia*> media) {
const auto document = media->owner();
if (const auto voice = document->voice()) { if (const auto voice = document->voice()) {
if (_localLoader) { if (_localLoader) {
voice->waveform.resize(1 + sizeof(TaskId)); voice->waveform.resize(1 + sizeof(TaskId));
voice->waveform[0] = -1; // counting voice->waveform[0] = -1; // counting
TaskId taskId = _localLoader->addTask( TaskId taskId = _localLoader->addTask(
std::make_unique<CountWaveformTask>(document)); std::make_unique<CountWaveformTask>(media));
memcpy(voice->waveform.data() + 1, &taskId, sizeof(taskId)); memcpy(voice->waveform.data() + 1, &taskId, sizeof(taskId));
} }
} }

View File

@ -18,6 +18,7 @@ class History;
namespace Data { namespace Data {
class WallPaper; class WallPaper;
class DocumentMedia;
} // namespace Data } // namespace Data
namespace Lang { namespace Lang {
@ -129,7 +130,7 @@ Storage::EncryptionKey cacheBigFileKey();
QString cacheBigFilePath(); QString cacheBigFilePath();
Storage::Cache::Database::Settings cacheBigFileSettings(); Storage::Cache::Database::Settings cacheBigFileSettings();
void countVoiceWaveform(DocumentData *document); void countVoiceWaveform(not_null<Data::DocumentMedia*> media);
void cancelTask(TaskId id); void cancelTask(TaskId id);

View File

@ -15,6 +15,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_cloud_themes.h" #include "data/data_cloud_themes.h"
#include "data/data_file_origin.h" #include "data/data_file_origin.h"
#include "data/data_document.h" #include "data/data_document.h"
#include "data/data_document_media.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "ui/image/image_prepare.h" #include "ui/image/image_prepare.h"
#include "ui/widgets/popup_menu.h" #include "ui/widgets/popup_menu.h"
@ -560,6 +561,7 @@ void CloudList::refreshColors(Element &element) {
&& (!document || !document->isTheme()))) { && (!document || !document->isTheme()))) {
element.check->setColors(ColorsFromCurrentTheme()); element.check->setColors(ColorsFromCurrentTheme());
} else if (document) { } else if (document) {
element.media = document ? document->createMediaView() : nullptr;
document->save( document->save(
Data::FileOriginTheme(theme.id, theme.accessHash), Data::FileOriginTheme(theme.id, theme.accessHash),
QString()); QString());
@ -639,7 +641,9 @@ void CloudList::refreshColorsFromDocument(
not_null<DocumentData*> document) { not_null<DocumentData*> document) {
const auto id = element.id(); const auto id = element.id();
const auto path = document->filepath(); const auto path = document->filepath();
const auto data = document->data(); const auto data = element.media
? base::take(element.media)->bytes()
: QByteArray();
crl::async([=, guard = element.generating.make_guard()]() mutable { crl::async([=, guard = element.generating.make_guard()]() mutable {
crl::on_main(std::move(guard), [ crl::on_main(std::move(guard), [
=, =,

View File

@ -88,6 +88,7 @@ private:
Data::CloudTheme theme; Data::CloudTheme theme;
not_null<CloudListCheck*> check; not_null<CloudListCheck*> check;
std::unique_ptr<Ui::Radiobutton> button; std::unique_ptr<Ui::Radiobutton> button;
std::shared_ptr<Data::DocumentMedia> media;
base::binary_guard generating; base::binary_guard generating;
bool waiting = false; bool waiting = false;

View File

@ -226,7 +226,7 @@ void MediaPreviewWidget::setupLottie() {
Expects(_document != nullptr); Expects(_document != nullptr);
_lottie = std::make_unique<Lottie::SinglePlayer>( _lottie = std::make_unique<Lottie::SinglePlayer>(
Lottie::ReadContent(_document->data(), _document->filepath()), Lottie::ReadContent(_documentMedia->bytes(), _document->filepath()),
Lottie::FrameRequest{ currentDimensions() * cIntRetinaFactor() }, Lottie::FrameRequest{ currentDimensions() * cIntRetinaFactor() },
Lottie::Quality::High); Lottie::Quality::High);
@ -266,7 +266,7 @@ QPixmap MediaPreviewWidget::currentImage() const {
if (_document->loaded()) { if (_document->loaded()) {
if (!_gif && !_gif.isBad()) { if (!_gif && !_gif.isBad()) {
auto that = const_cast<MediaPreviewWidget*>(this); auto that = const_cast<MediaPreviewWidget*>(this);
that->_gif = Media::Clip::MakeReader(_document, FullMsgId(), [=](Media::Clip::Notification notification) { that->_gif = Media::Clip::MakeReader(_documentMedia.get(), FullMsgId(), [=](Media::Clip::Notification notification) {
that->clipCallback(notification); that->clipCallback(notification);
}); });
if (_gif) _gif->setAutoplay(); if (_gif) _gif->setAutoplay();