Fix preloading in media viewer.

This commit is contained in:
John Preston 2020-05-27 13:01:25 +04:00
parent 056945d9f5
commit 50e0c3ee4d
7 changed files with 209 additions and 174 deletions

View File

@ -17,6 +17,7 @@ void UpdateCloudFile(
CloudFile &file, CloudFile &file,
const ImageWithLocation &data, const ImageWithLocation &data,
Storage::Cache::Database &cache, Storage::Cache::Database &cache,
uint8 cacheTag,
Fn<void(FileOrigin)> restartLoader, Fn<void(FileOrigin)> restartLoader,
Fn<void(QImage)> usePreloaded) { Fn<void(QImage)> usePreloaded) {
if (!data.location.valid()) { if (!data.location.valid()) {
@ -42,7 +43,7 @@ void UpdateCloudFile(
cacheKey, cacheKey,
Storage::Cache::Database::TaggedValue( Storage::Cache::Database::TaggedValue(
std::move(cacheBytes), std::move(cacheBytes),
Data::kImageCacheTag)); cacheTag));
} }
} }
file.location = data.location; file.location = data.location;
@ -58,4 +59,132 @@ void UpdateCloudFile(
} }
} }
void LoadCloudFile(
CloudFile &file,
FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading,
uint8 cacheTag,
Fn<bool()> finalCheck,
Fn<void(CloudFile&)> done,
Fn<void(bool)> fail,
Fn<void()> progress) {
if (file.loader) {
if (fromCloud == LoadFromCloudOrLocal) {
file.loader->permitLoadFromCloud();
}
return;
} else if ((file.flags & CloudFile::Flag::Failed)
|| !file.location.valid()
|| (finalCheck && !finalCheck())) {
return;
}
file.flags &= ~CloudFile::Flag::Cancelled;
file.loader = CreateFileLoader(
file.location.file(),
origin,
QString(),
file.byteSize,
UnknownFileLocation,
LoadToCacheAsWell,
fromCloud,
autoLoading,
cacheTag);
const auto finish = [done](CloudFile &file) {
if (!file.loader || file.loader->cancelled()) {
file.flags |= CloudFile::Flag::Cancelled;
} else {
done(file);
}
// NB! file.loader may be in ~FileLoader() already.
if (const auto loader = base::take(file.loader)) {
if (file.flags & CloudFile::Flag::Cancelled) {
loader->cancel();
}
}
};
file.loader->updates(
) | rpl::start_with_next_error_done([=] {
if (const auto onstack = progress) {
onstack();
}
}, [=, &file](bool started) {
finish(file);
file.flags |= CloudFile::Flag::Failed;
if (const auto onstack = fail) {
onstack(started);
}
}, [=, &file] {
finish(file);
}, file.loader->lifetime());
file.loader->start();
}
void LoadCloudFile(
CloudFile &file,
FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading,
uint8 cacheTag,
Fn<bool()> finalCheck,
Fn<void(QImage)> done,
Fn<void(bool)> fail,
Fn<void()> progress) {
const auto callback = [=](CloudFile &file) {
if (auto read = file.loader->imageData(); read.isNull()) {
file.flags |= CloudFile::Flag::Failed;
if (const auto onstack = fail) {
onstack(true);
}
} else if (const auto onstack = done) {
onstack(std::move(read));
}
};
LoadCloudFile(
file,
origin,
fromCloud,
autoLoading,
cacheTag,
finalCheck,
callback,
std::move(fail),
std::move(progress));
}
void LoadCloudFile(
CloudFile &file,
FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading,
uint8 cacheTag,
Fn<bool()> finalCheck,
Fn<void(QByteArray)> done,
Fn<void(bool)> fail,
Fn<void()> progress) {
const auto callback = [=](CloudFile &file) {
if (auto bytes = file.loader->bytes(); bytes.isEmpty()) {
file.flags |= Data::CloudFile::Flag::Failed;
if (const auto onstack = fail) {
onstack(true);
}
} else if (const auto onstack = done) {
onstack(std::move(bytes));
}
};
LoadCloudFile(
file,
origin,
fromCloud,
autoLoading,
cacheTag,
finalCheck,
callback,
std::move(fail),
std::move(progress));
}
} // namespace Data } // namespace Data

