Remove legacy image-related code.

This commit is contained in:
John Preston 2020-05-29 18:08:18 +04:00
parent f066e0f05a
commit 6513422e40
60 changed files with 161 additions and 1509 deletions

View File

@ -331,7 +331,6 @@ PRIVATE
core/launcher.h core/launcher.h
core/local_url_handlers.cpp core/local_url_handlers.cpp
core/local_url_handlers.h core/local_url_handlers.h
core/media_active_cache.h
core/mime_type.cpp core/mime_type.cpp
core/mime_type.h core/mime_type.h
core/sandbox.cpp core/sandbox.cpp

View File

@ -227,8 +227,6 @@ namespace App {
clearCorners(); clearCorners();
Data::clearGlobalStructures(); Data::clearGlobalStructures();
Images::ClearAll();
} }
void hoveredItem(HistoryView::Element *item) { void hoveredItem(HistoryView::Element *item) {

View File

@ -266,10 +266,8 @@ void BackgroundBox::Inner::resizeToContentAndPreload() {
const auto preload = kBackgroundsInRow * 3; const auto preload = kBackgroundsInRow * 3;
for (const auto &paper : _papers | ranges::view::take(preload)) { for (const auto &paper : _papers | ranges::view::take(preload)) {
if (paper.data.localThumbnail()) { if (!paper.data.localThumbnail() && !paper.dataMedia) {
paper.data.loadLocalThumbnail(); if (const auto document = paper.data.document()) {
} else if (const auto document = paper.data.document()) {
if (!paper.dataMedia) {
paper.dataMedia = document->createMediaView(); paper.dataMedia = document->createMediaView();
paper.dataMedia->thumbnailWanted(paper.data.fileOrigin()); paper.dataMedia->thumbnailWanted(paper.data.fileOrigin());
} }
@ -323,9 +321,6 @@ void BackgroundBox::Inner::validatePaperThumbnail(
if (!paper.dataMedia || !paper.dataMedia->thumbnail()) { if (!paper.dataMedia || !paper.dataMedia->thumbnail()) {
return; return;
} }
} else if (!localThumbnail->loaded()) {
localThumbnail->load(paper.data.fileOrigin());
return;
} }
const auto thumbnail = localThumbnail const auto thumbnail = localThumbnail
? localThumbnail ? localThumbnail

View File

@ -434,7 +434,6 @@ void BackgroundPreviewBox::prepare() {
} }
updateServiceBg(_paper.backgroundColor()); updateServiceBg(_paper.backgroundColor());
_paper.loadLocalThumbnail();
_paper.loadDocument(); _paper.loadDocument();
const auto document = _paper.document(); const auto document = _paper.document();
if (document && document->loading()) { if (document && document->loading()) {

View File

@ -12,6 +12,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/wrap/slide_wrap.h" #include "ui/wrap/slide_wrap.h"
#include "mtproto/mtproto_rpc_sender.h" #include "mtproto/mtproto_rpc_sender.h"
class Image;
namespace ChatHelpers { namespace ChatHelpers {
class TabbedPanel; class TabbedPanel;
} // namespace ChatHelpers } // namespace ChatHelpers

View File

@ -14,6 +14,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/effects/animations.h" #include "ui/effects/animations.h"
#include "ui/rp_widget.h" #include "ui/rp_widget.h"
class Image;
namespace Data { namespace Data {
class PhotoMedia; class PhotoMedia;
class CloudImageView; class CloudImageView;

View File

@ -452,25 +452,13 @@ void GifsListWidget::processPanelHideFinished() {
} }
void GifsListWidget::clearHeavyData() { void GifsListWidget::clearHeavyData() {
const auto itemForget = [](const auto &item) {
if (const auto document = item->getDocument()) {
document->unload();
}
if (const auto photo = item->getPhoto()) {
photo->unload();
}
if (const auto result = item->getResult()) {
result->unload();
}
item->unloadHeavyPart();
};
// Preserve panel state through visibility toggles. // Preserve panel state through visibility toggles.
//clearInlineRows(false); //clearInlineRows(false);
for (const auto &[document, layout] : _gifLayouts) { for (const auto &[document, layout] : _gifLayouts) {
itemForget(layout); layout->unloadHeavyPart();
} }
for (const auto &[document, layout] : _inlineLayouts) { for (const auto &[document, layout] : _inlineLayouts) {
itemForget(layout); layout->unloadHeavyPart();
} }
} }

View File

@ -123,31 +123,11 @@ public:
EmojiPtr emoji, EmojiPtr emoji,
not_null<crl::object_on_queue<EmojiImageLoader>*> loader); not_null<crl::object_on_queue<EmojiImageLoader>*> loader);
void load(Data::FileOrigin origin) override; void load() override;
void loadEvenCancelled(Data::FileOrigin origin) override;
QImage takeLoaded() override; QImage takeLoaded() override;
void unload() override;
bool loading() override;
bool displayLoading() override;
void cancel() override;
float64 progress() override;
int loadOffset() override;
const StorageImageLocation &location() override;
void refreshFileReference(const QByteArray &data) override;
Storage::Cache::Key cacheKey() override;
void setDelayedStorageLocation(
const StorageImageLocation &location) override;
void performDelayedLoad(Data::FileOrigin origin) override;
void setImageBytes(const QByteArray &bytes) override;
int width() override; int width() override;
int height() override; int height() override;
int bytesSize() override;
void setInformation(int size, int width, int height) override;
QByteArray bytesForCache() override;
private: private:
// While HistoryView::Element-s are almost never destroyed // While HistoryView::Element-s are almost never destroyed
@ -155,8 +135,6 @@ private:
not_null<crl::object_on_queue<EmojiImageLoader>*> _loader; not_null<crl::object_on_queue<EmojiImageLoader>*> _loader;
EmojiPtr _emoji = nullptr; EmojiPtr _emoji = nullptr;
QImage _data; QImage _data;
QByteArray _format;
QByteArray _bytes;
QSize _size; QSize _size;
base::binary_guard _loading; base::binary_guard _loading;
@ -170,92 +148,29 @@ ImageSource::ImageSource(
, _size(SingleSize()) { , _size(SingleSize()) {
} }
void ImageSource::load(Data::FileOrigin origin) { void ImageSource::load() {
if (!_data.isNull()) { if (!_data.isNull() || _loading) {
return; return;
} }
if (_bytes.isEmpty()) { _loader->with([
_loader->with([ this,
this, emoji = _emoji,
emoji = _emoji, guard = _loading.make_guard()
guard = _loading.make_guard() ](EmojiImageLoader &loader) mutable {
](EmojiImageLoader &loader) mutable { if (!guard) {
if (!guard) { return;
return; }
} crl::on_main(std::move(guard), [this, image = loader.prepare(emoji)]{
crl::on_main(std::move(guard), [this, image = loader.prepare(emoji)]{ _data = image;
_data = image; Auth().downloaderTaskFinished().notify();
Auth().downloaderTaskFinished().notify();
});
}); });
} else { });
_data = App::readImage(_bytes, &_format, false);
}
}
void ImageSource::loadEvenCancelled(Data::FileOrigin origin) {
load(origin);
} }
QImage ImageSource::takeLoaded() { QImage ImageSource::takeLoaded() {
load({});
return _data; return _data;
} }
void ImageSource::unload() {
if (_bytes.isEmpty() && !_data.isNull()) {
if (_format != "JPG") {
_format = "PNG";
}
{
QBuffer buffer(&_bytes);
_data.save(&buffer, _format);
}
Assert(!_bytes.isEmpty());
}
_data = QImage();
}
bool ImageSource::loading() {
return _data.isNull() && _bytes.isEmpty();
}
bool ImageSource::displayLoading() {
return false;
}
void ImageSource::cancel() {
}
float64 ImageSource::progress() {
return 1.;
}
int ImageSource::loadOffset() {
return 0;
}
const StorageImageLocation &ImageSource::location() {
return StorageImageLocation::Invalid();
}
void ImageSource::refreshFileReference(const QByteArray &data) {
}
Storage::Cache::Key ImageSource::cacheKey() {
return {};
}
void ImageSource::setDelayedStorageLocation(
const StorageImageLocation &location) {
}
void ImageSource::performDelayedLoad(Data::FileOrigin origin) {
}
void ImageSource::setImageBytes(const QByteArray &bytes) {
}
int ImageSource::width() { int ImageSource::width() {
return _size.width(); return _size.width();
} }
@ -264,29 +179,6 @@ int ImageSource::height() {
return _size.height(); return _size.height();
} }
int ImageSource::bytesSize() {
return _bytes.size();
}
void ImageSource::setInformation(int size, int width, int height) {
if (width && height) {
_size = QSize(width, height);
}
}
QByteArray ImageSource::bytesForCache() {
auto result = QByteArray();
{
QBuffer buffer(&result);
if (!_data.save(&buffer, _format)) {
if (_data.save(&buffer, "PNG")) {
_format = "PNG";
}
}
}
return result;
}
} // namespace } // namespace
EmojiImageLoader::EmojiImageLoader( EmojiImageLoader::EmojiImageLoader(

View File

@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include <crl/crl_object_on_queue.h> #include <crl/crl_object_on_queue.h>
class Image;
class HistoryItem; class HistoryItem;
class DocumentData; class DocumentData;

View File

@ -31,7 +31,7 @@ void SetThumbnailView::set(
_content = std::move(content); _content = std::move(content);
} else { } else {
_image = std::make_unique<Image>( _image = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(std::move(image), "PNG")); std::make_unique<Images::ImageSource>(std::move(image)));
} }
session->downloaderTaskFinished().notify(); session->downloaderTaskFinished().notify();
} }

View File

@ -1,83 +0,0 @@
/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#pragma once
#include "base/last_used_cache.h"
namespace Core {
template <typename Type>
class MediaActiveCache {
public:
template <typename Unload>
MediaActiveCache(int64 limit, Unload &&unload);
void up(Type *entry);
void remove(Type *entry);
void clear();
void increment(int64 amount);
void decrement(int64 amount);
private:
template <typename Unload>
void check(Unload &&unload);
base::last_used_cache<Type*> _cache;
SingleQueuedInvokation _delayed;
int64 _usage = 0;
int64 _limit = 0;
};
template <typename Type>
template <typename Unload>
MediaActiveCache<Type>::MediaActiveCache(int64 limit, Unload &&unload)
: _delayed([=] { check(unload); })
, _limit(limit) {
}
template <typename Type>
void MediaActiveCache<Type>::up(Type *entry) {
_cache.up(entry);
_delayed.call();
}
template <typename Type>
void MediaActiveCache<Type>::remove(Type *entry) {
_cache.remove(entry);
}
template <typename Type>
void MediaActiveCache<Type>::clear() {
_cache.clear();
}
template <typename Type>
void MediaActiveCache<Type>::increment(int64 amount) {
_usage += amount;
}
template <typename Type>
void MediaActiveCache<Type>::decrement(int64 amount) {
_usage -= amount;
}
template <typename Type>
template <typename Unload>
void MediaActiveCache<Type>::check(Unload &&unload) {
while (_usage > _limit) {
if (const auto entry = _cache.take_lowest()) {
unload(entry);
} else {
break;
}
}
}
} // namespace Core

View File

@ -175,7 +175,6 @@ public:
std::deque<not_null<UserData*>> lastAuthors; std::deque<not_null<UserData*>> lastAuthors;
base::flat_set<not_null<PeerData*>> markupSenders; base::flat_set<not_null<PeerData*>> markupSenders;
int botStatus = 0; // -1 - no bots, 0 - unknown, 1 - one bot, that sees all history, 2 - other int botStatus = 0; // -1 - no bots, 0 - unknown, 1 - one bot, that sees all history, 2 - other
// ImagePtr photoFull;
private: private:
Flags _flags; Flags _flags;

View File

@ -22,7 +22,7 @@ void CloudImageView::set(
not_null<Main::Session*> session, not_null<Main::Session*> session,
QImage image) { QImage image) {
_image = std::make_unique<Image>( _image = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(std::move(image), "PNG")); std::make_unique<Images::ImageSource>(std::move(image)));
session->downloaderTaskFinished().notify(); session->downloaderTaskFinished().notify();
} }

View File

@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/image/image_location.h" #include "ui/image/image_location.h"
class FileLoader; class FileLoader;
class Image;
namespace Storage { namespace Storage {
namespace Cache { namespace Cache {

View File

@ -16,7 +16,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "main/main_session.h" #include "main/main_session.h"
#include "mainwidget.h" #include "mainwidget.h"
#include "core/file_utilities.h" #include "core/file_utilities.h"
#include "core/media_active_cache.h"
#include "core/mime_type.h" #include "core/mime_type.h"
#include "chat_helpers/stickers.h" #include "chat_helpers/stickers.h"
#include "chat_helpers/stickers_set.h" #include "chat_helpers/stickers_set.h"
@ -457,7 +456,6 @@ DocumentData::~DocumentData() {
base::take(_thumbnail.loader).reset(); base::take(_thumbnail.loader).reset();
base::take(_videoThumbnail.loader).reset(); base::take(_videoThumbnail.loader).reset();
destroyLoader(); destroyLoader();
unload();
} }
Data::Session &DocumentData::owner() const { Data::Session &DocumentData::owner() const {
@ -803,10 +801,6 @@ bool DocumentData::saveToCache() const {
|| isTheme()); || isTheme());
} }
void DocumentData::unload() {
_replyPreview = nullptr;
}
void DocumentData::automaticLoadSettingsChanged() { void DocumentData::automaticLoadSettingsChanged() {
if (!cancelled() || status != FileReady) { if (!cancelled() || status != FileReady) {
return; return;

View File

@ -125,7 +125,6 @@ public:
[[nodiscard]] bool saveToCache() const; [[nodiscard]] bool saveToCache() const;
void unload();
[[nodiscard]] Image *getReplyPreview(Data::FileOrigin origin); [[nodiscard]] Image *getReplyPreview(Data::FileOrigin origin);
[[nodiscard]] StickerData *sticker() const; [[nodiscard]] StickerData *sticker() const;

View File

@ -166,7 +166,7 @@ void DocumentMedia::setGoodThumbnail(QImage thumbnail) {
return; return;
} }
_goodThumbnail = std::make_unique<Image>( _goodThumbnail = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(std::move(thumbnail), "PNG")); std::make_unique<Images::ImageSource>(std::move(thumbnail)));
_owner->session().downloaderTaskFinished().notify(); _owner->session().downloaderTaskFinished().notify();
} }
@ -175,9 +175,7 @@ Image *DocumentMedia::thumbnailInline() const {
auto image = Images::FromInlineBytes(_owner->inlineThumbnailBytes()); auto image = Images::FromInlineBytes(_owner->inlineThumbnailBytes());
if (!image.isNull()) { if (!image.isNull()) {
_inlineThumbnail = std::make_unique<Image>( _inlineThumbnail = std::make_unique<Image>(
std::make_unique<Images::ImageSource>( std::make_unique<Images::ImageSource>(std::move(image)));
std::move(image),
"PNG"));
} }
} }
return _inlineThumbnail.get(); return _inlineThumbnail.get();
@ -203,7 +201,7 @@ QSize DocumentMedia::thumbnailSize() const {
void DocumentMedia::setThumbnail(QImage thumbnail) { void DocumentMedia::setThumbnail(QImage thumbnail) {
_thumbnail = std::make_unique<Image>( _thumbnail = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(std::move(thumbnail), "PNG")); std::make_unique<Images::ImageSource>(std::move(thumbnail)));
_owner->session().downloaderTaskFinished().notify(); _owner->session().downloaderTaskFinished().notify();
} }
@ -243,18 +241,12 @@ void DocumentMedia::checkStickerLarge() {
const auto &loc = _owner->location(true); const auto &loc = _owner->location(true);
if (loc.accessEnable()) { if (loc.accessEnable()) {
_sticker = std::make_unique<Image>( _sticker = std::make_unique<Image>(
std::make_unique<Images::LocalFileSource>(loc.name())); std::make_unique<Images::ImageSource>(loc.name()));
loc.accessDisable(); loc.accessDisable();
} }
} else { } else {
auto format = QByteArray();
auto image = App::readImage(_bytes, &format, false);
_sticker = std::make_unique<Image>( _sticker = std::make_unique<Image>(
std::make_unique<Images::LocalFileSource>( std::make_unique<Images::ImageSource>(_bytes));
QString(),
_bytes,
format,
std::move(image)));
} }
} }
@ -297,19 +289,19 @@ void DocumentMedia::automaticLoad(
void DocumentMedia::collectLocalData(not_null<DocumentMedia*> local) { void DocumentMedia::collectLocalData(not_null<DocumentMedia*> local) {
if (const auto image = local->_goodThumbnail.get()) { if (const auto image = local->_goodThumbnail.get()) {
_goodThumbnail = std::make_unique<Image>( _goodThumbnail = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(image->original(), "PNG")); std::make_unique<Images::ImageSource>(image->original()));
} }
if (const auto image = local->_inlineThumbnail.get()) { if (const auto image = local->_inlineThumbnail.get()) {
_inlineThumbnail = std::make_unique<Image>( _inlineThumbnail = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(image->original(), "PNG")); std::make_unique<Images::ImageSource>(image->original()));
} }
if (const auto image = local->_thumbnail.get()) { if (const auto image = local->_thumbnail.get()) {
_thumbnail = std::make_unique<Image>( _thumbnail = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(image->original(), "PNG")); std::make_unique<Images::ImageSource>(image->original()));
} }
if (const auto image = local->_sticker.get()) { if (const auto image = local->_sticker.get()) {
_sticker = std::make_unique<Image>( _sticker = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(image->original(), "PNG")); std::make_unique<Images::ImageSource>(image->original()));
} }
_bytes = local->_bytes; _bytes = local->_bytes;
_videoThumbnailBytes = local->_videoThumbnailBytes; _videoThumbnailBytes = local->_videoThumbnailBytes;
@ -380,14 +372,9 @@ Image *DocumentMedia::getStickerSmall() {
void DocumentMedia::checkStickerLarge(not_null<FileLoader*> loader) { void DocumentMedia::checkStickerLarge(not_null<FileLoader*> loader) {
if (_owner->sticker() if (_owner->sticker()
&& !_sticker && !_sticker
&& !loader->imageData().isNull() && !loader->imageData().isNull()) {
&& !_bytes.isEmpty()) {
_sticker = std::make_unique<Image>( _sticker = std::make_unique<Image>(
std::make_unique<Images::LocalFileSource>( std::make_unique<Images::ImageSource>(loader->imageData()));
QString(),
_bytes,
loader->imageFormat(),
loader->imageData()));
} }
} }

View File

@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "base/flags.h" #include "base/flags.h"
class Image;
class FileLoader; class FileLoader;
namespace Media { namespace Media {

View File

@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_location.h" #include "data/data_location.h"
class Image;
class HistoryItem; class HistoryItem;
namespace base { namespace base {

View File

@ -28,6 +28,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "mainwindow.h" #include "mainwindow.h"
#include "window/window_session_controller.h" #include "window/window_session_controller.h"
#include "ui/image/image.h" #include "ui/image/image.h"
#include "ui/image/image_source.h"
#include "ui/empty_userpic.h" #include "ui/empty_userpic.h"
#include "ui/text_options.h" #include "ui/text_options.h"
#include "history/history.h" #include "history/history.h"
@ -217,6 +218,13 @@ Image *PeerData::currentUserpic(
const auto image = view ? view->image() : nullptr; const auto image = view ? view->image() : nullptr;
if (image) { if (image) {
_userpicEmpty = nullptr; _userpicEmpty = nullptr;
} else if (isNotificationsUser()) {
static auto result = Image(
std::make_unique<Images::ImageSource>(
Core::App().logoNoMargin().scaledToWidth(
kUserpicSize,
Qt::SmoothTransformation)));
return &result;
} }
return image; return image;
} }
@ -370,17 +378,6 @@ void PeerData::updateUserpic(
} }
void PeerData::clearUserpic() { void PeerData::clearUserpic() {
//const auto photo = [&] { // #TODO optimize
// if (isNotificationsUser()) {
// auto image = Core::App().logoNoMargin().scaledToWidth(
// kUserpicSize,
// Qt::SmoothTransformation);
// return _userpic
// ? _userpic
// : Images::Create(std::move(image), "PNG");
// }
// return ImagePtr();
//}();
setUserpicChecked(PhotoId(), ImageLocation()); setUserpicChecked(PhotoId(), ImageLocation());
} }

View File

@ -169,10 +169,6 @@ bool PhotoData::uploading() const {
return (uploadingData != nullptr); return (uploadingData != nullptr);
} }
void PhotoData::unload() {
_replyPreview = nullptr;
}
Image *PhotoData::getReplyPreview(Data::FileOrigin origin) { Image *PhotoData::getReplyPreview(Data::FileOrigin origin) {
if (!_replyPreview) { if (!_replyPreview) {
_replyPreview = std::make_unique<Data::ReplyPreview>(this); _replyPreview = std::make_unique<Data::ReplyPreview>(this);

View File

@ -59,7 +59,6 @@ public:
void setWaitingForAlbum(); void setWaitingForAlbum();
[[nodiscard]] bool waitingForAlbum() const; [[nodiscard]] bool waitingForAlbum() const;
void unload();
[[nodiscard]] Image *getReplyPreview(Data::FileOrigin origin); [[nodiscard]] Image *getReplyPreview(Data::FileOrigin origin);
void setRemoteLocation( void setRemoteLocation(

View File

@ -38,8 +38,7 @@ Image *PhotoMedia::thumbnailInline() const {
if (!image.isNull()) { if (!image.isNull()) {
_inlineThumbnail = std::make_unique<Image>( _inlineThumbnail = std::make_unique<Image>(
std::make_unique<Images::ImageSource>( std::make_unique<Images::ImageSource>(
std::move(image), std::move(image)));
"PNG"));
} }
} }
return _inlineThumbnail.get(); return _inlineThumbnail.get();
@ -79,7 +78,7 @@ void PhotoMedia::set(PhotoSize size, QImage image) {
Qt::SmoothTransformation); Qt::SmoothTransformation);
} }
_images[index] = std::make_unique<Image>( _images[index] = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(std::move(image), "PNG")); std::make_unique<Images::ImageSource>(std::move(image)));
_owner->session().downloaderTaskFinished().notify(); _owner->session().downloaderTaskFinished().notify();
} }
@ -114,14 +113,12 @@ void PhotoMedia::automaticLoad(
void PhotoMedia::collectLocalData(not_null<PhotoMedia*> local) { void PhotoMedia::collectLocalData(not_null<PhotoMedia*> local) {
if (const auto image = local->_inlineThumbnail.get()) { if (const auto image = local->_inlineThumbnail.get()) {
_inlineThumbnail = std::make_unique<Image>( _inlineThumbnail = std::make_unique<Image>(
std::make_unique<Images::ImageSource>(image->original(), "PNG")); std::make_unique<Images::ImageSource>(image->original()));
} }
for (auto i = 0; i != kPhotoSizeCount; ++i) { for (auto i = 0; i != kPhotoSizeCount; ++i) {
if (const auto image = local->_images[i].get()) { if (const auto image = local->_images[i].get()) {
_images[i] = std::make_unique<Image>( _images[i] = std::make_unique<Image>(
std::make_unique<Images::ImageSource>( std::make_unique<Images::ImageSource>(image->original()));
image->original(),
"PNG"));
} }
} }
} }

View File

@ -25,6 +25,8 @@ ReplyPreview::ReplyPreview(not_null<PhotoData*> photo)
: _photo(photo) { : _photo(photo) {
} }
ReplyPreview::~ReplyPreview() = default;
void ReplyPreview::prepare(not_null<Image*> image, Images::Options options) { void ReplyPreview::prepare(not_null<Image*> image, Images::Options options) {
if (image->isNull() || !image->loaded()) { if (image->isNull() || !image->loaded()) {
return; return;
@ -53,8 +55,7 @@ void ReplyPreview::prepare(not_null<Image*> image, Images::Options options) {
outerSize); outerSize);
_image = std::make_unique<Image>( _image = std::make_unique<Image>(
std::make_unique<Images::ImageSource>( std::make_unique<Images::ImageSource>(
bitmap.toImage(), bitmap.toImage()));
"PNG"));
_good = ((options & Images::Option::Blurred) == 0); _good = ((options & Images::Option::Blurred) == 0);
} }

View File

@ -7,6 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/ */
#pragma once #pragma once
class Image;
class DocumentData; class DocumentData;
class PhotoData; class PhotoData;
@ -20,6 +21,7 @@ class ReplyPreview {
public: public:
explicit ReplyPreview(not_null<DocumentData*> document); explicit ReplyPreview(not_null<DocumentData*> document);
explicit ReplyPreview(not_null<PhotoData*> photo); explicit ReplyPreview(not_null<PhotoData*> photo);
~ReplyPreview();
[[nodiscard]] Image *image(Data::FileOrigin origin); [[nodiscard]] Image *image(Data::FileOrigin origin);

View File

@ -16,7 +16,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "core/mime_type.h" // Core::IsMimeSticker #include "core/mime_type.h" // Core::IsMimeSticker
#include "core/crash_reports.h" // CrashReports::SetAnnotation #include "core/crash_reports.h" // CrashReports::SetAnnotation
#include "ui/image/image.h" #include "ui/image/image.h"
#include "ui/image/image_source.h" // Images::LocalFileSource #include "ui/image/image_source.h" // Images::ImageSource
#include "ui/image/image_location_factory.h" // Images::FromPhotoSize #include "ui/image/image_location_factory.h" // Images::FromPhotoSize
#include "export/export_controller.h" #include "export/export_controller.h"
#include "export/view/export_view_panel_controller.h" #include "export/view/export_view_panel_controller.h"
@ -1080,7 +1080,6 @@ Session::~Session() {
_session->notifications().clearAllFast(); _session->notifications().clearAllFast();
clear(); clear();
Images::ClearRemote();
} }
template <typename Method> template <typename Method>
@ -3823,10 +3822,8 @@ void Session::setWallpapers(const QVector<MTPWallPaper> &data, int32 hash) {
_wallpapers.push_back(Data::Legacy1DefaultWallPaper()); _wallpapers.push_back(Data::Legacy1DefaultWallPaper());
_wallpapers.back().setLocalImageAsThumbnail(std::make_shared<Image>( _wallpapers.back().setLocalImageAsThumbnail(std::make_shared<Image>(
std::make_unique<Images::LocalFileSource>( std::make_unique<Images::ImageSource>(
qsl(":/gui/art/bg_initial.jpg"), u":/gui/art/bg_initial.jpg"_q)));
QByteArray(),
"JPG")));
for (const auto &paper : data) { for (const auto &paper : data) {
if (const auto parsed = Data::WallPaper::Create(paper)) { if (const auto parsed = Data::WallPaper::Create(paper)) {
_wallpapers.push_back(*parsed); _wallpapers.push_back(*parsed);
@ -3838,10 +3835,8 @@ void Session::setWallpapers(const QVector<MTPWallPaper> &data, int32 hash) {
if (defaultFound == end(_wallpapers)) { if (defaultFound == end(_wallpapers)) {
_wallpapers.push_back(Data::DefaultWallPaper()); _wallpapers.push_back(Data::DefaultWallPaper());
_wallpapers.back().setLocalImageAsThumbnail(std::make_shared<Image>( _wallpapers.back().setLocalImageAsThumbnail(std::make_shared<Image>(
std::make_unique<Images::LocalFileSource>( std::make_unique<Images::ImageSource>(
qsl(":/gui/arg/bg.jpg"), u":/gui/arg/bg.jpg"_q)));
QByteArray(),
"JPG")));
} }
} }

View File

@ -183,12 +183,6 @@ QString WallPaper::shareUrl() const {
: base + '?' + params.join('&'); : base + '?' + params.join('&');
} }
void WallPaper::loadLocalThumbnail() const {
if (_thumbnail) {
_thumbnail->load(fileOrigin());
}
}
void WallPaper::loadDocumentThumbnail() const { void WallPaper::loadDocumentThumbnail() const {
if (_document) { if (_document) {
_document->loadThumbnail(fileOrigin()); _document->loadThumbnail(fileOrigin());

View File

@ -34,7 +34,6 @@ public:
[[nodiscard]] QString shareUrl() const; [[nodiscard]] QString shareUrl() const;
void loadDocument() const; void loadDocument() const;
void loadLocalThumbnail() const;
void loadDocumentThumbnail() const; void loadDocumentThumbnail() const;
[[nodiscard]] FileOrigin fileOrigin() const; [[nodiscard]] FileOrigin fileOrigin() const;

View File

@ -6921,7 +6921,6 @@ void HistoryWidget::drawPinnedBar(Painter &p) {
auto top = _topBar->bottomNoMargins(); auto top = _topBar->bottomNoMargins();
bool serviceColor = false, hasForward = readyToForward(); bool serviceColor = false, hasForward = readyToForward();
ImagePtr preview;
p.fillRect(myrtlrect(0, top, width(), st::historyReplyHeight), st::historyPinnedBg); p.fillRect(myrtlrect(0, top, width(), st::historyReplyHeight), st::historyPinnedBg);
top += st::msgReplyPadding.top(); top += st::msgReplyPadding.top();

View File

@ -80,6 +80,10 @@ Contact::Contact(
Contact::~Contact() { Contact::~Contact() {
history()->owner().unregisterContactView(_userId, _parent); history()->owner().unregisterContactView(_userId, _parent);
if (_userpic) {
_userpic = nullptr;
_parent->checkHeavyPart();
}
} }
void Contact::updateSharedContactUserId(UserId userId) { void Contact::updateSharedContactUserId(UserId userId) {

View File

@ -421,13 +421,9 @@ void Gif::draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms
} }
} else { } else {
ensureDataMediaCreated(); ensureDataMediaCreated();
const auto good = _dataMedia->goodThumbnail(); if (const auto good = _dataMedia->goodThumbnail()) {
if (good && good->loaded()) {
p.drawPixmap(rthumb.topLeft(), good->pixSingle({}, _thumbw, _thumbh, usew, painth, roundRadius, roundCorners)); p.drawPixmap(rthumb.topLeft(), good->pixSingle({}, _thumbw, _thumbh, usew, painth, roundRadius, roundCorners));
} else { } else {
if (good) {
good->load({});
}
const auto normal = _dataMedia->thumbnail(); const auto normal = _dataMedia->thumbnail();
if (normal) { if (normal) {
if (normal->width() >= kUseNonBlurredThreshold if (normal->width() >= kUseNonBlurredThreshold
@ -657,8 +653,7 @@ void Gif::validateVideoThumbnail() const {
std::make_unique<Images::ImageSource>( std::make_unique<Images::ImageSource>(
(info.thumbnail.isNull() (info.thumbnail.isNull()
? Image::BlankMedia()->original() ? Image::BlankMedia()->original()
: info.thumbnail), : info.thumbnail)));
"PNG"));
} }
void Gif::drawCornerStatus(Painter &p, bool selected, QPoint position) const { void Gif::drawCornerStatus(Painter &p, bool selected, QPoint position) const {
@ -1179,23 +1174,18 @@ void Gif::validateGroupedCache(
ensureDataMediaCreated(); ensureDataMediaCreated();
const auto good = _dataMedia->goodThumbnail(); const auto good = _dataMedia->goodThumbnail();
const auto useGood = (good && good->loaded());
const auto thumb = _dataMedia->thumbnail(); const auto thumb = _dataMedia->thumbnail();
const auto useThumb = (thumb != nullptr); const auto image = good
const auto image = useGood
? good ? good
: useThumb : thumb
? thumb ? thumb
: _dataMedia->thumbnailInline(); : _dataMedia->thumbnailInline();
const auto blur = !useGood const auto blur = !good
&& (!useThumb && (!thumb
|| (thumb->width() < kUseNonBlurredThreshold || (thumb->width() < kUseNonBlurredThreshold
&& thumb->height() < kUseNonBlurredThreshold)); && thumb->height() < kUseNonBlurredThreshold));
if (good && !useGood) {
good->load({});
}
const auto loadLevel = useGood ? 3 : useThumb ? 2 : image ? 1 : 0; const auto loadLevel = good ? 3 : thumb ? 2 : image ? 1 : 0;
const auto width = geometry.width(); const auto width = geometry.width();
const auto height = geometry.height(); const auto height = geometry.height();
const auto options = Option::Smooth const auto options = Option::Smooth
@ -1312,6 +1302,7 @@ bool Gif::hasHeavyPart() const {
void Gif::unloadHeavyPart() { void Gif::unloadHeavyPart() {
stopAnimation(); stopAnimation();
_dataMedia = nullptr; _dataMedia = nullptr;
_videoThumbnailFrame = nullptr;
} }
void Gif::refreshParentId(not_null<HistoryItem*> realParent) { void Gif::refreshParentId(not_null<HistoryItem*> realParent) {
@ -1498,7 +1489,6 @@ void Gif::stopAnimation() {
if (_streamed) { if (_streamed) {
setStreamed(nullptr); setStreamed(nullptr);
history()->owner().requestViewResize(_parent); history()->owner().requestViewResize(_parent);
_data->unload();
} }
} }

View File

@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/view/media/history_view_file.h" #include "history/view/media/history_view_file.h"
#include "media/streaming/media_streaming_common.h" #include "media/streaming/media_streaming_common.h"
class Image;
struct HistoryMessageVia; struct HistoryMessageVia;
struct HistoryMessageReply; struct HistoryMessageReply;
struct HistoryMessageForwarded; struct HistoryMessageForwarded;

View File

@ -72,7 +72,7 @@ void LargeEmoji::draw(Painter &p, const QRect &r, bool selected) {
const auto o = Data::FileOrigin(); const auto o = Data::FileOrigin();
const auto skip = st::largeEmojiSkip - 2 * st::largeEmojiOutline; const auto skip = st::largeEmojiSkip - 2 * st::largeEmojiOutline;
for (const auto &image : images) { for (const auto &image : images) {
image->load(Data::FileOrigin()); image->load();
const auto w = image->width() / cIntRetinaFactor(); const auto w = image->width() / cIntRetinaFactor();
if (image->loaded()) { if (image->loaded()) {
const auto h = image->height() / cIntRetinaFactor(); const auto h = image->height() / cIntRetinaFactor();

View File

@ -10,6 +10,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/view/media/history_view_media_unwrapped.h" #include "history/view/media/history_view_media_unwrapped.h"
#include "ui/text/text_isolated_emoji.h" #include "ui/text/text_isolated_emoji.h"
class Image;
namespace Data { namespace Data {
struct FileOrigin; struct FileOrigin;
} // namespace Data } // namespace Data

View File

@ -51,6 +51,13 @@ Location::Location(
} }
} }
Location::~Location() {
if (_media) {
_media = nullptr;
_parent->checkHeavyPart();
}
}
void Location::ensureMediaCreated() const { void Location::ensureMediaCreated() const {
if (_media) { if (_media) {
return; return;

View File

@ -25,6 +25,7 @@ public:
Data::LocationPoint point, Data::LocationPoint point,
const QString &title = QString(), const QString &title = QString(),
const QString &description = QString()); const QString &description = QString());
~Location();
void draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms) const override; void draw(Painter &p, const QRect &r, TextSelection selection, crl::time ms) const override;
TextState textState(QPoint point, StateRequest request) const override; TextState textState(QPoint point, StateRequest request) const override;

View File

@ -1523,6 +1523,10 @@ void Poll::toggleLinkRipple(bool pressed) {
Poll::~Poll() { Poll::~Poll() {
history()->owner().unregisterPollView(_poll, _parent); history()->owner().unregisterPollView(_poll, _parent);
if (hasHeavyPart()) {
unloadHeavyPart();
_parent->checkHeavyPart();
}
} }
} // namespace HistoryView } // namespace HistoryView

View File

@ -213,9 +213,6 @@ QPixmap Sticker::paintedPixmap(bool selected) const {
const auto h = _size.height(); const auto h = _size.height();
const auto &c = st::msgStickerOverlay; const auto &c = st::msgStickerOverlay;
const auto good = _dataMedia->goodThumbnail(); const auto good = _dataMedia->goodThumbnail();
if (good && !good->loaded()) {
good->load({});
}
if (const auto image = _dataMedia->getStickerLarge()) { if (const auto image = _dataMedia->getStickerLarge()) {
return selected return selected
? image->pixColored(o, c, w, h) ? image->pixColored(o, c, w, h)
@ -227,7 +224,7 @@ QPixmap Sticker::paintedPixmap(bool selected) const {
// return selected // return selected
// ? blurred->pixBlurredColored(o, c, w, h) // ? blurred->pixBlurredColored(o, c, w, h)
// : blurred->pixBlurred(o, w, h); // : blurred->pixBlurred(o, w, h);
} else if (good && good->loaded()) { } else if (good) {
return selected return selected
? good->pixColored(o, c, w, h) ? good->pixColored(o, c, w, h)
: good->pix(o, w, h); : good->pix(o, w, h);

View File

@ -207,14 +207,10 @@ void ThemeDocument::validateThumbnail() const {
} }
ensureDataMediaCreated(); ensureDataMediaCreated();
if (const auto good = _dataMedia->goodThumbnail()) { if (const auto good = _dataMedia->goodThumbnail()) {
if (good->loaded()) { prepareThumbnailFrom(good, 1);
prepareThumbnailFrom(good, 1); return;
return;
} else {
good->load({});
}
} }
if (_thumbnailGood >= 0 || !_dataMedia->thumbnail()) { if (_thumbnailGood >= 0) {
return; return;
} }
if (const auto normal = _dataMedia->thumbnail()) { if (const auto normal = _dataMedia->thumbnail()) {

View File

@ -9,6 +9,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/view/media/history_view_file.h" #include "history/view/media/history_view_file.h"
class Image;
namespace Data { namespace Data {
class DocumentMedia; class DocumentMedia;
} // namespace Data } // namespace Data

View File

@ -315,9 +315,6 @@ void Gif::validateThumbnail(
bool good) const { bool good) const {
if (!image || (_thumbGood && !good)) { if (!image || (_thumbGood && !good)) {
return; return;
} else if (!image->loaded()) {
image->load(fileOrigin());
return;
} else if ((_thumb.size() == size * cIntRetinaFactor()) } else if ((_thumb.size() == size * cIntRetinaFactor())
&& (_thumbGood || !good)) { && (_thumbGood || !good)) {
return; return;
@ -393,7 +390,6 @@ void Gif::radialAnimationCallback(crl::time now) const {
void Gif::unloadHeavyPart() { void Gif::unloadHeavyPart() {
_gif.reset(); _gif.reset();
getShownDocument()->unload();
_dataMedia = nullptr; _dataMedia = nullptr;
} }
@ -404,7 +400,6 @@ void Gif::clipCallback(Media::Clip::Notification notification) {
if (_gif) { if (_gif) {
if (_gif->state() == State::Error) { if (_gif->state() == State::Error) {
_gif.setBad(); _gif.setBad();
getShownDocument()->unload();
} else if (_gif->ready() && !_gif->started()) { } else if (_gif->ready() && !_gif->started()) {
if (_gif->width() * _gif->height() > kMaxInlineArea) { if (_gif->width() * _gif->height() > kMaxInlineArea) {
getShownDocument()->dimensions = QSize( getShownDocument()->dimensions = QSize(
@ -608,7 +603,6 @@ TextState Photo::getState(
} }
void Photo::unloadHeavyPart() { void Photo::unloadHeavyPart() {
getShownPhoto()->unload();
_photoMedia = nullptr; _photoMedia = nullptr;
} }
@ -653,9 +647,6 @@ void Photo::validateThumbnail(
bool good) const { bool good) const {
if (!image || (_thumbGood && !good)) { if (!image || (_thumbGood && !good)) {
return; return;
} else if (!image->loaded()) {
image->load(fileOrigin());
return;
} else if ((_thumb.size() == size * cIntRetinaFactor()) } else if ((_thumb.size() == size * cIntRetinaFactor())
&& (_thumbGood || !good)) { && (_thumbGood || !good)) {
return; return;
@ -802,28 +793,23 @@ void Video::prepareThumbnail(QSize size) const {
if (!thumb) { if (!thumb) {
return; return;
} }
const auto origin = fileOrigin(); if (_thumb.size() != size * cIntRetinaFactor()) {
if (thumb->loaded()) { const auto width = size.width();
if (_thumb.size() != size * cIntRetinaFactor()) { const auto height = size.height();
const auto width = size.width(); auto w = qMax(style::ConvertScale(thumb->width()), 1);
const auto height = size.height(); auto h = qMax(style::ConvertScale(thumb->height()), 1);
auto w = qMax(style::ConvertScale(thumb->width()), 1); if (w * height > h * width) {
auto h = qMax(style::ConvertScale(thumb->height()), 1); if (height < h) {
if (w * height > h * width) { w = w * height / h;
if (height < h) { h = height;
w = w * height / h; }
h = height; } else {
} if (width < w) {
} else { h = h * width / w;
if (width < w) { w = width;
h = h * width / w;
w = width;
}
} }
_thumb = thumb->pixNoCache(origin, w * cIntRetinaFactor(), h * cIntRetinaFactor(), Images::Option::Smooth, width, height);
} }
} else { _thumb = thumb->pixNoCache({}, w * cIntRetinaFactor(), h * cIntRetinaFactor(), Images::Option::Smooth, width, height);
thumb->load(origin);
} }
} }
@ -1497,9 +1483,6 @@ void Game::ensureDataMediaCreated(not_null<PhotoData*> photo) const {
void Game::validateThumbnail(Image *image, QSize size, bool good) const { void Game::validateThumbnail(Image *image, QSize size, bool good) const {
if (!image || (_thumbGood && !good)) { if (!image || (_thumbGood && !good)) {
return; return;
} else if (!image->loaded()) {
image->load(fileOrigin());
return;
} else if ((_thumb.size() == size * cIntRetinaFactor()) } else if ((_thumb.size() == size * cIntRetinaFactor())
&& (_thumbGood || !good)) { && (_thumbGood || !good)) {
return; return;
@ -1565,14 +1548,8 @@ void Game::radialAnimationCallback(crl::time now) const {
void Game::unloadHeavyPart() { void Game::unloadHeavyPart() {
_gif.reset(); _gif.reset();
if (const auto document = getResultDocument()) { _documentMedia = nullptr;
document->unload(); _photoMedia = nullptr;
_documentMedia = nullptr;
}
if (const auto photo = getResultPhoto()) {
photo->unload();
_photoMedia = nullptr;
}
} }
void Game::clipCallback(Media::Clip::Notification notification) { void Game::clipCallback(Media::Clip::Notification notification) {
@ -1582,7 +1559,6 @@ void Game::clipCallback(Media::Clip::Notification notification) {
if (_gif) { if (_gif) {
if (_gif->state() == State::Error) { if (_gif->state() == State::Error) {
_gif.setBad(); _gif.setBad();
getResultDocument()->unload();
} else if (_gif->ready() && !_gif->started()) { } else if (_gif->ready() && !_gif->started()) {
if (_gif->width() * _gif->height() > kMaxInlineArea) { if (_gif->width() * _gif->height() > kMaxInlineArea) {
getResultDocument()->dimensions = QSize( getResultDocument()->dimensions = QSize(

View File

@ -10,6 +10,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "layout.h" #include "layout.h"
#include "ui/text/text.h" #include "ui/text/text.h"
class Image;
namespace Data { namespace Data {
class CloudImageView; class CloudImageView;
} // namespace Data } // namespace Data

View File

@ -329,15 +329,6 @@ bool Result::onChoose(Layout::ItemBase *layout) {
return true; return true;
} }
void Result::unload() {
if (_document) {
_document->unload();
}
if (_photo) {
_photo->unload();
}
}
void Result::openFile() { void Result::openFile() {
if (_document) { if (_document) {
DocumentOpenClickHandler(_document).onClick({}); DocumentOpenClickHandler(_document).onClick({});

View File

@ -52,7 +52,6 @@ public:
// inline bot result. If it returns true you need to send this result. // inline bot result. If it returns true you need to send this result.
bool onChoose(Layout::ItemBase *layout); bool onChoose(Layout::ItemBase *layout);
void unload();
void openFile(); void openFile();
void cancelFile(); void cancelFile();

View File

@ -297,21 +297,9 @@ void Inner::hideFinished() {
} }
void Inner::clearHeavyData() { void Inner::clearHeavyData() {
const auto unload = [](const auto &item) {
if (const auto document = item->getDocument()) {
document->unload();
}
if (const auto photo = item->getPhoto()) {
photo->unload();
}
if (const auto result = item->getResult()) {
result->unload();
}
item->unloadHeavyPart();
};
clearInlineRows(false); clearInlineRows(false);
for (const auto &[result, layout] : _inlineLayouts) { for (const auto &[result, layout] : _inlineLayouts) {
unload(layout); layout->unloadHeavyPart();
} }
} }

View File

@ -442,7 +442,6 @@ void Account::loggedOut() {
Local::reset(); Local::reset();
cSetOtherOnline(0); cSetOtherOnline(0);
Images::ClearRemote();
} }
void Account::destroyMtpKeys(MTP::AuthKeysList &&keys) { void Account::destroyMtpKeys(MTP::AuthKeysList &&keys) {

View File

@ -1383,8 +1383,6 @@ float64 MainWidget::chatBackgroundProgress() const {
return 1.; return 1.;
} else if (const auto document = _background->data.document()) { } else if (const auto document = _background->data.document()) {
return _background->dataMedia->progress(); return _background->dataMedia->progress();
} else if (const auto thumbnail = _background->data.localThumbnail()) {
return thumbnail->progress();
} }
} }
return 1.; return 1.;

View File

@ -22,6 +22,7 @@ class HistoryWidget;
class StackItem; class StackItem;
struct FileLoadResult; struct FileLoadResult;
class History; class History;
class Image;
namespace Api { namespace Api {
struct SendAction; struct SendAction;

View File

@ -241,10 +241,6 @@ void GroupThumbs::Thumb::validateImage() {
if (!_full.isNull() || !_image) { if (!_full.isNull() || !_image) {
return; return;
} }
_image->load(_origin);
if (!_image->loaded()) {
return;
}
const auto pixSize = wantedPixSize(); const auto pixSize = wantedPixSize();
if (pixSize.width() > st::mediaviewGroupWidthMax) { if (pixSize.width() > st::mediaviewGroupWidthMax) {

View File

@ -2214,15 +2214,10 @@ void OverlayWidget::initStreamingThumbnail() {
Expects(_document != nullptr); Expects(_document != nullptr);
const auto good = _documentMedia->goodThumbnail(); const auto good = _documentMedia->goodThumbnail();
const auto useGood = (good && good->loaded()); const auto useGood = (good != nullptr);
const auto thumbnail = _documentMedia->thumbnail(); const auto thumbnail = _documentMedia->thumbnail();
const auto useThumb = (thumbnail != nullptr); const auto useThumb = (thumbnail != nullptr);
const auto blurred = _documentMedia->thumbnailInline(); const auto blurred = _documentMedia->thumbnailInline();
if (good && !useGood) {
good->load({});
} else if (thumbnail) {
thumbnail->load(fileOrigin());
}
const auto size = useGood ? good->size() : _document->dimensions; const auto size = useGood ? good->size() : _document->dimensions;
if (!useGood && !thumbnail && !blurred) { if (!useGood && !thumbnail && !blurred) {
return; return;

View File

@ -1379,16 +1379,14 @@ QImage Pip::videoFrame(const FrameRequest &request) const {
? nullptr ? nullptr
: _data->createMediaView(); : _data->createMediaView();
const auto good = use ? use->goodThumbnail() : nullptr; const auto good = use ? use->goodThumbnail() : nullptr;
const auto useGood = (good && good->loaded());
const auto thumb = use ? use->thumbnail() : nullptr; const auto thumb = use ? use->thumbnail() : nullptr;
const auto useThumb = (thumb && thumb->loaded());
const auto blurred = use ? use->thumbnailInline() : nullptr; const auto blurred = use ? use->thumbnailInline() : nullptr;
const auto state = !cover.isNull() const auto state = !cover.isNull()
? ThumbState::Cover ? ThumbState::Cover
: useGood : good
? ThumbState::Good ? ThumbState::Good
: useThumb : thumb
? ThumbState::Thumb ? ThumbState::Thumb
: blurred : blurred
? ThumbState::Inline ? ThumbState::Inline
@ -1406,14 +1404,9 @@ QImage Pip::videoFrame(const FrameRequest &request) const {
request, request,
std::move(_preparedCoverStorage)); std::move(_preparedCoverStorage));
} else if (!request.resize.isEmpty()) { } else if (!request.resize.isEmpty()) {
if (good && !useGood) {
good->load({});
} else if (thumb && !useThumb) {
thumb->load(_contextId);
}
using Option = Images::Option; using Option = Images::Option;
const auto options = Option::Smooth const auto options = Option::Smooth
| (useGood ? Option(0) : Option::Blurred) | (good ? Option(0) : Option::Blurred)
| Option::RoundedLarge | Option::RoundedLarge
| ((request.corners & RectPart::TopLeft) | ((request.corners & RectPart::TopLeft)
? Option::RoundedTopLeft ? Option::RoundedTopLeft
@ -1427,9 +1420,9 @@ QImage Pip::videoFrame(const FrameRequest &request) const {
| ((request.corners & RectPart::BottomLeft) | ((request.corners & RectPart::BottomLeft)
? Option::RoundedBottomLeft ? Option::RoundedBottomLeft
: Option(0)); : Option(0));
_preparedCoverStorage = (useGood _preparedCoverStorage = (good
? good ? good
: useThumb : thumb
? thumb ? thumb
: blurred : blurred
? blurred ? blurred

View File

@ -358,8 +358,6 @@ void Photo::paint(Painter &p, const QRect &clip, TextSelection selection, const
} }
void Photo::setPixFrom(not_null<Image*> image) { void Photo::setPixFrom(not_null<Image*> image) {
Expects(image->loaded());
const auto size = _width * cIntRetinaFactor(); const auto size = _width * cIntRetinaFactor();
auto img = image->original(); auto img = image->original();
if (!_goodLoaded) { if (!_goodLoaded) {

View File

@ -13,6 +13,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/effects/radial_animation.h" #include "ui/effects/radial_animation.h"
#include "styles/style_overview.h" #include "styles/style_overview.h"
class Image;
namespace style { namespace style {
struct RoundCheckbox; struct RoundCheckbox;
} // namespace style } // namespace style

View File

@ -630,9 +630,7 @@ void ChooseFromFile(
} }
auto local = Data::CustomWallPaper(); auto local = Data::CustomWallPaper();
local.setLocalImageAsThumbnail(std::make_shared<Image>( local.setLocalImageAsThumbnail(std::make_shared<Image>(
std::make_unique<Images::ImageSource>( std::make_unique<Images::ImageSource>(std::move(image))));
std::move(image),
"JPG")));
Ui::show(Box<BackgroundPreviewBox>(session, local)); Ui::show(Box<BackgroundPreviewBox>(session, local));
}); });
FileDialog::GetOpenPath( FileDialog::GetOpenPath(

View File

@ -8,7 +8,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/image/image.h" #include "ui/image/image.h"
#include "ui/image/image_source.h" #include "ui/image/image_source.h"
#include "core/media_active_cache.h"
#include "storage/cache/storage_cache_database.h" #include "storage/cache/storage_cache_database.h"
#include "data/data_session.h" #include "data/data_session.h"
#include "data/data_file_origin.h" #include "data/data_file_origin.h"
@ -21,31 +20,6 @@ using namespace Images;
namespace Images { namespace Images {
namespace { namespace {
// After 128 MB of unpacked images we try to clear some memory.
constexpr auto kMemoryForCache = 128 * 1024 * 1024;
std::unordered_map<InMemoryKey, std::unique_ptr<Image>> StorageImages;
std::unordered_map<InMemoryKey, std::unique_ptr<Image>> GeoPointImages;
int64 ComputeUsage(QSize size) {
return int64(size.width()) * size.height() * 4;
}
int64 ComputeUsage(const QPixmap &image) {
return ComputeUsage(image.size());
}
int64 ComputeUsage(const QImage &image) {
return ComputeUsage(image.size());
}
[[nodiscard]] Core::MediaActiveCache<const Image> &ActiveCache() {
static auto Instance = Core::MediaActiveCache<const Image>(
kMemoryForCache,
[](const Image *image) { image->unload(); });
return Instance;
}
uint64 PixKey(int width, int height, Options options) { uint64 PixKey(int width, int height, Options options) {
return static_cast<uint64>(width) return static_cast<uint64>(width)
| (static_cast<uint64>(height) << 24) | (static_cast<uint64>(height) << 24)
@ -111,54 +85,6 @@ QImage FromInlineBytes(const QByteArray &bytes) {
return App::readImage(ExpandInlineBytes(bytes)); return App::readImage(ExpandInlineBytes(bytes));
} }
void ClearRemote() {
base::take(StorageImages);
base::take(GeoPointImages);
}
void ClearAll() {
ActiveCache().clear();
ClearRemote();
}
ImagePtr Create(QImage &&image, QByteArray format) {
return ImagePtr(new Image(std::make_unique<ImageSource>(
std::move(image),
format)));
}
template <typename SourceType>
ImagePtr Create(
const StorageImageLocation &location,
int size,
const QByteArray &bytes) {
if (!location.valid()) {
return ImagePtr();
}
const auto key = inMemoryKey(location);
const auto i = StorageImages.find(key);
const auto found = (i != end(StorageImages));
const auto image = found
? i->second.get()
: StorageImages.emplace(
key,
std::make_unique<Image>(
std::make_unique<SourceType>(location, size))
).first->second.get();
if (found) {
image->refreshFileReference(location.fileReference());
}
if (!bytes.isEmpty()) {
image->setImageBytes(bytes);
}
return ImagePtr(image);
}
ImagePtr Create(const StorageImageLocation &location, int size) {
return Create<StorageSource>(location, size, QByteArray());
}
QSize GetSizeForDocument(const QVector<MTPDocumentAttribute> &attributes) { QSize GetSizeForDocument(const QVector<MTPDocumentAttribute> &attributes) {
for (const auto &attribute : attributes) { for (const auto &attribute : attributes) {
if (attribute.type() == mtpc_documentAttributeImageSize) { if (attribute.type() == mtpc_documentAttributeImageSize) {
@ -169,33 +95,13 @@ QSize GetSizeForDocument(const QVector<MTPDocumentAttribute> &attributes) {
return QSize(); return QSize();
} }
ImagePtr Create(const GeoPointLocation &location) {
const auto key = inMemoryKey(location);
const auto i = GeoPointImages.find(key);
const auto image = (i != end(GeoPointImages))
? i->second.get()
: GeoPointImages.emplace(
key,
std::make_unique<Image>(
std::make_unique<GeoPointSource>(location))
).first->second.get();
return ImagePtr(image);
}
} // namespace Images } // namespace Images
Image::Image(std::unique_ptr<Source> &&source) Image::Image(std::unique_ptr<Source> &&source)
: _source(std::move(source)) { : _source(std::move(source)) {
} }
void Image::replaceSource(std::unique_ptr<Source> &&source) { Image::~Image() = default;
const auto width = _source->width();
const auto height = _source->height();
if (width > 0 && height > 0) {
source->setInformation(_source->bytesSize(), width, height);
}
_source = std::move(source);
}
not_null<Image*> Image::Empty() { not_null<Image*> Image::Empty() {
static auto result = [] { static auto result = [] {
@ -206,7 +112,7 @@ not_null<Image*> Image::Empty() {
QImage::Format_ARGB32_Premultiplied); QImage::Format_ARGB32_Premultiplied);
data.fill(Qt::transparent); data.fill(Qt::transparent);
data.setDevicePixelRatio(cRetinaFactor()); data.setDevicePixelRatio(cRetinaFactor());
return Image(std::make_unique<ImageSource>(std::move(data), "GIF")); return Image(std::make_unique<ImageSource>(std::move(data)));
}(); }();
return &result; return &result;
} }
@ -220,7 +126,7 @@ not_null<Image*> Image::BlankMedia() {
QImage::Format_ARGB32_Premultiplied); QImage::Format_ARGB32_Premultiplied);
data.fill(Qt::black); data.fill(Qt::black);
data.setDevicePixelRatio(cRetinaFactor()); data.setDevicePixelRatio(cRetinaFactor());
return Image(std::make_unique<ImageSource>(std::move(data), "GIF")); return Image(std::make_unique<ImageSource>(std::move(data)));
}(); }();
return &result; return &result;
} }
@ -248,7 +154,6 @@ const QPixmap &Image::pix(
auto p = pixNoCache(origin, w, h, options); auto p = pixNoCache(origin, w, h, options);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -287,7 +192,6 @@ const QPixmap &Image::pixRounded(
auto p = pixNoCache(origin, w, h, options); auto p = pixNoCache(origin, w, h, options);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -311,7 +215,6 @@ const QPixmap &Image::pixCircled(
auto p = pixNoCache(origin, w, h, options); auto p = pixNoCache(origin, w, h, options);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -335,7 +238,6 @@ const QPixmap &Image::pixBlurredCircled(
auto p = pixNoCache(origin, w, h, options); auto p = pixNoCache(origin, w, h, options);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -359,7 +261,6 @@ const QPixmap &Image::pixBlurred(
auto p = pixNoCache(origin, w, h, options); auto p = pixNoCache(origin, w, h, options);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -384,7 +285,6 @@ const QPixmap &Image::pixColored(
auto p = pixColoredNoCache(origin, add, w, h, true); auto p = pixColoredNoCache(origin, add, w, h, true);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -409,7 +309,6 @@ const QPixmap &Image::pixBlurredColored(
auto p = pixBlurredColoredNoCache(origin, add, w, h); auto p = pixBlurredColoredNoCache(origin, add, w, h);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -453,13 +352,9 @@ const QPixmap &Image::pixSingle(
auto k = SinglePixKey(options); auto k = SinglePixKey(options);
auto i = _sizesCache.constFind(k); auto i = _sizesCache.constFind(k);
if (i == _sizesCache.cend() || i->width() != (outerw * cIntRetinaFactor()) || i->height() != (outerh * cIntRetinaFactor())) { if (i == _sizesCache.cend() || i->width() != (outerw * cIntRetinaFactor()) || i->height() != (outerh * cIntRetinaFactor())) {
if (i != _sizesCache.cend()) {
ActiveCache().decrement(ComputeUsage(*i));
}
auto p = pixNoCache(origin, w, h, options, outerw, outerh, colored); auto p = pixNoCache(origin, w, h, options, outerw, outerh, colored);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -499,13 +394,9 @@ const QPixmap &Image::pixBlurredSingle(
auto k = SinglePixKey(options); auto k = SinglePixKey(options);
auto i = _sizesCache.constFind(k); auto i = _sizesCache.constFind(k);
if (i == _sizesCache.cend() || i->width() != (outerw * cIntRetinaFactor()) || i->height() != (outerh * cIntRetinaFactor())) { if (i == _sizesCache.cend() || i->width() != (outerw * cIntRetinaFactor()) || i->height() != (outerh * cIntRetinaFactor())) {
if (i != _sizesCache.cend()) {
ActiveCache().decrement(ComputeUsage(*i));
}
auto p = pixNoCache(origin, w, h, options, outerw, outerh); auto p = pixNoCache(origin, w, h, options, outerw, outerh);
p.setDevicePixelRatio(cRetinaFactor()); p.setDevicePixelRatio(cRetinaFactor());
i = _sizesCache.insert(k, p); i = _sizesCache.insert(k, p);
ActiveCache().increment(ComputeUsage(*i));
} }
return i.value(); return i.value();
} }
@ -518,9 +409,6 @@ QPixmap Image::pixNoCache(
int outerw, int outerw,
int outerh, int outerh,
const style::color *colored) const { const style::color *colored) const {
if (!loading()) {
const_cast<Image*>(this)->load(origin);
}
checkSource(); checkSource();
if (_data.isNull()) { if (_data.isNull()) {
@ -579,9 +467,6 @@ QPixmap Image::pixColoredNoCache(
int32 w, int32 w,
int32 h, int32 h,
bool smooth) const { bool smooth) const {
if (!loading()) {
const_cast<Image*>(this)->load(origin);
}
checkSource(); checkSource();
if (_data.isNull()) { if (_data.isNull()) {
@ -603,9 +488,6 @@ QPixmap Image::pixBlurredColoredNoCache(
style::color add, style::color add,
int32 w, int32 w,
int32 h) const { int32 h) const {
if (!loading()) {
const_cast<Image*>(this)->load(origin);
}
checkSource(); checkSource();
if (_data.isNull()) { if (_data.isNull()) {
@ -627,22 +509,12 @@ QImage Image::original() const {
return _data; return _data;
} }
void Image::load(Data::FileOrigin origin) { void Image::load() {
if (!loaded()) { if (!loaded()) {
_source->load(origin); _source->load();
} }
} }
void Image::loadEvenCancelled(Data::FileOrigin origin) {
if (!loaded()) {
_source->loadEvenCancelled(origin);
}
}
Storage::Cache::Key Image::cacheKey() const {
return _source->cacheKey();
}
bool Image::loaded() const { bool Image::loaded() const {
checkSource(); checkSource();
return !_data.isNull(); return !_data.isNull();
@ -653,46 +525,9 @@ void Image::checkSource() const {
if (_data.isNull() && !data.isNull()) { if (_data.isNull() && !data.isNull()) {
invalidateSizeCache(); invalidateSizeCache();
_data = std::move(data); _data = std::move(data);
ActiveCache().increment(ComputeUsage(_data));
} }
ActiveCache().up(this);
}
void Image::unload() const {
_source->unload();
invalidateSizeCache();
ActiveCache().decrement(ComputeUsage(_data));
_data = QImage();
}
void Image::setDelayedStorageLocation(
Data::FileOrigin origin,
const StorageImageLocation &location) {
_source->setDelayedStorageLocation(location);
if (!loaded()) {
_source->performDelayedLoad(origin);
}
}
void Image::setImageBytes(const QByteArray &bytes) {
_source->setImageBytes(bytes);
checkSource();
} }
void Image::invalidateSizeCache() const { void Image::invalidateSizeCache() const {
auto &cache = ActiveCache();
for (const auto &image : std::as_const(_sizesCache)) {
cache.decrement(ComputeUsage(image));
}
_sizesCache.clear(); _sizesCache.clear();
} }
Image::~Image() {
if (this != Empty() && this != BlankMedia()) {
invalidateSizeCache();
ActiveCache().decrement(ComputeUsage(_data));
_data = QImage();
ActiveCache().remove(this);
}
}

View File

@ -16,16 +16,9 @@ namespace Images {
[[nodiscard]] QByteArray ExpandInlineBytes(const QByteArray &bytes); [[nodiscard]] QByteArray ExpandInlineBytes(const QByteArray &bytes);
[[nodiscard]] QImage FromInlineBytes(const QByteArray &bytes); [[nodiscard]] QImage FromInlineBytes(const QByteArray &bytes);
void ClearRemote();
void ClearAll();
[[nodiscard]] QSize GetSizeForDocument( [[nodiscard]] QSize GetSizeForDocument(
const QVector<MTPDocumentAttribute> &attributes); const QVector<MTPDocumentAttribute> &attributes);
ImagePtr Create(QImage &&data, QByteArray format);
ImagePtr Create(const StorageImageLocation &location, int size = 0);
ImagePtr Create(const GeoPointLocation &location);
class Source { class Source {
public: public:
Source() = default; Source() = default;
@ -35,31 +28,11 @@ public:
Source &operator=(Source &&other) = delete; Source &operator=(Source &&other) = delete;
virtual ~Source() = default; virtual ~Source() = default;
virtual void load(Data::FileOrigin origin) = 0; virtual void load() = 0;
virtual void loadEvenCancelled(Data::FileOrigin origin) = 0;
virtual QImage takeLoaded() = 0; virtual QImage takeLoaded() = 0;
virtual void unload() = 0;
virtual bool loading() = 0;
virtual bool displayLoading() = 0;
virtual void cancel() = 0;
virtual float64 progress() = 0;
virtual int loadOffset() = 0;
virtual const StorageImageLocation &location() = 0;
virtual void refreshFileReference(const QByteArray &data) = 0;
virtual Storage::Cache::Key cacheKey() = 0;
virtual void setDelayedStorageLocation(
const StorageImageLocation &location) = 0;
virtual void performDelayedLoad(Data::FileOrigin origin) = 0;
virtual void setImageBytes(const QByteArray &bytes) = 0;
virtual int width() = 0; virtual int width() = 0;
virtual int height() = 0; virtual int height() = 0;
virtual int bytesSize() = 0;
virtual void setInformation(int size, int width, int height) = 0;
virtual QByteArray bytesForCache() = 0;
}; };
@ -69,8 +42,6 @@ class Image final {
public: public:
explicit Image(std::unique_ptr<Images::Source> &&source); explicit Image(std::unique_ptr<Images::Source> &&source);
void replaceSource(std::unique_ptr<Images::Source> &&source);
static not_null<Image*> Empty(); // 1x1 transparent static not_null<Image*> Empty(); // 1x1 transparent
static not_null<Image*> BlankMedia(); // 1x1 black static not_null<Image*> BlankMedia(); // 1x1 black
@ -145,21 +116,6 @@ public:
int32 w, int32 w,
int32 h = 0) const; int32 h = 0) const;
bool loading() const {
return _source->loading();
}
bool displayLoading() const {
return _source->displayLoading();
}
void cancel() {
_source->cancel();
}
float64 progress() const {
return loaded() ? 1. : _source->progress();
}
int loadOffset() const {
return _source->loadOffset();
}
int width() const { int width() const {
return _source->width(); return _source->width();
} }
@ -169,32 +125,10 @@ public:
QSize size() const { QSize size() const {
return { width(), height() }; return { width(), height() };
} }
int bytesSize() const { void load();
return _source->bytesSize();
}
void setInformation(int size, int width, int height) {
_source->setInformation(size, width, height);
}
void load(Data::FileOrigin origin);
void loadEvenCancelled(Data::FileOrigin origin);
const StorageImageLocation &location() const {
return _source->location();
}
void refreshFileReference(const QByteArray &data) {
_source->refreshFileReference(data);
}
Storage::Cache::Key cacheKey() const;
QByteArray bytesForCache() const {
return _source->bytesForCache();
}
bool loaded() const; bool loaded() const;
bool isNull() const; bool isNull() const;
void unload() const;
void setDelayedStorageLocation(
Data::FileOrigin origin,
const StorageImageLocation &location);
void setImageBytes(const QByteArray &bytes);
~Image(); ~Image();

View File

@ -68,23 +68,6 @@ MTPInputPeer GenerateInputPeer(
} // namespace } // namespace
ImagePtr::ImagePtr() : _data(Image::Empty()) {
}
ImagePtr::ImagePtr(not_null<Image*> data) : _data(data) {
}
Image *ImagePtr::operator->() const {
return _data;
}
Image *ImagePtr::get() const {
return _data;
}
ImagePtr::operator bool() const {
return !_data->isNull();
}
WebFileLocation WebFileLocation::Null; WebFileLocation WebFileLocation::Null;
StorageFileLocation::StorageFileLocation( StorageFileLocation::StorageFileLocation(

View File

@ -616,22 +616,6 @@ struct ImageWithLocation {
QImage preloaded; QImage preloaded;
}; };
class Image;
class ImagePtr {
public:
ImagePtr();
explicit ImagePtr(not_null<Image*> data);
Image *operator->() const;
Image *get() const;
explicit operator bool() const;
private:
not_null<Image*> _data;
};
InMemoryKey inMemoryKey(const StorageFileLocation &location); InMemoryKey inMemoryKey(const StorageFileLocation &location);
inline InMemoryKey inMemoryKey(const StorageImageLocation &location) { inline InMemoryKey inMemoryKey(const StorageImageLocation &location) {

View File

@ -20,597 +20,45 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include <QtCore/QBuffer> #include <QtCore/QBuffer>
namespace Images { namespace Images {
namespace {
ImageSource::ImageSource(QImage &&data, const QByteArray &format) [[nodiscard]] QByteArray ReadContent(const QString &path) {
: _data(std::move(data)) auto file = QFile(path);
, _format(format) const auto good = (file.size() <= App::kImageSizeLimit)
, _width(_data.width()) && file.open(QIODevice::ReadOnly);
, _height(_data.height()) { return good ? file.readAll() : QByteArray();
} }
void ImageSource::load(Data::FileOrigin origin) { [[nodiscard]] QImage ReadImage(const QByteArray &content) {
if (_data.isNull() && !_bytes.isEmpty()) { return App::readImage(content, nullptr, false, nullptr);
_data = App::readImage(_bytes, &_format, false);
}
} }
void ImageSource::loadEvenCancelled(Data::FileOrigin origin) { } // namespace
load(origin);
ImageSource::ImageSource(const QString &path)
: ImageSource(ReadContent(path)) {
}
ImageSource::ImageSource(const QByteArray &content)
: ImageSource(ReadImage(content)) {
}
ImageSource::ImageSource(QImage &&data) : _data(std::move(data)) {
}
void ImageSource::load() {
} }
QImage ImageSource::takeLoaded() { QImage ImageSource::takeLoaded() {
load({});
return _data; return _data;
} }
void ImageSource::unload() {
if (_bytes.isEmpty() && !_data.isNull()) {
if (_format != "JPG") {
_format = "PNG";
}
{
QBuffer buffer(&_bytes);
_data.save(&buffer, _format);
}
Assert(!_bytes.isEmpty());
}
_data = QImage();
}
bool ImageSource::loading() {
return false;
}
bool ImageSource::displayLoading() {
return false;
}
void ImageSource::cancel() {
}
float64 ImageSource::progress() {
return 1.;
}
int ImageSource::loadOffset() {
return 0;
}
const StorageImageLocation &ImageSource::location() {
return StorageImageLocation::Invalid();
}
void ImageSource::refreshFileReference(const QByteArray &data) {
}
Storage::Cache::Key ImageSource::cacheKey() {
return Storage::Cache::Key();
}
void ImageSource::setDelayedStorageLocation(
const StorageImageLocation &location) {
}
void ImageSource::performDelayedLoad(Data::FileOrigin origin) {
}
void ImageSource::setImageBytes(const QByteArray &bytes) {
}
int ImageSource::width() { int ImageSource::width() {
return _width; return _data.width();
} }
int ImageSource::height() { int ImageSource::height() {
return _height; return _data.height();
}
int ImageSource::bytesSize() {
return _bytes.size();
}
void ImageSource::setInformation(int size, int width, int height) {
if (width && height) {
_width = width;
_height = height;
}
}
QByteArray ImageSource::bytesForCache() {
auto result = QByteArray();
{
QBuffer buffer(&result);
if (!_data.save(&buffer, _format)) {
if (_data.save(&buffer, "PNG")) {
_format = "PNG";
}
}
}
return result;
}
LocalFileSource::LocalFileSource(
const QString &path,
const QByteArray &content,
const QByteArray &format,
QImage &&data)
: _path(path)
, _bytes(content)
, _format(format)
, _data(std::move(data))
, _width(_data.width())
, _height(_data.height()) {
}
void LocalFileSource::load(Data::FileOrigin origin) {
if (!_data.isNull()) {
return;
}
if (_bytes.isEmpty()) {
QFile f(_path);
if (f.size() <= App::kImageSizeLimit && f.open(QIODevice::ReadOnly)) {
_bytes = f.readAll();
}
if (_bytes.isEmpty()) {
_bytes = "(bad)";
}
}
if (_bytes != "(bad)") {
_data = App::readImage(_bytes, &_format, false, nullptr);
}
_width = std::max(_data.width(), 1);
_height = std::max(_data.height(), 1);
}
void LocalFileSource::loadEvenCancelled(Data::FileOrigin origin) {
load(origin);
}
QImage LocalFileSource::takeLoaded() {
return std::move(_data);
}
void LocalFileSource::unload() {
_data = QImage();
}
bool LocalFileSource::loading() {
return false;
}
bool LocalFileSource::displayLoading() {
return false;
}
void LocalFileSource::cancel() {
}
float64 LocalFileSource::progress() {
return 1.;
}
int LocalFileSource::loadOffset() {
return 0;
}
const StorageImageLocation &LocalFileSource::location() {
return StorageImageLocation::Invalid();
}
void LocalFileSource::refreshFileReference(const QByteArray &data) {
}
Storage::Cache::Key LocalFileSource::cacheKey() {
return Storage::Cache::Key();
}
void LocalFileSource::setDelayedStorageLocation(
const StorageImageLocation &location) {
}
void LocalFileSource::performDelayedLoad(Data::FileOrigin origin) {
}
void LocalFileSource::setImageBytes(const QByteArray &bytes) {
_bytes = bytes;
load({});
}
int LocalFileSource::width() {
ensureDimensionsKnown();
return _width;
}
int LocalFileSource::height() {
ensureDimensionsKnown();
return _height;
}
int LocalFileSource::bytesSize() {
ensureDimensionsKnown();
return _bytes.size();
}
void LocalFileSource::setInformation(int size, int width, int height) {
ensureDimensionsKnown(); // First load _bytes.
if (width && height) {
_width = width;
_height = height;
}
}
void LocalFileSource::ensureDimensionsKnown() {
if (!_width || !_height) {
load({});
}
}
QByteArray LocalFileSource::bytesForCache() {
ensureDimensionsKnown();
return (_bytes == "(bad)") ? QByteArray() : _bytes;
}
QImage RemoteSource::takeLoaded() {
if (!_loader || !_loader->finished()) {
return QImage();
}
if (_loader->cancelled()) {
_cancelled = true;
destroyLoader();
return QImage();
}
auto data = _loader->imageData(shrinkBox());
if (data.isNull()) {
// Bad content in the image.
data = Image::Empty()->original();
}
setInformation(_loader->bytes().size(), data.width(), data.height());
destroyLoader();
return data;
}
void RemoteSource::destroyLoader() {
if (!_loader) {
return;
}
const auto loader = base::take(_loader);
if (cancelled()) {
loader->cancel();
}
}
void RemoteSource::loadLocal() {
if (_loader) {
return;
}
_loader = createLoader(Data::FileOrigin(), LoadFromLocalOnly, true);
if (_loader) {
_loader->start();
}
}
void RemoteSource::setImageBytes(const QByteArray &bytes) {
if (bytes.isEmpty()) {
return;
} else if (_loader) {
unload();
}
_loader = createLoader({}, LoadFromLocalOnly, true);
_loader->finishWithBytes(bytes);
const auto location = this->location();
if (location.valid()
&& !bytes.isEmpty()
&& bytes.size() <= Storage::kMaxFileInMemory) {
Auth().data().cache().putIfEmpty(
location.file().cacheKey(),
Storage::Cache::Database::TaggedValue(
base::duplicate(bytes),
Data::kImageCacheTag));
}
}
bool RemoteSource::loading() {
return (_loader != nullptr);
}
void RemoteSource::load(Data::FileOrigin origin) {
if (!_loader) {
_loader = createLoader(origin, LoadFromCloudOrLocal, false);
}
if (_loader) {
_loader->start();
}
}
bool RemoteSource::cancelled() const {
return _cancelled;
}
void RemoteSource::loadEvenCancelled(Data::FileOrigin origin) {
_cancelled = false;
return load(origin);
}
bool RemoteSource::displayLoading() {
return _loader && (!_loader->loadingLocal() || !_loader->autoLoading());
}
void RemoteSource::cancel() {
if (!_loader) {
return;
}
_cancelled = true;
destroyLoader();
}
void RemoteSource::unload() {
base::take(_loader);
}
float64 RemoteSource::progress() {
return _loader ? _loader->currentProgress() : 0.;
}
int RemoteSource::loadOffset() {
return _loader ? _loader->currentOffset() : 0;
}
RemoteSource::~RemoteSource() {
unload();
}
const StorageImageLocation &RemoteSource::location() {
return StorageImageLocation::Invalid();
}
void RemoteSource::refreshFileReference(const QByteArray &data) {
}
void RemoteSource::setDelayedStorageLocation(
const StorageImageLocation &location) {
}
void RemoteSource::performDelayedLoad(Data::FileOrigin origin) {
}
QByteArray RemoteSource::bytesForCache() {
return QByteArray();
}
StorageSource::StorageSource(const StorageImageLocation &location, int size)
: _location(location)
, _size(size) {
}
void StorageSource::refreshFileReference(const QByteArray &data) {
_location.refreshFileReference(data);
}
const StorageImageLocation &StorageSource::location() {
return _location;
}
Storage::Cache::Key StorageSource::cacheKey() {
return _location.valid()
? _location.file().cacheKey()
: Storage::Cache::Key();
}
int StorageSource::width() {
return _location.width();
}
int StorageSource::height() {
return _location.height();
}
int StorageSource::bytesSize() {
return _size;
}
void StorageSource::setInformation(int size, int width, int height) {
if (size) {
_size = size;
}
if (width && height) {
_location.setSize(width, height);
}
}
QSize StorageSource::shrinkBox() const {
return QSize();
}
std::unique_ptr<FileLoader> StorageSource::createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) {
return _location.valid()
? std::make_unique<mtpFileLoader>(
_location.file(),
origin,
UnknownFileLocation,
QString(),
_size,
LoadToCacheAsWell,
fromCloud,
autoLoading,
Data::kImageCacheTag)
: nullptr;
}
WebCachedSource::WebCachedSource(
const WebFileLocation &location,
QSize box,
int size)
: _location(location)
, _box(box)
, _size(size) {
}
WebCachedSource::WebCachedSource(
const WebFileLocation &location,
int width,
int height,
int size)
: _location(location)
, _width(width)
, _height(height)
, _size(size) {
}
Storage::Cache::Key WebCachedSource::cacheKey() {
return _location.isNull()
? Storage::Cache::Key()
: Data::WebDocumentCacheKey(_location);
}
int WebCachedSource::width() {
return _width;
}
int WebCachedSource::height() {
return _height;
}
int WebCachedSource::bytesSize() {
return _size;
}
void WebCachedSource::setInformation(int size, int width, int height) {
if (size) {
_size = size;
}
if (width && height) {
_width = width;
_height = height;
}
}
QSize WebCachedSource::shrinkBox() const {
return _box;
}
std::unique_ptr<FileLoader> WebCachedSource::createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) {
return !_location.isNull()
? std::make_unique<mtpFileLoader>(
_location,
_size,
fromCloud,
autoLoading,
Data::kImageCacheTag)
: nullptr;
}
GeoPointSource::GeoPointSource(const GeoPointLocation &location)
: _location(location) {
}
Storage::Cache::Key GeoPointSource::cacheKey() {
return Data::GeoPointCacheKey(_location);
}
int GeoPointSource::width() {
return _location.width * _location.scale;
}
int GeoPointSource::height() {
return _location.height * _location.scale;
}
int GeoPointSource::bytesSize() {
return _size;
}
void GeoPointSource::setInformation(int size, int width, int height) {
Expects(_location.scale != 0);
if (size) {
_size = size;
}
if (width && height) {
_location.width = width / _location.scale;
_location.height = height / _location.scale;
}
}
QSize GeoPointSource::shrinkBox() const {
return QSize();
}
std::unique_ptr<FileLoader> GeoPointSource::createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) {
return std::make_unique<mtpFileLoader>(
_location,
_size,
fromCloud,
autoLoading,
Data::kImageCacheTag);
}
WebUrlSource::WebUrlSource(const QString &url, QSize box)
: _url(url)
, _box(box) {
}
WebUrlSource::WebUrlSource(const QString &url, int width, int height)
: _url(url)
, _width(width)
, _height(height) {
}
Storage::Cache::Key WebUrlSource::cacheKey() {
return Data::UrlCacheKey(_url);
}
int WebUrlSource::width() {
return _width;
}
int WebUrlSource::height() {
return _height;
}
int WebUrlSource::bytesSize() {
return _size;
}
void WebUrlSource::setInformation(int size, int width, int height) {
if (size) {
_size = size;
}
if (width && height) {
_width = width;
_height = height;
}
}
QSize WebUrlSource::shrinkBox() const {
return _box;
}
std::unique_ptr<FileLoader> WebUrlSource::createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) {
return std::make_unique<webFileLoader>(
_url,
QString(),
fromCloud,
autoLoading,
Data::kImageCacheTag);
} }
} // namespace Images } // namespace Images

View File

@ -13,244 +13,18 @@ namespace Images {
class ImageSource : public Source { class ImageSource : public Source {
public: public:
ImageSource(QImage &&data, const QByteArray &format); explicit ImageSource(const QString &path);
explicit ImageSource(const QByteArray &content);
explicit ImageSource(QImage &&data);
void load(Data::FileOrigin origin) override; void load() override;
void loadEvenCancelled(Data::FileOrigin origin) override;
QImage takeLoaded() override; QImage takeLoaded() override;
void unload() override;
bool loading() override;
bool displayLoading() override;
void cancel() override;
float64 progress() override;
int loadOffset() override;
const StorageImageLocation &location() override;
void refreshFileReference(const QByteArray &data) override;
Storage::Cache::Key cacheKey() override;
void setDelayedStorageLocation(
const StorageImageLocation &location) override;
void performDelayedLoad(Data::FileOrigin origin) override;
void setImageBytes(const QByteArray &bytes) override;
int width() override; int width() override;
int height() override; int height() override;
int bytesSize() override;
void setInformation(int size, int width, int height) override;
QByteArray bytesForCache() override;
private: private:
QImage _data; QImage _data;
QByteArray _format;
QByteArray _bytes;
int _width = 0;
int _height = 0;
};
class LocalFileSource : public Source {
public:
explicit LocalFileSource(
const QString &path,
const QByteArray &content = QByteArray(),
const QByteArray &format = QByteArray(),
QImage &&data = QImage());
void load(Data::FileOrigin origin) override;
void loadEvenCancelled(Data::FileOrigin origin) override;
QImage takeLoaded() override;
void unload() override;
bool loading() override;
bool displayLoading() override;
void cancel() override;
float64 progress() override;
int loadOffset() override;
const StorageImageLocation &location() override;
void refreshFileReference(const QByteArray &data) override;
Storage::Cache::Key cacheKey() override;
void setDelayedStorageLocation(
const StorageImageLocation &location) override;
void performDelayedLoad(Data::FileOrigin origin) override;
void setImageBytes(const QByteArray &bytes) override;
int width() override;
int height() override;
int bytesSize() override;
void setInformation(int size, int width, int height) override;
QByteArray bytesForCache() override;
private:
void ensureDimensionsKnown();
QString _path;
QByteArray _bytes;
QByteArray _format;
QImage _data;
int _width = 0;
int _height = 0;
};
class RemoteSource : public Source {
public:
void load(Data::FileOrigin origin) override;
void loadEvenCancelled(Data::FileOrigin origin) override;
QImage takeLoaded() override;
void unload() override;
bool loading() override;
bool displayLoading() override;
void cancel() override;
float64 progress() override;
int loadOffset() override;
const StorageImageLocation &location() override;
void refreshFileReference(const QByteArray &data) override;
void setDelayedStorageLocation(
const StorageImageLocation &location) override;
void performDelayedLoad(Data::FileOrigin origin) override;
void setImageBytes(const QByteArray &bytes) override;
QByteArray bytesForCache() override;
~RemoteSource();
protected:
// If after loading the image we need to shrink it to fit into a
// specific size, you can return this size here.
virtual QSize shrinkBox() const = 0;
virtual std::unique_ptr<FileLoader> createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) = 0;
void loadLocal();
FileLoader *currentLoader() const {
return _loader.get();
}
private:
bool cancelled() const;
void destroyLoader();
std::unique_ptr<FileLoader> _loader;
bool _cancelled = false;
};
class StorageSource : public RemoteSource {
public:
StorageSource(
const StorageImageLocation &location,
int size);
const StorageImageLocation &location() override;
Storage::Cache::Key cacheKey() override;
void refreshFileReference(const QByteArray &data) override;
int width() override;
int height() override;
int bytesSize() override;
void setInformation(int size, int width, int height) override;
protected:
QSize shrinkBox() const override;
std::unique_ptr<FileLoader> createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) override;
StorageImageLocation _location;
int _size = 0;
};
class WebCachedSource : public RemoteSource {
public:
WebCachedSource(const WebFileLocation &location, QSize box, int size = 0);
WebCachedSource(
const WebFileLocation &location,
int width,
int height,
int size = 0);
Storage::Cache::Key cacheKey() override;
int width() override;
int height() override;
int bytesSize() override;
void setInformation(int size, int width, int height) override;
protected:
QSize shrinkBox() const override;
std::unique_ptr<FileLoader> createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) override;
WebFileLocation _location;
QSize _box;
int _width = 0;
int _height = 0;
int _size = 0;
};
class GeoPointSource : public RemoteSource {
public:
GeoPointSource(const GeoPointLocation &location);
Storage::Cache::Key cacheKey() override;
int width() override;
int height() override;
int bytesSize() override;
void setInformation(int size, int width, int height) override;
protected:
QSize shrinkBox() const override;
std::unique_ptr<FileLoader> createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) override;
GeoPointLocation _location;
int _size = 0;
};
class WebUrlSource : public RemoteSource {
public:
// If !box.isEmpty() then resize the image to fit in this box.
explicit WebUrlSource(const QString &url, QSize box = QSize());
WebUrlSource(const QString &url, int width, int height);
Storage::Cache::Key cacheKey() override;
int width() override;
int height() override;
int bytesSize() override;
void setInformation(int size, int width, int height) override;
protected:
QSize shrinkBox() const override;
std::unique_ptr<FileLoader> createLoader(
Data::FileOrigin origin,
LoadFromCloudSetting fromCloud,
bool autoLoading) override;
private:
QString _url;
QSize _box;
int _size = 0;
int _width = 0;
int _height = 0;
}; };