2017-04-27 18:24:53 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2015-11-06 18:39:59 +00:00
|
|
|
#include "dive.h"
|
|
|
|
#include "metrics.h"
|
|
|
|
#include "divelist.h"
|
|
|
|
#include "qthelper.h"
|
|
|
|
#include "imagedownloader.h"
|
2018-03-10 13:15:50 +00:00
|
|
|
#include "qt-models/divepicturemodel.h"
|
|
|
|
#include "metadata.h"
|
2015-11-06 18:39:59 +00:00
|
|
|
#include <unistd.h>
|
2016-04-30 22:00:29 +00:00
|
|
|
#include <QString>
|
2018-03-30 05:57:30 +00:00
|
|
|
#include <QImageReader>
|
2018-03-10 13:15:50 +00:00
|
|
|
#include <QDataStream>
|
2018-05-09 21:08:44 +00:00
|
|
|
#include <QSvgRenderer>
|
|
|
|
#include <QPainter>
|
2015-11-06 18:39:59 +00:00
|
|
|
|
|
|
|
#include <QtConcurrent>
|
|
|
|
|
2018-03-11 11:58:55 +00:00
|
|
|
// Note: this is a global instead of a function-local variable on purpose.
|
|
|
|
// We don't want this to be generated in a different thread context if
|
|
|
|
// ImageDownloader::instance() is called from a worker thread.
|
|
|
|
static ImageDownloader imageDownloader;
|
|
|
|
ImageDownloader *ImageDownloader::instance()
|
2015-11-06 18:39:59 +00:00
|
|
|
{
|
2018-03-11 11:58:55 +00:00
|
|
|
return &imageDownloader;
|
2016-03-15 20:31:59 +00:00
|
|
|
}
|
|
|
|
|
2018-03-11 11:58:55 +00:00
|
|
|
ImageDownloader::ImageDownloader()
|
2018-02-08 21:45:55 +00:00
|
|
|
{
|
2018-03-11 11:58:55 +00:00
|
|
|
connect(&manager, &QNetworkAccessManager::finished, this, &ImageDownloader::saveImage);
|
2018-02-08 21:45:55 +00:00
|
|
|
}
|
|
|
|
|
2018-06-12 10:50:34 +00:00
|
|
|
void ImageDownloader::load(QUrl url, QString filename)
|
2018-02-08 21:45:55 +00:00
|
|
|
{
|
2018-03-11 11:58:55 +00:00
|
|
|
QNetworkRequest request(url);
|
|
|
|
request.setAttribute(QNetworkRequest::User, filename);
|
|
|
|
manager.get(request);
|
2015-11-06 18:39:59 +00:00
|
|
|
}
|
|
|
|
|
2018-03-11 11:58:55 +00:00
|
|
|
void ImageDownloader::saveImage(QNetworkReply *reply)
|
2015-11-06 18:39:59 +00:00
|
|
|
{
|
2018-03-11 11:58:55 +00:00
|
|
|
QString filename = reply->request().attribute(QNetworkRequest::User).toString();
|
|
|
|
|
|
|
|
if (reply->error() != QNetworkReply::NoError) {
|
2018-05-23 16:24:08 +00:00
|
|
|
emit failed(filename);
|
2018-03-11 11:58:55 +00:00
|
|
|
} else {
|
|
|
|
QByteArray imageData = reply->readAll();
|
|
|
|
QCryptographicHash hash(QCryptographicHash::Sha1);
|
|
|
|
hash.addData(imageData);
|
|
|
|
QString path = QStandardPaths::standardLocations(QStandardPaths::CacheLocation).first();
|
|
|
|
QDir dir(path);
|
|
|
|
if (!dir.exists())
|
|
|
|
dir.mkpath(path);
|
|
|
|
QFile imageFile(path.append("/").append(hash.result().toHex()));
|
|
|
|
if (imageFile.open(QIODevice::WriteOnly)) {
|
|
|
|
qDebug() << "Write image to" << imageFile.fileName();
|
|
|
|
QDataStream stream(&imageFile);
|
|
|
|
stream.writeRawData(imageData.data(), imageData.length());
|
|
|
|
imageFile.waitForBytesWritten(-1);
|
|
|
|
imageFile.close();
|
|
|
|
learnHash(filename, imageFile.fileName(), hash.result());
|
|
|
|
}
|
|
|
|
emit loaded(filename);
|
2015-11-06 18:39:59 +00:00
|
|
|
}
|
2016-03-15 21:45:17 +00:00
|
|
|
|
2018-03-11 11:58:55 +00:00
|
|
|
reply->deleteLater();
|
2015-11-06 18:39:59 +00:00
|
|
|
}
|
|
|
|
|
2018-06-12 10:50:34 +00:00
|
|
|
static void loadPicture(QUrl url, QString filename)
|
2015-11-06 18:39:59 +00:00
|
|
|
{
|
2018-03-11 11:58:55 +00:00
|
|
|
// This has to be done in UI main thread, because QNetworkManager refuses
|
|
|
|
// to treat requests from other threads.
|
2018-06-12 10:50:34 +00:00
|
|
|
QMetaObject::invokeMethod(ImageDownloader::instance(), "load", Qt::AutoConnection, Q_ARG(QUrl, url), Q_ARG(QString, filename));
|
2018-03-30 05:57:30 +00:00
|
|
|
}
|
|
|
|
|
2018-03-11 11:58:55 +00:00
|
|
|
// Returns: thumbnail, still loading
|
2018-05-15 22:17:04 +00:00
|
|
|
static std::pair<QImage,bool> getHashedImage(const QString &file_in, bool tryDownload)
|
2015-11-06 18:39:59 +00:00
|
|
|
{
|
2018-05-15 22:17:04 +00:00
|
|
|
QString file = file_in.startsWith("file://", Qt::CaseInsensitive) ? file_in.mid(7) : file_in;
|
2018-03-11 11:58:55 +00:00
|
|
|
QImage thumb;
|
|
|
|
bool stillLoading = false;
|
2018-03-07 15:37:31 +00:00
|
|
|
QUrl url = QUrl::fromUserInput(localFilePath(file));
|
|
|
|
if (url.isLocalFile())
|
2018-06-12 10:50:34 +00:00
|
|
|
thumb.load(url.toLocalFile());
|
2018-05-15 22:17:04 +00:00
|
|
|
if (!thumb.isNull()) {
|
|
|
|
// We loaded successfully. Now, make sure hash is up to date.
|
|
|
|
hashPicture(file);
|
|
|
|
} else if (tryDownload) {
|
2016-01-09 15:29:49 +00:00
|
|
|
// This did not load anything. Let's try to get the image from other sources
|
2018-06-12 10:50:34 +00:00
|
|
|
QString filenameLocal = localFilePath(file);
|
2018-05-23 16:24:08 +00:00
|
|
|
// Load locally from translated file name if it is different
|
|
|
|
if (filenameLocal != file)
|
2018-06-12 10:50:34 +00:00
|
|
|
thumb.load(filenameLocal);
|
2018-05-23 16:24:08 +00:00
|
|
|
if (!thumb.isNull()) {
|
|
|
|
// Make sure the hash still matches the image file
|
|
|
|
hashPicture(filenameLocal);
|
2015-11-06 18:39:59 +00:00
|
|
|
} else {
|
2018-05-23 16:24:08 +00:00
|
|
|
// Interpret filename as URL
|
2018-06-12 10:50:34 +00:00
|
|
|
QUrl url = QUrl::fromUserInput(filenameLocal);
|
|
|
|
if (!url.isLocalFile() && url.isValid()) {
|
|
|
|
loadPicture(url, file);
|
|
|
|
stillLoading = true;
|
|
|
|
}
|
2015-11-06 18:39:59 +00:00
|
|
|
}
|
|
|
|
}
|
2018-06-12 10:50:34 +00:00
|
|
|
if (thumb.isNull() && !stillLoading)
|
|
|
|
qInfo() << "Error loading image" << file << "[local:" << localFilePath(file) << "]";
|
2018-03-11 11:58:55 +00:00
|
|
|
return { thumb, stillLoading };
|
2015-11-06 18:39:59 +00:00
|
|
|
}
|
2018-03-10 13:15:50 +00:00
|
|
|
|
2018-05-09 21:08:44 +00:00
|
|
|
static QImage renderIcon(const char *id, int size)
|
|
|
|
{
|
|
|
|
QImage res(size, size, QImage::Format_ARGB32);
|
|
|
|
QSvgRenderer svg{QString(id)};
|
|
|
|
QPainter painter(&res);
|
|
|
|
svg.render(&painter);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
Thumbnailer::Thumbnailer() : failImage(renderIcon(":filter-close", maxThumbnailSize())), // TODO: Don't misuse filter close icon
|
|
|
|
dummyImage(renderIcon(":camera-icon", maxThumbnailSize()))
|
2018-03-10 13:15:50 +00:00
|
|
|
{
|
|
|
|
// Currently, we only process one image at a time. Stefan Fuchs reported problems when
|
|
|
|
// calculating multiple thumbnails at once and this hopefully helps.
|
|
|
|
pool.setMaxThreadCount(1);
|
2018-03-11 11:58:55 +00:00
|
|
|
connect(ImageDownloader::instance(), &ImageDownloader::loaded, this, &Thumbnailer::imageDownloaded);
|
|
|
|
connect(ImageDownloader::instance(), &ImageDownloader::failed, this, &Thumbnailer::imageDownloadFailed);
|
2018-03-10 13:15:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Thumbnailer *Thumbnailer::instance()
|
|
|
|
{
|
|
|
|
static Thumbnailer self;
|
|
|
|
return &self;
|
|
|
|
}
|
|
|
|
|
|
|
|
static QImage getThumbnailFromCache(const QString &picture_filename)
|
|
|
|
{
|
|
|
|
QString filename = thumbnailFileName(picture_filename);
|
|
|
|
if (filename.isEmpty())
|
|
|
|
return QImage();
|
|
|
|
QFile file(filename);
|
2018-05-27 13:42:22 +00:00
|
|
|
|
|
|
|
if (prefs.auto_recalculate_thumbnails) {
|
|
|
|
// Check if thumbnails is older than the (local) image file
|
|
|
|
QString filenameLocal = localFilePath(qPrintable(picture_filename));
|
|
|
|
QFileInfo pictureInfo(filenameLocal);
|
|
|
|
QFileInfo thumbnailInfo(file);
|
|
|
|
if (pictureInfo.exists() && thumbnailInfo.exists()) {
|
|
|
|
QDateTime pictureTime = pictureInfo.lastModified();
|
|
|
|
QDateTime thumbnailTime = thumbnailInfo.lastModified();
|
|
|
|
if (pictureTime.isValid() && thumbnailTime.isValid() && thumbnailTime < pictureTime) {
|
|
|
|
// Both files exist, have valid timestamps and thumbnail was calculated before picture.
|
|
|
|
// Return an empty thumbnail to signal recalculation of the thumbnail
|
|
|
|
return QImage();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-10 13:15:50 +00:00
|
|
|
if (!file.open(QIODevice::ReadOnly))
|
|
|
|
return QImage();
|
|
|
|
QDataStream stream(&file);
|
|
|
|
|
|
|
|
// Each thumbnail file is composed of a media-type and an image file.
|
|
|
|
// Currently, the type is ignored. This will be used to mark videos.
|
|
|
|
quint32 type;
|
|
|
|
QImage res;
|
|
|
|
stream >> type;
|
|
|
|
stream >> res;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void addThumbnailToCache(const QImage &thumbnail, const QString &picture_filename)
|
|
|
|
{
|
|
|
|
if (thumbnail.isNull())
|
|
|
|
return;
|
|
|
|
|
|
|
|
QString filename = thumbnailFileName(picture_filename);
|
|
|
|
QSaveFile file(filename);
|
|
|
|
if (!file.open(QIODevice::WriteOnly))
|
|
|
|
return;
|
|
|
|
QDataStream stream(&file);
|
|
|
|
|
|
|
|
// For format of the file, see comments in getThumnailForCache
|
|
|
|
quint32 type = MEDIATYPE_PICTURE;
|
|
|
|
stream << type;
|
|
|
|
stream << thumbnail;
|
|
|
|
file.commit();
|
|
|
|
}
|
|
|
|
|
2018-05-27 12:03:29 +00:00
|
|
|
void Thumbnailer::recalculate(QString filename)
|
|
|
|
{
|
|
|
|
auto res = getHashedImage(filename, true);
|
|
|
|
|
|
|
|
// If we couldn't load the image from disk -> leave old thumbnail.
|
|
|
|
// The case "load from web" is a bit inconsistent: it will call into processItem() later
|
|
|
|
// and therefore a "broken" image symbol may be shown.
|
|
|
|
if (res.second || res.first.isNull())
|
|
|
|
return;
|
|
|
|
QImage thumbnail = res.first;
|
|
|
|
addThumbnailToCache(thumbnail, filename);
|
|
|
|
|
|
|
|
QMutexLocker l(&lock);
|
|
|
|
emit thumbnailChanged(filename, thumbnail);
|
|
|
|
workingOn.remove(filename);
|
|
|
|
}
|
|
|
|
|
2018-05-15 22:17:04 +00:00
|
|
|
void Thumbnailer::processItem(QString filename, bool tryDownload)
|
2018-03-10 13:15:50 +00:00
|
|
|
{
|
|
|
|
QImage thumbnail = getThumbnailFromCache(filename);
|
|
|
|
|
|
|
|
if (thumbnail.isNull()) {
|
2018-05-15 22:17:04 +00:00
|
|
|
auto res = getHashedImage(filename, tryDownload);
|
2018-03-11 11:58:55 +00:00
|
|
|
if (res.second)
|
|
|
|
return;
|
|
|
|
thumbnail = res.first;
|
|
|
|
|
2018-03-10 13:15:50 +00:00
|
|
|
if (thumbnail.isNull()) {
|
2018-03-11 09:19:08 +00:00
|
|
|
thumbnail = failImage;
|
2018-03-10 13:15:50 +00:00
|
|
|
} else {
|
2018-03-11 09:19:08 +00:00
|
|
|
int size = maxThumbnailSize();
|
2018-03-10 13:15:50 +00:00
|
|
|
thumbnail = thumbnail.scaled(size, size, Qt::KeepAspectRatio);
|
|
|
|
addThumbnailToCache(thumbnail, filename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QMutexLocker l(&lock);
|
|
|
|
emit thumbnailChanged(filename, thumbnail);
|
|
|
|
workingOn.remove(filename);
|
|
|
|
}
|
|
|
|
|
2018-03-11 11:58:55 +00:00
|
|
|
void Thumbnailer::imageDownloaded(QString filename)
|
|
|
|
{
|
|
|
|
// Image was downloaded and the filename connected with a hash.
|
|
|
|
// Try thumbnailing again.
|
|
|
|
QMutexLocker l(&lock);
|
2018-05-15 22:17:04 +00:00
|
|
|
workingOn[filename] = QtConcurrent::run(&pool, [this, filename]() { processItem(filename, false); });
|
2018-03-11 11:58:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Thumbnailer::imageDownloadFailed(QString filename)
|
|
|
|
{
|
|
|
|
emit thumbnailChanged(filename, failImage);
|
|
|
|
QMutexLocker l(&lock);
|
|
|
|
workingOn.remove(filename);
|
|
|
|
}
|
|
|
|
|
2018-03-11 09:19:08 +00:00
|
|
|
QImage Thumbnailer::fetchThumbnail(PictureEntry &entry)
|
2018-03-10 13:15:50 +00:00
|
|
|
{
|
|
|
|
QMutexLocker l(&lock);
|
|
|
|
|
|
|
|
// We are not currently fetching this thumbnail - add it to the list.
|
|
|
|
const QString &filename = entry.filename;
|
|
|
|
if (!workingOn.contains(filename)) {
|
|
|
|
workingOn.insert(filename,
|
2018-05-15 22:17:04 +00:00
|
|
|
QtConcurrent::run(&pool, [this, filename]() { processItem(filename, true); }));
|
2018-03-10 13:15:50 +00:00
|
|
|
}
|
2018-03-11 09:19:08 +00:00
|
|
|
return dummyImage;
|
2018-03-10 13:15:50 +00:00
|
|
|
}
|
|
|
|
|
2018-05-27 12:03:29 +00:00
|
|
|
void Thumbnailer::calculateThumbnails(const QVector<QString> &filenames)
|
|
|
|
{
|
|
|
|
QMutexLocker l(&lock);
|
|
|
|
for (const QString &filename: filenames) {
|
|
|
|
if (!workingOn.contains(filename)) {
|
|
|
|
workingOn.insert(filename,
|
|
|
|
QtConcurrent::run(&pool, [this, filename]() { recalculate(filename); }));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-10 13:15:50 +00:00
|
|
|
void Thumbnailer::clearWorkQueue()
|
|
|
|
{
|
|
|
|
QMutexLocker l(&lock);
|
|
|
|
for (auto it = workingOn.begin(); it != workingOn.end(); ++it)
|
|
|
|
it->cancel();
|
|
|
|
workingOn.clear();
|
|
|
|
}
|
2018-03-11 09:19:08 +00:00
|
|
|
|
|
|
|
static const int maxZoom = 3; // Maximum zoom: thrice of standard size
|
|
|
|
|
|
|
|
int Thumbnailer::defaultThumbnailSize()
|
|
|
|
{
|
|
|
|
return defaultIconMetrics().sz_pic;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Thumbnailer::maxThumbnailSize()
|
|
|
|
{
|
|
|
|
return defaultThumbnailSize() * maxZoom;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Thumbnailer::thumbnailSize(double zoomLevel)
|
|
|
|
{
|
|
|
|
// Calculate size of thumbnails. The standard size is defaultIconMetrics().sz_pic.
|
|
|
|
// We use exponential scaling so that the central point is the standard
|
|
|
|
// size and the minimum and maximum extreme points are a third respectively
|
|
|
|
// three times the standard size.
|
|
|
|
// Naturally, these three zoom levels are then represented by
|
|
|
|
// -1.0 (minimum), 0 (standard) and 1.0 (maximum). The actual size is
|
|
|
|
// calculated as standard_size*3.0^zoomLevel.
|
|
|
|
return static_cast<int>(round(defaultThumbnailSize() * pow(maxZoom, zoomLevel)));
|
|
|
|
}
|