qt5base-lts/examples/qtconcurrent/imagescaling/imagescaling.cpp

220 lines
5.7 KiB
C++
Raw Normal View History

// Copyright (C) 2020 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
#include "imagescaling.h"
#include "downloaddialog.h"
#include <QNetworkReply>
#include <qmath.h>
#include <functional>
Images::Images(QWidget *parent) : QWidget(parent), downloadDialog(new DownloadDialog())
{
setWindowTitle(tr("Image downloading and scaling example"));
resize(800, 600);
addUrlsButton = new QPushButton(tr("Add URLs"));
//! [1]
connect(addUrlsButton, &QPushButton::clicked, this, &Images::process);
//! [1]
cancelButton = new QPushButton(tr("Cancel"));
cancelButton->setEnabled(false);
//! [2]
connect(cancelButton, &QPushButton::clicked, this, &Images::cancel);
//! [2]
QHBoxLayout *buttonLayout = new QHBoxLayout();
buttonLayout->addWidget(addUrlsButton);
buttonLayout->addWidget(cancelButton);
buttonLayout->addStretch();
statusBar = new QStatusBar();
imagesLayout = new QGridLayout();
mainLayout = new QVBoxLayout();
mainLayout->addLayout(buttonLayout);
mainLayout->addLayout(imagesLayout);
mainLayout->addStretch();
mainLayout->addWidget(statusBar);
setLayout(mainLayout);
}
Images::~Images()
{
cancel();
}
//! [3]
void Images::process()
{
// Clean previous state
replies.clear();
if (downloadDialog->exec() == QDialog::Accepted) {
const auto urls = downloadDialog->getUrls();
if (urls.empty())
return;
cancelButton->setEnabled(true);
initLayout(urls.size());
downloadFuture = download(urls);
statusBar->showMessage(tr("Downloading..."));
//! [3]
//! [4]
downloadFuture.then([this](auto) { cancelButton->setEnabled(false); })
.then(QtFuture::Launch::Async,
[this] {
QMetaObject::invokeMethod(this,
[this] { updateStatus(tr("Scaling...")); });
return scaled();
})
//! [4]
//! [5]
.then(this, [this](const QList<QImage> &scaled) {
showImages(scaled);
updateStatus(tr("Finished"));
})
//! [5]
//! [6]
.onCanceled([this] { updateStatus(tr("Download has been canceled.")); })
.onFailed([this](QNetworkReply::NetworkError error) {
updateStatus(tr("Download finished with error: %1").arg(error));
// Abort all pending requests
abortDownload();
})
.onFailed([this](const std::exception& ex) {
updateStatus(tr(ex.what()));
});
//! [6]
}
}
//! [7]
void Images::cancel()
{
statusBar->showMessage(tr("Canceling..."));
downloadFuture.cancel();
abortDownload();
}
//! [7]
//! [8]
QFuture<QByteArray> Images::download(const QList<QUrl> &urls)
//! [8]
{
//! [9]
QSharedPointer<QPromise<QByteArray>> promise(new QPromise<QByteArray>());
promise->start();
//! [9]
//! [10]
for (auto url : urls) {
QSharedPointer<QNetworkReply> reply(qnam.get(QNetworkRequest(url)));
replies.push_back(reply);
//! [10]
//! [11]
QtFuture::connect(reply.get(), &QNetworkReply::finished).then([=] {
if (promise->isCanceled()) {
if (!promise->future().isFinished())
promise->finish();
return;
}
if (reply->error() != QNetworkReply::NoError) {
if (!promise->future().isFinished())
throw reply->error();
}
//! [12]
promise->addResult(reply->readAll());
// Report finished on the last download
if (promise->future().resultCount() == urls.size()) {
promise->finish();
}
//! [12]
}).onFailed([=] (QNetworkReply::NetworkError error) {
promise->setException(std::make_exception_ptr(error));
promise->finish();
}).onFailed([=] {
const auto ex = std::make_exception_ptr(
std::runtime_error("Unknown error occurred while downloading."));
promise->setException(ex);
promise->finish();
});
}
//! [11]
//! [13]
return promise->future();
}
//! [13]
//! [14]
QList<QImage> Images::scaled() const
{
QList<QImage> scaled;
const auto data = downloadFuture.results();
for (auto imgData : data) {
QImage image;
image.loadFromData(imgData);
if (image.isNull())
throw std::runtime_error("Failed to load image.");
scaled.push_back(image.scaled(100, 100, Qt::KeepAspectRatio));
}
return scaled;
}
//! [14]
void Images::showImages(const QList<QImage> &images)
{
for (int i = 0; i < images.size(); ++i) {
labels[i]->setAlignment(Qt::AlignCenter);
labels[i]->setPixmap(QPixmap::fromImage(images[i]));
}
}
void Images::initLayout(qsizetype count)
{
// Clean old images
QLayoutItem *child;
while ((child = imagesLayout->takeAt(0)) != nullptr) {
child->widget()->setParent(nullptr);
delete child;
}
labels.clear();
// Init the images layout for the new images
const auto dim = int(qSqrt(qreal(count))) + 1;
for (int i = 0; i < dim; ++i) {
for (int j = 0; j < dim; ++j) {
QLabel *imageLabel = new QLabel;
imageLabel->setFixedSize(100, 100);
imagesLayout->addWidget(imageLabel, i, j);
labels.append(imageLabel);
}
}
}
void Images::updateStatus(const QString &msg)
{
statusBar->showMessage(msg);
}
void Images::abortDownload()
{
for (auto reply : replies)
reply->abort();
}