View File

@ -39,7 +39,30 @@ void UpdateCloudFile(
CloudFile &file, CloudFile &file,
const ImageWithLocation &data, const ImageWithLocation &data,
Storage::Cache::Database &cache, Storage::Cache::Database &cache,
uint8 cacheTag,
Fn<void(FileOrigin)> restartLoader, Fn<void(FileOrigin)> restartLoader,
Fn<void(QImage)> usePreloaded = nullptr); Fn<void(QImage)> usePreloaded = nullptr);
void LoadCloudFile(
CloudFile &file,
FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading,
uint8 cacheTag,
Fn<bool()> finalCheck,
Fn<void(QImage)> done,
Fn<void(bool)> fail = nullptr,
Fn<void()> progress = nullptr);
void LoadCloudFile(
CloudFile &file,
FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading,
uint8 cacheTag,
Fn<bool()> finalCheck,
Fn<void(QByteArray)> done,
Fn<void(bool)> fail = nullptr,
Fn<void()> progress = nullptr);
} // namespace Data } // namespace Data

View File

@ -617,6 +617,7 @@ void DocumentData::updateThumbnails(
_thumbnail, _thumbnail,
thumbnail, thumbnail,
owner().cache(), owner().cache(),
Data::kImageCacheTag,
[&](Data::FileOrigin origin) { loadThumbnail(origin); }, [&](Data::FileOrigin origin) { loadThumbnail(origin); },
[&](QImage preloaded) { [&](QImage preloaded) {
if (const auto media = activeMediaView()) { if (const auto media = activeMediaView()) {
@ -627,6 +628,7 @@ void DocumentData::updateThumbnails(
_videoThumbnail, _videoThumbnail,
videoThumbnail, videoThumbnail,
owner().cache(), owner().cache(),
Data::kAnimationCacheTag,
[&](Data::FileOrigin origin) { loadVideoThumbnail(origin); }); [&](Data::FileOrigin origin) { loadVideoThumbnail(origin); });
} }
@ -647,47 +649,24 @@ bool DocumentData::thumbnailLoading() const {
} }
bool DocumentData::thumbnailFailed() const { bool DocumentData::thumbnailFailed() const {
return (_flags & Flag::ThumbnailFailed); return (_thumbnail.flags & Data::CloudFile::Flag::Failed);
} }
void DocumentData::loadThumbnail(Data::FileOrigin origin) { void DocumentData::loadThumbnail(Data::FileOrigin origin) {
if (_thumbnail.loader auto &file = _thumbnail;
|| (_flags & Flag::ThumbnailFailed) const auto fromCloud = LoadFromCloudOrLocal;
|| !_thumbnail.location.valid()) { const auto cacheTag = Data::kImageCacheTag;
return;
} else if (const auto active = activeMediaView()) {
if (active->thumbnail()) {
return;
}
}
const auto autoLoading = false; const auto autoLoading = false;
_thumbnail.loader = CreateFileLoader( Data::LoadCloudFile(file, origin, fromCloud, autoLoading, cacheTag, [=] {
_thumbnail.location.file(), if (const auto active = activeMediaView()) {
origin, return !active->thumbnail();
QString(),
_thumbnail.byteSize,
UnknownFileLocation,
LoadToCacheAsWell,
LoadFromCloudOrLocal,
autoLoading,
Data::kImageCacheTag);
_thumbnail.loader->updates(
) | rpl::start_with_error_done([=](bool started) {
_thumbnail.loader = nullptr;
_flags |= Flag::ThumbnailFailed;
}, [=] {
if (_thumbnail.loader && !_thumbnail.loader->cancelled()) {
if (auto read = _thumbnail.loader->imageData(); read.isNull()) {
_flags |= Flag::ThumbnailFailed;
} else if (const auto active = activeMediaView()) {
active->setThumbnail(std::move(read));
}
} }
_thumbnail.loader = nullptr; return true;
}, _thumbnail.loader->lifetime()); }, [=](QImage result) {
if (const auto active = activeMediaView()) {
_thumbnail.loader->start(); active->setThumbnail(std::move(result));
}
});
} }
const ImageLocation &DocumentData::thumbnailLocation() const { const ImageLocation &DocumentData::thumbnailLocation() const {
@ -707,48 +686,24 @@ bool DocumentData::videoThumbnailLoading() const {
} }
bool DocumentData::videoThumbnailFailed() const { bool DocumentData::videoThumbnailFailed() const {
return (_flags & Flag::VideoThumbnailFailed); return (_videoThumbnail.flags & Data::CloudFile::Flag::Failed);
} }
void DocumentData::loadVideoThumbnail(Data::FileOrigin origin) { void DocumentData::loadVideoThumbnail(Data::FileOrigin origin) {
if (_videoThumbnail.loader auto &file = _videoThumbnail;
|| (_flags & Flag::VideoThumbnailFailed) const auto fromCloud = LoadFromCloudOrLocal;
|| !_videoThumbnail.location.valid()) { const auto cacheTag = Data::kAnimationCacheTag;
return;
} else if (const auto active = activeMediaView()) {
if (!active->videoThumbnailContent().isEmpty()) {
return;
}
}
const auto autoLoading = false; const auto autoLoading = false;
_videoThumbnail.loader = CreateFileLoader( Data::LoadCloudFile(file, origin, fromCloud, autoLoading, cacheTag, [=] {
_videoThumbnail.location.file(), if (const auto active = activeMediaView()) {
origin, return active->videoThumbnailContent().isEmpty();
QString(),
_videoThumbnail.byteSize,
UnknownFileLocation,
LoadToCacheAsWell,
LoadFromCloudOrLocal,
autoLoading,
Data::kAnimationCacheTag);
_videoThumbnail.loader->updates(
) | rpl::start_with_error_done([=](bool started) {
_videoThumbnail.loader = nullptr;
_flags |= Flag::VideoThumbnailFailed;
}, [=] {
if (_videoThumbnail.loader && !_videoThumbnail.loader->cancelled()) {
auto bytes = _videoThumbnail.loader->bytes();
if (bytes.isEmpty()) {
_flags |= Flag::VideoThumbnailFailed;
} else if (const auto active = activeMediaView()) {
active->setVideoThumbnail(std::move(bytes));
}
} }
_videoThumbnail.loader = nullptr; return true;
}, _videoThumbnail.loader->lifetime()); }, [=](QByteArray result) {
if (const auto active = activeMediaView()) {
_videoThumbnail.loader->start(); active->setVideoThumbnail(std::move(result));
}
});
} }
const ImageLocation &DocumentData::videoThumbnailLocation() const { const ImageLocation &DocumentData::videoThumbnailLocation() const {

View File

@ -257,8 +257,6 @@ private:
ImageType = 0x08, ImageType = 0x08,
DownloadCancelled = 0x10, DownloadCancelled = 0x10,
LoadedInMediaCache = 0x20, LoadedInMediaCache = 0x20,
ThumbnailFailed = 0x40,
VideoThumbnailFailed = 0x80,
}; };
using Flags = base::flags<Flag>; using Flags = base::flags<Flag>;
friend constexpr bool is_flag_type(Flag) { return true; }; friend constexpr bool is_flag_type(Flag) { return true; };

View File

@ -127,14 +127,9 @@ bool PhotoData::displayLoading() const {
} }
void PhotoData::cancel() { void PhotoData::cancel() {
if (!loading()) { if (loading()) {
return; _images[PhotoSizeIndex(PhotoSize::Large)].loader->cancel();
} }
const auto index = PhotoSizeIndex(PhotoSize::Large);
_images[index].flags |= Data::CloudFile::Flag::Cancelled;
destroyLoader(PhotoSize::Large);
_owner->photoLoadDone(this);
} }
float64 PhotoData::progress() const { float64 PhotoData::progress() const {
@ -244,91 +239,37 @@ void PhotoData::load(
bool autoLoading) { bool autoLoading) {
const auto index = validSizeIndex(size); const auto index = validSizeIndex(size);
auto &image = _images[index]; auto &image = _images[index];
if (image.loader) {
if (fromCloud == LoadFromCloudOrLocal) {
image.loader->permitLoadFromCloud();
}
return;
} else if ((image.flags & Data::CloudFile::Flag::Failed)
|| !image.location.valid()) {
return;
} else if (const auto active = activeMediaView()) {
if (active->image(size)) {
return;
}
}
// Could've changed, if the requested size didn't have a location. // Could've changed, if the requested size didn't have a location.
size = static_cast<PhotoSize>(index); const auto loadingSize = static_cast<PhotoSize>(index);
const auto cacheTag = Data::kImageCacheTag;
image.flags &= ~Data::CloudFile::Flag::Cancelled; Data::LoadCloudFile(image, origin, fromCloud, autoLoading, cacheTag, [=] {
image.loader = CreateFileLoader( if (const auto active = activeMediaView()) {
image.location.file(), return !active->image(size);
origin,
QString(),
image.byteSize,
UnknownFileLocation,
LoadToCacheAsWell,
fromCloud,
autoLoading,
Data::kImageCacheTag);
image.loader->updates(
) | rpl::start_with_next_error_done([=] {
if (size == PhotoSize::Large) {
_owner->photoLoadProgress(this);
} }
}, [=, &image](bool started) { return true;
finishLoad(size); }, [=](QImage result) {
image.flags |= Data::CloudFile::Flag::Failed; if (const auto active = activeMediaView()) {
if (size == PhotoSize::Large) { active->set(loadingSize, std::move(result));
}
if (loadingSize == PhotoSize::Large) {
_owner->photoLoadDone(this);
}
}, [=](bool started) {
if (loadingSize == PhotoSize::Large) {
_owner->photoLoadFail(this, started); _owner->photoLoadFail(this, started);
} }
}, [=] { }, [=] {
finishLoad(size); if (loadingSize == PhotoSize::Large) {
if (size == PhotoSize::Large) { _owner->photoLoadProgress(this);
_owner->photoLoadDone(this);
} }
}, image.loader->lifetime()); });
image.loader->start();
if (size == PhotoSize::Large) { if (size == PhotoSize::Large) {
_owner->notifyPhotoLayoutChanged(this); _owner->notifyPhotoLayoutChanged(this);
} }
} }
void PhotoData::finishLoad(PhotoSize size) {
const auto index = PhotoSizeIndex(size);
auto &image = _images[index];
// NB! image.loader may be in ~FileLoader() already.
const auto guard = gsl::finally([&] {
destroyLoader(size);
});
if (!image.loader || image.loader->cancelled()) {
image.flags |= Data::CloudFile::Flag::Cancelled;
return;
} else if (auto read = image.loader->imageData(); read.isNull()) {
image.flags |= Data::CloudFile::Flag::Failed;
} else if (const auto active = activeMediaView()) {
active->set(size, std::move(read));
}
}
void PhotoData::destroyLoader(PhotoSize size) {
const auto index = PhotoSizeIndex(size);
auto &image = _images[index];
// NB! image.loader may be in ~FileLoader() already.
if (!image.loader) {
return;
}
const auto loader = base::take(image.loader);
if (image.flags & Data::CloudFile::Flag::Cancelled) {
loader->cancel();
}
}
std::shared_ptr<PhotoMedia> PhotoData::createMediaView() { std::shared_ptr<PhotoMedia> PhotoData::createMediaView() {
if (auto result = activeMediaView()) { if (auto result = activeMediaView()) {
return result; return result;
@ -356,6 +297,7 @@ void PhotoData::updateImages(
_images[PhotoSizeIndex(size)], _images[PhotoSizeIndex(size)],
data, data,
owner().cache(), owner().cache(),
Data::kImageCacheTag,
[=](Data::FileOrigin origin) { load(size, origin); }, [=](Data::FileOrigin origin) { load(size, origin); },
[=](QImage preloaded) { [=](QImage preloaded) {
if (const auto media = activeMediaView()) { if (const auto media = activeMediaView()) {

View File

@ -124,9 +124,6 @@ public:
std::unique_ptr<Data::UploadState> uploadingData; std::unique_ptr<Data::UploadState> uploadingData;
private: private:
void finishLoad(Data::PhotoSize size);
void destroyLoader(Data::PhotoSize size);
QByteArray _inlineThumbnailBytes; QByteArray _inlineThumbnailBytes;
std::array<Data::CloudFile, Data::kPhotoSizeCount> _images; std::array<Data::CloudFile, Data::kPhotoSizeCount> _images;

View File

@ -71,7 +71,7 @@ namespace Media {
namespace View { namespace View {
namespace { namespace {
constexpr auto kPreloadCount = 4; constexpr auto kPreloadCount = 3;
constexpr auto kMaxZoomLevel = 7; // x8 constexpr auto kMaxZoomLevel = 7; // x8
constexpr auto kZoomToScreenLevel = 1024; constexpr auto kZoomToScreenLevel = 1024;
constexpr auto kOverlayLoaderPriority = 2; constexpr auto kOverlayLoaderPriority = 2;
@ -939,7 +939,7 @@ bool OverlayWidget::radialLoading() const {
if (_document) { if (_document) {
return _document->loading() && !_streamed; return _document->loading() && !_streamed;
} else if (_photo) { } else if (_photo) {
return _photo->loading(); return _photo->displayLoading();
} }
return false; return false;
} }
@ -1122,8 +1122,8 @@ void OverlayWidget::assignMediaPointer(not_null<PhotoData*> photo) {
if (_photo != photo) { if (_photo != photo) {
_photo = photo; _photo = photo;
_photoMedia = _photo->createMediaView(); _photoMedia = _photo->createMediaView();
_photoMedia->wanted(Data::PhotoSize::Large, fileOrigin());
_photoMedia->wanted(Data::PhotoSize::Small, fileOrigin()); _photoMedia->wanted(Data::PhotoSize::Small, fileOrigin());
_photo->load(fileOrigin(), LoadFromCloudOrLocal, true);
} }
} }
@ -1739,7 +1739,6 @@ void OverlayWidget::refreshMediaViewer() {
} }
findCurrent(); findCurrent();
updateControls(); updateControls();
preloadData(0);
} }
void OverlayWidget::refreshFromLabel(HistoryItem *item) { void OverlayWidget::refreshFromLabel(HistoryItem *item) {
@ -3463,7 +3462,7 @@ bool OverlayWidget::moveToNext(int delta) {
return false; return false;
} }
auto newIndex = *_index + delta; auto newIndex = *_index + delta;
return moveToEntity(entityByIndex(newIndex)); return moveToEntity(entityByIndex(newIndex), delta);
} }
bool OverlayWidget::moveToEntity(const Entity &entity, int preloadDelta) { bool OverlayWidget::moveToEntity(const Entity &entity, int preloadDelta) {
@ -3498,22 +3497,14 @@ void OverlayWidget::preloadData(int delta) {
auto till = *_index + (delta ? delta * kPreloadCount : 1); auto till = *_index + (delta ? delta * kPreloadCount : 1);
if (from > till) std::swap(from, till); if (from > till) std::swap(from, till);
if (delta != 0) {
auto forgetIndex = *_index - delta * 2;
auto entity = entityByIndex(forgetIndex);
if (auto photo = base::get_if<not_null<PhotoData*>>(&entity.data)) {
(*photo)->unload();
}
}
auto photos = base::flat_set<std::shared_ptr<Data::PhotoMedia>>(); auto photos = base::flat_set<std::shared_ptr<Data::PhotoMedia>>();
auto documents = base::flat_set<std::shared_ptr<Data::DocumentMedia>>(); auto documents = base::flat_set<std::shared_ptr<Data::DocumentMedia>>();
for (auto index = from; index != till; ++index) { for (auto index = from; index != till + 1; ++index) {
auto entity = entityByIndex(index); auto entity = entityByIndex(index);
if (auto photo = base::get_if<not_null<PhotoData*>>(&entity.data)) { if (auto photo = base::get_if<not_null<PhotoData*>>(&entity.data)) {
const auto [i, ok] = photos.emplace((*photo)->createMediaView()); const auto [i, ok] = photos.emplace((*photo)->createMediaView());
(*i)->wanted(Data::PhotoSize::Small, fileOrigin()); (*i)->wanted(Data::PhotoSize::Small, fileOrigin());
(*i)->wanted(Data::PhotoSize::Large, fileOrigin()); (*photo)->load(fileOrigin(), LoadFromCloudOrLocal, true);
} else if (auto document = base::get_if<not_null<DocumentData*>>( } else if (auto document = base::get_if<not_null<DocumentData*>>(
&entity.data)) { &entity.data)) {
const auto [i, ok] = documents.emplace( const auto [i, ok] = documents.emplace(