Source code for silx.gui.plot.PlotInteraction

# /*##########################################################################
#
# Copyright (c) 2014-2023 European Synchrotron Radiation Facility
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
# THE SOFTWARE.
#
# ###########################################################################*/
"""Implementation of the interaction for the :class:`Plot`."""

from __future__ import annotations

__authors__ = ["T. Vincent"]
__license__ = "MIT"
__date__ = "15/02/2019"


import math
import numpy
import time
import weakref
from typing import NamedTuple, Optional

from silx.gui import qt
from .. import colors
from . import items
from .Interaction import (
    ClickOrDrag,
    LEFT_BTN,
    RIGHT_BTN,
    MIDDLE_BTN,
    State,
    StateMachine,
)
from .PlotEvents import (
    prepareCurveSignal,
    prepareDrawingSignal,
    prepareHoverSignal,
    prepareImageSignal,
    prepareMarkerSignal,
    prepareMouseSignal,
)

from .backends.BackendBase import (
    CURSOR_POINTING,
    CURSOR_SIZE_HOR,
    CURSOR_SIZE_VER,
    CURSOR_SIZE_ALL,
)

from ._utils import (
    FLOAT32_SAFE_MIN,
    FLOAT32_MINPOS,
    FLOAT32_SAFE_MAX,
    applyZoomToPlot,
    EnabledAxes,
)


# Base class ##################################################################


class _PlotInteraction(object):
    """Base class for interaction handler.

    It provides a weakref to the plot and methods to set/reset overlay.
    """

    def __init__(self, plot):
        """Init.

        :param plot: The plot to apply modifications to.
        """
        self._needReplot = False
        self._selectionAreas = set()
        self._plot = weakref.ref(plot)  # Avoid cyclic-ref

    @property
    def plot(self):
        plot = self._plot()
        assert plot is not None
        return plot

    def setSelectionArea(self, points, fill, color, name="", shape="polygon"):
        """Set a polygon selection area overlaid on the plot.
        Multiple simultaneous areas are supported through the name parameter.

        :param points: The 2D coordinates of the points of the polygon
        :type points: An iterable of (x, y) coordinates
        :param str fill: The fill mode: 'hatch', 'solid' or 'none'
        :param color: RGBA color to use or None to disable display
        :type color: list or tuple of 4 float in the range [0, 1]
        :param name: The key associated with this selection area
        :param str shape: Shape of the area in 'polygon', 'polylines'
        """
        assert shape in ("polygon", "polylines")

        if color is None:
            return

        points = numpy.asarray(points)

        # TODO Not very nice, but as is for now
        legend = "__SELECTION_AREA__" + name

        fill = fill != "none"  # TODO not very nice either

        greyed = colors.greyed(color)[0]
        if greyed < 0.5:
            color2 = "white"
        else:
            color2 = "black"

        self.plot.addShape(
            points[:, 0],
            points[:, 1],
            legend=legend,
            replace=False,
            shape=shape,
            fill=fill,
            color=color,
            gapcolor=color2,
            linestyle="--",
            overlay=True,
        )

        self._selectionAreas.add(legend)

    def resetSelectionArea(self):
        """Remove all selection areas set by setSelectionArea."""
        for legend in self._selectionAreas:
            self.plot.remove(legend, kind="item")
        self._selectionAreas = set()


# Zoom/Pan ####################################################################


class _PlotInteractionWithClickEvents(ClickOrDrag, _PlotInteraction):
    """:class:`ClickOrDrag` state machine emitting click and double click events.

    Base class for :class:`Pan` and :class:`Zoom`
    """

    _DOUBLE_CLICK_TIMEOUT = 0.4

    def click(self, x, y, btn):
        """Handle clicks by sending events

        :param int x: Mouse X position in pixels
        :param int y: Mouse Y position in pixels
        :param btn: Clicked mouse button
        """
        if btn == LEFT_BTN:
            lastClickTime, lastClickPos = self._lastClick

            # Signal mouse double clicked event first
            if (time.time() - lastClickTime) <= self._DOUBLE_CLICK_TIMEOUT:
                # Use position of first click
                eventDict = prepareMouseSignal(
                    "mouseDoubleClicked", "left", *lastClickPos
                )
                self.plot.notify(**eventDict)

                self._lastClick = 0.0, None
            else:
                # Signal mouse clicked event
                dataPos = self.plot.pixelToData(x, y)
                assert dataPos is not None
                eventDict = prepareMouseSignal(
                    "mouseClicked", "left", dataPos[0], dataPos[1], x, y
                )
                self.plot.notify(**eventDict)

                self._lastClick = time.time(), (dataPos[0], dataPos[1], x, y)

        elif btn == RIGHT_BTN:
            # Signal mouse clicked event
            dataPos = self.plot.pixelToData(x, y)
            assert dataPos is not None
            eventDict = prepareMouseSignal(
                "mouseClicked", "right", dataPos[0], dataPos[1], x, y
            )
            self.plot.notify(**eventDict)

    def __init__(self, plot, **kwargs):
        """Init.

        :param plot: The plot to apply modifications to.
        """
        self._lastClick = 0.0, None

        _PlotInteraction.__init__(self, plot)
        ClickOrDrag.__init__(self, **kwargs)


# Pan #########################################################################


