Read background image async.

This commit is contained in:
John Preston 2019-01-16 16:39:35 +04:00
parent 0f9c2a62fe
commit b8cb792831
2 changed files with 65 additions and 36 deletions

View File

@ -220,6 +220,11 @@ StackItemSection::StackItemSection(
, _memento(std::move(memento)) { , _memento(std::move(memento)) {
} }
struct MainWidget::SettingBackground {
Data::WallPaper data;
base::binary_guard generating;
};
MainWidget::MainWidget( MainWidget::MainWidget(
QWidget *parent, QWidget *parent,
not_null<Window::Controller*> controller) not_null<Window::Controller*> controller)
@ -1414,11 +1419,12 @@ void MainWidget::updateScrollColors() {
} }
void MainWidget::setChatBackground(const Data::WallPaper &background) { void MainWidget::setChatBackground(const Data::WallPaper &background) {
_background = std::make_unique<Data::WallPaper>(background); _background = std::make_unique<SettingBackground>();
if (_background->document) { _background->data = background;
_background->document->save(Data::FileOrigin(), QString()); if (_background->data.document) {
} else if (_background->thumb) { _background->data.document->save(Data::FileOrigin(), QString());
_background->thumb->loadEvenCancelled(Data::FileOrigin()); } else if (_background->data.thumb) {
_background->data.thumb->loadEvenCancelled(Data::FileOrigin());
} }
checkChatBackground(); checkChatBackground();
@ -1433,71 +1439,91 @@ bool MainWidget::chatBackgroundLoading() {
float64 MainWidget::chatBackgroundProgress() const { float64 MainWidget::chatBackgroundProgress() const {
if (_background) { if (_background) {
if (_background->document) { if (_background->generating.alive()) {
return _background->document->progress(); return 1.;
} else if (_background->thumb) { } else if (_background->data.document) {
return _background->thumb->progress(); return _background->data.document->progress();
} else if (_background->data.thumb) {
return _background->data.thumb->progress();
} }
} }
return 1.; return 1.;
} }
void MainWidget::checkChatBackground() { void MainWidget::checkChatBackground() {
using namespace Window::Theme; if (!_background || _background->generating.alive()) {
if (!_background) {
return; return;
} }
const auto document = _background->document; const auto document = _background->data.document;
if (document && !document->loaded()) { if (document && !document->loaded()) {
return; return;
} else if (!document && !_background->thumb->loaded()) { } else if (!document && !_background->data.thumb->loaded()) {
return; return;
} }
auto image = [&] { if (!document) {
if (!document) { const auto &thumb = _background->data.thumb;
const auto &thumb = _background->thumb; setGeneratedBackground(thumb
return thumb ? thumb->pixNoCache(Data::FileOrigin()).toImage()
? thumb->pixNoCache(Data::FileOrigin()).toImage() : QImage());
: QImage(); return;
} }
auto bytes = document->data(); auto [left, right] = base::make_binary_guard();
_background->generating = std::move(left);
crl::async([
this,
bytes = document->data(),
path = document->filepath(),
guard = std::move(right)
]() mutable {
auto format = QByteArray(); auto format = QByteArray();
const auto path = document->filepath();
if (bytes.isEmpty()) { if (bytes.isEmpty()) {
QFile f(document->filepath()); QFile f(path);
if (f.size() <= App::kImageSizeLimit if (f.size() <= App::kImageSizeLimit
&& f.open(QIODevice::ReadOnly)) { && f.open(QIODevice::ReadOnly)) {
bytes = f.readAll(); bytes = f.readAll();
} }
} }
return bytes.isEmpty() auto image = bytes.isEmpty()
? QImage() ? QImage()
: App::readImage(bytes, &format, false, nullptr); : App::readImage(bytes, &format, false, nullptr);
}(); crl::on_main([
this,
guard = std::move(guard),
image = std::move(image)
]() mutable {
if (!guard.alive()) {
return;
}
setGeneratedBackground(std::move(image));
});
});
}
ImagePtr MainWidget::newBackgroundThumb() {
return _background ? _background->data.thumb : ImagePtr();
}
void MainWidget::setGeneratedBackground(QImage &&image) {
using namespace Window::Theme;
if (image.isNull()) { if (image.isNull()) {
Background()->setImage(kDefaultBackground); Background()->setImage(kDefaultBackground);
} else if (false } else if (false
|| _background->id == kInitialBackground || _background->data.id == kInitialBackground
|| _background->id == kDefaultBackground) { || _background->data.id == kDefaultBackground) {
Background()->setImage(_background->id); Background()->setImage(_background->data.id);
} else { } else {
Background()->setImage( Background()->setImage(
_background->id, _background->data.id,
std::move(image)); std::move(image));
} }
const auto tile = (_background->id == kInitialBackground); const auto tile = (_background->data.id == kInitialBackground);
Background()->setTile(tile); Background()->setTile(tile);
_background = nullptr; _background = nullptr;
crl::on_main(this, [=] { update(); }); crl::on_main(this, [=] { update(); });
} }
ImagePtr MainWidget::newBackgroundThumb() {
return _background ? _background->thumb : ImagePtr();
}
void MainWidget::messageDataReceived(ChannelData *channel, MsgId msgId) { void MainWidget::messageDataReceived(ChannelData *channel, MsgId msgId) {
_history->messageDataReceived(channel, msgId); _history->messageDataReceived(channel, msgId);
} }

View File

@ -454,6 +454,8 @@ private:
void ensureFirstColumnResizeAreaCreated(); void ensureFirstColumnResizeAreaCreated();
void ensureThirdColumnResizeAreaCreated(); void ensureThirdColumnResizeAreaCreated();
void setGeneratedBackground(QImage &&image);
not_null<Window::Controller*> _controller; not_null<Window::Controller*> _controller;
bool _started = false; bool _started = false;
@ -543,7 +545,8 @@ private:
QMap<mtpRequestId, PeerData*> _viewsIncrementByRequest; QMap<mtpRequestId, PeerData*> _viewsIncrementByRequest;
SingleTimer _viewsIncrementTimer; SingleTimer _viewsIncrementTimer;
std::unique_ptr<Data::WallPaper> _background; struct SettingBackground;
std::unique_ptr<SettingBackground> _background;
bool _firstColumnResizing = false; bool _firstColumnResizing = false;
int _firstColumnResizingShift = 0; int _firstColumnResizingShift = 0;