mirror of
https://github.com/arsenetar/dupeguru.git
synced 2024-11-17 20:49:02 +00:00
6213d50670
commitac941037ff
Author: glubsy <glubsy@users.noreply.github.com> Date: Thu Jul 16 22:21:24 2020 +0200 Fix resize of top frame not updating scaled pixmap * Also limit viewing features such as zoom levels when files have different dimensions * GraphicsViewImageViewer is still a bit buggy: the scrollbars are toggled on when the pixmap is null in the reference viewer (we do not use that class right anyway) commit733b3b0ed4
Author: glubsy <glubsy@users.noreply.github.com> Date: Thu Jul 16 01:31:24 2020 +0200 Prevent zoom for images of differing dimensions * If images are not the same size, prevent zooming features from being used by disabling the normal size button, only enable swap commit9168d72f38
Author: glubsy <glubsy@users.noreply.github.com> Date: Wed Jul 15 22:47:32 2020 +0200 Update preferences on show(), not in constructor * If the dialog window shouldn't have a titlebar during construction, update accordingly only when showing to fix Windows displaying a window without titlebar on first show * Only save geometry if the window is floating. Otherwise geometry while docked is saved whih gives weird results on subsequent starts, since it may be floating by default anyway (at least on Linux where titlebar being disabled is allowed while floating) * Vertical title bar doesn't seem to work on Windows, add note in preferences dialog commit75621cc816
Author: glubsy <glubsy@users.noreply.github.com> Date: Wed Jul 15 22:04:19 2020 +0200 Prevent Windows from floating if no decoration * Windows users cannot move a window which has no native decorations. Toggling a dock widget's titlebar off also removes native decorations on a floating window. Until we implement a replacement titlebar by overriding paintEvents, simply force the floating window to go back to docked state after we toggled the titlebar off. commit3c816b2f11
Author: glubsy <glubsy@users.noreply.github.com> Date: Wed Jul 15 21:43:01 2020 +0200 Fix computing and setting offset to 0 for tableview commit85d6e05cd4
Merge:66127d02
3eddeb6a
Author: glubsy <glubsy@users.noreply.github.com> Date: Wed Jul 15 21:25:44 2020 +0200 Merge branch 'dockable_windows' into details_dialog_improvements_dev commit66127d025e
Author: glubsy <glubsy@users.noreply.github.com> Date: Wed Jul 15 20:22:13 2020 +0200 Add credit for icons used, upscale exchange icon * Jason Cho gave his express permission to use the icon (it was made 10 years ago and he doesn't have the source files anymore) * Used waifu2x to upscale the icon * Used GIMP to draw dark outline around the icon * Source files are included commit58c675d1fa
Author: glubsy <glubsy@users.noreply.github.com> Date: Wed Jul 15 05:25:47 2020 +0200 Add custom icons * Use custom icons on platforms which do not provide theme * Old zoom icons credits to "schollidesign" from icon pack Office and Entertainment (GPL licence). * Exchange icon credit to Jason Cho (Unknown license). * Use hack to resize viewers on first show() as well commit95b8406c7b
Author: glubsy <glubsy@users.noreply.github.com> Date: Wed Jul 15 04:14:24 2020 +0200 Fix scrollbar displayed while splitter maxed out * For some reason the table's height is a few pixel longer on Windows so we work around the issue by adding a small offset to the maximum height hint. * No idea about MacOS yet but this might need the same treatment. commit3eddeb6aeb
Author: glubsy <glubsy@users.noreply.github.com> Date: Tue Jul 14 17:37:48 2020 +0200 Fix ME/SE details dialogs, add preferences * Fix ME and SE versions of details dialog not displaying their content properly after change to QDockWidget * Add option to toggle titlebar and orientation of titlebar in preferences dialog * Fix setting layout on PE details dialog window while layout already set, by removing the self (parent) reference in constructing the QSplitter commit56912a7108
Author: glubsy <glubsy@users.noreply.github.com> Date: Mon Jul 13 05:06:04 2020 +0200 Make details dialog dockable
1347 lines
50 KiB
Python
1347 lines
50 KiB
Python
# This software is licensed under the "GPLv3" License as described in the "LICENSE" file,
|
|
# which should be included with this package. The terms are also available at
|
|
# http://www.gnu.org/licenses/gpl-3.0.html
|
|
|
|
from PyQt5.QtCore import (
|
|
QObject, Qt, QSize, QRectF, QPointF, QPoint, pyqtSlot, pyqtSignal, QEvent)
|
|
from PyQt5.QtGui import QPixmap, QPainter, QPalette, QCursor, QIcon, QKeySequence
|
|
from PyQt5.QtWidgets import (
|
|
QGraphicsView, QGraphicsScene, QGraphicsPixmapItem,
|
|
QToolBar, QToolButton, QAction, QWidget, QScrollArea,
|
|
QApplication, QAbstractScrollArea, QStyle)
|
|
from hscommon.trans import trget
|
|
from hscommon.plat import ISLINUX
|
|
tr = trget("ui")
|
|
|
|
MAX_SCALE = 12.0
|
|
MIN_SCALE = 0.1
|
|
|
|
|
|
def createActions(actions, target):
|
|
# actions = [(name, shortcut, icon, desc, func)]
|
|
for name, shortcut, icon, desc, func in actions:
|
|
action = QAction(target)
|
|
if icon:
|
|
action.setIcon(icon)
|
|
if shortcut:
|
|
action.setShortcut(shortcut)
|
|
action.setText(desc)
|
|
action.triggered.connect(func)
|
|
setattr(target, name, action)
|
|
|
|
|
|
class ViewerToolBar(QToolBar):
|
|
def __init__(self, parent, controller):
|
|
super().__init__(parent)
|
|
self.parent = parent
|
|
self.controller = controller
|
|
self.setupActions(controller)
|
|
self.createButtons()
|
|
self.buttonImgSwap.setEnabled(False)
|
|
self.buttonZoomIn.setEnabled(False)
|
|
self.buttonZoomOut.setEnabled(False)
|
|
self.buttonNormalSize.setEnabled(False)
|
|
self.buttonBestFit.setEnabled(False)
|
|
|
|
def setupActions(self, controller):
|
|
# actions = [(name, shortcut, icon, desc, func)]
|
|
ACTIONS = [
|
|
(
|
|
"actionZoomIn",
|
|
QKeySequence.ZoomIn,
|
|
QIcon.fromTheme("zoom-in") if ISLINUX
|
|
else QIcon(QPixmap(":/" + "zoom_in")),
|
|
tr("Increase zoom"),
|
|
controller.zoomIn,
|
|
),
|
|
(
|
|
"actionZoomOut",
|
|
QKeySequence.ZoomOut,
|
|
QIcon.fromTheme("zoom-out") if ISLINUX
|
|
else QIcon(QPixmap(":/" + "zoom_out")),
|
|
tr("Decrease zoom"),
|
|
controller.zoomOut,
|
|
),
|
|
(
|
|
"actionNormalSize",
|
|
tr("Ctrl+/"),
|
|
QIcon.fromTheme("zoom-original") if ISLINUX
|
|
else QIcon(QPixmap(":/" + "zoom_original")),
|
|
tr("Normal size"),
|
|
controller.zoomNormalSize,
|
|
),
|
|
(
|
|
"actionBestFit",
|
|
tr("Ctrl+*"),
|
|
QIcon.fromTheme("zoom-best-fit") if ISLINUX
|
|
else QIcon(QPixmap(":/" + "zoom_best_fit")),
|
|
tr("Best fit"),
|
|
controller.zoomBestFit,
|
|
)
|
|
]
|
|
# TODO try with QWidgetAction() instead in order to have
|
|
# the popup menu work in the toolbar (if resized below minimum height)
|
|
createActions(ACTIONS, self)
|
|
|
|
def createButtons(self):
|
|
self.buttonImgSwap = QToolButton(self)
|
|
self.buttonImgSwap.setToolButtonStyle(Qt.ToolButtonIconOnly)
|
|
self.buttonImgSwap.setIcon(
|
|
QIcon.fromTheme('view-refresh',
|
|
self.style().standardIcon(QStyle.SP_BrowserReload))
|
|
if ISLINUX
|
|
else QIcon(QPixmap(":/" + "exchange")))
|
|
self.buttonImgSwap.setText('Swap images')
|
|
self.buttonImgSwap.setToolTip('Swap images')
|
|
self.buttonImgSwap.pressed.connect(self.controller.swapImages)
|
|
self.buttonImgSwap.released.connect(self.controller.swapImages)
|
|
|
|
self.buttonZoomIn = QToolButton(self)
|
|
self.buttonZoomIn.setToolButtonStyle(Qt.ToolButtonIconOnly)
|
|
self.buttonZoomIn.setDefaultAction(self.actionZoomIn)
|
|
# self.buttonZoomIn.setText('ZoomIn')
|
|
# self.buttonZoomIn.setIcon(QIcon.fromTheme('zoom-in'))
|
|
self.buttonZoomIn.setEnabled(False)
|
|
|
|
self.buttonZoomOut = QToolButton(self)
|
|
self.buttonZoomOut.setToolButtonStyle(Qt.ToolButtonIconOnly)
|
|
self.buttonZoomOut.setDefaultAction(self.actionZoomOut)
|
|
# self.buttonZoomOut.setText('ZoomOut')
|
|
# self.buttonZoomOut.setIcon(QIcon.fromTheme('zoom-out'))
|
|
self.buttonZoomOut.setEnabled(False)
|
|
|
|
self.buttonNormalSize = QToolButton(self)
|
|
self.buttonNormalSize.setToolButtonStyle(Qt.ToolButtonIconOnly)
|
|
self.buttonNormalSize.setDefaultAction(self.actionNormalSize)
|
|
# self.buttonNormalSize.setText('Normal Size')
|
|
# self.buttonNormalSize.setIcon(QIcon.fromTheme('zoom-original'))
|
|
self.buttonNormalSize.setEnabled(True)
|
|
|
|
self.buttonBestFit = QToolButton(self)
|
|
self.buttonBestFit.setToolButtonStyle(Qt.ToolButtonIconOnly)
|
|
self.buttonBestFit.setDefaultAction(self.actionBestFit)
|
|
# self.buttonBestFit.setText('BestFit')
|
|
# self.buttonBestFit.setIcon(QIcon.fromTheme('zoom-best-fit'))
|
|
self.buttonBestFit.setEnabled(False)
|
|
|
|
self.addWidget(self.buttonImgSwap)
|
|
self.addWidget(self.buttonZoomIn)
|
|
self.addWidget(self.buttonZoomOut)
|
|
self.addWidget(self.buttonNormalSize)
|
|
self.addWidget(self.buttonBestFit)
|
|
|
|
|
|
class BaseController(QObject):
|
|
"""Abstract Base class. Singleton.
|
|
Base proxy interface to keep image viewers synchronized.
|
|
Relays function calls, keep tracks of things."""
|
|
|
|
def __init__(self, parent):
|
|
super().__init__()
|
|
self.selectedViewer = None
|
|
self.referenceViewer = None
|
|
# cached pixmaps
|
|
self.selectedPixmap = QPixmap()
|
|
self.referencePixmap = QPixmap()
|
|
self.scaledSelectedPixmap = QPixmap()
|
|
self.scaledReferencePixmap = QPixmap()
|
|
self.current_scale = 1.0
|
|
self.bestFit = True
|
|
self.parent = parent # To change buttons' states
|
|
self.cached_group = None
|
|
self.same_dimensions = True
|
|
|
|
def setupViewers(self, selectedViewer, referenceViewer):
|
|
self.selectedViewer = selectedViewer
|
|
self.referenceViewer = referenceViewer
|
|
self.selectedViewer.controller = self
|
|
self.referenceViewer.controller = self
|
|
self._setupConnections()
|
|
|
|
def _setupConnections(self):
|
|
self.selectedViewer.connectMouseSignals()
|
|
self.referenceViewer.connectMouseSignals()
|
|
|
|
def updateView(self, ref, dupe, group):
|
|
# To keep current scale accross dupes from the same group
|
|
previous_same_dimensions = self.same_dimensions
|
|
self.same_dimensions = True
|
|
same_group = True
|
|
if group != self.cached_group:
|
|
same_group = False
|
|
self.resetState()
|
|
self.cached_group = group
|
|
|
|
self.selectedPixmap = QPixmap(str(dupe.path))
|
|
if ref is dupe: # currently selected file is the actual reference file
|
|
# self.same_dimensions = False
|
|
self.referencePixmap = QPixmap()
|
|
self.scaledReferencePixmap = QPixmap()
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
else:
|
|
self.referencePixmap = QPixmap(str(ref.path))
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(True)
|
|
if ref.dimensions != dupe.dimensions:
|
|
self.same_dimensions = False
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
self.updateButtonsAsPerDimensions(previous_same_dimensions)
|
|
self.updateBothImages(same_group)
|
|
self.centerViews(same_group and self.referencePixmap.isNull())
|
|
|
|
def updateBothImages(self, same_group=False):
|
|
# WARNING this is called on every resize event,
|
|
ignore_update = self.referencePixmap.isNull()
|
|
if ignore_update:
|
|
self.selectedViewer.ignore_signal = True
|
|
# the SelectedImageViewer widget sometimes ends up being bigger
|
|
# than the ReferenceImageViewer by one pixel, which distorts the
|
|
# scaled down pixmap for the reference, hence we'll reuse its size here.
|
|
selected_size = self._updateImage(
|
|
self.selectedPixmap, self.scaledSelectedPixmap,
|
|
self.selectedViewer, None, same_group)
|
|
self._updateImage(
|
|
self.referencePixmap, self.scaledReferencePixmap,
|
|
self.referenceViewer, selected_size, same_group)
|
|
if ignore_update:
|
|
self.selectedViewer.ignore_signal = False
|
|
|
|
def _updateImage(self, pixmap, scaledpixmap, viewer, target_size=None, same_group=False):
|
|
# WARNING this is called on every resize event, might need to split
|
|
# into a separate function depending on the implementation used
|
|
if pixmap.isNull():
|
|
# This should disable the blank widget
|
|
viewer.setImage(pixmap)
|
|
return
|
|
target_size = viewer.size()
|
|
if not viewer.bestFit:
|
|
if same_group:
|
|
viewer.setImage(pixmap)
|
|
return target_size
|
|
# zoomed in state, expand
|
|
# only if not same_group, we need full update
|
|
scaledpixmap = pixmap.scaled(
|
|
target_size, Qt.KeepAspectRatioByExpanding, Qt.FastTransformation)
|
|
else:
|
|
# best fit, keep ratio always
|
|
scaledpixmap = pixmap.scaled(
|
|
target_size, Qt.KeepAspectRatio, Qt.FastTransformation)
|
|
viewer.setImage(scaledpixmap)
|
|
return target_size
|
|
|
|
def resetState(self):
|
|
"""Only called when the group of dupes has changed. We reset our
|
|
controller internal state and buttons, center view on viewers."""
|
|
self.selectedPixmap = QPixmap()
|
|
self.scaledSelectedPixmap = QPixmap()
|
|
self.referencePixmap = QPixmap()
|
|
self.scaledReferencePixmap = QPixmap()
|
|
self.setBestFit(True)
|
|
self.current_scale = 1.0
|
|
self.selectedViewer.current_scale = 1.0
|
|
self.referenceViewer.current_scale = 1.0
|
|
self.selectedViewer.resetCenter()
|
|
self.referenceViewer.resetCenter()
|
|
self.selectedViewer.scaleAt(1.0)
|
|
self.referenceViewer.scaleAt(1.0)
|
|
self.centerViews()
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
self.parent.verticalToolBar.buttonBestFit.setEnabled(False) # active mode by default
|
|
|
|
def resetViewersState(self):
|
|
"""No item from the model, disable and clear everything."""
|
|
# only called by the details dialog
|
|
self.selectedPixmap = QPixmap()
|
|
self.scaledSelectedPixmap = QPixmap()
|
|
self.referencePixmap = QPixmap()
|
|
self.scaledReferencePixmap = QPixmap()
|
|
self.setBestFit(True)
|
|
self.current_scale = 1.0
|
|
self.selectedViewer.current_scale = 1.0
|
|
self.referenceViewer.current_scale = 1.0
|
|
self.selectedViewer.resetCenter()
|
|
self.referenceViewer.resetCenter()
|
|
self.selectedViewer.scaleAt(1.0)
|
|
self.referenceViewer.scaleAt(1.0)
|
|
self.centerViews()
|
|
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonBestFit.setEnabled(False) # active mode by default
|
|
|
|
self.selectedViewer.setImage(self.selectedPixmap) # null
|
|
self.selectedViewer.setEnabled(False)
|
|
self.referenceViewer.setImage(self.referencePixmap) # null
|
|
self.referenceViewer.setEnabled(False)
|
|
|
|
@pyqtSlot()
|
|
def zoomIn(self):
|
|
self.scaleImagesBy(1.25)
|
|
|
|
@pyqtSlot()
|
|
def zoomOut(self):
|
|
self.scaleImagesBy(0.8)
|
|
|
|
@pyqtSlot(float)
|
|
def scaleImagesBy(self, factor):
|
|
"""Compute new scale from factor and scale."""
|
|
self.current_scale *= factor
|
|
self.selectedViewer.scaleBy(factor)
|
|
self.referenceViewer.scaleBy(factor)
|
|
self.updateButtons()
|
|
|
|
@pyqtSlot(float)
|
|
def scaleImagesAt(self, scale):
|
|
"""Scale at a pre-computed scale."""
|
|
self.current_scale = scale
|
|
self.selectedViewer.scaleAt(scale)
|
|
self.referenceViewer.scaleAt(scale)
|
|
self.updateButtons()
|
|
|
|
def updateButtons(self):
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(self.current_scale < MAX_SCALE)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(self.current_scale > MIN_SCALE)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(round(self.current_scale, 1) != 1.0)
|
|
self.parent.verticalToolBar.buttonBestFit.setEnabled(self.bestFit is False)
|
|
|
|
def updateButtonsAsPerDimensions(self, previous_same_dimensions):
|
|
if not self.same_dimensions:
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(False)
|
|
if not self.bestFit:
|
|
self.zoomBestFit()
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
if not self.referencePixmap.isNull():
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(True)
|
|
return
|
|
if not self.bestFit and not previous_same_dimensions:
|
|
self.zoomBestFit()
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
if self.referencePixmap.isNull():
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(False)
|
|
|
|
@pyqtSlot()
|
|
def zoomBestFit(self):
|
|
"""Setup before scaling to bestfit"""
|
|
self.setBestFit(True)
|
|
self.current_scale = 1.0
|
|
self.selectedViewer.current_scale = 1.0
|
|
self.referenceViewer.current_scale = 1.0
|
|
|
|
self.selectedViewer.scaleAt(1.0)
|
|
self.referenceViewer.scaleAt(1.0)
|
|
|
|
self.selectedViewer.resetCenter()
|
|
self.referenceViewer.resetCenter()
|
|
|
|
target_size = self._updateImage(
|
|
self.selectedPixmap, self.scaledSelectedPixmap,
|
|
self.selectedViewer, None, True)
|
|
self._updateImage(
|
|
self.referencePixmap, self.scaledReferencePixmap,
|
|
self.referenceViewer, target_size, True)
|
|
self.centerViews()
|
|
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
self.parent.verticalToolBar.buttonBestFit.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(True)
|
|
|
|
def setBestFit(self, value):
|
|
self.bestFit = value
|
|
self.selectedViewer.bestFit = value
|
|
self.referenceViewer.bestFit = value
|
|
|
|
@pyqtSlot()
|
|
def zoomNormalSize(self):
|
|
self.setBestFit(False)
|
|
self.current_scale = 1.0
|
|
|
|
self.selectedViewer.setImage(self.selectedPixmap)
|
|
self.referenceViewer.setImage(self.referencePixmap)
|
|
|
|
self.centerViews()
|
|
|
|
self.selectedViewer.scaleToNormalSize()
|
|
self.referenceViewer.scaleToNormalSize()
|
|
|
|
if self.same_dimensions:
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(True)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(True)
|
|
else:
|
|
# we can't allow swapping pixmaps of different dimensions
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonBestFit.setEnabled(True)
|
|
|
|
def centerViews(self, only_selected=False):
|
|
self.selectedViewer.centerViewAndUpdate()
|
|
if only_selected:
|
|
return
|
|
self.referenceViewer.centerViewAndUpdate()
|
|
|
|
@pyqtSlot()
|
|
def swapImages(self):
|
|
# swap the columns in the details table as well
|
|
self.parent.tableView.horizontalHeader().swapSections(1, 2)
|
|
|
|
|
|
class QWidgetController(BaseController):
|
|
"""Specialized version for QWidget-based viewers."""
|
|
def __init__(self, parent):
|
|
super().__init__(parent)
|
|
|
|
def _updateImage(self, *args):
|
|
ret = super()._updateImage(*args)
|
|
# Fix alignment when resizing window
|
|
self.centerViews()
|
|
return ret
|
|
|
|
@pyqtSlot(QPointF)
|
|
def onDraggedMouse(self, delta):
|
|
if not self.same_dimensions:
|
|
return
|
|
if self.sender() is self.referenceViewer:
|
|
self.selectedViewer.onDraggedMouse(delta)
|
|
else:
|
|
self.referenceViewer.onDraggedMouse(delta)
|
|
|
|
@pyqtSlot()
|
|
def swapImages(self):
|
|
self.selectedViewer._pixmap.swap(self.referenceViewer._pixmap)
|
|
self.selectedViewer.centerViewAndUpdate()
|
|
self.referenceViewer.centerViewAndUpdate()
|
|
super().swapImages()
|
|
|
|
|
|
class ScrollAreaController(BaseController):
|
|
"""Specialized version fro QLabel-based viewers."""
|
|
def __init__(self, parent):
|
|
super().__init__(parent)
|
|
|
|
def _setupConnections(self):
|
|
super()._setupConnections()
|
|
self.selectedViewer.connectScrollBars()
|
|
self.referenceViewer.connectScrollBars()
|
|
|
|
def updateBothImages(self, same_group=False):
|
|
super().updateBothImages(same_group)
|
|
if not self.referenceViewer.isEnabled():
|
|
return
|
|
self.referenceViewer._horizontalScrollBar.setValue(
|
|
self.selectedViewer._horizontalScrollBar.value())
|
|
self.referenceViewer._verticalScrollBar.setValue(
|
|
self.selectedViewer._verticalScrollBar.value())
|
|
|
|
@pyqtSlot(QPoint)
|
|
def onDraggedMouse(self, delta):
|
|
self.selectedViewer.ignore_signal = True
|
|
self.referenceViewer.ignore_signal = True
|
|
|
|
if self.same_dimensions:
|
|
self.selectedViewer.onDraggedMouse(delta)
|
|
self.referenceViewer.onDraggedMouse(delta)
|
|
else:
|
|
if self.sender() is self.selectedViewer:
|
|
self.selectedViewer.onDraggedMouse(delta)
|
|
else:
|
|
self.referenceViewer.onDraggedMouse(delta)
|
|
|
|
self.selectedViewer.ignore_signal = False
|
|
self.referenceViewer.ignore_signal = False
|
|
|
|
@pyqtSlot()
|
|
def swapImages(self):
|
|
self.referenceViewer._pixmap.swap(self.selectedViewer._pixmap)
|
|
self.referenceViewer.setCachedPixmap()
|
|
self.selectedViewer.setCachedPixmap()
|
|
super().swapImages()
|
|
|
|
@pyqtSlot(float, QPointF)
|
|
def onMouseWheel(self, scale, delta):
|
|
self.scaleImagesAt(scale)
|
|
self.selectedViewer.adjustScrollBarsScaled(delta)
|
|
# Signal from scrollbars will automatically change the other:
|
|
# self.referenceViewer.adjustScrollBarsScaled(delta)
|
|
|
|
@pyqtSlot(int)
|
|
def onVScrollBarChanged(self, value):
|
|
if not self.same_dimensions:
|
|
return
|
|
if self.sender() is self.referenceViewer._verticalScrollBar:
|
|
if not self.selectedViewer.ignore_signal:
|
|
self.selectedViewer._verticalScrollBar.setValue(value)
|
|
else:
|
|
if not self.referenceViewer.ignore_signal:
|
|
self.referenceViewer._verticalScrollBar.setValue(value)
|
|
|
|
@pyqtSlot(int)
|
|
def onHScrollBarChanged(self, value):
|
|
if not self.same_dimensions:
|
|
return
|
|
if self.sender() is self.referenceViewer._horizontalScrollBar:
|
|
if not self.selectedViewer.ignore_signal:
|
|
self.selectedViewer._horizontalScrollBar.setValue(value)
|
|
else:
|
|
if not self.referenceViewer.ignore_signal:
|
|
self.referenceViewer._horizontalScrollBar.setValue(value)
|
|
|
|
@pyqtSlot(float)
|
|
def scaleImagesBy(self, factor):
|
|
super().scaleImagesBy(factor)
|
|
# The other is automatically updated via sigals
|
|
self.selectedViewer.adjustScrollBarsFactor(factor)
|
|
|
|
@pyqtSlot()
|
|
def zoomBestFit(self):
|
|
# Disable scrollbars to avoid GridLayout size rounding glitch
|
|
super().zoomBestFit()
|
|
if self.referencePixmap.isNull():
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(False)
|
|
self.selectedViewer.toggleScrollBars()
|
|
self.referenceViewer.toggleScrollBars()
|
|
|
|
|
|
class GraphicsViewController(BaseController):
|
|
"""Specialized version fro QGraphicsView-based viewers."""
|
|
def __init__(self, parent):
|
|
super().__init__(parent)
|
|
|
|
def _setupConnections(self):
|
|
super()._setupConnections()
|
|
self.selectedViewer.connectScrollBars()
|
|
self.referenceViewer.connectScrollBars()
|
|
# Special case for mouse wheel event conflicting with scrollbar adjustments
|
|
self.selectedViewer.other_viewer = self.referenceViewer
|
|
self.referenceViewer.other_viewer = self.selectedViewer
|
|
|
|
@pyqtSlot()
|
|
def syncCenters(self):
|
|
if self.sender() is self.referenceViewer:
|
|
self.selectedViewer.setCenter(self.referenceViewer._centerPoint)
|
|
else:
|
|
self.referenceViewer.setCenter(self.selectedViewer._centerPoint)
|
|
|
|
@pyqtSlot(float, QPointF)
|
|
def onMouseWheel(self, factor, newCenter):
|
|
self.current_scale *= factor
|
|
if self.sender() is self.referenceViewer:
|
|
self.selectedViewer.scaleBy(factor)
|
|
self.selectedViewer.setCenter(newCenter)
|
|
else:
|
|
self.referenceViewer.scaleBy(factor)
|
|
self.referenceViewer.setCenter(newCenter)
|
|
|
|
@pyqtSlot(int)
|
|
def onVScrollBarChanged(self, value):
|
|
if not self.same_dimensions:
|
|
return
|
|
if self.sender() is self.referenceViewer._verticalScrollBar:
|
|
if not self.selectedViewer.ignore_signal:
|
|
self.selectedViewer._verticalScrollBar.setValue(value)
|
|
else:
|
|
if not self.referenceViewer.ignore_signal:
|
|
self.referenceViewer._verticalScrollBar.setValue(value)
|
|
|
|
@pyqtSlot(int)
|
|
def onHScrollBarChanged(self, value):
|
|
if not self.same_dimensions:
|
|
return
|
|
if self.sender() is self.referenceViewer._horizontalScrollBar:
|
|
if not self.selectedViewer.ignore_signal:
|
|
self.selectedViewer._horizontalScrollBar.setValue(value)
|
|
else:
|
|
if not self.referenceViewer.ignore_signal:
|
|
self.referenceViewer._horizontalScrollBar.setValue(value)
|
|
|
|
@pyqtSlot()
|
|
def swapImages(self):
|
|
self.referenceViewer._pixmap.swap(self.selectedViewer._pixmap)
|
|
self.referenceViewer.setCachedPixmap()
|
|
self.selectedViewer.setCachedPixmap()
|
|
super().swapImages()
|
|
|
|
@pyqtSlot()
|
|
def zoomBestFit(self):
|
|
"""Setup before scaling to bestfit"""
|
|
self.setBestFit(True)
|
|
self.current_scale = 1.0
|
|
self.selectedViewer.fitScale()
|
|
self.referenceViewer.fitScale()
|
|
self.parent.verticalToolBar.buttonBestFit.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
if not self.referencePixmap.isNull():
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(True)
|
|
# else:
|
|
# self.referenceViewer.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
# self.referenceViewer.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
|
|
def updateView(self, ref, dupe, group):
|
|
# Keep current scale accross dupes from the same group
|
|
previous_same_dimensions = self.same_dimensions
|
|
self.same_dimensions = True
|
|
same_group = True
|
|
if group != self.cached_group:
|
|
same_group = False
|
|
self.resetState()
|
|
self.cached_group = group
|
|
|
|
self.selectedPixmap = QPixmap(str(dupe.path))
|
|
if ref is dupe: # currently selected file is the actual reference file
|
|
self.same_dimensions = False
|
|
self.referencePixmap = QPixmap()
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
else:
|
|
self.referencePixmap = QPixmap(str(ref.path))
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(True)
|
|
if ref.dimensions != dupe.dimensions:
|
|
self.same_dimensions = False
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
# self.selectedViewer.setImage(self.selectedPixmap)
|
|
# self.referenceViewer.setImage(self.referencePixmap)
|
|
self.updateButtonsAsPerDimensions(previous_same_dimensions)
|
|
self.updateBothImages(same_group)
|
|
|
|
def updateBothImages(self, same_group=False):
|
|
"""This is called only during resize events and while bestFit."""
|
|
ignore_update = self.referencePixmap.isNull()
|
|
if ignore_update:
|
|
self.selectedViewer.ignore_signal = True
|
|
|
|
self._updateFitImage(
|
|
self.selectedPixmap, self.selectedViewer)
|
|
self._updateFitImage(
|
|
self.referencePixmap, self.referenceViewer)
|
|
|
|
if ignore_update:
|
|
self.selectedViewer.ignore_signal = False
|
|
|
|
def _updateFitImage(self, pixmap, viewer):
|
|
# If not same_group, we need full update"""
|
|
viewer.setImage(pixmap)
|
|
if pixmap.isNull():
|
|
# viewer._item = None
|
|
return
|
|
if viewer.bestFit:
|
|
viewer.fitScale()
|
|
|
|
def resetState(self):
|
|
"""Only called when the group of dupes has changed. We reset our
|
|
controller internal state and buttons, center view on viewers."""
|
|
self.selectedPixmap = QPixmap()
|
|
self.referencePixmap = QPixmap()
|
|
self.setBestFit(True)
|
|
self.current_scale = 1.0
|
|
self.selectedViewer.current_scale = 1.0
|
|
self.referenceViewer.current_scale = 1.0
|
|
|
|
self.selectedViewer.resetCenter()
|
|
self.referenceViewer.resetCenter()
|
|
|
|
self.selectedViewer.fitScale()
|
|
self.referenceViewer.fitScale()
|
|
# self.centerViews()
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonBestFit.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(True)
|
|
|
|
def resetViewersState(self):
|
|
"""No item from the model, disable and clear everything."""
|
|
# only called by the details dialog
|
|
self.selectedPixmap = QPixmap()
|
|
self.scaledSelectedPixmap = QPixmap()
|
|
self.referencePixmap = QPixmap()
|
|
self.scaledReferencePixmap = QPixmap()
|
|
self.setBestFit(True)
|
|
self.current_scale = 1.0
|
|
self.selectedViewer.current_scale = 1.0
|
|
self.referenceViewer.current_scale = 1.0
|
|
self.selectedViewer.resetCenter()
|
|
self.referenceViewer.resetCenter()
|
|
# self.centerViews()
|
|
self.parent.verticalToolBar.buttonZoomIn.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonZoomOut.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonBestFit.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonImgSwap.setEnabled(False)
|
|
self.parent.verticalToolBar.buttonNormalSize.setEnabled(False)
|
|
|
|
self.selectedViewer.setImage(self.selectedPixmap) # null
|
|
self.selectedViewer.setEnabled(False)
|
|
self.referenceViewer.setImage(self.referencePixmap) # null
|
|
self.referenceViewer.setEnabled(False)
|
|
|
|
@pyqtSlot(float)
|
|
def scaleImagesBy(self, factor):
|
|
self.selectedViewer.updateCenterPoint()
|
|
self.referenceViewer.updateCenterPoint()
|
|
super().scaleImagesBy(factor)
|
|
self.selectedViewer.centerOn(self.selectedViewer._centerPoint)
|
|
# Scrollbars sync themselves here
|
|
|
|
|
|
class QWidgetImageViewer(QWidget):
|
|
"""Use a QPixmap, but no scrollbars and no keyboard key sequence for navigation."""
|
|
# FIXME: panning while zoomed-in is broken (due to delta not interpolated right?
|
|
mouseDragged = pyqtSignal(QPointF)
|
|
mouseWheeled = pyqtSignal(float)
|
|
|
|
def __init__(self, parent, name=""):
|
|
super().__init__(parent)
|
|
self._app = QApplication
|
|
self._pixmap = QPixmap()
|
|
self._rect = QRectF()
|
|
self._lastMouseClickPoint = QPointF()
|
|
self._mousePanningDelta = QPointF()
|
|
self.current_scale = 1.0
|
|
self._drag = False
|
|
self._dragConnection = None
|
|
self._wheelConnection = None
|
|
self._instance_name = name
|
|
self.bestFit = True
|
|
self.controller = None
|
|
self.setMouseTracking(False)
|
|
|
|
def __repr__(self):
|
|
return f'{self._instance_name}'
|
|
|
|
def connectMouseSignals(self):
|
|
if not self._dragConnection:
|
|
self._dragConnection = self.mouseDragged.connect(
|
|
self.controller.onDraggedMouse)
|
|
if not self._wheelConnection:
|
|
self._wheelConnection = self.mouseWheeled.connect(
|
|
self.controller.scaleImagesBy)
|
|
|
|
def disconnectMouseSignals(self):
|
|
if self._dragConnection:
|
|
self.mouseDragged.disconnect()
|
|
self._dragConnection = None
|
|
if self._wheelConnection:
|
|
self.mouseWheeled.disconnect()
|
|
self._wheelConnection = None
|
|
|
|
def paintEvent(self, event):
|
|
painter = QPainter(self)
|
|
painter.translate(self.rect().center())
|
|
painter.scale(self.current_scale, self.current_scale)
|
|
painter.translate(self._mousePanningDelta)
|
|
painter.drawPixmap(self._rect.topLeft(), self._pixmap)
|
|
|
|
def resetCenter(self):
|
|
""" Resets origin """
|
|
# Make sure we are not still panning around
|
|
self._mousePanningDelta = QPointF()
|
|
self.update()
|
|
|
|
def changeEvent(self, event):
|
|
if event.type() == QEvent.EnabledChange:
|
|
if self.isEnabled():
|
|
self.connectMouseSignals()
|
|
return
|
|
self.disconnectMouseSignals()
|
|
|
|
def mousePressEvent(self, event):
|
|
if self.bestFit or not self.isEnabled():
|
|
event.ignore()
|
|
return
|
|
if event.button() == Qt.LeftButton:
|
|
self._drag = True
|
|
else:
|
|
self._drag = False
|
|
event.ignore()
|
|
return
|
|
|
|
self._lastMouseClickPoint = event.pos()
|
|
self._app.setOverrideCursor(Qt.ClosedHandCursor)
|
|
self.setMouseTracking(True)
|
|
event.accept()
|
|
|
|
def mouseMoveEvent(self, event):
|
|
if self.bestFit or not self.isEnabled():
|
|
event.ignore()
|
|
return
|
|
|
|
self._mousePanningDelta += (
|
|
event.pos() - self._lastMouseClickPoint) * 1.0 / self.current_scale
|
|
self._lastMouseClickPoint = event.pos()
|
|
if self._drag:
|
|
self.mouseDragged.emit(self._mousePanningDelta)
|
|
self.update()
|
|
|
|
def mouseReleaseEvent(self, event):
|
|
if self.bestFit or not self.isEnabled():
|
|
event.ignore()
|
|
return
|
|
if event.button() == Qt.LeftButton:
|
|
self._drag = False
|
|
|
|
self._app.restoreOverrideCursor()
|
|
self.setMouseTracking(False)
|
|
|
|
def wheelEvent(self, event):
|
|
if self.bestFit or not self.controller.same_dimensions or not self.isEnabled():
|
|
event.ignore()
|
|
return
|
|
|
|
if event.angleDelta().y() > 0:
|
|
if self.current_scale > MAX_SCALE:
|
|
return
|
|
self.mouseWheeled.emit(1.25) # zoom-in
|
|
else:
|
|
if self.current_scale < MIN_SCALE:
|
|
return
|
|
self.mouseWheeled.emit(0.8) # zoom-out
|
|
|
|
def setImage(self, pixmap):
|
|
if pixmap.isNull():
|
|
if not self._pixmap.isNull():
|
|
self._pixmap = pixmap
|
|
self.disconnectMouseSignals()
|
|
self.setEnabled(False)
|
|
self.update()
|
|
return
|
|
elif not self.isEnabled():
|
|
self.setEnabled(True)
|
|
self.connectMouseSignals()
|
|
self._pixmap = pixmap
|
|
|
|
def centerViewAndUpdate(self):
|
|
self._rect = self._pixmap.rect()
|
|
self._rect.translate(-self._rect.center())
|
|
self.update()
|
|
|
|
def shouldBeActive(self):
|
|
return True if not self.pixmap.isNull() else False
|
|
|
|
def scaleBy(self, factor):
|
|
self.current_scale *= factor
|
|
self.update()
|
|
|
|
def scaleAt(self, scale):
|
|
self.current_scale = scale
|
|
self.update()
|
|
|
|
def sizeHint(self):
|
|
return QSize(400, 400)
|
|
|
|
@pyqtSlot()
|
|
def scaleToNormalSize(self):
|
|
"""Called when the pixmap is set back to original size."""
|
|
self.current_scale = 1.0
|
|
self.update()
|
|
|
|
@pyqtSlot(QPointF)
|
|
def onDraggedMouse(self, delta):
|
|
self._mousePanningDelta = delta
|
|
self.update()
|
|
|
|
|
|
class ScalablePixmap(QWidget):
|
|
"""Container for a pixmap that scales up very fast, used in ScrollAreaImageViewer."""
|
|
def __init__(self, parent):
|
|
super().__init__(parent)
|
|
self._pixmap = QPixmap()
|
|
self.current_scale = 1.0
|
|
|
|
def paintEvent(self, event):
|
|
painter = QPainter(self)
|
|
painter.scale(self.current_scale, self.current_scale)
|
|
# painter.drawPixmap(self.rect().topLeft(), self._pixmap)
|
|
# should be the same as:
|
|
painter.drawPixmap(0, 0, self._pixmap)
|
|
|
|
def sizeHint(self):
|
|
return self._pixmap.size() * self.current_scale
|
|
|
|
def minimumSizeHint(self):
|
|
return self.sizeHint()
|
|
|
|
|
|
class ScrollAreaImageViewer(QScrollArea):
|
|
"""Implementation using a pixmap container in a simple scroll area."""
|
|
mouseDragged = pyqtSignal(QPoint)
|
|
mouseWheeled = pyqtSignal(float, QPointF)
|
|
|
|
def __init__(self, parent, name=""):
|
|
super().__init__(parent)
|
|
self._parent = parent
|
|
self._app = QApplication
|
|
self._pixmap = QPixmap()
|
|
self._scaledpixmap = None
|
|
self._rect = QRectF()
|
|
self._lastMouseClickPoint = QPointF()
|
|
self._mousePanningDelta = QPoint()
|
|
self.current_scale = 1.0
|
|
self._drag = False
|
|
self._dragConnection = None
|
|
self._wheelConnection = None
|
|
self._instance_name = name
|
|
self.wantScrollBars = True
|
|
self.bestFit = True
|
|
self.controller = None
|
|
self.label = ScalablePixmap(self)
|
|
# This is to avoid sending signals twice on scrollbar updates
|
|
self.ignore_signal = False
|
|
self.setBackgroundRole(QPalette.Dark)
|
|
self.setWidgetResizable(False)
|
|
self.setSizeAdjustPolicy(QAbstractScrollArea.AdjustToContents)
|
|
self.setAlignment(Qt.AlignCenter)
|
|
self._verticalScrollBar = self.verticalScrollBar()
|
|
self._horizontalScrollBar = self.horizontalScrollBar()
|
|
if self.wantScrollBars:
|
|
self.toggleScrollBars()
|
|
else:
|
|
self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
self.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
self.setWidget(self.label)
|
|
self.setVisible(True)
|
|
|
|
def __repr__(self):
|
|
return f'{self._instance_name}'
|
|
|
|
def toggleScrollBars(self, forceOn=False):
|
|
if not self.wantScrollBars:
|
|
return
|
|
# Ensure that it's off on the first run
|
|
if self.horizontalScrollBarPolicy() == Qt.ScrollBarAsNeeded:
|
|
if forceOn:
|
|
return
|
|
self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
self.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
else:
|
|
self.setVerticalScrollBarPolicy(Qt.ScrollBarAsNeeded)
|
|
self.setHorizontalScrollBarPolicy(Qt.ScrollBarAsNeeded)
|
|
|
|
def connectMouseSignals(self):
|
|
if not self._dragConnection:
|
|
self._dragConnection = self.mouseDragged.connect(
|
|
self.controller.onDraggedMouse)
|
|
if not self._wheelConnection:
|
|
self._wheelConnection = self.mouseWheeled.connect(
|
|
self.controller.onMouseWheel)
|
|
|
|
def disconnectMouseSignals(self):
|
|
if self._dragConnection:
|
|
self.mouseDragged.disconnect()
|
|
self._dragConnection = None
|
|
if self._wheelConnection:
|
|
self.mouseWheeled.disconnect()
|
|
self._wheelConnection = None
|
|
|
|
def connectScrollBars(self):
|
|
"""Only call once controller is connected."""
|
|
# Cyclic connections are handled by Qt
|
|
self._verticalScrollBar.valueChanged.connect(
|
|
self.controller.onVScrollBarChanged, Qt.UniqueConnection)
|
|
self._horizontalScrollBar.valueChanged.connect(
|
|
self.controller.onHScrollBarChanged, Qt.UniqueConnection)
|
|
|
|
def mousePressEvent(self, event):
|
|
if self.bestFit:
|
|
event.ignore()
|
|
return
|
|
if event.button() == Qt.LeftButton:
|
|
self._drag = True
|
|
else:
|
|
self._drag = False
|
|
event.ignore()
|
|
return
|
|
self._lastMouseClickPoint = event.pos()
|
|
self._app.setOverrideCursor(Qt.ClosedHandCursor)
|
|
self.setMouseTracking(True)
|
|
super().mousePressEvent(event)
|
|
|
|
def mouseMoveEvent(self, event):
|
|
if self.bestFit:
|
|
event.ignore()
|
|
return
|
|
if self._drag:
|
|
delta = (event.pos() - self._lastMouseClickPoint)
|
|
self._lastMouseClickPoint = event.pos()
|
|
self.mouseDragged.emit(delta)
|
|
super().mouseMoveEvent(event)
|
|
|
|
def mouseReleaseEvent(self, event):
|
|
if self.bestFit:
|
|
event.ignore()
|
|
return
|
|
if event.button() == Qt.LeftButton:
|
|
self._drag = False
|
|
self._app.restoreOverrideCursor()
|
|
self.setMouseTracking(False)
|
|
super().mouseReleaseEvent(event)
|
|
|
|
def wheelEvent(self, event):
|
|
if self.bestFit or not self.controller.same_dimensions:
|
|
event.ignore()
|
|
return
|
|
oldScale = self.current_scale
|
|
if event.angleDelta().y() > 0: # zoom-in
|
|
if oldScale < MAX_SCALE:
|
|
self.current_scale *= 1.25
|
|
else:
|
|
if oldScale > MIN_SCALE: # zoom-out
|
|
self.current_scale *= 0.8
|
|
if oldScale == self.current_scale:
|
|
return
|
|
|
|
deltaToPos = (event.position() / oldScale) - (self.label.pos() / oldScale)
|
|
delta = (deltaToPos * self.current_scale) - (deltaToPos * oldScale)
|
|
self.mouseWheeled.emit(self.current_scale, delta)
|
|
|
|
def setImage(self, pixmap):
|
|
self._pixmap = pixmap
|
|
self.label._pixmap = pixmap
|
|
self.label.update()
|
|
self.label.adjustSize()
|
|
if pixmap.isNull():
|
|
self.setEnabled(False)
|
|
self.disconnectMouseSignals()
|
|
elif not self.isEnabled():
|
|
self.setEnabled(True)
|
|
self.connectMouseSignals()
|
|
|
|
def centerViewAndUpdate(self):
|
|
self._rect = self.label.rect()
|
|
self.label.rect().translate(-self._rect.center())
|
|
self.label.current_scale = self.current_scale
|
|
self.label.update()
|
|
# self.viewport().update()
|
|
|
|
def setCachedPixmap(self):
|
|
"""In case we have changed the cached pixmap, reset it."""
|
|
self.label._pixmap = self._pixmap
|
|
self.label.update()
|
|
|
|
def shouldBeActive(self):
|
|
return True if not self.pixmap.isNull() else False
|
|
|
|
def scaleBy(self, factor):
|
|
self.current_scale *= factor
|
|
# factor has to be either 1.25 or 0.8 here
|
|
self.label.resize(self.label.size().__imul__(factor))
|
|
self.label.current_scale = self.current_scale
|
|
self.label.update()
|
|
|
|
def scaleAt(self, scale):
|
|
self.current_scale = scale
|
|
self.label.resize(self._pixmap.size().__imul__(scale))
|
|
self.label.current_scale = scale
|
|
self.label.update()
|
|
# self.label.adjustSize()
|
|
|
|
def adjustScrollBarsFactor(self, factor):
|
|
"""After scaling, no mouse position, default to center."""
|
|
# scrollBar.setMaximum(scrollBar.maximum() - scrollBar.minimum() + scrollBar.pageStep())
|
|
self._horizontalScrollBar.setValue(
|
|
int(factor * self._horizontalScrollBar.value()
|
|
+ ((factor - 1) * self._horizontalScrollBar.pageStep() / 2)))
|
|
self._verticalScrollBar.setValue(
|
|
int(factor * self._verticalScrollBar.value()
|
|
+ ((factor - 1) * self._verticalScrollBar.pageStep() / 2)))
|
|
|
|
def adjustScrollBarsScaled(self, delta):
|
|
"""After scaling with the mouse, update relative to mouse position."""
|
|
self._horizontalScrollBar.setValue(
|
|
self._horizontalScrollBar.value() + delta.x())
|
|
self._verticalScrollBar.setValue(
|
|
self._verticalScrollBar.value() + delta.y())
|
|
|
|
def adjustScrollBarsAuto(self):
|
|
"""After panning, update accordingly."""
|
|
self.horizontalScrollBar().setValue(
|
|
self.horizontalScrollBar().value() - self._mousePanningDelta.x())
|
|
self.verticalScrollBar().setValue(
|
|
self.verticalScrollBar().value() - self._mousePanningDelta.y())
|
|
|
|
def adjustScrollBarCentered(self):
|
|
"""Just center in the middle."""
|
|
self._horizontalScrollBar.setValue(
|
|
int(self._horizontalScrollBar.maximum() / 2))
|
|
self._verticalScrollBar.setValue(
|
|
int(self._verticalScrollBar.maximum() / 2))
|
|
|
|
def resetCenter(self):
|
|
""" Resets origin """
|
|
self._mousePanningDelta = QPoint()
|
|
self.current_scale = 1.0
|
|
# self.scaleAt(1.0)
|
|
|
|
def setCenter(self, point):
|
|
self._lastMouseClickPoint = point
|
|
|
|
def sizeHint(self):
|
|
return self.viewport().rect().size()
|
|
|
|
@pyqtSlot()
|
|
def scaleToNormalSize(self):
|
|
"""Called when the pixmap is set back to original size."""
|
|
self.scaleAt(1.0)
|
|
self.ensureWidgetVisible(self.label) # needed for centering
|
|
self.toggleScrollBars(True)
|
|
|
|
@pyqtSlot(QPoint)
|
|
def onDraggedMouse(self, delta):
|
|
"""Update position from mouse delta sent by the other panel."""
|
|
self._mousePanningDelta = delta
|
|
# Signal from scrollbars had already synced the values here
|
|
self.adjustScrollBarsAuto()
|
|
|
|
# def viewportSizeHint(self):
|
|
# return self.viewport().rect().size()
|
|
|
|
# def changeEvent(self, event):
|
|
# if event.type() == QEvent.EnabledChange:
|
|
# print(f"{self} is now {'enabled' if self.isEnabled() else 'disabled'}")
|
|
|
|
|
|
class GraphicsViewViewer(QGraphicsView):
|
|
"""Re-Implementation a full-fledged GraphicsView but is a bit buggy."""
|
|
mouseDragged = pyqtSignal()
|
|
mouseWheeled = pyqtSignal(float, QPointF)
|
|
|
|
def __init__(self, parent, name=""):
|
|
super().__init__(parent)
|
|
self._parent = parent
|
|
self._app = QApplication
|
|
self._pixmap = QPixmap()
|
|
self._scaledpixmap = None
|
|
self._rect = QRectF()
|
|
self._lastMouseClickPoint = QPointF()
|
|
self._mousePanningDelta = QPointF()
|
|
self._scaleFactor = 1.3
|
|
self.zoomInFactor = self._scaleFactor
|
|
self.zoomOutFactor = 1.0 / self._scaleFactor
|
|
self.current_scale = 1.0
|
|
self._drag = False
|
|
self._dragConnection = None
|
|
self._wheelConnection = None
|
|
self._instance_name = name
|
|
self.wantScrollBars = True
|
|
self.bestFit = True
|
|
self.controller = None
|
|
self._centerPoint = QPointF()
|
|
self.centerOn(self._centerPoint)
|
|
self.other_viewer = None
|
|
# specific to this class
|
|
self._scene = QGraphicsScene()
|
|
self._scene.setBackgroundBrush(Qt.black)
|
|
self._item = QGraphicsPixmapItem()
|
|
self.setScene(self._scene)
|
|
self._scene.addItem(self._item)
|
|
self.setDragMode(QGraphicsView.DragMode.ScrollHandDrag)
|
|
# self.matrix = QTransform()
|
|
self._horizontalScrollBar = self.horizontalScrollBar()
|
|
self._verticalScrollBar = self.verticalScrollBar()
|
|
self.ignore_signal = False
|
|
|
|
if self.wantScrollBars:
|
|
self.toggleScrollBars()
|
|
else:
|
|
self.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
|
|
self.setResizeAnchor(QGraphicsView.AnchorViewCenter)
|
|
self.setAlignment(Qt.AlignCenter)
|
|
self.setViewportUpdateMode(QGraphicsView.FullViewportUpdate)
|
|
self.setMouseTracking(True)
|
|
|
|
def connectMouseSignals(self):
|
|
if not self._dragConnection:
|
|
self._dragConnection = self.mouseDragged.connect(
|
|
self.controller.syncCenters)
|
|
if not self._wheelConnection:
|
|
self._wheelConnection = self.mouseWheeled.connect(
|
|
self.controller.onMouseWheel)
|
|
|
|
def disconnectMouseSignals(self):
|
|
if self._dragConnection:
|
|
self.mouseDragged.disconnect()
|
|
self._dragConnection = None
|
|
if self._wheelConnection:
|
|
self.mouseWheeled.disconnect()
|
|
self._wheelConnection = None
|
|
|
|
def connectScrollBars(self):
|
|
"""Only call once controller is connected."""
|
|
# Cyclic connections are handled by Qt
|
|
self._verticalScrollBar.valueChanged.connect(
|
|
self.controller.onVScrollBarChanged, Qt.UniqueConnection)
|
|
self._horizontalScrollBar.valueChanged.connect(
|
|
self.controller.onHScrollBarChanged, Qt.UniqueConnection)
|
|
|
|
def toggleScrollBars(self, forceOn=False):
|
|
if not self.wantScrollBars:
|
|
return
|
|
# Ensure that it's off on the first run
|
|
if self.horizontalScrollBarPolicy() == Qt.ScrollBarAsNeeded:
|
|
if forceOn:
|
|
return
|
|
self.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
self.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
else:
|
|
self.setVerticalScrollBarPolicy(Qt.ScrollBarAsNeeded)
|
|
self.setHorizontalScrollBarPolicy(Qt.ScrollBarAsNeeded)
|
|
|
|
def mousePressEvent(self, event):
|
|
if self.bestFit:
|
|
event.ignore()
|
|
return
|
|
if event.button() == Qt.LeftButton:
|
|
self._drag = True
|
|
else:
|
|
self._drag = False
|
|
event.ignore()
|
|
return
|
|
self._lastMouseClickPoint = event.pos()
|
|
self._app.setOverrideCursor(Qt.ClosedHandCursor)
|
|
self.setMouseTracking(True)
|
|
# We need to propagate to scrollbars, so we send back up
|
|
super().mousePressEvent(event)
|
|
|
|
def mouseReleaseEvent(self, event):
|
|
if self.bestFit:
|
|
event.ignore()
|
|
return
|
|
if event.button() == Qt.LeftButton:
|
|
self._drag = False
|
|
self._app.restoreOverrideCursor()
|
|
self.setMouseTracking(False)
|
|
self.updateCenterPoint()
|
|
super().mouseReleaseEvent(event)
|
|
|
|
def mouseMoveEvent(self, event):
|
|
if self.bestFit:
|
|
event.ignore()
|
|
return
|
|
if self._drag:
|
|
self._lastMouseClickPoint = event.pos()
|
|
# We can simply rely on the scrollbar updating each other here
|
|
# self.mouseDragged.emit()
|
|
self.updateCenterPoint()
|
|
super().mouseMoveEvent(event)
|
|
|
|
def updateCenterPoint(self):
|
|
self._centerPoint = self.mapToScene(self.rect().center())
|
|
|
|
def wheelEvent(self, event):
|
|
if self.bestFit or MIN_SCALE > self.current_scale > MAX_SCALE or not self.controller.same_dimensions:
|
|
event.ignore()
|
|
return
|
|
pointBeforeScale = QPointF(self.mapToScene(self.mapFromGlobal(QCursor.pos())))
|
|
# Get the original screen centerpoint
|
|
screenCenter = QPointF(self.mapToScene(self.rect().center()))
|
|
if event.angleDelta().y() > 0:
|
|
factor = self.zoomInFactor
|
|
else:
|
|
factor = self.zoomOutFactor
|
|
# Avoid scrollbars conflict:
|
|
self.other_viewer.ignore_signal = True
|
|
self.scaleBy(factor)
|
|
pointAfterScale = QPointF(self.mapToScene(self.mapFromGlobal(QCursor.pos())))
|
|
# Get the offset of how the screen moved
|
|
offset = pointBeforeScale - pointAfterScale
|
|
# Adjust to the new center for correct zooming
|
|
newCenter = screenCenter + offset
|
|
self.setCenter(newCenter)
|
|
self.mouseWheeled.emit(factor, newCenter)
|
|
self.other_viewer.ignore_signal = False
|
|
|
|
def setImage(self, pixmap):
|
|
if pixmap.isNull():
|
|
self.ignore_signal = True
|
|
elif self.ignore_signal:
|
|
self.ignore_signal = False
|
|
self._pixmap = pixmap
|
|
self._item.setPixmap(pixmap)
|
|
self.translate(1, 1)
|
|
|
|
def centerViewAndUpdate(self):
|
|
# Called from the base controller for Normal Size
|
|
pass
|
|
|
|
def setCenter(self, point):
|
|
self._centerPoint = point
|
|
self.centerOn(self._centerPoint)
|
|
|
|
def resetCenter(self):
|
|
""" Resets origin """
|
|
self._mousePanningDelta = QPointF()
|
|
self.current_scale = 1.0
|
|
|
|
def setNewCenter(self, position):
|
|
self._centerPoint = position
|
|
self.centerOn(self._centerPoint)
|
|
|
|
def setCachedPixmap(self):
|
|
"""In case we have changed the cached pixmap, reset it."""
|
|
self._item.setPixmap(self._pixmap)
|
|
self._item.update()
|
|
|
|
def scaleAt(self, scale):
|
|
# self.current_scale = scale
|
|
if scale == 1.0:
|
|
self.resetScale()
|
|
# self.setTransform( QTransform() )
|
|
self.scale(scale, scale)
|
|
|
|
def getScale(self):
|
|
return self.transform().m22()
|
|
|
|
def scaleBy(self, factor):
|
|
self.current_scale *= factor
|
|
super().scale(factor, factor)
|
|
|
|
def resetScale(self):
|
|
# self.setTransform( QTransform() )
|
|
self.resetTransform() # probably same as above
|
|
self.setCenter(self.scene().sceneRect().center())
|
|
|
|
def fitScale(self):
|
|
self.bestFit = True
|
|
super().fitInView(self._scene.sceneRect(), Qt.KeepAspectRatio)
|
|
self.setNewCenter(self._scene.sceneRect().center())
|
|
|
|
@pyqtSlot()
|
|
def scaleToNormalSize(self):
|
|
"""Called when the pixmap is set back to original size."""
|
|
self.bestFit = False
|
|
self.scaleAt(1.0)
|
|
self.toggleScrollBars(True)
|
|
self.update()
|
|
|
|
def adjustScrollBarsScaled(self, delta):
|
|
"""After scaling with the mouse, update relative to mouse position."""
|
|
self._horizontalScrollBar.setValue(
|
|
self._horizontalScrollBar.value() + delta.x())
|
|
self._verticalScrollBar.setValue(
|
|
self._verticalScrollBar.value() + delta.y())
|
|
|
|
def sizeHint(self):
|
|
return self.viewport().rect().size()
|
|
|
|
def adjustScrollBarsFactor(self, factor):
|
|
"""After scaling, no mouse position, default to center."""
|
|
self._horizontalScrollBar.setValue(
|
|
int(factor * self._horizontalScrollBar.value()
|
|
+ ((factor - 1) * self._horizontalScrollBar.pageStep() / 2)))
|
|
self._verticalScrollBar.setValue(
|
|
int(factor * self._verticalScrollBar.value()
|
|
+ ((factor - 1) * self._verticalScrollBar.pageStep() / 2)))
|
|
|
|
def adjustScrollBarsAuto(self):
|
|
"""After panning, update accordingly."""
|
|
self.horizontalScrollBar().setValue(
|
|
self.horizontalScrollBar().value() - self._mousePanningDelta.x())
|
|
self.verticalScrollBar().setValue(
|
|
self.verticalScrollBar().value() - self._mousePanningDelta.y())
|