[docs] class Pan(_PlotInteractionWithClickEvents): """Pan plot content and zoom on wheel state machine.""" def _pixelToData(self, x, y): xData, yData = self.plot.pixelToData(x, y) _, y2Data = self.plot.pixelToData(x, y, axis="right") return xData, yData, y2Data
[docs] def beginDrag(self, x, y, btn): self._previousDataPos = self._pixelToData(x, y)
[docs] def drag(self, x, y, btn): xData, yData, y2Data = self._pixelToData(x, y) lastX, lastY, lastY2 = self._previousDataPos xMin, xMax = self.plot.getXAxis().getLimits() yMin, yMax = self.plot.getYAxis().getLimits() y2Min, y2Max = self.plot.getYAxis(axis="right").getLimits() if self.plot.getXAxis()._isLogarithmic(): try: dx = math.log10(xData) - math.log10(lastX) newXMin = pow(10.0, (math.log10(xMin) - dx)) newXMax = pow(10.0, (math.log10(xMax) - dx)) except (ValueError, OverflowError): newXMin, newXMax = xMin, xMax # Makes sure both values stays in positive float32 range if newXMin < FLOAT32_MINPOS or newXMax > FLOAT32_SAFE_MAX: newXMin, newXMax = xMin, xMax else: dx = xData - lastX newXMin, newXMax = xMin - dx, xMax - dx # Makes sure both values stays in float32 range if newXMin < FLOAT32_SAFE_MIN or newXMax > FLOAT32_SAFE_MAX: newXMin, newXMax = xMin, xMax if self.plot.getYAxis()._isLogarithmic(): try: dy = math.log10(yData) - math.log10(lastY) newYMin = pow(10.0, math.log10(yMin) - dy) newYMax = pow(10.0, math.log10(yMax) - dy) dy2 = math.log10(y2Data) - math.log10(lastY2) newY2Min = pow(10.0, math.log10(y2Min) - dy2) newY2Max = pow(10.0, math.log10(y2Max) - dy2) except (ValueError, OverflowError): newYMin, newYMax = yMin, yMax newY2Min, newY2Max = y2Min, y2Max # Makes sure y and y2 stays in positive float32 range if ( newYMin < FLOAT32_MINPOS or newYMax > FLOAT32_SAFE_MAX or newY2Min < FLOAT32_MINPOS or newY2Max > FLOAT32_SAFE_MAX ): newYMin, newYMax = yMin, yMax newY2Min, newY2Max = y2Min, y2Max else: dy = yData - lastY dy2 = y2Data - lastY2 newYMin, newYMax = yMin - dy, yMax - dy newY2Min, newY2Max = y2Min - dy2, y2Max - dy2 # Makes sure y and y2 stays in float32 range if ( newYMin < FLOAT32_SAFE_MIN or newYMax > FLOAT32_SAFE_MAX or newY2Min < FLOAT32_SAFE_MIN or newY2Max > FLOAT32_SAFE_MAX ): newYMin, newYMax = yMin, yMax newY2Min, newY2Max = y2Min, y2Max self.plot.setLimits(newXMin, newXMax, newYMin, newYMax, newY2Min, newY2Max) self._previousDataPos = self._pixelToData(x, y)
[docs] def endDrag(self, startPos, endPos, btn): del self._previousDataPos
def cancel(self): pass
# Zoom ########################################################################
[docs] class AxesExtent(NamedTuple): xmin: float xmax: float ymin: float ymax: float y2min: float y2max: float
[docs] class Zoom(_PlotInteractionWithClickEvents): """Zoom-in/out state machine. Zoom-in on selected area, zoom-out on right click, and zoom on mouse wheel. """ SURFACE_THRESHOLD = 5 def __init__(self, plot, color): self.color = color self.enabledAxes = EnabledAxes() super(Zoom, self).__init__(plot) self.plot.getLimitsHistory().clear() def _getAxesExtent( self, x0: float, y0: float, x1: float, y1: float, enabledAxes: Optional[EnabledAxes] = None, ) -> AxesExtent: """Convert selection coordinates (pixels) to axes coordinates (data) This takes into account axes selected for zoom and aspect ratio. """ if enabledAxes is None: enabledAxes = self.enabledAxes y2_0, y2_1 = y0, y1 left, top, width, height = self.plot.getPlotBoundsInPixels() if not all(enabledAxes) and not self.plot.isKeepDataAspectRatio(): # Handle axes disabled for zoom if plot is not keeping aspec ratio if not enabledAxes.xaxis: x0, x1 = left, left + width if not enabledAxes.yaxis: y0, y1 = top, top + height if not enabledAxes.y2axis: y2_0, y2_1 = top, top + height if self.plot.isKeepDataAspectRatio() and height != 0 and width != 0: ratio = width / height xextent, yextent = math.fabs(x1 - x0), math.fabs(y1 - y0) if xextent != 0 and yextent != 0: if xextent / yextent > ratio: areaHeight = xextent / ratio center = 0.5 * (y0 + y1) y0 = center - numpy.sign(y1 - y0) * 0.5 * areaHeight y1 = center + numpy.sign(y1 - y0) * 0.5 * areaHeight else: areaWidth = yextent * ratio center = 0.5 * (x0 + x1) x0 = center - numpy.sign(x1 - x0) * 0.5 * areaWidth x1 = center + numpy.sign(x1 - x0) * 0.5 * areaWidth # Convert to data space x0, y0 = self.plot.pixelToData(x0, y0, check=False) x1, y1 = self.plot.pixelToData(x1, y1, check=False) y2_0 = self.plot.pixelToData(None, y2_0, axis="right", check=False)[1] y2_1 = self.plot.pixelToData(None, y2_1, axis="right", check=False)[1] return AxesExtent( min(x0, x1), max(x0, x1), min(y0, y1), max(y0, y1), min(y2_0, y2_1), max(y2_0, y2_1), )
[docs] def beginDrag(self, x, y, btn): dataPos = self.plot.pixelToData(x, y) assert dataPos is not None self.x0, self.y0 = x, y
[docs] def drag(self, x1, y1, btn): if self.color is None: return # Do not draw zoom area dataPos = self.plot.pixelToData(x1, y1) assert dataPos is not None if self.plot.isKeepDataAspectRatio() or not all(self.enabledAxes): # Patch enabledAxes to display the right Y axis area on the left Y axis # since the selection area is always displayed on the left Y axis isY2Visible = self.plot.getYAxis("right").isVisible() areaZoomEnabledAxes = EnabledAxes( self.enabledAxes.xaxis, self.enabledAxes.yaxis and (not isY2Visible or self.enabledAxes.y2axis), self.enabledAxes.y2axis, ) extents = self._getAxesExtent(self.x0, self.y0, x1, y1, areaZoomEnabledAxes) areaCorners = ( (extents.xmin, extents.ymin), (extents.xmax, extents.ymin), (extents.xmax, extents.ymax), (extents.xmin, extents.ymax), ) if self.color != "video inverted": areaColor = list(self.color) areaColor[3] *= 0.25 else: areaColor = [1.0, 1.0, 1.0, 1.0] self.setSelectionArea( areaCorners, fill="none", color=areaColor, name="zoomedArea" ) corners = ((self.x0, self.y0), (self.x0, y1), (x1, y1), (x1, self.y0)) corners = numpy.array( [self.plot.pixelToData(x, y, check=False) for (x, y) in corners] ) self.setSelectionArea(corners, fill="none", color=self.color)
def _zoom(self, x0, y0, x1, y1): """Zoom to the rectangle view x0,y0 x1,y1.""" # Store current zoom state in stack self.plot.getLimitsHistory().push() extents = self._getAxesExtent(x0, y0, x1, y1) self.plot.setLimits( extents.xmin, extents.xmax, extents.ymin, extents.ymax, extents.y2min, extents.y2max, )
[docs] def endDrag(self, startPos, endPos, btn): x0, y0 = startPos x1, y1 = endPos if abs(x0 - x1) * abs(y0 - y1) >= self.SURFACE_THRESHOLD: # Avoid empty zoom area self._zoom(x0, y0, x1, y1) self.resetSelectionArea()
def cancel(self): if isinstance(self.state, self.states["drag"]): self.resetSelectionArea()
# Select ######################################################################
[docs] class Select(StateMachine, _PlotInteraction): """Base class for drawing selection areas.""" def __init__(self, plot, parameters, states, state): """Init a state machine. :param plot: The plot to apply changes to. :param dict parameters: A dict of parameters such as color. :param dict states: The states of the state machine. :param str state: The name of the initial state. """ _PlotInteraction.__init__(self, plot) self.parameters = parameters StateMachine.__init__(self, states, state) @property def color(self): return self.parameters.get("color", None)
[docs] class SelectPolygon(Select): """Drawing selection polygon area state machine.""" DRAG_THRESHOLD_DIST = 4
[docs] class Idle(State): def onPress(self, x, y, btn): if btn == LEFT_BTN: self.goto("select", x, y) return True
[docs] class Select(State):
[docs] def enterState(self, x, y): dataPos = self.machine.plot.pixelToData(x, y) assert dataPos is not None self._firstPos = dataPos self.points = [dataPos, dataPos] self.updateFirstPoint()
[docs] def updateFirstPoint(self): """Update drawing first point, using self._firstPos""" x, y = self.machine.plot.dataToPixel(*self._firstPos, check=False) offset = self.machine.getDragThreshold() points = [ (x - offset, y - offset), (x - offset, y + offset), (x + offset, y + offset), (x + offset, y - offset), ] points = [ self.machine.plot.pixelToData(xpix, ypix, check=False) for xpix, ypix in points ] self.machine.setSelectionArea( points, fill=None, color=self.machine.color, name="first_point" )
[docs] def updateSelectionArea(self): """Update drawing selection area using self.points""" self.machine.setSelectionArea( self.points, fill="hatch", color=self.machine.color ) eventDict = prepareDrawingSignal( "drawingProgress", "polygon", self.points, self.machine.parameters ) self.machine.plot.notify(**eventDict)
[docs] def validate(self): if len(self.points) > 2: self.closePolygon() else: # It would be nice to have a cancel event. # The plot is not aware that the interaction was cancelled self.machine.cancel()
def closePolygon(self): self.machine.resetSelectionArea() self.points[-1] = self.points[0] eventDict = prepareDrawingSignal( "drawingFinished", "polygon", self.points, self.machine.parameters ) self.machine.plot.notify(**eventDict) self.goto("idle") def onWheel(self, x, y, angle): self.machine.onWheel(x, y, angle) self.updateFirstPoint() def onRelease(self, x, y, btn): if btn == LEFT_BTN: # checking if the position is close to the first point # if yes : closing the "loop" firstPos = self.machine.plot.dataToPixel(*self._firstPos, check=False) dx, dy = abs(firstPos[0] - x), abs(firstPos[1] - y) threshold = self.machine.getDragThreshold() # Only allow to close polygon after first point if len(self.points) > 2 and dx <= threshold and dy <= threshold: self.closePolygon() return False # Update polygon last point not too close to previous one dataPos = self.machine.plot.pixelToData(x, y) assert dataPos is not None self.updateSelectionArea() # checking that the new points isnt the same (within range) # of the previous one # This has to be done because sometimes the mouse release event # is caught right after entering the Select state (i.e : press # in Idle state, but with a slightly different position that # the mouse press. So we had the two first vertices that were # almost identical. previousPos = self.machine.plot.dataToPixel( *self.points[-2], check=False ) dx, dy = abs(previousPos[0] - x), abs(previousPos[1] - y) if dx >= threshold or dy >= threshold: self.points.append(dataPos) else: self.points[-1] = dataPos return True return False def onMove(self, x, y): firstPos = self.machine.plot.dataToPixel(*self._firstPos, check=False) dx, dy = abs(firstPos[0] - x), abs(firstPos[1] - y) threshold = self.machine.getDragThreshold() if dx <= threshold and dy <= threshold: x, y = firstPos # Snap to first point dataPos = self.machine.plot.pixelToData(x, y) assert dataPos is not None self.points[-1] = dataPos self.updateSelectionArea()
def __init__(self, plot, parameters): states = {"idle": SelectPolygon.Idle, "select": SelectPolygon.Select} super(SelectPolygon, self).__init__(plot, parameters, states, "idle") def cancel(self): if isinstance(self.state, self.states["select"]): self.resetSelectionArea()
[docs] def getDragThreshold(self): """Return dragging ratio with device to pixel ratio applied. :rtype: float """ ratio = self.plot.window().windowHandle().devicePixelRatio() return self.DRAG_THRESHOLD_DIST * ratio
[docs] class Select2Points(Select): """Base class for drawing selection based on 2 input points."""
[docs] class Idle(State): def onPress(self, x, y, btn): if btn == LEFT_BTN: self.goto("start", x, y) return True
[docs] class Start(State):
[docs] def enterState(self, x, y): self.machine.beginSelect(x, y)
def onMove(self, x, y): self.goto("select", x, y) def onRelease(self, x, y, btn): if btn == LEFT_BTN: self.goto("select", x, y) return True
[docs] class Select(State):
[docs] def enterState(self, x, y): self.onMove(x, y)
def onMove(self, x, y): self.machine.select(x, y) def onRelease(self, x, y, btn): if btn == LEFT_BTN: self.machine.endSelect(x, y) self.goto("idle")
def __init__(self, plot, parameters): states = { "idle": Select2Points.Idle, "start": Select2Points.Start, "select": Select2Points.Select, } super(Select2Points, self).__init__(plot, parameters, states, "idle") def beginSelect(self, x, y): pass def select(self, x, y): pass def endSelect(self, x, y): pass def cancelSelect(self): pass def cancel(self): if isinstance(self.state, self.states["select"]): self.cancelSelect()
[docs] class SelectEllipse(Select2Points): """Drawing ellipse selection area state machine.""" def beginSelect(self, x, y): self.center = self.plot.pixelToData(x, y) assert self.center is not None def _getEllipseSize(self, pointInEllipse): """ Returns the size from the center to the bounding box of the ellipse. :param Tuple[float,float] pointInEllipse: A point of the ellipse :rtype: Tuple[float,float] """ x = abs(self.center[0] - pointInEllipse[0]) y = abs(self.center[1] - pointInEllipse[1]) if x == 0 or y == 0: return x, y # Ellipse definitions # e: eccentricity # a: length fron center to bounding box width # b: length fron center to bounding box height # Equations # (1) b < a # (2) For x,y a point in the ellipse: x^2/a^2 + y^2/b^2 = 1 # (3) b = a * sqrt(1-e^2) # (4) e = sqrt(a^2 - b^2) / a # The eccentricity of the ellipse defined by a,b=x,y is the same # as the one we are searching for. swap = x < y if swap: x, y = y, x e = math.sqrt(x**2 - y**2) / x # From (2) using (3) to replace b # a^2 = x^2 + y^2 / (1-e^2) a = math.sqrt(x**2 + y**2 / (1.0 - e**2)) b = a * math.sqrt(1 - e**2) if swap: a, b = b, a return a, b def select(self, x, y): dataPos = self.plot.pixelToData(x, y) assert dataPos is not None width, height = self._getEllipseSize(dataPos) # Circle used for circle preview nbpoints = 27.0 angles = numpy.arange(nbpoints) * numpy.pi * 2.0 / nbpoints circleShape = numpy.array( (numpy.cos(angles) * width, numpy.sin(angles) * height) ).T circleShape += numpy.array(self.center) self.setSelectionArea( circleShape, shape="polygon", fill="hatch", color=self.color ) eventDict = prepareDrawingSignal( "drawingProgress", "ellipse", (self.center, (width, height)), self.parameters, ) self.plot.notify(**eventDict) def endSelect(self, x, y): self.resetSelectionArea() dataPos = self.plot.pixelToData(x, y) assert dataPos is not None width, height = self._getEllipseSize(dataPos) eventDict = prepareDrawingSignal( "drawingFinished", "ellipse", (self.center, (width, height)), self.parameters, ) self.plot.notify(**eventDict) def cancelSelect(self): self.resetSelectionArea()
[docs] class SelectRectangle(Select2Points): """Drawing rectangle selection area state machine.""" def beginSelect(self, x, y): self.startPt = self.plot.pixelToData(x, y) assert self.startPt is not None def select(self, x, y): dataPos = self.plot.pixelToData(x, y) assert dataPos is not None self.setSelectionArea( ( self.startPt, (self.startPt[0], dataPos[1]), dataPos, (dataPos[0], self.startPt[1]), ), fill="hatch", color=self.color, ) eventDict = prepareDrawingSignal( "drawingProgress", "rectangle", (self.startPt, dataPos), self.parameters ) self.plot.notify(**eventDict) def endSelect(self, x, y): self.resetSelectionArea() dataPos = self.plot.pixelToData(x, y) assert dataPos is not None eventDict = prepareDrawingSignal( "drawingFinished", "rectangle", (self.startPt, dataPos), self.parameters ) self.plot.notify(**eventDict) def cancelSelect(self): self.resetSelectionArea()
[docs] class SelectLine(Select2Points): """Drawing line selection area state machine.""" def beginSelect(self, x, y): self.startPt = self.plot.pixelToData(x, y) assert self.startPt is not None def select(self, x, y): dataPos = self.plot.pixelToData(x, y) assert dataPos is not None self.setSelectionArea((self.startPt, dataPos), fill="hatch", color=self.color) eventDict = prepareDrawingSignal( "drawingProgress", "line", (self.startPt, dataPos), self.parameters ) self.plot.notify(**eventDict) def endSelect(self, x, y): self.resetSelectionArea() dataPos = self.plot.pixelToData(x, y) assert dataPos is not None eventDict = prepareDrawingSignal( "drawingFinished", "line", (self.startPt, dataPos), self.parameters ) self.plot.notify(**eventDict) def cancelSelect(self): self.resetSelectionArea()
[docs] class Select1Point(Select): """Base class for drawing selection area based on one input point."""
[docs] class Idle(State): def onPress(self, x, y, btn): if btn == LEFT_BTN: self.goto("select", x, y) return True
[docs] class Select(State):
[docs] def enterState(self, x, y): self.onMove(x, y)
def onMove(self, x, y): self.machine.select(x, y) def onRelease(self, x, y, btn): if btn == LEFT_BTN: self.machine.endSelect(x, y) self.goto("idle") def onWheel(self, x, y, angle): self.machine.onWheel(x, y, angle) # Call select default wheel self.machine.select(x, y)
def __init__(self, plot, parameters): states = {"idle": Select1Point.Idle, "select": Select1Point.Select} super(Select1Point, self).__init__(plot, parameters, states, "idle") def select(self, x, y): pass def endSelect(self, x, y): pass def cancelSelect(self): pass def cancel(self): if isinstance(self.state, self.states["select"]): self.cancelSelect()
[docs] class SelectHLine(Select1Point): """Drawing a horizontal line selection area state machine.""" def _hLine(self, y): """Return points in data coords of the segment visible in the plot. Supports non-orthogonal axes. """ left, _top, width, _height = self.plot.getPlotBoundsInPixels() dataPos1 = self.plot.pixelToData(left, y, check=False) dataPos2 = self.plot.pixelToData(left + width, y, check=False) return dataPos1, dataPos2 def select(self, x, y): points = self._hLine(y) self.setSelectionArea(points, fill="hatch", color=self.color) eventDict = prepareDrawingSignal( "drawingProgress", "hline", points, self.parameters ) self.plot.notify(**eventDict) def endSelect(self, x, y): self.resetSelectionArea() eventDict = prepareDrawingSignal( "drawingFinished", "hline", self._hLine(y), self.parameters ) self.plot.notify(**eventDict) def cancelSelect(self): self.resetSelectionArea()
[docs] class SelectVLine(Select1Point): """Drawing a vertical line selection area state machine.""" def _vLine(self, x): """Return points in data coords of the segment visible in the plot. Supports non-orthogonal axes. """ _left, top, _width, height = self.plot.getPlotBoundsInPixels() dataPos1 = self.plot.pixelToData(x, top, check=False) dataPos2 = self.plot.pixelToData(x, top + height, check=False) return dataPos1, dataPos2 def select(self, x, y): points = self._vLine(x) self.setSelectionArea(points, fill="hatch", color=self.color) eventDict = prepareDrawingSignal( "drawingProgress", "vline", points, self.parameters ) self.plot.notify(**eventDict) def endSelect(self, x, y): self.resetSelectionArea() eventDict = prepareDrawingSignal( "drawingFinished", "vline", self._vLine(x), self.parameters ) self.plot.notify(**eventDict) def cancelSelect(self): self.resetSelectionArea()
[docs] class DrawFreeHand(Select): """Interaction for drawing pencil. It display the preview of the pencil before pressing the mouse. """
[docs] class Idle(State): def onPress(self, x, y, btn): if btn == LEFT_BTN: self.goto("select", x, y) return True def onMove(self, x, y): self.machine.updatePencilShape(x, y) def onLeave(self): self.machine.cancel()
[docs] class Select(State):
[docs] def enterState(self, x, y): self.__isOut = False self.machine.setFirstPoint(x, y)
def onMove(self, x, y): self.machine.updatePencilShape(x, y) self.machine.select(x, y) def onRelease(self, x, y, btn): if btn == LEFT_BTN: if self.__isOut: self.machine.resetSelectionArea() self.machine.endSelect(x, y) self.goto("idle") def onEnter(self): self.__isOut = False def onLeave(self): self.__isOut = True
def __init__(self, plot, parameters): # Circle used for pencil preview angle = numpy.arange(13.0) * numpy.pi * 2.0 / 13.0 size = parameters.get("width", 1.0) * 0.5 self._circle = size * numpy.array((numpy.cos(angle), numpy.sin(angle))).T states = {"idle": DrawFreeHand.Idle, "select": DrawFreeHand.Select} super(DrawFreeHand, self).__init__(plot, parameters, states, "idle") @property def width(self): return self.parameters.get("width", None) def setFirstPoint(self, x, y): self._points = [] self.select(x, y) def updatePencilShape(self, x, y): center = self.plot.pixelToData(x, y, check=False) assert center is not None polygon = center + self._circle self.setSelectionArea(polygon, fill="none", color=self.color) def select(self, x, y): pos = self.plot.pixelToData(x, y, check=False) if len(self._points) > 0: if self._points[-1] == pos: # Skip same points return self._points.append(pos) eventDict = prepareDrawingSignal( "drawingProgress", "polylines", self._points, self.parameters ) self.plot.notify(**eventDict) def endSelect(self, x, y): pos = self.plot.pixelToData(x, y, check=False) if len(self._points) > 0: if self._points[-1] != pos: # Append if different self._points.append(pos) eventDict = prepareDrawingSignal( "drawingFinished", "polylines", self._points, self.parameters ) self.plot.notify(**eventDict) self._points = None def cancelSelect(self): self.resetSelectionArea() def cancel(self): self.resetSelectionArea()
[docs] class SelectFreeLine(ClickOrDrag, _PlotInteraction): """Base class for drawing free lines with tools such as pencil.""" def __init__(self, plot, parameters): """Init a state machine. :param plot: The plot to apply changes to. :param dict parameters: A dict of parameters such as color. """ # self.DRAG_THRESHOLD_SQUARE_DIST = 1 # Disable first move threshold self._points = [] ClickOrDrag.__init__(self) _PlotInteraction.__init__(self, plot) self.parameters = parameters @property def color(self): return self.parameters.get("color", None)
[docs] def click(self, x, y, btn): if btn == LEFT_BTN: self._processEvent(x, y, isLast=True)
[docs] def beginDrag(self, x, y, btn): self._processEvent(x, y, isLast=False)
[docs] def drag(self, x, y, btn): self._processEvent(x, y, isLast=False)
[docs] def endDrag(self, startPos, endPos, btn): x, y = endPos self._processEvent(x, y, isLast=True)
def cancel(self): self.resetSelectionArea() self._points = [] def _processEvent(self, x, y, isLast): dataPos = self.plot.pixelToData(x, y, check=False) isNewPoint = not self._points or dataPos != self._points[-1] if isNewPoint: self._points.append(dataPos) if isNewPoint or isLast: eventDict = prepareDrawingSignal( "drawingFinished" if isLast else "drawingProgress", "polylines", self._points, self.parameters, ) self.plot.notify(**eventDict) if not isLast: self.setSelectionArea( self._points, fill="none", color=self.color, shape="polylines" ) else: self.cancel()
# ItemInteraction #############################################################
[docs] class ItemsInteraction(ClickOrDrag, _PlotInteraction): """Interaction with items (markers, curves and images). This class provides selection and dragging of plot primitives that support those interaction. It is also meant to be combined with the zoom interaction. """
[docs] class Idle(ClickOrDrag.Idle): def __init__(self, *args, **kw): super(ItemsInteraction.Idle, self).__init__(*args, **kw) self._hoverMarker = None
[docs] def enterState(self): widget = self.machine.plot.getWidgetHandle() if widget is None or not widget.isVisible(): return position = widget.mapFromGlobal(qt.QCursor.pos()) self.onMove(position.x(), position.y())
def onMove(self, x, y): marker = self.machine.plot._getMarkerAt(x, y) if marker is not None: dataPos = self.machine.plot.pixelToData(x, y) assert dataPos is not None eventDict = prepareHoverSignal( marker.getName(), "marker", dataPos, (x, y), marker.isDraggable(), marker.isSelectable(), ) self.machine.plot.notify(**eventDict) if marker != self._hoverMarker: self._hoverMarker = marker self.machine._setCursorForMarker(marker) return True
def __init__(self, plot): self._pan = Pan(plot) _PlotInteraction.__init__(self, plot) ClickOrDrag.__init__( self, clickButtons=(LEFT_BTN, RIGHT_BTN), dragButtons=(LEFT_BTN, MIDDLE_BTN) ) def _setCursorForMarker(self, marker: Optional[items.MarkerBase] = None): """Set mouse cursor for given marker""" if marker is None: cursor = None elif marker.isDraggable(): if isinstance(marker, items.YMarker): cursor = CURSOR_SIZE_VER elif isinstance(marker, items.XMarker): cursor = CURSOR_SIZE_HOR else: cursor = CURSOR_SIZE_ALL elif marker.isSelectable(): cursor = CURSOR_POINTING else: cursor = None self.plot.setGraphCursorShape(cursor)
[docs] def click(self, x, y, btn): """Handle mouse click :param x: X position of the mouse in pixels :param y: Y position of the mouse in pixels :param btn: Pressed button id :return: True if click is catched by an item, False otherwise """ # Signal mouse clicked event dataPos = self.plot.pixelToData(x, y) assert dataPos is not None eventDict = prepareMouseSignal( "mouseClicked", btn, dataPos[0], dataPos[1], x, y ) self.plot.notify(**eventDict) eventDict = self._handleClick(x, y, btn) if eventDict is not None: self.plot.notify(**eventDict)
def _handleClick(self, x, y, btn): """Perform picking and prepare event if click is handled here :param x: X position of the mouse in pixels :param y: Y position of the mouse in pixels :param btn: Pressed button id :return: event description to send of None if not handling event. :rtype: dict or None """ if btn == LEFT_BTN: result = self.plot._pickTopMost(x, y, lambda i: i.isSelectable()) if result is None: return None item = result.getItem() if isinstance(item, items.MarkerBase): xData, yData = item.getPosition() if xData is None: xData = [0, 1] if yData is None: yData = [0, 1] eventDict = prepareMarkerSignal( "markerClicked", "left", item.getName(), "marker", item.isDraggable(), item.isSelectable(), (xData, yData), (x, y), None, ) return eventDict elif isinstance(item, items.Curve): dataPos = self.plot.pixelToData(x, y) assert dataPos is not None xData = item.getXData(copy=False) yData = item.getYData(copy=False) indices = result.getIndices(copy=False) eventDict = prepareCurveSignal( "left", item, xData[indices], yData[indices], dataPos[0], dataPos[1], x, y, ) return eventDict elif isinstance(item, items.ImageBase): dataPos = self.plot.pixelToData(x, y) assert dataPos is not None indices = result.getIndices(copy=False) row, column = indices[0][0], indices[1][0] eventDict = prepareImageSignal( "left", item, column, row, dataPos[0], dataPos[1], x, y ) return eventDict return None def _signalMarkerMovingEvent(self, eventType, marker, x, y): assert marker is not None xData, yData = marker.getPosition() if xData is None: xData = [0, 1] if yData is None: yData = [0, 1] posDataCursor = self.plot.pixelToData(x, y) assert posDataCursor is not None eventDict = prepareMarkerSignal( eventType, "left", marker.getName(), "marker", marker.isDraggable(), marker.isSelectable(), (xData, yData), (x, y), posDataCursor, ) self.plot.notify(**eventDict) @staticmethod def __isDraggableItem(item): return isinstance(item, items.DraggableMixIn) and item.isDraggable() def __terminateDrag(self, x, y): """Finalize a drag operation by reseting to initial state""" self._setCursorForMarker(self.plot._getMarkerAt(x, y)) self.draggedItemRef = None
[docs] def beginDrag(self, x, y, btn): """Handle begining of drag interaction :param x: X position of the mouse in pixels :param y: Y position of the mouse in pixels :param str btn: The mouse button for which a drag is starting. :return: True if drag is catched by an item, False otherwise """ if btn == LEFT_BTN: self._lastPos = self.plot.pixelToData(x, y) assert self._lastPos is not None result = self.plot._pickTopMost(x, y, self.__isDraggableItem) item = result.getItem() if result is not None else None self.draggedItemRef = None if item is None else weakref.ref(item) if item is None: self.__terminateDrag(x, y) return False if isinstance(item, items.MarkerBase): self._signalMarkerMovingEvent("markerMoving", item, x, y) item._startDrag() return True elif btn == MIDDLE_BTN: self._pan.beginDrag(x, y, btn) return True
[docs] def drag(self, x, y, btn): if btn == LEFT_BTN: dataPos = self.plot.pixelToData(x, y) assert dataPos is not None item = None if self.draggedItemRef is None else self.draggedItemRef() if item is not None: item.drag(self._lastPos, dataPos) if isinstance(item, items.MarkerBase): self._signalMarkerMovingEvent("markerMoving", item, x, y) self._lastPos = dataPos elif btn == MIDDLE_BTN: self._pan.drag(x, y, btn)
[docs] def endDrag(self, startPos, endPos, btn): if btn == LEFT_BTN: item = None if self.draggedItemRef is None else self.draggedItemRef() if isinstance(item, items.MarkerBase): posData = list(item.getPosition()) if posData[0] is None: posData[0] = 1.0 if posData[1] is None: posData[1] = 1.0 eventDict = prepareMarkerSignal( "markerMoved", "left", item.getLegend(), "marker", item.isDraggable(), item.isSelectable(), posData, ) self.plot.notify(**eventDict) item._endDrag() self.__terminateDrag(*endPos) elif btn == MIDDLE_BTN: self._pan.endDrag(startPos, endPos, btn)
def cancel(self): self._pan.cancel() widget = self.plot.getWidgetHandle() if widget is None or not widget.isVisible(): return position = widget.mapFromGlobal(qt.QCursor.pos()) self.__terminateDrag(position.x(), position.y())
[docs] class ItemsInteractionForCombo(ItemsInteraction): """Interaction with items to combine through :class:`FocusManager`."""
[docs] class Idle(ItemsInteraction.Idle): @staticmethod def __isItemSelectableOrDraggable(item): return item.isSelectable() or ( isinstance(item, items.DraggableMixIn) and item.isDraggable() ) def onPress(self, x, y, btn): if btn == LEFT_BTN: result = self.machine.plot._pickTopMost( x, y, self.__isItemSelectableOrDraggable ) if result is not None: # Request focus and handle interaction self.goto("clickOrDrag", x, y, btn) return True else: # Do not request focus return False else: return super().onPress(x, y, btn)
# FocusManager ################################################################
[docs] class FocusManager(StateMachine): """Manages focus across multiple event handlers On press an event handler can acquire focus. By default it looses focus when all buttons are released. """
[docs] class Idle(State): def onPress(self, x, y, btn): if btn == LEFT_BTN: for eventHandler in self.machine.eventHandlers: requestFocus = eventHandler.handleEvent("press", x, y, btn) if requestFocus: self.goto("focus", eventHandler, btn) break def _processEvent(self, *args): for eventHandler in self.machine.eventHandlers: consumeEvent = eventHandler.handleEvent(*args) if consumeEvent: break def onMove(self, x, y): self._processEvent("move", x, y) def onRelease(self, x, y, btn): if btn == LEFT_BTN: self._processEvent("release", x, y, btn) def onWheel(self, x, y, angle): self._processEvent("wheel", x, y, angle)
[docs] class Focus(State):
[docs] def enterState(self, eventHandler, btn): self.eventHandler = eventHandler self.focusBtns = {btn}
[docs] def validate(self): self.eventHandler.validate() self.goto("idle")
def onPress(self, x, y, btn): if btn == LEFT_BTN: self.focusBtns.add(btn) self.eventHandler.handleEvent("press", x, y, btn) def onMove(self, x, y): self.eventHandler.handleEvent("move", x, y) def onRelease(self, x, y, btn): if btn == LEFT_BTN: self.focusBtns.discard(btn) requestFocus = self.eventHandler.handleEvent("release", x, y, btn) if len(self.focusBtns) == 0 and not requestFocus: self.goto("idle") def onWheel(self, x, y, angleInDegrees): self.eventHandler.handleEvent("wheel", x, y, angleInDegrees)
def __init__(self, eventHandlers=()): self.eventHandlers = list(eventHandlers) states = {"idle": FocusManager.Idle, "focus": FocusManager.Focus} super(FocusManager, self).__init__(states, "idle") def cancel(self): for handler in self.eventHandlers: handler.cancel()
[docs] class ZoomAndSelect(ItemsInteraction): """Combine Zoom and ItemInteraction state machine. :param plot: The Plot to which this interaction is attached :param color: The color to use for the zoom area bounding box """ def __init__(self, plot, color): super(ZoomAndSelect, self).__init__(plot) self._zoom = Zoom(plot, color) self._doZoom = False @property def color(self): """Color of the zoom area""" return self._zoom.color @property def zoomEnabledAxes(self) -> EnabledAxes: """Whether or not to apply zoom for each axis""" return self._zoom.enabledAxes @zoomEnabledAxes.setter def zoomEnabledAxes(self, enabledAxes: EnabledAxes): self._zoom.enabledAxes = enabledAxes
[docs] def click(self, x, y, btn): """Handle mouse click :param x: X position of the mouse in pixels :param y: Y position of the mouse in pixels :param btn: Pressed button id :return: True if click is catched by an item, False otherwise """ eventDict = self._handleClick(x, y, btn) if eventDict is not None: # Signal mouse clicked event dataPos = self.plot.pixelToData(x, y) assert dataPos is not None clickedEventDict = prepareMouseSignal( "mouseClicked", btn, dataPos[0], dataPos[1], x, y ) self.plot.notify(**clickedEventDict) self.plot.notify(**eventDict) else: self._zoom.click(x, y, btn)
[docs] def beginDrag(self, x, y, btn): """Handle start drag and switching between zoom and item drag. :param x: X position in pixels :param y: Y position in pixels :param str btn: The mouse button for which a drag is starting. """ self._doZoom = not super(ZoomAndSelect, self).beginDrag(x, y, btn) if self._doZoom: self._zoom.beginDrag(x, y, btn)
[docs] def drag(self, x, y, btn): """Handle drag, eventually forwarding to zoom. :param x: X position in pixels :param y: Y position in pixels :param str btn: The mouse button for which a drag is in progress. """ if self._doZoom: return self._zoom.drag(x, y, btn) else: return super(ZoomAndSelect, self).drag(x, y, btn)
[docs] def endDrag(self, startPos, endPos, btn): """Handle end of drag, eventually forwarding to zoom. :param startPos: (x, y) position at the beginning of the drag :param endPos: (x, y) position at the end of the drag :param str btn: The mouse button for which a drag is done. """ if self._doZoom: return self._zoom.endDrag(startPos, endPos, btn) else: return super(ZoomAndSelect, self).endDrag(startPos, endPos, btn)
[docs] class PanAndSelect(ItemsInteraction): """Combine Pan and ItemInteraction state machine. :param plot: The Plot to which this interaction is attached """ def __init__(self, plot): super(PanAndSelect, self).__init__(plot) self._pan = Pan(plot) self._doPan = False
[docs] def click(self, x, y, btn): """Handle mouse click :param x: X position of the mouse in pixels :param y: Y position of the mouse in pixels :param btn: Pressed button id :return: True if click is catched by an item, False otherwise """ eventDict = self._handleClick(x, y, btn) if eventDict is not None: # Signal mouse clicked event dataPos = self.plot.pixelToData(x, y) assert dataPos is not None clickedEventDict = prepareMouseSignal( "mouseClicked", btn, dataPos[0], dataPos[1], x, y ) self.plot.notify(**clickedEventDict) self.plot.notify(**eventDict) else: self._pan.click(x, y, btn)
[docs] def beginDrag(self, x, y, btn): """Handle start drag and switching between zoom and item drag. :param x: X position in pixels :param y: Y position in pixels :param str btn: The mouse button for which a drag is starting. """ self._doPan = not super(PanAndSelect, self).beginDrag(x, y, btn) if self._doPan: self._pan.beginDrag(x, y, btn)
[docs] def drag(self, x, y, btn): """Handle drag, eventually forwarding to zoom. :param x: X position in pixels :param y: Y position in pixels :param str btn: The mouse button for which a drag is in progress. """ if self._doPan: return self._pan.drag(x, y, btn) else: return super(PanAndSelect, self).drag(x, y, btn)
[docs] def endDrag(self, startPos, endPos, btn): """Handle end of drag, eventually forwarding to zoom. :param startPos: (x, y) position at the beginning of the drag :param endPos: (x, y) position at the end of the drag :param str btn: The mouse button for which a drag is done. """ if self._doPan: return self._pan.endDrag(startPos, endPos, btn) else: return super(PanAndSelect, self).endDrag(startPos, endPos, btn)
# Interaction mode control #################################################### # Mapping of draw modes: event handler _DRAW_MODES = { "polygon": SelectPolygon, "rectangle": SelectRectangle, "ellipse": SelectEllipse, "line": SelectLine, "vline": SelectVLine, "hline": SelectHLine, "polylines": SelectFreeLine, "pencil": DrawFreeHand, }
[docs] class DrawMode(FocusManager): """Interactive mode for draw and select""" def __init__(self, plot, shape, label, color, width): eventHandlerClass = _DRAW_MODES[shape] parameters = { "shape": shape, "label": label, "color": color, "width": width, } super().__init__( ( Pan(plot, clickButtons=(), dragButtons=(MIDDLE_BTN,)), eventHandlerClass(plot, parameters), ) )
[docs] def getDescription(self): """Returns the dict describing this interactive mode""" params = self.eventHandlers[1].parameters.copy() params["mode"] = "draw" return params
[docs] class DrawSelectMode(FocusManager): """Interactive mode for draw and select""" def __init__(self, plot, shape, label, color, width): eventHandlerClass = _DRAW_MODES[shape] self._pan = Pan(plot) self._panStart = None parameters = { "shape": shape, "label": label, "color": color, "width": width, } super().__init__( (ItemsInteractionForCombo(plot), eventHandlerClass(plot, parameters)) )
[docs] def handleEvent(self, eventName, *args, **kwargs): # Hack to add pan interaction to select-draw # See issue Refactor PlotWidget interaction #3292 if eventName == "press" and args[2] == MIDDLE_BTN: self._panStart = args[:2] self._pan.beginDrag(*args) return # Consume middle click events elif eventName == "release" and args[2] == MIDDLE_BTN: self._panStart = None self._pan.endDrag(self._panStart, args[:2], MIDDLE_BTN) return # Consume middle click events elif self._panStart is not None and eventName == "move": x, y = args[:2] self._pan.drag(x, y, MIDDLE_BTN) super().handleEvent(eventName, *args, **kwargs)
[docs] def getDescription(self): """Returns the dict describing this interactive mode""" params = self.eventHandlers[1].parameters.copy() params["mode"] = "select-draw" return params
[docs] class PlotInteraction(qt.QObject): """PlotWidget user interaction handler. :param plot: The :class:`PlotWidget` to apply interaction to """ sigChanged = qt.Signal() """Signal emitted when the interaction configuration has changed""" _DRAW_MODES = { "polygon": SelectPolygon, "rectangle": SelectRectangle, "ellipse": SelectEllipse, "line": SelectLine, "vline": SelectVLine, "hline": SelectHLine, "polylines": SelectFreeLine, "pencil": DrawFreeHand, } def __init__(self, parent): super().__init__(parent) self.__zoomOnWheel = True self.__zoomEnabledAxes = EnabledAxes() # Default event handler self._eventHandler = ItemsInteraction(parent)
[docs] def isZoomOnWheelEnabled(self) -> bool: """Returns whether or not wheel interaction triggers zoom""" return self.__zoomOnWheel
[docs] def setZoomOnWheelEnabled(self, enabled: bool): """Toggle zoom on wheel interaction""" if enabled != self.__zoomOnWheel: self.__zoomOnWheel = enabled self.sigChanged.emit()
[docs] def setZoomEnabledAxes(self, xaxis: bool, yaxis: bool, y2axis: bool): """Toggle zoom interaction for each axis This is taken into account only if the plot does not keep aspect ratio. """ zoomEnabledAxes = EnabledAxes(xaxis, yaxis, y2axis) if zoomEnabledAxes != self.__zoomEnabledAxes: self.__zoomEnabledAxes = zoomEnabledAxes if isinstance(self._eventHandler, ZoomAndSelect): self._eventHandler.zoomEnabledAxes = zoomEnabledAxes self.sigChanged.emit()
[docs] def getZoomEnabledAxes(self) -> EnabledAxes: """Returns axes for which zoom is enabled""" return self.__zoomEnabledAxes
def _getInteractiveMode(self): """Returns the current interactive mode as a dict. The returned dict contains at least the key 'mode'. Mode can be: 'draw', 'pan', 'select', 'select-draw', 'zoom'. It can also contains extra keys (e.g., 'color') specific to a mode as provided to :meth:`_setInteractiveMode`. """ if isinstance(self._eventHandler, ZoomAndSelect): return {"mode": "zoom", "color": self._eventHandler.color} elif isinstance(self._eventHandler, (DrawMode, DrawSelectMode)): return self._eventHandler.getDescription() elif isinstance(self._eventHandler, PanAndSelect): return {"mode": "pan"} else: return {"mode": "select"} def _validate(self): """Validate the current interaction if possible If was designed to close the polygon interaction. """ self._eventHandler.validate() def _setInteractiveMode( self, mode, color="black", shape="polygon", label=None, width=None ): """Switch the interactive mode. :param str mode: The name of the interactive mode. In 'draw', 'pan', 'select', 'select-draw', 'zoom'. :param color: Only for 'draw' and 'zoom' modes. Color to use for drawing selection area. Default black. If None, selection area is not drawn. :type color: Color description: The name as a str or a tuple of 4 floats or None. :param str shape: Only for 'draw' mode. The kind of shape to draw. In 'polygon', 'rectangle', 'line', 'vline', 'hline', 'polylines'. Default is 'polygon'. :param str label: Only for 'draw' mode. :param float width: Width of the pencil. Only for draw pencil mode. """ assert mode in ("draw", "pan", "select", "select-draw", "zoom") plotWidget = self.parent() assert plotWidget is not None if isinstance(color, numpy.ndarray) or color not in (None, "video inverted"): color = colors.rgba(color) if mode in ("draw", "select-draw"): self._eventHandler.cancel() handlerClass = DrawMode if mode == "draw" else DrawSelectMode self._eventHandler = handlerClass(plotWidget, shape, label, color, width) elif mode == "pan": # Ignores color, shape and label self._eventHandler.cancel() self._eventHandler = PanAndSelect(plotWidget) elif mode == "zoom": # Ignores shape and label self._eventHandler.cancel() self._eventHandler = ZoomAndSelect(plotWidget, color) self._eventHandler.zoomEnabledAxes = self.getZoomEnabledAxes() else: # Default mode: interaction with plot objects # Ignores color, shape and label self._eventHandler.cancel() self._eventHandler = ItemsInteraction(plotWidget) self.sigChanged.emit()
[docs] def handleEvent(self, event, *args, **kwargs): """Forward event to current interactive mode state machine.""" if event == "wheel": # Handle wheel events directly self._onWheel(*args, **kwargs) return self._eventHandler.handleEvent(event, *args, **kwargs)
def _onWheel(self, x: float, y: float, angle: float): """Handle wheel events""" if not self.isZoomOnWheelEnabled(): return plotWidget = self.parent() if plotWidget is None: return # All axes are enabled if keep aspect ratio is on enabledAxes = ( EnabledAxes() if plotWidget.isKeepDataAspectRatio() else self.getZoomEnabledAxes() ) if enabledAxes.isDisabled(): return scale = 1.1 if angle > 0 else 1.0 / 1.1 applyZoomToPlot(plotWidget, scale, (x, y), enabledAxes)