mirror of
https://bitbucket.org/smil3y/kde-extraapps.git
synced 2025-02-24 02:42:52 +00:00
484 lines
14 KiB
C++
484 lines
14 KiB
C++
// vim: set tabstop=4 shiftwidth=4 expandtab:
|
|
/*
|
|
Gwenview: an image viewer
|
|
Copyright 2011 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 "moc_rasterimageview.cpp"
|
|
|
|
// Local
|
|
#include <lib/documentview/abstractrasterimageviewtool.h>
|
|
#include <lib/imagescaler.h>
|
|
#include <lib/gvdebug.h>
|
|
|
|
// KDE
|
|
#include <KDebug>
|
|
|
|
// Qt
|
|
#include <QtGui/qgraphicssceneevent.h>
|
|
#include <QPainter>
|
|
#include <QTimer>
|
|
#include <QtCore/qsharedpointer.h>
|
|
|
|
|
|
namespace Gwenview
|
|
{
|
|
|
|
struct RasterImageViewPrivate
|
|
{
|
|
RasterImageView* q;
|
|
ImageScaler* mScaler;
|
|
QPixmap mBackgroundTexture;
|
|
bool mEmittedCompleted;
|
|
|
|
// Config
|
|
RasterImageView::AlphaBackgroundMode mAlphaBackgroundMode;
|
|
QColor mAlphaBackgroundColor;
|
|
bool mEnlargeSmallerImages;
|
|
// /Config
|
|
|
|
bool mBufferIsEmpty;
|
|
QPixmap mCurrentBuffer;
|
|
// The alternate buffer is useful when scrolling: existing content is copied
|
|
// to mAlternateBuffer and buffers are swapped. This avoids allocating a new
|
|
// QPixmap every time the image is scrolled.
|
|
QPixmap mAlternateBuffer;
|
|
|
|
QTimer* mUpdateTimer;
|
|
|
|
QWeakPointer<AbstractRasterImageViewTool> mTool;
|
|
|
|
void createBackgroundTexture()
|
|
{
|
|
mBackgroundTexture = QPixmap(32, 32);
|
|
QPainter painter(&mBackgroundTexture);
|
|
painter.fillRect(mBackgroundTexture.rect(), QColor(128, 128, 128));
|
|
QColor light = QColor(192, 192, 192);
|
|
painter.fillRect(0, 0, 16, 16, light);
|
|
painter.fillRect(16, 16, 16, 16, light);
|
|
}
|
|
|
|
void setupUpdateTimer()
|
|
{
|
|
mUpdateTimer = new QTimer(q);
|
|
mUpdateTimer->setInterval(500);
|
|
mUpdateTimer->setSingleShot(true);
|
|
QObject::connect(mUpdateTimer, SIGNAL(timeout()),
|
|
q, SLOT(updateBuffer()));
|
|
}
|
|
|
|
void startAnimationIfNecessary()
|
|
{
|
|
if (q->document() && q->isVisible()) {
|
|
q->document()->startAnimation();
|
|
}
|
|
}
|
|
|
|
QRectF mapViewportToZoomedImage(const QRectF& viewportRect) const
|
|
{
|
|
return QRectF(
|
|
viewportRect.topLeft() - q->imageOffset() + q->scrollPos(),
|
|
viewportRect.size()
|
|
);
|
|
}
|
|
|
|
void setScalerRegionToVisibleRect()
|
|
{
|
|
QRectF rect = mapViewportToZoomedImage(q->boundingRect());
|
|
mScaler->setDestinationRegion(QRegion(rect.toRect()));
|
|
}
|
|
|
|
void resizeBuffer()
|
|
{
|
|
QSize size = q->visibleImageSize().toSize();
|
|
if (size == mCurrentBuffer.size()) {
|
|
return;
|
|
}
|
|
if (!size.isValid()) {
|
|
mAlternateBuffer = QPixmap();
|
|
mCurrentBuffer = QPixmap();
|
|
return;
|
|
}
|
|
|
|
mAlternateBuffer = QPixmap(size);
|
|
mAlternateBuffer.fill(Qt::transparent);
|
|
{
|
|
QPainter painter(&mAlternateBuffer);
|
|
painter.drawPixmap(0, 0, mCurrentBuffer);
|
|
}
|
|
qSwap(mAlternateBuffer, mCurrentBuffer);
|
|
|
|
mAlternateBuffer = QPixmap();
|
|
}
|
|
|
|
void drawAlphaBackground(QPainter* painter, const QRect& viewportRect, const QPoint& zoomedImageTopLeft)
|
|
{
|
|
if (mAlphaBackgroundMode == RasterImageView::AlphaBackgroundCheckBoard) {
|
|
QPoint textureOffset(
|
|
zoomedImageTopLeft.x() % mBackgroundTexture.width(),
|
|
zoomedImageTopLeft.y() % mBackgroundTexture.height()
|
|
);
|
|
painter->drawTiledPixmap(
|
|
viewportRect,
|
|
mBackgroundTexture,
|
|
textureOffset);
|
|
} else {
|
|
painter->fillRect(viewportRect, mAlphaBackgroundColor);
|
|
}
|
|
}
|
|
};
|
|
|
|
RasterImageView::RasterImageView(QGraphicsItem* parent)
|
|
: AbstractImageView(parent)
|
|
, d(new RasterImageViewPrivate)
|
|
{
|
|
d->q = this;
|
|
d->mEmittedCompleted = false;
|
|
|
|
d->mAlphaBackgroundMode = AlphaBackgroundCheckBoard;
|
|
d->mAlphaBackgroundColor = Qt::black;
|
|
d->mEnlargeSmallerImages = false;
|
|
|
|
d->mBufferIsEmpty = true;
|
|
d->mScaler = new ImageScaler(this);
|
|
connect(d->mScaler, SIGNAL(scaledRect(int,int,QImage)),
|
|
SLOT(updateFromScaler(int,int,QImage)));
|
|
|
|
d->createBackgroundTexture();
|
|
d->setupUpdateTimer();
|
|
}
|
|
|
|
RasterImageView::~RasterImageView()
|
|
{
|
|
delete d;
|
|
}
|
|
|
|
void RasterImageView::setAlphaBackgroundMode(AlphaBackgroundMode mode)
|
|
{
|
|
d->mAlphaBackgroundMode = mode;
|
|
if (document() && document()->hasAlphaChannel()) {
|
|
d->mCurrentBuffer = QPixmap();
|
|
updateBuffer();
|
|
}
|
|
}
|
|
|
|
void RasterImageView::setAlphaBackgroundColor(const QColor& color)
|
|
{
|
|
d->mAlphaBackgroundColor = color;
|
|
if (document() && document()->hasAlphaChannel()) {
|
|
d->mCurrentBuffer = QPixmap();
|
|
updateBuffer();
|
|
}
|
|
}
|
|
|
|
void RasterImageView::loadFromDocument()
|
|
{
|
|
Document::Ptr doc = document();
|
|
if (!doc) {
|
|
return;
|
|
}
|
|
|
|
connect(doc.data(), SIGNAL(metaInfoLoaded(KUrl)),
|
|
SLOT(slotDocumentMetaInfoLoaded()));
|
|
connect(doc.data(), SIGNAL(isAnimatedUpdated()),
|
|
SLOT(slotDocumentIsAnimatedUpdated()));
|
|
|
|
const Document::LoadingState state = doc->loadingState();
|
|
if (state == Document::MetaInfoLoaded || state == Document::Loaded) {
|
|
slotDocumentMetaInfoLoaded();
|
|
}
|
|
}
|
|
|
|
void RasterImageView::slotDocumentMetaInfoLoaded()
|
|
{
|
|
if (document()->size().isValid()) {
|
|
QMetaObject::invokeMethod(this, "finishSetDocument", Qt::QueuedConnection);
|
|
} else {
|
|
// Could not retrieve image size from meta info, we need to load the
|
|
// full image now.
|
|
connect(document().data(), SIGNAL(loaded(KUrl)),
|
|
SLOT(finishSetDocument()));
|
|
document()->startLoadingFullImage();
|
|
}
|
|
}
|
|
|
|
void RasterImageView::finishSetDocument()
|
|
{
|
|
GV_RETURN_IF_FAIL(document()->size().isValid());
|
|
|
|
d->mScaler->setDocument(document());
|
|
d->resizeBuffer();
|
|
applyPendingScrollPos();
|
|
|
|
connect(document().data(), SIGNAL(imageRectUpdated(QRect)),
|
|
SLOT(updateImageRect(QRect)));
|
|
|
|
if (zoomToFit()) {
|
|
// Force the update otherwise if computeZoomToFit() returns 1, setZoom()
|
|
// will think zoom has not changed and won't update the image
|
|
setZoom(computeZoomToFit(), QPointF(-1, -1), ForceUpdate);
|
|
} else {
|
|
updateBuffer();
|
|
}
|
|
|
|
d->startAnimationIfNecessary();
|
|
update();
|
|
}
|
|
|
|
void RasterImageView::updateImageRect(const QRect& imageRect)
|
|
{
|
|
QRectF viewRect = mapToView(imageRect);
|
|
if (!viewRect.intersects(boundingRect())) {
|
|
return;
|
|
}
|
|
|
|
if (zoomToFit()) {
|
|
setZoom(computeZoomToFit());
|
|
}
|
|
d->setScalerRegionToVisibleRect();
|
|
update();
|
|
}
|
|
|
|
void RasterImageView::slotDocumentIsAnimatedUpdated()
|
|
{
|
|
d->startAnimationIfNecessary();
|
|
}
|
|
|
|
void RasterImageView::updateFromScaler(int zoomedImageLeft, int zoomedImageTop, const QImage& image)
|
|
{
|
|
d->resizeBuffer();
|
|
int viewportLeft = zoomedImageLeft - scrollPos().x();
|
|
int viewportTop = zoomedImageTop - scrollPos().y();
|
|
d->mBufferIsEmpty = false;
|
|
{
|
|
QPainter painter(&d->mCurrentBuffer);
|
|
if (document()->hasAlphaChannel()) {
|
|
d->drawAlphaBackground(
|
|
&painter, QRect(viewportLeft, viewportTop, image.width(), image.height()),
|
|
QPoint(zoomedImageLeft, zoomedImageTop)
|
|
);
|
|
} else {
|
|
painter.setCompositionMode(QPainter::CompositionMode_Source);
|
|
}
|
|
painter.drawImage(viewportLeft, viewportTop, image);
|
|
}
|
|
update();
|
|
|
|
if (!d->mEmittedCompleted) {
|
|
d->mEmittedCompleted = true;
|
|
completed();
|
|
}
|
|
}
|
|
|
|
void RasterImageView::onZoomChanged()
|
|
{
|
|
// If we zoom more than twice, then assume the user wants to see the real
|
|
// pixels, for example to fine tune a crop operation
|
|
if (zoom() < 2.) {
|
|
d->mScaler->setTransformationMode(Qt::SmoothTransformation);
|
|
} else {
|
|
d->mScaler->setTransformationMode(Qt::FastTransformation);
|
|
}
|
|
if (!d->mUpdateTimer->isActive()) {
|
|
updateBuffer();
|
|
}
|
|
}
|
|
|
|
void RasterImageView::onImageOffsetChanged()
|
|
{
|
|
update();
|
|
}
|
|
|
|
void RasterImageView::onScrollPosChanged(const QPointF& oldPos)
|
|
{
|
|
QPointF delta = scrollPos() - oldPos;
|
|
|
|
// Scroll existing
|
|
{
|
|
if (d->mAlternateBuffer.size() != d->mCurrentBuffer.size()) {
|
|
d->mAlternateBuffer = QPixmap(d->mCurrentBuffer.size());
|
|
}
|
|
QPainter painter(&d->mAlternateBuffer);
|
|
painter.drawPixmap(-delta, d->mCurrentBuffer);
|
|
}
|
|
qSwap(d->mCurrentBuffer, d->mAlternateBuffer);
|
|
|
|
// Scale missing parts
|
|
QRegion bufferRegion = QRegion(d->mCurrentBuffer.rect().translated(scrollPos().toPoint()));
|
|
QRegion updateRegion = bufferRegion - bufferRegion.translated(-delta.toPoint());
|
|
updateBuffer(updateRegion);
|
|
update();
|
|
}
|
|
|
|
void RasterImageView::paint(QPainter* painter, const QStyleOptionGraphicsItem* /*option*/, QWidget* /*widget*/)
|
|
{
|
|
QPointF topLeft = imageOffset();
|
|
if (zoomToFit()) {
|
|
// In zoomToFit mode, scale crudely the buffer to fit the screen. This
|
|
// provide an approximate rendered which will be replaced when the scheduled
|
|
// proper scale is ready.
|
|
QSizeF size = documentSize() * zoom();
|
|
painter->drawPixmap(topLeft.x(), topLeft.y(), size.width(), size.height(), d->mCurrentBuffer);
|
|
} else {
|
|
painter->drawPixmap(topLeft, d->mCurrentBuffer);
|
|
}
|
|
|
|
if (d->mTool) {
|
|
d->mTool.data()->paint(painter);
|
|
}
|
|
|
|
// Debug
|
|
#if 0
|
|
QSizeF visibleSize = documentSize() * zoom();
|
|
painter->setPen(Qt::red);
|
|
painter->drawRect(topLeft.x(), topLeft.y(), visibleSize.width() - 1, visibleSize.height() - 1);
|
|
|
|
painter->setPen(Qt::blue);
|
|
painter->drawRect(topLeft.x(), topLeft.y(), d->mCurrentBuffer.width() - 1, d->mCurrentBuffer.height() - 1);
|
|
#endif
|
|
}
|
|
|
|
void RasterImageView::resizeEvent(QGraphicsSceneResizeEvent* event)
|
|
{
|
|
// If we are in zoomToFit mode and have something in our buffer, delay the
|
|
// update: paint() will paint a scaled version of the buffer until resizing
|
|
// is done. This is much faster than rescaling the whole image for each
|
|
// resize event we receive.
|
|
// mUpdateTimer must be started before calling AbstractImageView::resizeEvent()
|
|
// because AbstractImageView::resizeEvent() will call onZoomChanged(), which
|
|
// will trigger an immediate update unless the mUpdateTimer is active.
|
|
if (zoomToFit() && !d->mBufferIsEmpty) {
|
|
d->mUpdateTimer->start();
|
|
}
|
|
AbstractImageView::resizeEvent(event);
|
|
if (!zoomToFit()) {
|
|
// Only update buffer if we are not in zoomToFit mode: if we are
|
|
// onZoomChanged() will have already updated the buffer.
|
|
updateBuffer();
|
|
}
|
|
}
|
|
|
|
void RasterImageView::updateBuffer(const QRegion& region)
|
|
{
|
|
d->mUpdateTimer->stop();
|
|
d->mScaler->setZoom(zoom());
|
|
if (region.isEmpty()) {
|
|
d->setScalerRegionToVisibleRect();
|
|
} else {
|
|
d->mScaler->setDestinationRegion(region);
|
|
}
|
|
}
|
|
|
|
void RasterImageView::setCurrentTool(AbstractRasterImageViewTool* tool)
|
|
{
|
|
if (d->mTool) {
|
|
d->mTool.data()->toolDeactivated();
|
|
d->mTool.data()->deleteLater();
|
|
}
|
|
d->mTool = tool;
|
|
if (d->mTool) {
|
|
d->mTool.data()->toolActivated();
|
|
}
|
|
updateCursor();
|
|
currentToolChanged(tool);
|
|
update();
|
|
}
|
|
|
|
AbstractRasterImageViewTool* RasterImageView::currentTool() const
|
|
{
|
|
return d->mTool.data();
|
|
}
|
|
|
|
void RasterImageView::mousePressEvent(QGraphicsSceneMouseEvent* event)
|
|
{
|
|
if (d->mTool) {
|
|
d->mTool.data()->mousePressEvent(event);
|
|
if (event->isAccepted()) {
|
|
return;
|
|
}
|
|
}
|
|
AbstractImageView::mousePressEvent(event);
|
|
}
|
|
|
|
void RasterImageView::mouseMoveEvent(QGraphicsSceneMouseEvent* event)
|
|
{
|
|
if (d->mTool) {
|
|
d->mTool.data()->mouseMoveEvent(event);
|
|
if (event->isAccepted()) {
|
|
return;
|
|
}
|
|
}
|
|
AbstractImageView::mouseMoveEvent(event);
|
|
}
|
|
|
|
void RasterImageView::mouseReleaseEvent(QGraphicsSceneMouseEvent* event)
|
|
{
|
|
if (d->mTool) {
|
|
d->mTool.data()->mouseReleaseEvent(event);
|
|
if (event->isAccepted()) {
|
|
return;
|
|
}
|
|
}
|
|
AbstractImageView::mouseReleaseEvent(event);
|
|
}
|
|
|
|
void RasterImageView::wheelEvent(QGraphicsSceneWheelEvent* event)
|
|
{
|
|
if (d->mTool) {
|
|
d->mTool.data()->wheelEvent(event);
|
|
if (event->isAccepted()) {
|
|
return;
|
|
}
|
|
}
|
|
AbstractImageView::wheelEvent(event);
|
|
}
|
|
|
|
void RasterImageView::keyPressEvent(QKeyEvent* event)
|
|
{
|
|
if (d->mTool) {
|
|
d->mTool.data()->keyPressEvent(event);
|
|
if (event->isAccepted()) {
|
|
return;
|
|
}
|
|
}
|
|
AbstractImageView::keyPressEvent(event);
|
|
}
|
|
|
|
void RasterImageView::keyReleaseEvent(QKeyEvent* event)
|
|
{
|
|
if (d->mTool) {
|
|
d->mTool.data()->keyReleaseEvent(event);
|
|
if (event->isAccepted()) {
|
|
return;
|
|
}
|
|
}
|
|
AbstractImageView::keyReleaseEvent(event);
|
|
}
|
|
|
|
void RasterImageView::hoverMoveEvent(QGraphicsSceneHoverEvent* event)
|
|
{
|
|
if (d->mTool) {
|
|
d->mTool.data()->hoverMoveEvent(event);
|
|
if (event->isAccepted()) {
|
|
return;
|
|
}
|
|
}
|
|
AbstractImageView::hoverMoveEvent(event);
|
|
}
|
|
|
|
} // namespace
|