Source code for pyqtgraph.graphicsItems.ImageItem

import os
import pathlib
import warnings
from collections.abc import Callable

import numpy as np
import numpy.typing as npt

from .. import colormap
from .. import debug as debug
from .. import functions as fn
from .. import functions_qimage
from .. import getConfigOption
from ..Point import Point
from ..Qt import QtCore, QtGui, QtWidgets
from ..util.cupy_helper import getCupy
from .GraphicsObject import GraphicsObject

translate = QtCore.QCoreApplication.translate

__all__ = ['ImageItem']


[docs] class ImageItem(GraphicsObject): """ Graphics object used to display image data. ImageItem can render images with 1, 3 or 4 channels, use lookup tables to apply false colors to images, and users can either set levels limits, or rely on the auto-sampling. Performance can vary wildly based on the attributes of the inputs provided, see :ref:`performance <ImageItem_performance>` for guidance if performance is an important factor. There is optional `numba` and `cupy` support. **Bases:** :class:`pyqtgraph.GraphicsObject` Parameters ---------- image : np.ndarray or None, default None Image data. **kargs : dict, optional Arguments directed to `setImage` and `setOpts`, refer to each method for documentation for possible arguments. Signals ------- sigImageChanged: :class:`Signal` Emitted when the image is changed. sigRemoveRequested: :class:`Signal` Emitted when there is a request to remove the image. Signal emits the instance of the :class:`~pyqtgraph.ImageItem` whose removal is requested. See Also -------- setImage : For descriptions of available keyword arguments. setOpts : For information on supported formats. """ sigImageChanged = QtCore.Signal() sigRemoveRequested = QtCore.Signal(object) def __init__(self, image: np.ndarray | None=None, **kargs): super().__init__() self.menu = None self.image = None ## original image data self.qimage = None ## rendered image for display self.paintMode = None self.levels = None ## [min, max] or [[redMin, redMax], ...] self.lut = None self.autoDownsample = False self._nanPolicy = 'propagate' self._colorMap = None # This is only set if a color map is assigned directly self._lastDownsample = (1, 1) self._processingBuffer = None self._displayBuffer = None self._renderRequired = True self._unrenderable = False self._xp = None # either numpy or cupy, to match the image data self._defferedLevels = None self._imageHasNans = None # None : not yet known self._imageNanLocations = None self._defaultAutoLevels = True self.axisOrder = getConfigOption('imageAxisOrder') self._dataTransform = self._inverseDataTransform = QtGui.QTransform() self._update_data_transforms( self.axisOrder ) # install initial transforms self.drawKernel = None self.border = None self.removable = False if image is not None: self.setImage(image, **kargs) else: self.setOpts(**kargs)
[docs] def setCompositionMode(self, mode: QtGui.QPainter.CompositionMode): """ Change the composition mode of the item, useful when overlaying multiple items. Parameters ---------- mode : :class:`QPainter.CompositionMode <QPainter.CompositionMode>` Composition of the item, often used when overlaying items. Common options include: * `QPainter.CompositionMode.CompositionMode_SourceOver` Image replaces the background if it is opaque. Otherwise, it uses the alpha channel to blend the image with the background, default. * `QPainter.CompositionMode.CompositionMode_Overlay` Image color is mixed with the background color to reflect the lightness or darkness of the background. * `QPainter.CompositionMode.CompositionMode_Plus` Both the alpha and color of the image and background pixels are added together. * `QPainter.CompositionMode.CompositionMode_Plus` The output is the image color multiplied by the background. See :class:`QPainter.CompositionMode <QPainter.CompositionMode>` in the Qt documentation for more options and details. See Also -------- :class:`QPainter.CompositionMode <QPainter.CompositionMode>` : Details all the possible composition mode options accepted. """ self.paintMode = mode self.update()
[docs] def setBorder(self, b): """ Define the color of the border drawn around the image. Parameters ---------- b : color_like Accepts all arguments supported by :func:`~pyqtgraph.mkPen`. """ self.border = fn.mkPen(b) self.update()
def width(self) -> int | None: if self.image is None: return None axis = 0 if self.axisOrder == 'col-major' else 1 return self.image.shape[axis] def height(self) -> int | None: if self.image is None: return None axis = 1 if self.axisOrder == 'col-major' else 0 return self.image.shape[axis] def channels(self) -> int | None: if self.image is None: return None return self.image.shape[2] if self.image.ndim == 3 else 1 def boundingRect(self) -> QtCore.QRectF: if self.image is None: return QtCore.QRectF(0., 0., 0., 0.) if (width := self.width()) is None: width = 0. if (height := self.height()) is None: height = 0. return QtCore.QRectF(0., 0., float(width), float(height))
[docs] def setAutoLevels(self, bState: bool): """ Controls whether automatic image scaling takes place for this ImageItem, if not otherwise overridden by ``autoLevels`` or ``levels`` keyword arguments in a call to :func:`~pyqtgraph.ImageItem.setImage`. """ self._defaultAutoLevels = bState
[docs] def setLevels(self, levels: npt.ArrayLike | None, update: bool=True): """ Set image scaling levels. Calling this method, even with ``levels=None`` will disable auto leveling which is equivalent to :meth:`setImage` with ``autoLevels=False``. Parameters ---------- levels : array_like or None Sets the numerical values that correspond to the limits of the color range. * ``[blackLevel, whiteLevel]`` sets black and white levels for monochrome data and can be used with a lookup table. * ``[[minR, maxR], [minG, maxG], [minB, maxB]]`` sets individual scaling for RGB values. Not compatible with lookup tables. * ``None`` Disables the application of levels, but setting to ``None`` prevents the auto-levels mechanism from sampling the image. Not compatible with images that use floating point dtypes. update : bool, default True Update the image immediately to reflect the new levels. See Also -------- pyqtgraph.functions.makeARGB For more details on how levels are applied. """ if self._xp is None: self.levels = levels self._defferedLevels = levels return if levels is not None: levels = self._xp.asarray(levels) self.levels = levels if update: self.updateImage()
[docs] def getLevels(self) -> np.ndarray | None: """ Return the array representing the current level settings. See :meth:`setLevels`. When `autoLevels` is active, the format is ``[blackLevel, whiteLevel]``. Returns ------- np.ndarray or None The value that the levels are set to. """ return self.levels
[docs] def setColorMap(self, colorMap: colormap.ColorMap | str): """ Set a color map for false color display of a monochrome image. Parameters ---------- colorMap : :class:`~pyqtgraph.ColorMap` or `str` A string argument will be passed to :func:`colormap.get() <pyqtgraph.colormap.get>`. Raises ------ TypeError Raised when `colorMap` is not of type `str` or :class:`~pyqtgraph.ColorMap`. """ if isinstance(colorMap, colormap.ColorMap): self._colorMap = colorMap elif isinstance(colorMap, str): self._colorMap = colormap.get(colorMap) else: raise TypeError("'colorMap' argument must be ColorMap or string") self.setLookupTable( self._colorMap.getLookupTable(nPts=256) )
[docs] def getColorMap(self) -> colormap.ColorMap | None: """ Retrieve the :class:`~pyqtgraph.ColorMap` object currently used. Returns ------- ColorMap or None The assigned :class:`~pyqtgraph.ColorMap`, or `None` if not available. """ return self._colorMap
[docs] def setLookupTable(self, lut: npt.ArrayLike | Callable, update: bool=True): """ Set lookup table `lut` to use for false color display of a monochrome image. Ordinarily, this table is supplied by a :class:`~pyqtgraph.HistogramLUTItem`, :class:`~pyqtgraph.GradientEditorItem` or :class:`~pyqtgraph.ColorBarItem`. Parameters ---------- lut : array_like or callable If `lut` is an np.ndarray, ensure the dtype is `np.uint8`. Alternatively can be a callable that accepts the current image as an argument and returns the lookup table to use. Support for callable will be removed in a future version of pyqtgraph. update : bool, default True Update the intermediate image. See Also -------- :func:`pyqtgraph.functions.makeARGB` See this function for more information on how this is used. :meth:`ColorMap.getLookupTable <pyqtgraph.ColorMap.getLookupTable>` Can construct a lookup table from a :class:`~pyqtgraph.ColorMap` object. Notes ----- For performance reasons, if not passing a callable, every effort should be made to keep the number of entries to `<= 256`. """ if lut is not self.lut: if self._xp is not None: lut = self._ensure_proper_substrate(lut, self._xp) self.lut = lut if update: self.updateImage()
@staticmethod def _ensure_proper_substrate(data: Callable | npt.ArrayLike, substrate) -> np.ndarray: if data is None or isinstance(data, (Callable, substrate.ndarray)): return data cupy = getCupy() if substrate == cupy and not isinstance(data, cupy.ndarray): data = cupy.asarray(data) elif substrate == np: if cupy is not None and isinstance(data, cupy.ndarray): data = data.get() else: data = np.asarray(data) return data
[docs] def setAutoDownsample(self, active: bool=True): """ Control automatic downsampling for this ImageItem. Parameters ---------- active : bool, default True If `active` is ``True``, the image is automatically downsampled to match the screen resolution. This improves performance for large images and reduces aliasing. If `autoDownsample` is not specified, then ImageItem will choose whether to downsample the image based on its size. ``False`` disables automatic downsampling. """ self.autoDownsample = active self._renderRequired = True self.update()
[docs] def nanPolicy(self) -> str: """ Retrieve the string representing the current NaN policy. See :meth:setNanPolicy. Returns ------- { 'propagate', 'omit' } The NaN policy that this ImageItem uses during downsampling. """ return self._nanPolicy
[docs] def setNanPolicy(self, nanPolicy: str): """ Control how NaN values are handled during downsampling for this ImageItem. Parameters ---------- nanPolicy : { 'propagate', 'omit' } If 'nanPolicy' is 'ignore', NaNs are automatically ignored during downsampling, at the expense of performance. If 'nanPolicy' is 'propagate', NaNs are kept during downsampling. Unless a different policy was specified, a new ImageItem is created with ``nanPolicy='propagate'``. """ if nanPolicy not in ['propagate', 'omit']: raise ValueError(f"{nanPolicy=} must be one of {'propagate', 'omit'}") self._nanPolicy = nanPolicy self._renderRequired = True self.update()
[docs] def setOpts(self, update: bool=True, **kwargs): """ Set display and processing options for this ImageItem. :class:`~pyqtgraph.ImageItem` and :meth:`setImage` support all keyword arguments listed here. Parameters ---------- update : bool, default True Controls if image immediately updates to reflect the new options. **kwargs : dict, optional Extra arguments that are directed to the respective methods. Expected keys include: * `autoDownsample` whose value is directed to :meth:`setAutoDownsample` * `nanPolicy` whose value is directed to :meth:`setNanPolicy` * `axisOrder`, which needs to be one of {'row-major', 'col-major'}, determines the relationship between the numpy axis and visual axis of the data. * `border`, whose value is directed to :meth:`setBorder` * `colorMap`, whose value is directed to :meth:`setColorMap` * `compositionMode`, whose value is directed to :meth:`setCompositionMode` * `levels` whose value is directed to :meth:`setLevels` * `lut`, whose value is directed to :meth:`setLookupTable` * `opacify` whose value is directed to :meth:`QGraphicsItem.setOpacity <QGraphicsItem.setOpacity>` * `rect` whose value is directed to :meth:`setRect` * `removable` boolean, determines if the context menu is available See Also -------- :meth:`setAutoDownsample` Accepts the value of ``kwargs['autoDownsample']``. :meth:`setAutoLevels` Accepts the value of ``kwargs['autoLevels']``. :meth:`setNanPolicy` Accepts the value of ``kwargs['nanPolicy']``. :meth:`setBorder` Accepts the value of ``kwargs['border']``. :meth:`setColorMap` Accepts the value of ``kwargs['colorMap']``. :meth:`setCompositionMode` Accepts the value of ``kwargs['compositionMode']``. :meth:`setImage` Accepts the value of ``kwargs['image']``. :meth:`setLevels` Accepts the value of ``kwargs['levels']``. :meth:`setLookupTable` Accepts the value of ``kwargs['lut']``. :meth:`QGraphicsItem.setOpacity <QGraphicsItem.setOpacity>` Accepts the value of ``kwargs['opacity']``. :meth:`setRect` Accepts the value of ``kwargs['rect']``. """ if 'axisOrder' in kwargs: val = kwargs['axisOrder'] if val not in ('row-major', 'col-major'): raise ValueError("axisOrder must be either 'row-major' or 'col-major'") self.axisOrder = val self._update_data_transforms(self.axisOrder) # update cached transforms if 'colorMap' in kwargs: self.setColorMap(kwargs['colorMap']) if 'lut' in kwargs: self.setLookupTable(kwargs['lut'], update=update) if 'levels' in kwargs: self.setLevels(kwargs['levels'], update=update) #if 'clipLevel' in kargs: #self.setClipLevel(kargs['clipLevel']) if 'opacity' in kwargs: self.setOpacity(kwargs['opacity']) if 'compositionMode' in kwargs: self.setCompositionMode(kwargs['compositionMode']) if 'border' in kwargs: self.setBorder(kwargs['border']) if 'removable' in kwargs: self.removable = kwargs['removable'] self.menu = None if 'autoDownsample' in kwargs: self.setAutoDownsample(kwargs['autoDownsample']) if 'autoLevels' in kwargs: self.setAutoLevels(kwargs['autoLevels']) if 'nanPolicy' in kwargs: self.setNanPolicy(kwargs['nanPolicy']) if 'rect' in kwargs: self.setRect(kwargs['rect']) if update: self.update()
[docs] def setRect(self, *args): """ Set view rectangle for the :class:`~pyqtgraph.ImageItem` to occupy. In addition to accepting a :class:`QRectF`, you can pass the numerical values representing the `x, y, w, h`, where `x, y` represent the x, y coordinates of the top left corner, and `w` and `h` represent the width and height respectively. Parameters ---------- *args : tuple Contains one of :class:`QRectF`, :class:`QRect`, or arguments that can be used to construct :class:`QRectF`. See Also -------- :class:`QRectF` : See constructor methods for allowable `*args`. Notes ----- This method cannot be used before an image is assigned. See the :ref:`examples <ImageItem_examples>` for how to manually set transformations. """ if not args: # reset scaling and rotation when called without argument self.resetTransform() return if isinstance(args[0], (QtCore.QRectF, QtCore.QRect)): rect = args[0] # use QRectF or QRect directly else: if hasattr(args[0],'__len__'): args = args[0] # promote tuple or list of values # QRectF(x,y,w,h), but also accepts other initializers rect = QtCore.QRectF( *args ) tr = QtGui.QTransform() tr.translate(rect.left(), rect.top()) if (width := self.width()) is None: width = 1. if (height := self.height()) is None: height = 1. tr.scale(rect.width() / width, rect.height() / height) self.setTransform(tr)
[docs] def clear(self): """ Clear the assigned image. """ self.image = None self.prepareGeometryChange() self.informViewBoundsChanged() self.update()
def _buildQImageBuffer(self, shape: tuple[int, int, int]): self._displayBuffer = np.empty(shape[:2] + (4,), dtype=np.ubyte) if self._xp == getCupy(): self._processingBuffer = self._xp.empty( shape[:2] + (4,), dtype=self._xp.ubyte ) else: self._processingBuffer = self._displayBuffer self.qimage = None
[docs] def setImage( self, image: np.ndarray | None=None, autoLevels: bool | None=None, levelSamples: int = 65536, **kwargs ): """ Update the image displayed by this ImageItem. All keywords supported by :meth:`setOpts` are also allowed here. Parameters ---------- image : np.ndarray or None, default None Image data given as NumPy array with an integer or floating point dtype of any bit depth. A 2-dimensional array describes single-valued (monochromatic) data. A 3-dimensional array is used to give individual color components. The third dimension must be of length 3 (RGB) or 4 (RGBA). ``np.nan`` values are treated as transparent pixels. autoLevels : bool or None, default None If ``True``, ImageItem will automatically select levels based on the maximum and minimum values encountered in the data. For performance reasons, this search sub-samples the images and may miss individual bright or dark points in the data set. If ``False``, the search will be omitted. If ``None``, the value set by :func:`~pyqtgraph.ImageItem.setOpts` is used, unless a ``levels`` keyword argument is given, which implies `False`. levelSamples : int, default 65536 Only used when ``autoLevels is None``. When determining minimum and maximum values, ImageItem only inspects a subset of pixels no larger than this number. Setting this larger than the total number of pixels considers all values. See `quickMinMax`. **kwargs : dict, optional Extra arguments that are passed to `setOpts`. See Also -------- quickMinMax See this method for how levelSamples value is utilized. :func:`pyqtgraph.functions.makeARGB` See this function for how image data is modified prior to rendering. Notes ----- For backward compatibility, image data is assumed to be in column-major order (column, row) by default. However, most data is stored in row-major order (row, column). It can either be transposed before assignment .. code-block:: python imageitem.setImage(imagedata.T) or the interpretation of the data can be changed locally through the `axisOrder` keyword or by changing the `imageAxisOrder` :ref:`global configuration option <apiref_config>`. """ profile = debug.Profiler() gotNewData = False if image is None: if self.image is None: return else: old_xp = self._xp cp = getCupy() self._xp = cp.get_array_module(image) if cp else np gotNewData = True processingSubstrateChanged = old_xp != self._xp if processingSubstrateChanged: self._processingBuffer = None shapeChanged = ( processingSubstrateChanged or self.image is None or image.shape != self.image.shape ) image = image.view() self.image = image self._imageHasNans = None self._imageNanLocations = None if 'autoDownsample' not in kwargs and ( self.image.shape[0] > 2**15-1 or self.image.shape[1] > 2**15-1 ): kwargs['autoDownsample'] = True if shapeChanged: self.prepareGeometryChange() self.informViewBoundsChanged() profile() if autoLevels is None: autoLevels = False if 'levels' in kwargs else self._defaultAutoLevels if autoLevels: mn, mx = self.quickMinMax( targetSize=levelSamples ) # mn and mx can still be NaN if the data is all-NaN if mn == mx or self._xp.isnan(mn) or self._xp.isnan(mx): mn = 0 mx = 255 kwargs['levels'] = self._xp.asarray((mn,mx)) profile() self.setOpts(update=False, **kwargs) profile() self._renderRequired = True self.update() profile() if gotNewData: self.sigImageChanged.emit() if self._defferedLevels is not None: levels = self._defferedLevels self._defferedLevels = None self.setLevels((levels))
def _update_data_transforms(self, axisOrder: str='col-major'): """ Set up the transforms needed to map between input array and display. Parameters ---------- axisOrder : { 'col-major', 'row-major' } The axis order to update the data transformation to. """ self._dataTransform = QtGui.QTransform() self._inverseDataTransform = QtGui.QTransform() if self.axisOrder == 'row-major': # transpose both self._dataTransform.scale(1, -1) self._dataTransform.rotate(-90) self._inverseDataTransform.scale(1, -1) self._inverseDataTransform.rotate(-90) def dataTransform(self): """ Get the transform mapping image array to local coordinate system. This transform corrects for the transposition that occurs when image data is interpreted in row-major order. :meta private: Returns ------- :class:`QTransform` The transform that is used for mapping. """ # Might eventually need to account for downsampling / clipping here # transforms are updated in setOpts call. return self._dataTransform def inverseDataTransform(self) -> QtGui.QTransform: """ Get the transform mapping local coordinate system to image array. :meta private: Returns ------- :class:`QTransform` The transform that is used for mapping. See Also -------- dataTransform See dataTransform() for more information. """ # transforms are updated in setOpts call. return self._inverseDataTransform def mapToData(self, obj): return self._inverseDataTransform.map(obj) def mapFromData(self, obj): return self._dataTransform.map(obj)
[docs] def quickMinMax(self, targetSize: int=1_000_000) -> tuple[float, float]: """ Estimate the min and max values of the image data by sub-sampling. Sampling is performed at regular strides chosen to evaluate a number of samples equal to or less than `targetSize`. Returns the estimated min and max values of the image data. Parameters ---------- targetSize : int, default 1_000_000 The number of pixels to downsample the image to. Returns ------- float, float Estimated minimum and maximum values of the image data. """ data = self.image if data is None: # image hasn't been set yet return 0., 0. targetSize = max(targetSize, 2) # keep at least 2 pixels while True: h, w = data.shape[:2] if h * w <= targetSize: break data = data[::2, ::] if h > w else data[::, ::2] return self._xp.nanmin(data), self._xp.nanmax(data)
def updateImage(self, *args, **kargs): defaults = { 'autoLevels': False, } | kargs return self.setImage(*args, **defaults) def render(self): # Convert data to QImage for display. self._unrenderable = True if self.image is None or self.image.size == 0: return # Request a lookup table if this image has only one channel if self.image.ndim == 2 or self.image.shape[2] == 1: self.lut = self._ensure_proper_substrate(self.lut, self._xp) if isinstance(self.lut, Callable): lut = self._ensure_proper_substrate( self.lut(self.image, 256), self._xp ) else: lut = self.lut else: lut = None if self._imageHasNans is None: # awkward, but fastest numpy native nan evaluation self._imageHasNans = ( self.image.dtype.kind == 'f' and self._xp.isnan(self.image.min()) ) self._imageNanLocations = None image = self.image if self.autoDownsample: xds, yds = self._computeDownsampleFactors() if xds is None: return axes = [1, 0] if self.axisOrder == 'row-major' else [0, 1] nan_policy = self._nanPolicy if self._imageHasNans else 'propagate' image = fn.downsample(image, xds, axis=axes[0], nanPolicy=nan_policy) image = fn.downsample(image, yds, axis=axes[1], nanPolicy=nan_policy) self._lastDownsample = (xds, yds) # changes in view transform cause changes in downsampling factors, # which invalidates any previously calculated nan locations self._imageNanLocations = None # Check if downsampling reduced the image size to zero due to inf values. if image.size == 0: return # Convert single-channel image to 2D array if image.ndim == 3 and image.shape[-1] == 1: image = image[..., 0] # Assume images are in column-major order for backward compatibility # (most images are in row-major order) if self.axisOrder == 'col-major': image = image.swapaxes(0, 1) levels = self.levels qimage = None if lut is not None and lut.dtype != self._xp.uint8: # try_make_image() assumes that lut is of type uint8. # It is considered a usage error if that is not the case. # However, the makeARGB() code-path has previously allowed such # a usage to work. Rather than fail outright, we delegate this # case to makeARGB(). warnings.warn( ("Using non-uint8 LUTs is an undocumented accidental feature and may " "be removed at some point in the future. Please open an issue if you " "instead believe this to be worthy of protected inclusion in " "pyqtgraph."), DeprecationWarning, stacklevel=2 ) elif not self._imageHasNans: qimage = functions_qimage.try_make_qimage(image, levels=levels, lut=lut) elif image.ndim in (2, 3): # float images with nans if self._imageNanLocations is None: # the number of nans is expected to be small nanmask = self._xp.isnan(image) if nanmask.ndim == 3: nanmask = nanmask.any(axis=2) self._imageNanLocations = nanmask.nonzero() qimage = functions_qimage.try_make_qimage( image, levels=levels, lut=lut, transparentLocations=self._imageNanLocations ) if qimage is not None: self._processingBuffer = None self._displayBuffer = None self.qimage = qimage self._renderRequired = False self._unrenderable = False return if ( self._processingBuffer is None or self._processingBuffer.shape[:2] != image.shape[:2] ): self._buildQImageBuffer(image.shape) fn.makeARGB(image, lut=lut, levels=levels, output=self._processingBuffer) if self._xp == getCupy(): self._processingBuffer.get(out=self._displayBuffer) self.qimage = fn.ndarray_to_qimage( self._displayBuffer, QtGui.QImage.Format.Format_ARGB32 ) self._renderRequired = False self._unrenderable = False def paint(self, painter, *args): profile = debug.Profiler() if self.image is None: return if self._renderRequired: self.render() if self._unrenderable: return profile('render QImage') if self.paintMode is not None: painter.setCompositionMode(self.paintMode) profile('set comp mode') shape = ( self.image.shape[:2] if self.axisOrder == 'col-major' else self.image.shape[:2][::-1] ) painter.drawImage(QtCore.QRectF(0,0,*shape), self.qimage) profile('p.drawImage') if self.border is not None: painter.setPen(self.border) painter.drawRect(self.boundingRect())
[docs] def save(self, fileName: str | pathlib.Path, *args) -> None: """ Save this image to file. Note that this saves the visible image, after scale/color changes, not the original data. Parameters ---------- fileName : os.PathLike File path to save the image data to. *args : tuple Arguments that are passed to :meth:`QImage.save <QImage.save>`. See Also -------- :meth:`QImage.save <QImage.save>` : ``*args`` is relayed to this method. """ if self.qimage is None: return None if self._renderRequired: self.render() self.qimage.save(os.fsdecode(fileName), *args)
[docs] def getHistogram( self, bins: str | int='auto', step: str | np.generic='auto', perChannel: bool=False, targetImageSize: int=200, **kwargs ) -> list[tuple[np.ndarray, np.ndarray]] | tuple[np.ndarray, np.ndarray] | tuple[None, None]: """ Generate arrays containing the histogram values. Similar to :func:`numpy.histogram` Parameters ---------- bins : int or str, default 'auto' The `bins` argument and any extra keyword arguments are passed to :func:`numpy.histogram()`. If ``bins == 'auto'``, a bin number is automatically chosen based on the image characteristics. step : int or str, default 'auto' The `step` argument causes pixels to be skipped when computing the histogram to save time. If `step` is 'auto', then a step is chosen such that the analyzed data has dimensions approximating `targetImageSize` for each axis. perChannel : bool, default False If ``True``, then a histogram is computed for each channel, and the output is a list of the results. targetImageSize : int, default 200 This parameter is used if ``step == 'auto'``, If so, the `step` size is calculated by ``step = ceil(image.shape[0] / targetImageSize)``. **kwargs : dict, optional Dictionary of arguments passed to :func:`numpy.histogram()`. Returns ------- numpy.ndarray, numpy.ndarray or None, None or list of tuple of numpy.ndarray, numpy.ndarray Returns `x` and `y` arrays containing the histogram values for the current image. For an explanation of the return format, see :func:`numpy.histogram()`. Returns ``[(numpy.ndarray, numpy.ndarray),...]`` if ``perChannel=True``, one element per channel. Returns ``(None, None)`` is there is no image, or image size is 0. Warns ----- RuntimeWarning Emits when `targetHistogramSize` argument is passed in, which does nothing. See Also -------- numpy.histogram : Describes return format in greater detail. numpy.histogram_bin_edges: Details the different string values accepted as the `bins` parameter. """ if 'targetHistogramSize' in kwargs: warnings.warn( "'targetHistogramSize' option is not used", RuntimeWarning, stacklevel=2 ) # This method is also used when automatically computing levels. if self.image is None or self.image.size == 0: return None, None if step == 'auto': step = (max(1, int(self._xp.ceil(self.image.shape[0] / targetImageSize))), max(1, int(self._xp.ceil(self.image.shape[1] / targetImageSize)))) if self._xp.isscalar(step): step = (step, step) stepData = self.image[::step[0], ::step[1]] if isinstance(bins, str) and bins == 'auto': mn = self._xp.nanmin(stepData).item() mx = self._xp.nanmax(stepData).item() if mx == mn: # degenerate image, arange will fail mx += 1 if self._xp.isnan(mn) or self._xp.isnan(mx): # the data are all-nan return None, None if stepData.dtype.kind in "ui": # For integer data, we select the bins carefully to avoid aliasing step = int(self._xp.ceil((mx - mn) / 500.)) bins = [] if step > 0.0: bins = self._xp.arange(mn, mx + 1.01 * step, step, dtype=int) else: # for float data, let numpy select the bins. bins = self._xp.linspace(mn, mx, 500) if len(bins) == 0: bins = self._xp.asarray((mn, mx)) kwargs['bins'] = bins cp = getCupy() if perChannel: hist = [] for i in range(stepData.shape[-1]): stepChan = stepData[..., i] stepChan = stepChan[self._xp.isfinite(stepChan)] h = self._xp.histogram(stepChan, **kwargs) if cp: hist.append((cp.asnumpy(h[1][:-1]), cp.asnumpy(h[0]))) else: hist.append((h[1][:-1], h[0])) return hist else: stepData = stepData[self._xp.isfinite(stepData)] hist = self._xp.histogram(stepData, **kwargs) if cp: return cp.asnumpy(hist[1][:-1]), cp.asnumpy(hist[0]) else: return hist[1][:-1], hist[0]
[docs] def setPxMode(self, b: bool): """ Set whether item ignores transformations and draws directly to screen pixels. Parameters ---------- b : bool If ``True``, the item will not inherit any scale or rotation transformations from its parent items, but its position will be transformed as usual. See Also -------- :class:`QGraphicsItem.GraphicsItemFlag <QGraphicsItem.GraphicsItemFlag>` : Read the description of `ItemIgnoresTransformations` for more information. """ self.setFlag( QtWidgets.QGraphicsItem.GraphicsItemFlag.ItemIgnoresTransformations, b )
def setScaledMode(self): self.setPxMode(False) def getPixmap(self) -> QtGui.QPixmap | None: if self._renderRequired: self.render() if self._unrenderable: return None if self.qimage is None: return QtGui.QPixmap() return QtGui.QPixmap.fromImage(self.qimage)
[docs] def pixelSize(self) -> tuple[float, float]: """ Get the `x` and `y` size of each pixel in the view coordinate system. Returns ------- float, float The `x` and `y` size of each pixel in scene space. """ br = self.sceneBoundingRect() if self.image is None: return 1.,1. if (width := self.width()) is None: width = 0. if (height := self.height()) is None: height = 0. return br.width() / width, br.height() / height
def viewTransformChanged(self): if self.autoDownsample: xds, yds = self._computeDownsampleFactors() if xds is None: self._renderRequired = True self._unrenderable = True return if (xds, yds) != self._lastDownsample: self._renderRequired = True self.update() def _computeDownsampleFactors(self) -> tuple[int, int]: # reduce dimensions of image based on screen resolution o = self.mapToDevice(QtCore.QPointF(0, 0)) x = self.mapToDevice(QtCore.QPointF(1, 0)) y = self.mapToDevice(QtCore.QPointF(0, 1)) # scene may not be available yet if o is None: return 1, 1 w = Point(x - o).length() h = Point(y - o).length() if w == 0 or h == 0: return 1, 1 return max(1, int(1.0 / w)), max(1, int(1.0 / h)) def mouseDragEvent(self, ev): if ev.button() != QtCore.Qt.MouseButton.LeftButton: ev.ignore() return elif self.drawKernel is not None: ev.accept() self.drawAt(ev.pos(), ev) def mouseClickEvent(self, ev): if ( ev.button() == QtCore.Qt.MouseButton.RightButton and self.raiseContextMenu(ev) ): ev.accept() if self.drawKernel is not None and ev.button() == QtCore.Qt.MouseButton.LeftButton: self.drawAt(ev.pos(), ev) def raiseContextMenu(self, ev): menu = self.getMenu() if menu is None: return False if self.scene() is None: warnings.warn( ( "Attempting to raise a context menu with the GraphicsScene has " "not been set. Returning None" ), RuntimeWarning, stacklevel=2 ) return None menu = self.scene().addParentContextMenus(self, menu, ev) pos = ev.screenPos() menu.popup(QtCore.QPoint(int(pos.x()), int(pos.y()))) return True def getMenu(self): if self.menu is None: if not self.removable: return None self.menu = QtWidgets.QMenu() self.menu.setTitle(translate("ImageItem", "Image")) remAct = QtGui.QAction(translate("ImageItem", "Remove image"), self.menu) remAct.triggered.connect(self.removeClicked) self.menu.addAction(remAct) self.menu.remAct = remAct return self.menu def hoverEvent(self, ev): if not ev.isExit(): if self.drawKernel is not None and ev.acceptDrags( QtCore.Qt.MouseButton.LeftButton ): # we don't use the click, but we also don't want anyone else to use it ev.acceptClicks(QtCore.Qt.MouseButton.LeftButton) ev.acceptClicks(QtCore.Qt.MouseButton.RightButton) elif self.removable: # accept context menu clicks ev.acceptClicks(QtCore.Qt.MouseButton.RightButton) def tabletEvent(self, ev): pass def drawAt(self, pos, ev=None): if self.axisOrder == "col-major": pos = [int(pos.x()), int(pos.y())] else: pos = [int(pos.y()), int(pos.x())] dk = self.drawKernel kc = self.drawKernelCenter sx = [0,dk.shape[0]] sy = [0,dk.shape[1]] tx = [pos[0] - kc[0], pos[0] - kc[0]+ dk.shape[0]] ty = [pos[1] - kc[1], pos[1] - kc[1]+ dk.shape[1]] for i in [0,1]: dx1 = -min(0, tx[i]) dx2 = min(0, self.image.shape[0]-tx[i]) tx[i] += dx1+dx2 sx[i] += dx1+dx2 dy1 = -min(0, ty[i]) dy2 = min(0, self.image.shape[1]-ty[i]) ty[i] += dy1+dy2 sy[i] += dy1+dy2 ts = (slice(tx[0],tx[1]), slice(ty[0],ty[1])) ss = (slice(sx[0],sx[1]), slice(sy[0],sy[1])) mask = self.drawMask src = dk if isinstance(self.drawMode, Callable): self.drawMode(dk, self.image, mask, ss, ts, ev) else: src = src[ss] if self.drawMode == 'set': if mask is not None: mask = mask[ss] self.image[ts] = self.image[ts] * (1-mask) + src * mask else: self.image[ts] = src elif self.drawMode == 'add': self.image[ts] += src else: raise ValueError(f"Unknown draw mode '{self.drawMode}'") self.updateImage() def setDrawKernel(self, kernel=None, mask=None, center=(0,0), mode='set'): self.drawKernel = kernel self.drawKernelCenter = center self.drawMode = mode self.drawMask = mask def removeClicked(self): ## Send remove event only after we have exited the menu event handler self.removeTimer = QtCore.QTimer() self.removeTimer.timeout.connect(self.emitRemoveRequested) self.removeTimer.start(0) def emitRemoveRequested(self): self.removeTimer.timeout.disconnect(self.emitRemoveRequested) self.sigRemoveRequested.emit(self)