Source code for edifice.base_components.image_aspect

from __future__ import annotations

import typing as tp

from edifice.qt import QT_VERSION

if QT_VERSION == "PyQt6" and not tp.TYPE_CHECKING:
    from PyQt6 import QtCore, QtGui, QtWidgets
else:
    from PySide6 import QtCore, QtGui, QtWidgets

from .base_components import CommandType, Element, QtWidgetElement, _image_descriptor_to_pixmap, _WidgetTree


class _ScaledLabel(QtWidgets.QLabel):
    """
    https://stackoverflow.com/questions/72188903/pyside6-how-do-i-resize-a-qlabel-without-loosing-the-size-aspect-ratio
    """

    def __init__(self, *args, **kwargs):
        QtWidgets.QLabel.__init__(self)
        self._pixmap: QtGui.QPixmap | None = None
        self._aspect_ratio_mode: QtCore.Qt.AspectRatioMode | None = None
        self._rescale()

    def resizeEvent(self, event):
        self._rescale()

    def _rescale(self):
        if self._pixmap is not None:
            match self._aspect_ratio_mode:
                case None:
                    self.setPixmap(self._pixmap)
                case aspect_ratio_mode:
                    self.setPixmap(
                        self._pixmap.scaled(
                            self.frameSize(),
                            aspect_ratio_mode,
                            QtCore.Qt.TransformationMode.SmoothTransformation,
                        ),
                    )

    def _setPixmap(self, pixmap: QtGui.QPixmap):
        if not pixmap:
            return
        self._pixmap = pixmap
        self._rescale()

    def _setAspectRatioMode(self, aspect_ratio_mode: QtCore.Qt.AspectRatioMode | None):
        self._aspect_ratio_mode = aspect_ratio_mode
        self._rescale()


[docs] class Image(QtWidgetElement): """Render an image. .. highlights:: - Underlying Qt Widget `QLabel <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QLabel.html>`_ .. rubric:: Props All **props** from :class:`QtWidgetElement`, plus: Args: src: One of: * A path to an image file. * A `QImage <https://doc.qt.io/qtforpython-6/PySide6/QtGui/QImage.html>`_. * A `QPixmap <https://doc.qt.io/qtforpython-6/PySide6/QtGui/QPixmap.html>`_. aspect_ratio_mode: The aspect ratio mode of the image. * :code:`None` for a fixed image which doesn’t scale. * An `AspectRatioMode <https://doc.qt.io/qtforpython-6/PySide6/QtCore/Qt.html#PySide6.QtCore.Qt.AspectRatioMode>`_ to specify how the image should scale. .. rubric:: Usage To render a 3-dimensional :code:`uint8` :code:`numpy` array as an image, see :class:`edifice.extra.numpy_image.NumpyImage`. """
[docs] def __init__( self, src: str | QtGui.QImage | QtGui.QPixmap, aspect_ratio_mode: None | QtCore.Qt.AspectRatioMode = None, **kwargs, ): super().__init__(**kwargs) self._register_props( { "src": src, "aspect_ratio_mode": aspect_ratio_mode, }, ) self.underlying: _ScaledLabel | None = None
def _initialize(self): self.underlying = _ScaledLabel() self.underlying.setObjectName(str(id(self))) def _qt_update_commands( self, widget_trees: dict[Element, _WidgetTree], newprops, ): if self.underlying is None: self._initialize() assert self.underlying is not None commands = super()._qt_update_commands_super(widget_trees, newprops, self.underlying, None) if "src" in newprops: commands.append(CommandType(self.underlying._setPixmap, _image_descriptor_to_pixmap(newprops.src))) if "aspect_ratio_mode" in newprops: commands.append(CommandType(self.underlying._setAspectRatioMode, newprops.aspect_ratio_mode)) return commands