mirror of
https://bitbucket.org/smil3y/kde-extraapps.git
synced 2025-02-24 10:52:53 +00:00
314 lines
9.4 KiB
C++
314 lines
9.4 KiB
C++
// vim: set tabstop=4 shiftwidth=4 expandtab:
|
|
/*
|
|
Gwenview: an image viewer
|
|
Copyright 2012 Aurélien Gâteau <agateau@kde.org>
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Cambridge, MA 02110-1301, USA.
|
|
|
|
*/
|
|
// Self
|
|
#include <thumbnailgenerator.moc>
|
|
|
|
// Local
|
|
#include "imageutils.h"
|
|
#include "jpegcontent.h"
|
|
#include "gwenviewconfig.h"
|
|
#include "exiv2imageloader.h"
|
|
|
|
// KDE
|
|
#include <KDebug>
|
|
#include <libkdcraw/kdcraw.h>
|
|
|
|
// Qt
|
|
#include <QImageReader>
|
|
#include <QMatrix>
|
|
#include <QBuffer>
|
|
|
|
namespace Gwenview
|
|
{
|
|
|
|
#undef ENABLE_LOG
|
|
#undef LOG
|
|
//#define ENABLE_LOG
|
|
#ifdef ENABLE_LOG
|
|
#define LOG(x) kDebug() << x
|
|
#else
|
|
#define LOG(x) ;
|
|
#endif
|
|
|
|
const int MIN_PREV_SIZE = 1000;
|
|
|
|
//------------------------------------------------------------------------
|
|
//
|
|
// ThumbnailContext
|
|
//
|
|
//------------------------------------------------------------------------
|
|
bool ThumbnailContext::load(const QString &pixPath, int pixelSize)
|
|
{
|
|
mImage = QImage();
|
|
mNeedCaching = true;
|
|
Orientation orientation = NORMAL;
|
|
QImage originalImage;
|
|
QSize originalSize;
|
|
|
|
QByteArray formatHint = pixPath.section('.', -1).toAscii().toLower();
|
|
QImageReader reader(pixPath);
|
|
|
|
JpegContent content;
|
|
QByteArray format;
|
|
QByteArray data;
|
|
QBuffer buffer;
|
|
int previewRatio = 1;
|
|
|
|
// raw images deserve special treatment
|
|
if (KDcrawIface::KDcraw::rawFilesList().contains(QString(formatHint))) {
|
|
// use KDCraw to extract the preview
|
|
bool ret = KDcrawIface::KDcraw::loadEmbeddedPreview(data, pixPath);
|
|
|
|
// We need QImage. Loading JpegContent from QImage - exif lost
|
|
// Loading QImage from JpegContent - unimplemented, would go with loadFromData
|
|
if (!ret || !originalImage.loadFromData(data) || qMin(originalImage.width(), originalImage.height()) < MIN_PREV_SIZE) {
|
|
// if the emebedded preview loading failed or gets just a small image, load
|
|
// half preview instead. That's slower...
|
|
if (!KDcrawIface::KDcraw::loadHalfPreview(data, pixPath)) {
|
|
kWarning() << "unable to get preview for " << pixPath.toUtf8().constData();
|
|
return false;
|
|
}
|
|
previewRatio = 2;
|
|
}
|
|
|
|
// And we need JpegContent too because of EXIF (orientation!).
|
|
if (!content.loadFromData(data)) {
|
|
kWarning() << "unable to load preview for " << pixPath.toUtf8().constData();
|
|
return false;
|
|
}
|
|
|
|
buffer.setBuffer(&data);
|
|
buffer.open(QIODevice::ReadOnly);
|
|
reader.setDevice(&buffer);
|
|
reader.setFormat(formatHint);
|
|
} else {
|
|
if (!reader.canRead()) {
|
|
reader.setDecideFormatFromContent(true);
|
|
// Set filename again, otherwise QImageReader won't restart from scratch
|
|
reader.setFileName(pixPath);
|
|
}
|
|
|
|
if (reader.format() == "jpeg" && GwenviewConfig::applyExifOrientation()) {
|
|
content.load(pixPath);
|
|
}
|
|
}
|
|
|
|
// If there's jpeg content (from jpg or raw files), try to load an embedded thumbnail, if available.
|
|
// If applyExifOrientation is not set, don't use the
|
|
// embedded thumbnail since it might be rotated differently
|
|
// than the actual image
|
|
if (!content.rawData().isEmpty() && GwenviewConfig::applyExifOrientation()) {
|
|
QImage thumbnail = content.thumbnail();
|
|
orientation = content.orientation();
|
|
|
|
if (qMax(thumbnail.width(), thumbnail.height()) >= pixelSize) {
|
|
mImage = thumbnail;
|
|
if (orientation != NORMAL && orientation != NOT_AVAILABLE) {
|
|
QMatrix matrix = ImageUtils::transformMatrix(orientation);
|
|
mImage = mImage.transformed(matrix);
|
|
}
|
|
mOriginalWidth = content.size().width();
|
|
mOriginalHeight = content.size().height();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Generate thumbnail from full image
|
|
originalSize = reader.size();
|
|
if (originalSize.isValid() && reader.supportsOption(QImageIOHandler::ScaledSize)) {
|
|
QSizeF scaledSize = originalSize;
|
|
scaledSize.scale(pixelSize, pixelSize, Qt::KeepAspectRatio);
|
|
if (!scaledSize.isEmpty()) {
|
|
reader.setScaledSize(scaledSize.toSize());
|
|
}
|
|
}
|
|
|
|
// format() is empty after QImageReader::read() is called
|
|
format = reader.format();
|
|
if (!reader.read(&originalImage)) {
|
|
return false;
|
|
}
|
|
|
|
if (!originalSize.isValid()) {
|
|
originalSize = originalImage.size();
|
|
}
|
|
mOriginalWidth = originalSize.width() * previewRatio;
|
|
mOriginalHeight = originalSize.height() * previewRatio;
|
|
|
|
if (qMax(mOriginalWidth, mOriginalHeight) <= pixelSize) {
|
|
mImage = originalImage;
|
|
mNeedCaching = format != "png";
|
|
} else {
|
|
mImage = originalImage.scaled(pixelSize, pixelSize, Qt::KeepAspectRatio);
|
|
}
|
|
|
|
// Rotate if necessary
|
|
if (orientation != NORMAL && orientation != NOT_AVAILABLE && GwenviewConfig::applyExifOrientation()) {
|
|
QMatrix matrix = ImageUtils::transformMatrix(orientation);
|
|
mImage = mImage.transformed(matrix);
|
|
|
|
switch (orientation) {
|
|
case TRANSPOSE:
|
|
case ROT_90:
|
|
case TRANSVERSE:
|
|
case ROT_270:
|
|
qSwap(mOriginalWidth, mOriginalHeight);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
//------------------------------------------------------------------------
|
|
//
|
|
// ThumbnailGenerator
|
|
//
|
|
//------------------------------------------------------------------------
|
|
ThumbnailGenerator::ThumbnailGenerator()
|
|
: mCancel(false)
|
|
{}
|
|
|
|
void ThumbnailGenerator::load(
|
|
const QString& originalUri, time_t originalTime, KIO::filesize_t originalFileSize, const QString& originalMimeType,
|
|
const QString& pixPath,
|
|
const QString& thumbnailPath,
|
|
ThumbnailGroup::Enum group)
|
|
{
|
|
QMutexLocker lock(&mMutex);
|
|
Q_ASSERT(mPixPath.isNull());
|
|
|
|
mOriginalUri = originalUri;
|
|
mOriginalTime = originalTime;
|
|
mOriginalFileSize = originalFileSize;
|
|
mOriginalMimeType = originalMimeType;
|
|
mPixPath = pixPath;
|
|
mThumbnailPath = thumbnailPath;
|
|
mThumbnailGroup = group;
|
|
if (!isRunning()) start();
|
|
mCond.wakeOne();
|
|
}
|
|
|
|
QString ThumbnailGenerator::originalUri() const
|
|
{
|
|
return mOriginalUri;
|
|
}
|
|
|
|
time_t ThumbnailGenerator::originalTime() const
|
|
{
|
|
return mOriginalTime;
|
|
}
|
|
|
|
KIO::filesize_t ThumbnailGenerator::originalFileSize() const
|
|
{
|
|
return mOriginalFileSize;
|
|
}
|
|
|
|
QString ThumbnailGenerator::originalMimeType() const
|
|
{
|
|
return mOriginalMimeType;
|
|
}
|
|
|
|
bool ThumbnailGenerator::testCancel()
|
|
{
|
|
QMutexLocker lock(&mMutex);
|
|
return mCancel;
|
|
}
|
|
|
|
void ThumbnailGenerator::cancel()
|
|
{
|
|
QMutexLocker lock(&mMutex);
|
|
mCancel = true;
|
|
mCond.wakeOne();
|
|
}
|
|
|
|
void ThumbnailGenerator::run()
|
|
{
|
|
LOG("");
|
|
while (!testCancel()) {
|
|
QString pixPath;
|
|
int pixelSize;
|
|
{
|
|
QMutexLocker lock(&mMutex);
|
|
// empty mPixPath means nothing to do
|
|
LOG("Waiting for mPixPath");
|
|
if (mPixPath.isNull()) {
|
|
LOG("mPixPath.isNull");
|
|
mCond.wait(&mMutex);
|
|
}
|
|
}
|
|
if (testCancel()) {
|
|
return;
|
|
}
|
|
{
|
|
QMutexLocker lock(&mMutex);
|
|
pixPath = mPixPath;
|
|
pixelSize = ThumbnailGroup::pixelSize(mThumbnailGroup);
|
|
}
|
|
|
|
Q_ASSERT(!pixPath.isNull());
|
|
LOG("Loading" << pixPath);
|
|
ThumbnailContext context;
|
|
bool ok = context.load(pixPath, pixelSize);
|
|
|
|
{
|
|
QMutexLocker lock(&mMutex);
|
|
if (ok) {
|
|
mImage = context.mImage;
|
|
mOriginalWidth = context.mOriginalWidth;
|
|
mOriginalHeight = context.mOriginalHeight;
|
|
if (context.mNeedCaching) {
|
|
cacheThumbnail();
|
|
}
|
|
} else {
|
|
kWarning() << "Could not generate thumbnail for file" << mOriginalUri;
|
|
}
|
|
mPixPath.clear(); // done, ready for next
|
|
}
|
|
if (testCancel()) {
|
|
return;
|
|
}
|
|
{
|
|
QSize size(mOriginalWidth, mOriginalHeight);
|
|
LOG("emitting done signal, size=" << size);
|
|
QMutexLocker lock(&mMutex);
|
|
done(mImage, size);
|
|
LOG("Done");
|
|
}
|
|
}
|
|
LOG("Ending thread");
|
|
}
|
|
|
|
void ThumbnailGenerator::cacheThumbnail()
|
|
{
|
|
mImage.setText("Thumb::URI" , 0, mOriginalUri);
|
|
mImage.setText("Thumb::MTime" , 0, QString::number(mOriginalTime));
|
|
mImage.setText("Thumb::Size" , 0, QString::number(mOriginalFileSize));
|
|
mImage.setText("Thumb::Mimetype" , 0, mOriginalMimeType);
|
|
mImage.setText("Thumb::Image::Width" , 0, QString::number(mOriginalWidth));
|
|
mImage.setText("Thumb::Image::Height", 0, QString::number(mOriginalHeight));
|
|
mImage.setText("Software" , 0, "Gwenview");
|
|
|
|
emit thumbnailReadyToBeCached(mThumbnailPath, mImage);
|
|
}
|
|
|
|
} // namespace
|