forked from qt-creator/qt-creator
Since we also license under GPL-3.0 WITH Qt-GPL-exception-1.0,
this applies only to a hypothetical newer version of GPL, that doesn't
exist yet. If such a version emerges, we can still decide to relicense...
While at it, replace (deprecated) GPL-3.0 with more explicit GPL-3.0-only
Change was done by running
find . -type f -exec perl -pi -e "s/LicenseRef-Qt-Commercial OR GPL-3.0\+ OR GPL-3.0 WITH Qt-GPL-exception-1.0/LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0/g" {} \;
Change-Id: I5097e6ce8d10233993ee30d7e25120e2659eb10b
Reviewed-by: Eike Ziller <eike.ziller@qt.io>
250 lines
7.0 KiB
C++
250 lines
7.0 KiB
C++
// Copyright (C) 2016 Denis Mingulov.
|
|
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
#include "imageviewerfile.h"
|
|
|
|
#include "imageviewerconstants.h"
|
|
#include "imageviewertr.h"
|
|
|
|
#include <coreplugin/editormanager/documentmodel.h>
|
|
#include <coreplugin/editormanager/ieditor.h>
|
|
|
|
#include <utils/filepath.h>
|
|
#include <utils/mimeutils.h>
|
|
#include <utils/qtcassert.h>
|
|
|
|
#include <QFileInfo>
|
|
#include <QGraphicsPixmapItem>
|
|
#include <QImageReader>
|
|
#include <QMovie>
|
|
#include <QPainter>
|
|
#include <QPixmap>
|
|
|
|
#ifndef QT_NO_SVG
|
|
#include <QGraphicsSvgItem>
|
|
#endif
|
|
|
|
namespace ImageViewer::Internal {
|
|
|
|
class MovieItem : public QObject, public QGraphicsPixmapItem
|
|
{
|
|
public:
|
|
MovieItem(QMovie *movie)
|
|
: m_movie(movie)
|
|
{
|
|
setPixmap(m_movie->currentPixmap());
|
|
connect(m_movie, &QMovie::updated, this, [this](const QRectF &rect) {
|
|
update(rect);
|
|
});
|
|
}
|
|
|
|
void paint(QPainter *painter, const QStyleOptionGraphicsItem *, QWidget *) override
|
|
{
|
|
const bool smoothTransform = painter->worldTransform().m11() < 1;
|
|
painter->setRenderHint(QPainter::SmoothPixmapTransform, smoothTransform);
|
|
painter->drawPixmap(offset(), m_movie->currentPixmap());
|
|
}
|
|
|
|
private:
|
|
QMovie *m_movie;
|
|
};
|
|
|
|
ImageViewerFile::ImageViewerFile()
|
|
{
|
|
setId(Constants::IMAGEVIEWER_ID);
|
|
connect(this, &ImageViewerFile::mimeTypeChanged, this, &ImageViewerFile::changed);
|
|
}
|
|
|
|
ImageViewerFile::~ImageViewerFile()
|
|
{
|
|
cleanUp();
|
|
}
|
|
|
|
Core::IDocument::OpenResult ImageViewerFile::open(QString *errorString,
|
|
const Utils::FilePath &filePath,
|
|
const Utils::FilePath &realfilePath)
|
|
{
|
|
QTC_CHECK(filePath == realfilePath); // does not support auto save
|
|
OpenResult success = openImpl(errorString, filePath);
|
|
emit openFinished(success == OpenResult::Success);
|
|
return success;
|
|
}
|
|
|
|
Core::IDocument::OpenResult ImageViewerFile::openImpl(QString *errorString,
|
|
const Utils::FilePath &filePath)
|
|
{
|
|
cleanUp();
|
|
|
|
if (!filePath.isReadableFile())
|
|
return OpenResult::ReadError;
|
|
|
|
const QString &fileName = filePath.toString();
|
|
QByteArray format = QImageReader::imageFormat(fileName);
|
|
// if it is impossible to recognize a file format - file will not be open correctly
|
|
if (format.isEmpty()) {
|
|
if (errorString)
|
|
*errorString = Tr::tr("Image format not supported.");
|
|
return OpenResult::CannotHandle;
|
|
}
|
|
|
|
#ifndef QT_NO_SVG
|
|
if (format.startsWith("svg")) {
|
|
m_tempSvgItem = new QGraphicsSvgItem(fileName);
|
|
QRectF bound = m_tempSvgItem->boundingRect();
|
|
if (!bound.isValid() || (qFuzzyIsNull(bound.width()) && qFuzzyIsNull(bound.height()))) {
|
|
delete m_tempSvgItem;
|
|
m_tempSvgItem = nullptr;
|
|
if (errorString)
|
|
*errorString = Tr::tr("Failed to read SVG image.");
|
|
return OpenResult::CannotHandle;
|
|
}
|
|
m_type = TypeSvg;
|
|
emit imageSizeChanged(m_tempSvgItem->boundingRect().size().toSize());
|
|
} else
|
|
#endif
|
|
if (QMovie::supportedFormats().contains(format)) {
|
|
m_movie = new QMovie(fileName, QByteArray(), this);
|
|
// force reading movie/image data, so we can catch completely invalid movies/images early:
|
|
m_movie->jumpToNextFrame();
|
|
if (!m_movie->isValid()) {
|
|
if (errorString)
|
|
*errorString = Tr::tr("Failed to read image.");
|
|
delete m_movie;
|
|
m_movie = nullptr;
|
|
return OpenResult::CannotHandle;
|
|
}
|
|
m_type = TypeMovie;
|
|
m_movie->setCacheMode(QMovie::CacheAll);
|
|
connect(
|
|
m_movie,
|
|
&QMovie::finished,
|
|
m_movie,
|
|
[this] {
|
|
if (m_movie->isValid())
|
|
m_movie->start();
|
|
},
|
|
Qt::QueuedConnection);
|
|
connect(m_movie, &QMovie::resized, this, &ImageViewerFile::imageSizeChanged);
|
|
m_movie->start();
|
|
m_isPaused = false; // force update
|
|
setPaused(true);
|
|
} else {
|
|
m_pixmap = new QPixmap(fileName);
|
|
if (m_pixmap->isNull()) {
|
|
if (errorString)
|
|
*errorString = Tr::tr("Failed to read image.");
|
|
delete m_pixmap;
|
|
m_pixmap = nullptr;
|
|
return OpenResult::CannotHandle;
|
|
}
|
|
m_type = TypePixmap;
|
|
emit imageSizeChanged(m_pixmap->size());
|
|
}
|
|
|
|
setFilePath(filePath);
|
|
setMimeType(Utils::mimeTypeForFile(filePath).name());
|
|
return OpenResult::Success;
|
|
}
|
|
|
|
Core::IDocument::ReloadBehavior ImageViewerFile::reloadBehavior(ChangeTrigger state, ChangeType type) const
|
|
{
|
|
if (type == TypeRemoved)
|
|
return BehaviorSilent;
|
|
if (type == TypeContents && state == TriggerInternal && !isModified())
|
|
return BehaviorSilent;
|
|
return BehaviorAsk;
|
|
}
|
|
|
|
bool ImageViewerFile::reload(QString *errorString,
|
|
Core::IDocument::ReloadFlag flag,
|
|
Core::IDocument::ChangeType type)
|
|
{
|
|
Q_UNUSED(type)
|
|
if (flag == FlagIgnore)
|
|
return true;
|
|
emit aboutToReload();
|
|
bool success = (openImpl(errorString, filePath()) == OpenResult::Success);
|
|
emit reloadFinished(success);
|
|
return success;
|
|
}
|
|
|
|
bool ImageViewerFile::isPaused() const
|
|
{
|
|
return m_isPaused;
|
|
}
|
|
|
|
void ImageViewerFile::setPaused(bool paused)
|
|
{
|
|
if (!m_movie || m_isPaused == paused)
|
|
return;
|
|
m_isPaused = paused;
|
|
m_movie->setPaused(paused);
|
|
emit isPausedChanged(m_isPaused);
|
|
}
|
|
|
|
QGraphicsItem *ImageViewerFile::createGraphicsItem() const
|
|
{
|
|
QGraphicsItem *val = nullptr;
|
|
switch (m_type) {
|
|
case TypeInvalid:
|
|
break;
|
|
case TypeSvg:
|
|
#ifndef QT_NO_SVG
|
|
if (m_tempSvgItem) {
|
|
val = m_tempSvgItem;
|
|
m_tempSvgItem = nullptr;
|
|
} else {
|
|
val = new QGraphicsSvgItem(filePath().toString());
|
|
}
|
|
#endif
|
|
break;
|
|
case TypeMovie:
|
|
val = new MovieItem(m_movie);
|
|
break;
|
|
case TypePixmap: {
|
|
auto pixmapItem = new QGraphicsPixmapItem(*m_pixmap);
|
|
pixmapItem->setTransformationMode(Qt::SmoothTransformation);
|
|
val = pixmapItem;
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
return val;
|
|
}
|
|
|
|
ImageViewerFile::ImageType ImageViewerFile::type() const
|
|
{
|
|
return m_type;
|
|
}
|
|
|
|
void ImageViewerFile::updateVisibility()
|
|
{
|
|
if (!m_movie || m_isPaused)
|
|
return;
|
|
bool visible = false;
|
|
for (Core::IEditor *editor : Core::DocumentModel::editorsForDocument(this)) {
|
|
if (editor->widget()->isVisible()) {
|
|
visible = true;
|
|
break;
|
|
}
|
|
}
|
|
m_movie->setPaused(!visible);
|
|
}
|
|
|
|
void ImageViewerFile::cleanUp()
|
|
{
|
|
delete m_pixmap;
|
|
m_pixmap = nullptr;
|
|
delete m_movie;
|
|
m_movie = nullptr;
|
|
#ifndef QT_NO_SVG
|
|
delete m_tempSvgItem;
|
|
m_tempSvgItem = nullptr;
|
|
#endif
|
|
m_type = TypeInvalid;
|
|
}
|
|
|
|
} // ImageViewer::Internal
|