# /*##########################################################################
#
# Copyright (c) 2015-2020 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.
#
# ###########################################################################*/
"""This module provides a window to view a 3D scalar field.
It supports iso-surfaces, a cutting plane and the definition of
a region of interest.
"""
__authors__ = ["T. Vincent"]
__license__ = "MIT"
__date__ = "14/06/2018"
import re
import logging
import time
from collections import deque
import numpy
from silx._utils import NP_OPTIONAL_COPY
from silx.gui import qt, icons
from silx.gui.colors import rgba
from silx.gui.colors import Colormap
from silx.math.marchingcubes import MarchingCubes
from silx.math.combo import min_max
from .scene import axes, cutplane, interaction, primitives, transform
from . import scene
from .Plot3DWindow import Plot3DWindow
from .tools import InteractiveModeToolBar
_logger = logging.getLogger(__name__)
[docs]
class Isosurface(qt.QObject):
"""Class representing an iso-surface
:param parent: The View widget this iso-surface belongs to
"""
sigLevelChanged = qt.Signal(float)
"""Signal emitted when the iso-surface level has changed.
This signal provides the new level value (might be nan).
"""
sigColorChanged = qt.Signal()
"""Signal emitted when the iso-surface color has changed"""
sigVisibilityChanged = qt.Signal(bool)
"""Signal emitted when the iso-surface visibility has changed.
This signal provides the new visibility status.
"""
def __init__(self, parent):
super(Isosurface, self).__init__(parent=parent)
self._level = float("nan")
self._autoLevelFunction = None
self._color = rgba("#FFD700FF")
self._data = None
self._group = scene.Group()
def _setData(self, data, copy=True):
"""Set the data set from which to build the iso-surface.
:param numpy.ndarray data: The 3D dataset or None
:param bool copy: True to make a copy, False to use as is if possible
"""
if data is None:
self._data = None
else:
self._data = numpy.array(data, copy=copy or NP_OPTIONAL_COPY, order="C")
self._update()
def _get3DPrimitive(self):
"""Return the group containing the mesh of the iso-surface if any"""
return self._group
[docs]
def isVisible(self):
"""Returns True if iso-surface is visible, else False"""
return self._group.visible
[docs]
def setVisible(self, visible):
"""Set the visibility of the iso-surface in the view.
:param bool visible: True to show the iso-surface, False to hide
"""
visible = bool(visible)
if visible != self._group.visible:
self._group.visible = visible
self.sigVisibilityChanged.emit(visible)
[docs]
def getLevel(self):
"""Return the level of this iso-surface (float)"""
return self._level
[docs]
def setLevel(self, level):
"""Set the value at which to build the iso-surface.
Setting this value reset auto-level function
:param float level: The value at which to build the iso-surface
"""
self._autoLevelFunction = None
level = float(level)
if level != self._level:
self._level = level
self._update()
self.sigLevelChanged.emit(level)
[docs]
def isAutoLevel(self):
"""True if iso-level is rebuild for each data set."""
return self.getAutoLevelFunction() is not None
[docs]
def getAutoLevelFunction(self):
"""Return the function computing the iso-level (callable or None)"""
return self._autoLevelFunction
[docs]
def setAutoLevelFunction(self, autoLevel):
"""Set the function used to compute the iso-level.
WARNING: The function might get called in a thread.
:param callable autoLevel:
A function taking a 3D numpy.ndarray of float32 and returning
a float used as iso-level.
Example: numpy.mean(data) + numpy.std(data)
"""
assert callable(autoLevel)
self._autoLevelFunction = autoLevel
self._update()
[docs]
def getColor(self):
"""Return the color of this iso-surface (QColor)"""
return qt.QColor.fromRgbF(*self._color)
[docs]
def setColor(self, color):
"""Set the color of the iso-surface
:param color: RGBA color of the isosurface
:type color: QColor, str or array-like of 4 float in [0., 1.]
"""
color = rgba(color)
if color != self._color:
self._color = color
if len(self._group.children) != 0:
self._group.children[0].setAttribute("color", self._color)
self.sigColorChanged.emit()
def _update(self):
"""Update underlying mesh"""
self._group.children = []
if self._data is None:
if self.isAutoLevel():
self._level = float("nan")
else:
if self.isAutoLevel():
st = time.time()
try:
level = float(self.getAutoLevelFunction()(self._data))
except Exception:
module = self.getAutoLevelFunction().__module__
name = self.getAutoLevelFunction().__name__
_logger.error(
"Error while executing iso level function %s.%s",
module,
name,
exc_info=True,
)
level = float("nan")
else:
_logger.info("Computed iso-level in %f s.", time.time() - st)
if level != self._level:
self._level = level
self.sigLevelChanged.emit(level)
if not numpy.isfinite(self._level):
return
st = time.time()
vertices, normals, indices = MarchingCubes(self._data, isolevel=self._level)
_logger.info("Computed iso-surface in %f s.", time.time() - st)
if len(vertices) == 0:
return
else:
mesh = primitives.Mesh3D(
vertices,
colors=self._color,
normals=normals,
mode="triangles",
indices=indices,
)
self._group.children = [mesh]
[docs]
class SelectedRegion(object):
"""Selection of a 3D region aligned with the axis.
:param arrayRange: Range of the selection in the array
((zmin, zmax), (ymin, ymax), (xmin, xmax))
:param dataBBox: Bounding box of the selection in data coordinates
((xmin, xmax), (ymin, ymax), (zmin, zmax))
:param translation: Offset from array to data coordinates (ox, oy, oz)
:param scale: Scale from array to data coordinates (sx, sy, sz)
"""
def __init__(
self, arrayRange, dataBBox, translation=(0.0, 0.0, 0.0), scale=(1.0, 1.0, 1.0)
):
self._arrayRange = numpy.array(arrayRange, copy=True, dtype=numpy.int64)
assert self._arrayRange.shape == (3, 2)
assert numpy.all(self._arrayRange[:, 1] >= self._arrayRange[:, 0])
self._dataRange = dataBBox
self._translation = numpy.array(translation, dtype=numpy.float32)
assert self._translation.shape == (3,)
self._scale = numpy.array(scale, dtype=numpy.float32)
assert self._scale.shape == (3,)
[docs]
def getArrayRange(self):
"""Returns array ranges of the selection: 3x2 array of int
:return: A numpy array with ((zmin, zmax), (ymin, ymax), (xmin, xmax))
:rtype: numpy.ndarray
"""
return self._arrayRange.copy()
[docs]
def getArraySlices(self):
"""Slices corresponding to the selected range in the array
:return: A numpy array with (zslice, yslice, zslice)
:rtype: numpy.ndarray
"""
return (
slice(*self._arrayRange[0]),
slice(*self._arrayRange[1]),
slice(*self._arrayRange[2]),
)
[docs]
def getDataRange(self):
"""Range in the data coordinates of the selection: 3x2 array of float
When the transform matrix is not the identity matrix
(e.g., rotation, skew) the returned range is the one of the selected
region bounding box in data coordinates.
:return: A numpy array with ((xmin, xmax), (ymin, ymax), (zmin, zmax))
:rtype: numpy.ndarray
"""
return self._dataRange.copy()
[docs]
def getDataScale(self):
"""Scale from array to data coordinates: (sx, sy, sz)
:return: A numpy array with (sx, sy, sz)
:rtype: numpy.ndarray
"""
return self._scale.copy()
[docs]
def getDataTranslation(self):
"""Offset from array to data coordinates: (ox, oy, oz)
:return: A numpy array with (ox, oy, oz)
:rtype: numpy.ndarray
"""
return self._translation.copy()
[docs]
class CutPlane(qt.QObject):
"""Class representing a cutting plane
:param ~silx.gui.plot3d.ScalarFieldView.ScalarFieldView sfView:
Widget in which the cut plane is applied.
"""
sigVisibilityChanged = qt.Signal(bool)
"""Signal emitted when the cut visibility has changed.
This signal provides the new visibility status.
"""
sigDataChanged = qt.Signal()
"""Signal emitted when the data this plane is cutting has changed."""
sigPlaneChanged = qt.Signal()
"""Signal emitted when the cut plane has moved"""
sigColormapChanged = qt.Signal(Colormap)
"""Signal emitted when the colormap has changed
This signal provides the new colormap.
"""
sigTransparencyChanged = qt.Signal()
"""Signal emitted when the transparency of the plane has changed.
This signal is emitted when calling :meth:`setDisplayValuesBelowMin`.
"""
sigInterpolationChanged = qt.Signal(str)
"""Signal emitted when the cut plane interpolation has changed
This signal provides the new interpolation mode.
"""
def __init__(self, sfView):
super(CutPlane, self).__init__(parent=sfView)
self._dataRange = None
self._visible = False
self.__syncPlane = True
# Plane stroke on the outer bounding box
self._planeStroke = primitives.PlaneInGroup(normal=(0, 1, 0))
self._planeStroke.visible = self._visible
self._planeStroke.addListener(self._planeChanged)
self._planeStroke.plane.addListener(self._planePositionChanged)
# Plane with texture on the data bounding box
self._dataPlane = cutplane.CutPlane(normal=(0, 1, 0))
self._dataPlane.strokeVisible = False
self._dataPlane.alpha = 1.0
self._dataPlane.visible = self._visible
self._dataPlane.plane.addListener(self._planePositionChanged)
self._colormap = Colormap(
name="gray", normalization="linear", vmin=None, vmax=None
)
self.getColormap().sigChanged.connect(self._colormapChanged)
self._updateSceneColormap()
sfView.sigDataChanged.connect(self._sfViewDataChanged)
sfView.sigTransformChanged.connect(self._sfViewTransformChanged)
def _get3DPrimitives(self):
"""Return the cut plane scene node."""
return self._planeStroke, self._dataPlane
def _keepPlaneInBBox(self):
"""Makes sure the plane intersect its parent bounding box if any"""
bounds = self._planeStroke.parent.bounds(dataBounds=True)
if bounds is not None:
self._planeStroke.plane.point = numpy.clip(
self._planeStroke.plane.point, a_min=bounds[0], a_max=bounds[1]
)
@staticmethod
def _syncPlanes(master, slave):
"""Move slave PlaneInGroup so that it is coplanar with master.
:param PlaneInGroup master: Reference PlaneInGroup
:param PlaneInGroup slave: PlaneInGroup to align
"""
masterToSlave = transform.StaticTransformList(
[slave.objectToSceneTransform.inverse(), master.objectToSceneTransform]
)
point = masterToSlave.transformPoint(master.plane.point)
normal = masterToSlave.transformNormal(master.plane.normal)
slave.plane.setPlane(point, normal)
def _sfViewDataChanged(self):
"""Handle data change in the ScalarFieldView this plane belongs to"""
self._dataPlane.setData(self.sender().getData(), copy=False)
# Store data range info as 3-tuple of values
self._dataRange = self.sender().getDataRange()
self.sigDataChanged.emit()
# Update colormap range when autoscale
if self.getColormap().isAutoscale():
self._updateSceneColormap()
self._keepPlaneInBBox()
def _sfViewTransformChanged(self):
"""Handle transform changed in the ScalarFieldView"""
self._keepPlaneInBBox()
self._syncPlanes(master=self._planeStroke, slave=self._dataPlane)
self.sigPlaneChanged.emit()
def _planeChanged(self, source, *args, **kwargs):
"""Handle events from the plane primitive"""
# Using _visible for now, until scene as more info in events
if source.visible != self._visible:
self._visible = source.visible
self.sigVisibilityChanged.emit(source.visible)
def _planePositionChanged(self, source, *args, **kwargs):
"""Handle update of cut plane position and normal"""
if self.__syncPlane:
self.__syncPlane = False
if source is self._planeStroke.plane:
self._syncPlanes(master=self._planeStroke, slave=self._dataPlane)
elif source is self._dataPlane.plane:
self._syncPlanes(master=self._dataPlane, slave=self._planeStroke)
else:
_logger.error("Received an unknown object %s", str(source))
if self._planeStroke.visible or self._dataPlane.visible:
self.sigPlaneChanged.emit()
self.__syncPlane = True
# Plane position
[docs]
def moveToCenter(self):
"""Move cut plane to center of data set"""
self._planeStroke.moveToCenter()
[docs]
def isValid(self):
"""Returns whether the cut plane is defined or not (bool)"""
return self._planeStroke.isValid
def _plane(self, coordinates="array"):
"""Returns the scene plane to set.
:param str coordinates: The coordinate system to use:
Either 'scene' or 'array' (default)
:rtype: Plane
:raise ValueError: If coordinates is not correct
"""
if coordinates == "scene":
return self._planeStroke.plane
elif coordinates == "array":
return self._dataPlane.plane
else:
raise ValueError("Unsupported coordinates: %s" % str(coordinates))
[docs]
def getNormal(self, coordinates="array"):
"""Returns the normal of the plane (as a unit vector)
:param str coordinates: The coordinate system to use:
Either 'scene' or 'array' (default)
:return: Normal (nx, ny, nz), vector is 0 if no plane is defined
:rtype: numpy.ndarray
:raise ValueError: If coordinates is not correct
"""
return self._plane(coordinates).normal
[docs]
def setNormal(self, normal, coordinates="array"):
"""Set the normal of the plane.
:param normal: 3-tuple of float: nx, ny, nz
:param str coordinates: The coordinate system to use:
Either 'scene' or 'array' (default)
:raise ValueError: If coordinates is not correct
"""
self._plane(coordinates).normal = normal
[docs]
def getPoint(self, coordinates="array"):
"""Returns a point on the plane.
:param str coordinates: The coordinate system to use:
Either 'scene' or 'array' (default)
:return: (x, y, z)
:rtype: numpy.ndarray
:raise ValueError: If coordinates is not correct
"""
return self._plane(coordinates).point
[docs]
def setPoint(self, point, constraint=True, coordinates="array"):
"""Set a point contained in the plane.
Warning: The plane might not intersect the bounding box of the data.
:param point: (x, y, z) position
:type point: 3-tuple of float
:param bool constraint:
True (default) to make sure the plane intersect data bounding box,
False to set the plane without any constraint.
:raise ValueError: If coordinates is not correc
"""
self._plane(coordinates).point = point
if constraint:
self._keepPlaneInBBox()
[docs]
def getParameters(self, coordinates="array"):
"""Returns the plane equation parameters: a*x + b*y + c*z + d = 0
:param str coordinates: The coordinate system to use:
Either 'scene' or 'array' (default)
:return: Plane equation parameters: (a, b, c, d)
:rtype: numpy.ndarray
:raise ValueError: If coordinates is not correct
"""
return self._plane(coordinates).parameters
[docs]
def setParameters(self, parameters, constraint=True, coordinates="array"):
"""Set the plane equation parameters: a*x + b*y + c*z + d = 0
Warning: The plane might not intersect the bounding box of the data.
:param parameters: (a, b, c, d) plane equation parameters.
:type parameters: 4-tuple of float
:param bool constraint:
True (default) to make sure the plane intersect data bounding box,
False to set the plane without any constraint.
:raise ValueError: If coordinates is not correc
"""
self._plane(coordinates).parameters = parameters
if constraint:
self._keepPlaneInBBox()
# Visibility
[docs]
def isVisible(self):
"""Returns True if the plane is visible, False otherwise"""
return self._planeStroke.visible
[docs]
def setVisible(self, visible):
"""Set the visibility of the plane
:param bool visible: True to make plane visible
"""
visible = bool(visible)
self._planeStroke.visible = visible
self._dataPlane.visible = visible
# Border stroke
[docs]
def getStrokeColor(self):
"""Returns the color of the plane border (QColor)"""
return qt.QColor.fromRgbF(*self._planeStroke.color)
[docs]
def setStrokeColor(self, color):
"""Set the color of the plane border.
:param color: RGB color: name, #RRGGBB or RGB values
:type color:
QColor, str or array-like of 3 or 4 float in [0., 1.] or uint8
"""
color = rgba(color)
self._planeStroke.color = color
self._dataPlane.color = color
# Data
[docs]
def getImageData(self):
"""Returns the data and information corresponding to the cut plane.
The returned data is not interpolated,
it is a slice of the 3D scalar field.
Image data axes are so that plane normal is towards the point of view.
:return: An object containing the 2D data slice and information
"""
return _CutPlaneImage(self)
# Interpolation
[docs]
def getInterpolation(self):
"""Returns the interpolation used to display to cut plane.
:return: 'nearest' or 'linear'
:rtype: str
"""
return self._dataPlane.interpolation
[docs]
def setInterpolation(self, interpolation):
"""Set the interpolation used to display to cut plane
The default interpolation is 'linear'
:param str interpolation: 'nearest' or 'linear'
"""
if interpolation != self.getInterpolation():
self._dataPlane.interpolation = interpolation
self.sigInterpolationChanged.emit(interpolation)
# Colormap
# def getAlpha(self):
# """Returns the transparency of the plane as a float in [0., 1.]"""
# return self._plane.alpha
# def setAlpha(self, alpha):
# """Set the plane transparency.
#
# :param float alpha: Transparency in [0., 1]
# """
# self._plane.alpha = alpha
[docs]
def getDisplayValuesBelowMin(self):
"""Return whether values <= colormap min are displayed or not.
:rtype: bool
"""
return self._dataPlane.colormap.displayValuesBelowMin
[docs]
def setDisplayValuesBelowMin(self, display):
"""Set whether to display values <= colormap min.
:param bool display: True to show values below min,
False to discard them
"""
display = bool(display)
if display != self.getDisplayValuesBelowMin():
self._dataPlane.colormap.displayValuesBelowMin = display
self.sigTransparencyChanged.emit()
[docs]
def getColormap(self):
"""Returns the colormap set by :meth:`setColormap`.
:return: The colormap
:rtype: ~silx.gui.colors.Colormap
"""
return self._colormap
[docs]
def setColormap(self, name="gray", norm=None, vmin=None, vmax=None):
"""Set the colormap to use.
By either providing a :class:`Colormap` object or
its name, normalization and range.
:param name: Name of the colormap in
'gray', 'reversed gray', 'temperature', 'red', 'green', 'blue'.
Or Colormap object.
:type name: str or ~silx.gui.colors.Colormap
:param str norm: Colormap mapping: 'linear' or 'log'.
:param float vmin: The minimum value of the range or None for autoscale
:param float vmax: The maximum value of the range or None for autoscale
"""
_logger.debug(
"setColormap %s %s (%s, %s)", name, str(norm), str(vmin), str(vmax)
)
self._colormap.sigChanged.disconnect(self._colormapChanged)
if isinstance(name, Colormap): # Use it as it is
assert (norm, vmin, vmax) == (None, None, None)
self._colormap = name
else:
if norm is None:
norm = "linear"
self._colormap = Colormap(
name=name, normalization=norm, vmin=vmin, vmax=vmax
)
self._colormap.sigChanged.connect(self._colormapChanged)
self._colormapChanged()
[docs]
def getColormapEffectiveRange(self):
"""Returns the currently used range of the colormap.
This range is computed from the data set if colormap is in autoscale.
Range is clipped to positive values when using log scale.
:return: 2-tuple of float
"""
return self._dataPlane.colormap.range_
def _updateSceneColormap(self):
"""Synchronizes scene's colormap with Colormap object"""
colormap = self.getColormap()
sceneCMap = self._dataPlane.colormap
sceneCMap.colormap = colormap.getNColors()
sceneCMap.norm = colormap.getNormalization()
range_ = colormap.getColormapRange(data=self._dataRange)
sceneCMap.range_ = range_
def _colormapChanged(self):
"""Handle update of Colormap object"""
self._updateSceneColormap()
# Forward colormap changed event
self.sigColormapChanged.emit(self.getColormap())
class _CutPlaneImage(object):
"""Object representing the data sliced by a cut plane
:param CutPlane cutPlane: The CutPlane from which to generate image info
"""
def __init__(self, cutPlane):
# Init attributes with default values
self._isValid = False
self._data = numpy.zeros((0, 0), dtype=numpy.float32)
self._index = 0
self._xLabel = ""
self._yLabel = ""
self._normalLabel = ""
self._scale = float("nan"), float("nan")
self._translation = float("nan"), float("nan")
self._position = float("nan")
sfView = cutPlane.parent()
if not sfView or not cutPlane.isValid():
_logger.info("No plane available")
return
data = sfView.getData(copy=False)
if data is None:
_logger.info("No data available")
return
normal = cutPlane.getNormal(coordinates="array")
point = cutPlane.getPoint(coordinates="array")
if numpy.linalg.norm(numpy.cross(normal, (1.0, 0.0, 0.0))) < 0.0017:
if not 0 <= point[0] <= data.shape[2]:
_logger.info("Plane outside dataset")
return
index = max(0, min(int(point[0]), data.shape[2] - 1))
slice_ = data[:, :, index]
xAxisIndex, yAxisIndex, normalAxisIndex = 1, 2, 0 # y, z, x
elif numpy.linalg.norm(numpy.cross(normal, (0.0, 1.0, 0.0))) < 0.0017:
if not 0 <= point[1] <= data.shape[1]:
_logger.info("Plane outside dataset")
return
index = max(0, min(int(point[1]), data.shape[1] - 1))
slice_ = numpy.transpose(data[:, index, :])
xAxisIndex, yAxisIndex, normalAxisIndex = 2, 0, 1 # z, x, y
elif numpy.linalg.norm(numpy.cross(normal, (0.0, 0.0, 1.0))) < 0.0017:
if not 0 <= point[2] <= data.shape[0]:
_logger.info("Plane outside dataset")
return
index = max(0, min(int(point[2]), data.shape[0] - 1))
slice_ = data[index, :, :]
xAxisIndex, yAxisIndex, normalAxisIndex = 0, 1, 2 # x, y, z
else:
_logger.warning(
"Unsupported normal: (%f, %f, %f)", normal[0], normal[1], normal[2]
)
return
# Store cut plane image info
self._isValid = True
self._data = numpy.array(slice_, copy=True)
self._index = index
# Only store extra information when no transform matrix is set
# Otherwise this information can be meaningless
if numpy.all(
numpy.equal(
sfView.getTransformMatrix(), numpy.identity(3, dtype=numpy.float32)
)
):
labels = sfView.getAxesLabels()
self._xLabel = labels[xAxisIndex]
self._yLabel = labels[yAxisIndex]
self._normalLabel = labels[normalAxisIndex]
scale = sfView.getScale()
self._scale = scale[xAxisIndex], scale[yAxisIndex]
translation = sfView.getTranslation()
self._translation = translation[xAxisIndex], translation[yAxisIndex]
self._position = float(
index * scale[normalAxisIndex] + translation[normalAxisIndex]
)
def isValid(self):
"""Returns True if the cut plane image is defined (bool)"""
return self._isValid
def getData(self, copy=True):
"""Returns the image data sliced by the cut plane.
:param bool copy: True to get a copy, False otherwise
:return: The 2D image data corresponding to the cut plane
:rtype: numpy.ndarray
"""
return numpy.array(self._data, copy=copy or NP_OPTIONAL_COPY)
def getXLabel(self):
"""Returns the label associated to the X axis of the image (str)"""
return self._xLabel
def getYLabel(self):
"""Returns the label associated to the Y axis of the image (str)"""
return self._yLabel
def getNormalLabel(self):
"""Returns the label of the 3D axis of the plane normal (str)"""
return self._normalLabel
def getScale(self):
"""Returns the scales of the data as a 2-tuple of float (sx, sy)"""
return self._scale
def getTranslation(self):
"""Returns the offset of the data as a 2-tuple of float (ox, oy)"""
return self._translation
def getIndex(self):
"""Returns the index in the data array of the cut plane (int)"""
return self._index
def getPosition(self):
"""Returns the cut plane position along the normal axis (flaot)"""
return self._position
[docs]
class ScalarFieldView(Plot3DWindow):
"""Widget computing and displaying an iso-surface from a 3D scalar dataset.
Limitation: Currently, iso-surfaces are generated with higher values
than the iso-level 'inside' the surface.
:param parent: See :class:`QMainWindow`
"""
sigDataChanged = qt.Signal()
"""Signal emitted when the scalar data field has changed."""
sigTransformChanged = qt.Signal()
"""Signal emitted when the transformation has changed.
It is emitted by :meth:`setTranslation`, :meth:`setTransformMatrix`,
:meth:`setScale`.
"""
sigSelectedRegionChanged = qt.Signal(object)
"""Signal emitted when the selected region has changed.
This signal provides the new selected region.
"""
def __init__(self, parent=None):
super(ScalarFieldView, self).__init__(parent)
self._colormap = Colormap(
name="gray", normalization="linear", vmin=None, vmax=None
)
self._selectedRange = None
# Store iso-surfaces
self._isosurfaces = []
# Transformations
self._dataScale = transform.Scale()
self._dataTranslate = transform.Translate()
self._dataTransform = transform.Matrix() # default to identity
self._foregroundColor = 1.0, 1.0, 1.0, 1.0
self._highlightColor = 0.7, 0.7, 0.0, 1.0
self._data = None
self._dataRange = None
self._group = primitives.BoundedGroup()
self._group.transforms = [
self._dataTranslate,
self._dataTransform,
self._dataScale,
]
self._bbox = axes.LabelledAxes()
self._bbox.children = [self._group]
self._outerScale = transform.Scale(1.0, 1.0, 1.0)
self._bbox.transforms = [self._outerScale]
self.getPlot3DWidget().viewport.scene.children.append(self._bbox)
self._selectionBox = primitives.Box()
self._selectionBox.strokeSmooth = False
self._selectionBox.strokeWidth = 1.0
# self._selectionBox.fillColor = 1., 1., 1., 0.3
# self._selectionBox.fillCulling = 'back'
self._selectionBox.visible = False
self._group.children.append(self._selectionBox)
self._cutPlane = CutPlane(sfView=self)
self._cutPlane.sigVisibilityChanged.connect(self._planeVisibilityChanged)
planeStroke, dataPlane = self._cutPlane._get3DPrimitives()
self._bbox.children.append(planeStroke)
self._group.children.append(dataPlane)
self._isogroup = primitives.GroupDepthOffset()
self._isogroup.transforms = [
# Convert from z, y, x from marching cubes to x, y, z
transform.Matrix(
(
(0.0, 0.0, 1.0, 0.0),
(0.0, 1.0, 0.0, 0.0),
(1.0, 0.0, 0.0, 0.0),
(0.0, 0.0, 0.0, 1.0),
)
),
# Offset to match cutting plane coords
transform.Translate(0.5, 0.5, 0.5),
]
self._group.children.append(self._isogroup)
self._initPanPlaneAction()
self._updateColors()
self.getPlot3DWidget().viewport.light.shininess = 32
[docs]
def saveConfig(self, ioDevice):
"""
Saves this view state. Only isosurfaces at the moment. Does not save
the isosurface's function.
:param qt.QIODevice ioDevice: A `qt.QIODevice`.
"""
stream = qt.QDataStream(ioDevice)
stream.writeString("<ScalarFieldView>")
isoSurfaces = self.getIsosurfaces()
nIsoSurfaces = len(isoSurfaces)
# TODO : delegate the serialization to the serialized items
# isosurfaces
if nIsoSurfaces:
tagIn = "<IsoSurfaces nIso={0}>".format(nIsoSurfaces)
stream.writeString(tagIn)
for surface in isoSurfaces:
color = surface.getColor()
level = surface.getLevel()
visible = surface.isVisible()
stream << color
stream.writeDouble(level)
stream.writeBool(visible)
stream.writeString("</IsoSurfaces>")
stream.writeString("<Style>")
background = self.getBackgroundColor()
foreground = self.getForegroundColor()
highlight = self.getHighlightColor()
stream << background << foreground << highlight
stream.writeString("</Style>")
stream.writeString("</ScalarFieldView>")
[docs]
def loadConfig(self, ioDevice):
"""
Loads this view state.
See ScalarFieldView.saveView to know what is supported at the moment.
:param qt.QIODevice ioDevice: A `qt.QIODevice`.
"""
tagStack = deque()
tagInRegex = re.compile("<(?P<itemId>[^ /]*) *" "(?P<args>.*)>")
tagOutRegex = re.compile("</(?P<itemId>[^ ]*)>")
tagRootInRegex = re.compile("<ScalarFieldView>")
isoSurfaceArgsRegex = re.compile("nIso=(?P<nIso>[0-9]*)")
stream = qt.QDataStream(ioDevice)
tag = stream.readString()
tagMatch = tagRootInRegex.match(tag)
if tagMatch is None:
# TODO : explicit error
raise ValueError("Unknown data.")
itemId = "ScalarFieldView"
tagStack.append(itemId)
while True:
tag = stream.readString()
tagMatch = tagOutRegex.match(tag)
if tagMatch:
closeId = tagMatch.groupdict()["itemId"]
if closeId != itemId:
# TODO : explicit error
raise ValueError(
"Unexpected closing tag {0} "
"(expected {1})"
"".format(closeId, itemId)
)
if itemId == "ScalarFieldView":
# reached end
break
else:
itemId = tagStack.pop()
# fetching next tag
continue
tagMatch = tagInRegex.match(tag)
if tagMatch is None:
# TODO : explicit error
raise ValueError("Unknown data.")
tagStack.append(itemId)
matchDict = tagMatch.groupdict()
itemId = matchDict["itemId"]
# TODO : delegate the deserialization to the serialized items
if itemId == "IsoSurfaces":
argsMatch = isoSurfaceArgsRegex.match(matchDict["args"])
if not argsMatch:
# TODO : explicit error
raise ValueError(
'Failed to parse args "{0}".' "".format(matchDict["args"])
)
argsDict = argsMatch.groupdict()
nIso = int(argsDict["nIso"])
if nIso:
for surface in self.getIsosurfaces():
self.removeIsosurface(surface)
for isoIdx in range(nIso):
color = qt.QColor()
stream >> color
level = stream.readDouble()
visible = stream.readBool()
surface = self.addIsosurface(level, color=color)
surface.setVisible(visible)
elif itemId == "Style":
background = qt.QColor()
foreground = qt.QColor()
highlight = qt.QColor()
stream >> background >> foreground >> highlight
self.setBackgroundColor(background)
self.setForegroundColor(foreground)
self.setHighlightColor(highlight)
else:
raise ValueError("Unknown entry tag {0}." "".format(itemId))
def _initPanPlaneAction(self):
"""Creates and init the pan plane action"""
self._panPlaneAction = qt.QAction(self)
self._panPlaneAction.setIcon(icons.getQIcon("3d-plane-pan"))
self._panPlaneAction.setText("Pan plane")
self._panPlaneAction.setCheckable(True)
self._panPlaneAction.setToolTip(
"Pan the cutting plane. Press <b>Ctrl</b> to rotate the scene."
)
self._panPlaneAction.setEnabled(False)
self._panPlaneAction.triggered[bool].connect(self._planeActionTriggered)
self.getPlot3DWidget().sigInteractiveModeChanged.connect(
self._interactiveModeChanged
)
toolbar = self.findChild(InteractiveModeToolBar)
if toolbar is not None:
toolbar.addAction(self._panPlaneAction)
def _planeActionTriggered(self, checked=False):
self._panPlaneAction.setChecked(True)
self.setInteractiveMode("plane")
def _interactiveModeChanged(self):
self._panPlaneAction.setChecked(self.getInteractiveMode() == "plane")
self._updateColors()
def _planeVisibilityChanged(self, visible):
"""Handle visibility events from the plane"""
if visible != self._panPlaneAction.isEnabled():
self._panPlaneAction.setEnabled(visible)
if visible:
self.setInteractiveMode("plane")
elif self._panPlaneAction.isChecked():
self.setInteractiveMode("rotate")
[docs]
def setInteractiveMode(self, mode):
"""Choose the current interaction.
:param str mode: Either rotate, pan or plane
"""
if mode == self.getInteractiveMode():
return
sceneScale = self.getPlot3DWidget().viewport.scene.transforms[0]
if mode == "plane":
mode = interaction.PanPlaneZoomOnWheelControl(
self.getPlot3DWidget().viewport,
self._cutPlane._get3DPrimitives()[0],
mode="position",
orbitAroundCenter=False,
scaleTransform=sceneScale,
)
self.getPlot3DWidget().setInteractiveMode(mode)
self._updateColors()
[docs]
def getInteractiveMode(self):
"""Returns the current interaction mode, see :meth:`setInteractiveMode`"""
if isinstance(
self.getPlot3DWidget().eventHandler, interaction.PanPlaneZoomOnWheelControl
):
return "plane"
else:
return self.getPlot3DWidget().getInteractiveMode()
# Handle scalar field
[docs]
def setData(self, data, copy=True):
"""Set the 3D scalar data set to use for building the iso-surface.
Dataset order is zyx (i.e., first dimension is z).
:param data: scalar field from which to extract the iso-surface
:type data: 3D numpy.ndarray of float32 with shape at least (2, 2, 2)
:param bool copy:
True (default) to make a copy,
False to avoid copy (DO NOT MODIFY data afterwards)
"""
if data is None:
self._data = None
self._dataRange = None
self.setSelectedRegion(zrange=None, yrange=None, xrange_=None)
self._group.shape = None
self.centerScene()
else:
data = numpy.array(data, copy=copy or NP_OPTIONAL_COPY, dtype=numpy.float32, order="C")
assert data.ndim == 3
assert min(data.shape) >= 2
wasData = self._data is not None
previousSelectedRegion = self.getSelectedRegion()
self._data = data
# Store data range info
dataRange = min_max(self._data, min_positive=True, finite=True)
if dataRange.minimum is None: # Only non-finite data
dataRange = None
if dataRange is not None:
min_positive = dataRange.min_positive
if min_positive is None:
min_positive = float("nan")
dataRange = dataRange.minimum, min_positive, dataRange.maximum
self._dataRange = dataRange
if previousSelectedRegion is not None:
# Update selected region to ensure it is clipped to array range
self.setSelectedRegion(*previousSelectedRegion.getArrayRange())
self._group.shape = self._data.shape
if not wasData:
self.centerScene() # Reset viewpoint the first time only
# Update iso-surfaces
for isosurface in self.getIsosurfaces():
isosurface._setData(self._data, copy=False)
self.sigDataChanged.emit()
[docs]
def getData(self, copy=True):
"""Get the 3D scalar data currently used to build the iso-surface.
:param bool copy:
True (default) to get a copy,
False to get the internal data (DO NOT modify!)
:return: The data set (or None if not set)
"""
if self._data is None:
return None
else:
return numpy.array(self._data, copy=copy or NP_OPTIONAL_COPY)
[docs]
def getDataRange(self):
"""Return the range of the data as a 3-tuple of values.
positive min is NaN if no data is positive.
:return: (min, positive min, max) or None.
"""
return self._dataRange
# Transformations
[docs]
def setOuterScale(self, sx=1.0, sy=1.0, sz=1.0):
"""Set the scale to apply to the whole scene including the axes.
This is useful when axis lengths in data space are really different.
:param float sx: Scale factor along the X axis
:param float sy: Scale factor along the Y axis
:param float sz: Scale factor along the Z axis
"""
self._outerScale.setScale(sx, sy, sz)
self.centerScene()
[docs]
def getOuterScale(self):
"""Returns the scales provided by :meth:`setOuterScale`.
:rtype: numpy.ndarray
"""
return self._outerScale.scale
[docs]
def setScale(self, sx=1.0, sy=1.0, sz=1.0):
"""Set the scale of the 3D scalar field (i.e., size of a voxel).
:param float sx: Scale factor along the X axis
:param float sy: Scale factor along the Y axis
:param float sz: Scale factor along the Z axis
"""
scale = numpy.array((sx, sy, sz), dtype=numpy.float32)
if not numpy.all(numpy.equal(scale, self.getScale())):
self._dataScale.scale = scale
self.sigTransformChanged.emit()
self.centerScene() # Reset viewpoint
[docs]
def getScale(self):
"""Returns the scales provided by :meth:`setScale` as a numpy.ndarray."""
return self._dataScale.scale
[docs]
def setTranslation(self, x=0.0, y=0.0, z=0.0):
"""Set the translation of the origin of the data array in data coordinates.
:param float x: Offset of the data origin on the X axis
:param float y: Offset of the data origin on the Y axis
:param float z: Offset of the data origin on the Z axis
"""
translation = numpy.array((x, y, z), dtype=numpy.float32)
if not numpy.all(numpy.equal(translation, self.getTranslation())):
self._dataTranslate.translation = translation
self.sigTransformChanged.emit()
self.centerScene() # Reset viewpoint
[docs]
def getTranslation(self):
"""Returns the offset set by :meth:`setTranslation` as a numpy.ndarray."""
return self._dataTranslate.translation
# Axes labels
[docs]
def isBoundingBoxVisible(self):
"""Returns axes labels, grid and bounding box visibility.
:rtype: bool
"""
return self._bbox.boxVisible
[docs]
def setBoundingBoxVisible(self, visible):
"""Set axes labels, grid and bounding box visibility.
:param bool visible: True to show axes, False to hide
"""
visible = bool(visible)
self._bbox.boxVisible = visible
[docs]
def setAxesLabels(self, xlabel=None, ylabel=None, zlabel=None):
"""Set the text labels of the axes.
:param str xlabel: Label of the X axis, None to leave unchanged.
:param str ylabel: Label of the Y axis, None to leave unchanged.
:param str zlabel: Label of the Z axis, None to leave unchanged.
"""
if xlabel is not None:
self._bbox.xlabel = xlabel
if ylabel is not None:
self._bbox.ylabel = ylabel
if zlabel is not None:
self._bbox.zlabel = zlabel
class _Labels(tuple):
"""Return type of :meth:`getAxesLabels`"""
def getXLabel(self):
"""Label of the X axis (str)"""
return self[0]
def getYLabel(self):
"""Label of the Y axis (str)"""
return self[1]
def getZLabel(self):
"""Label of the Z axis (str)"""
return self[2]
[docs]
def getAxesLabels(self):
"""Returns the text labels of the axes
>>> widget = ScalarFieldView()
>>> widget.setAxesLabels(xlabel='X')
You can get the labels either as a 3-tuple:
>>> xlabel, ylabel, zlabel = widget.getAxesLabels()
Or as an object with methods getXLabel, getYLabel and getZLabel:
>>> labels = widget.getAxesLabels()
>>> labels.getXLabel()
... 'X'
:return: object describing the labels
"""
return self._Labels((self._bbox.xlabel, self._bbox.ylabel, self._bbox.zlabel))
# Colors
def _updateColors(self):
"""Update item depending on foreground/highlight color"""
self._bbox.tickColor = self._foregroundColor
self._selectionBox.strokeColor = self._foregroundColor
if self.getInteractiveMode() == "plane":
self._cutPlane.setStrokeColor(self._highlightColor)
self._bbox.color = self._foregroundColor
else:
self._cutPlane.setStrokeColor(self._foregroundColor)
self._bbox.color = self._highlightColor
[docs]
def getForegroundColor(self):
"""Return color used for text and bounding box (QColor)"""
return qt.QColor.fromRgbF(*self._foregroundColor)
[docs]
def setForegroundColor(self, color):
"""Set the foreground color.
:param color: RGB color: name, #RRGGBB or RGB values
:type color:
QColor, str or array-like of 3 or 4 float in [0., 1.] or uint8
"""
color = rgba(color)
if color != self._foregroundColor:
self._foregroundColor = color
self._updateColors()
[docs]
def getHighlightColor(self):
"""Return color used for highlighted item bounding box (QColor)"""
return qt.QColor.fromRgbF(*self._highlightColor)
[docs]
def setHighlightColor(self, color):
"""Set hightlighted item color.
:param color: RGB color: name, #RRGGBB or RGB values
:type color:
QColor, str or array-like of 3 or 4 float in [0., 1.] or uint8
"""
color = rgba(color)
if color != self._highlightColor:
self._highlightColor = color
self._updateColors()
# Cut Plane
[docs]
def getCutPlanes(self):
"""Return an iterable of all cut planes of the view.
This includes hidden cut planes.
For now, there is always one cut plane.
"""
return (self._cutPlane,)
# Selection
[docs]
def setSelectedRegion(self, zrange=None, yrange=None, xrange_=None):
"""Set the 3D selected region aligned with the axes.
Provided range are array indices range.
The provided ranges are clipped to the data.
If a range is None, the range of the array on this dimension is used.
:param zrange: (zmin, zmax) range of the selection
:param yrange: (ymin, ymax) range of the selection
:param xrange_: (xmin, xmax) range of the selection
"""
# No range given: unset selection
if zrange is None and yrange is None and xrange_ is None:
selectedRange = None
else:
# Handle default ranges
if self._data is not None:
if zrange is None:
zrange = 0, self._data.shape[0]
if yrange is None:
yrange = 0, self._data.shape[1]
if xrange_ is None:
xrange_ = 0, self._data.shape[2]
elif None in (xrange_, yrange, zrange):
# One of the range is None and no data available
raise RuntimeError("Data is not set, cannot get default range from it.")
# Clip selected region to data shape and make sure min <= max
selectedRange = numpy.array(
(
(max(0, min(*zrange)), min(self._data.shape[0], max(*zrange))),
(max(0, min(*yrange)), min(self._data.shape[1], max(*yrange))),
(max(0, min(*xrange_)), min(self._data.shape[2], max(*xrange_))),
),
dtype=numpy.int64,
)
# numpy.equal supports None
if not numpy.all(numpy.equal(selectedRange, self._selectedRange)):
self._selectedRange = selectedRange
# Update scene accordingly
if self._selectedRange is None:
self._selectionBox.visible = False
else:
self._selectionBox.visible = True
scales = self._selectedRange[:, 1] - self._selectedRange[:, 0]
self._selectionBox.size = scales[::-1]
self._selectionBox.transforms = [
transform.Translate(*self._selectedRange[::-1, 0])
]
self.sigSelectedRegionChanged.emit(self.getSelectedRegion())
[docs]
def getSelectedRegion(self):
"""Returns the currently selected region or None."""
if self._selectedRange is None:
return None
else:
dataBBox = self._group.transforms.transformBounds(
self._selectedRange[::-1].T
).T
return SelectedRegion(
self._selectedRange,
dataBBox,
translation=self.getTranslation(),
scale=self.getScale(),
)
# Handle iso-surfaces
sigIsosurfaceAdded = qt.Signal(object)
"""Signal emitted when a new iso-surface is added to the view.
The newly added iso-surface is provided by this signal
"""
sigIsosurfaceRemoved = qt.Signal(object)
"""Signal emitted when an iso-surface is removed from the view
The removed iso-surface is provided by this signal.
"""
[docs]
def addIsosurface(self, level, color):
"""Add an iso-surface to the view.
:param level:
The value at which to build the iso-surface or a callable
(e.g., a function) taking a 3D numpy.ndarray as input and
returning a float.
Example: numpy.mean(data) + numpy.std(data)
:type level: float or callable
:param color: RGBA color of the isosurface
:type color: str or array-like of 4 float in [0., 1.]
:return: Isosurface object describing this isosurface
"""
isosurface = Isosurface(parent=self)
isosurface.setColor(color)
if callable(level):
isosurface.setAutoLevelFunction(level)
else:
isosurface.setLevel(level)
isosurface._setData(self._data, copy=False)
isosurface.sigLevelChanged.connect(self._updateIsosurfaces)
self._isosurfaces.append(isosurface)
self._updateIsosurfaces()
self.sigIsosurfaceAdded.emit(isosurface)
return isosurface
[docs]
def getIsosurfaces(self):
"""Return an iterable of all iso-surfaces of the view"""
return tuple(self._isosurfaces)
[docs]
def removeIsosurface(self, isosurface):
"""Remove an iso-surface from the view.
:param isosurface: The isosurface object to remove"""
if isosurface not in self.getIsosurfaces():
_logger.warning(
"Try to remove isosurface that is not in the list: %s", str(isosurface)
)
else:
isosurface.sigLevelChanged.disconnect(self._updateIsosurfaces)
self._isosurfaces.remove(isosurface)
self._updateIsosurfaces()
self.sigIsosurfaceRemoved.emit(isosurface)
[docs]
def clearIsosurfaces(self):
"""Remove all iso-surfaces from the view."""
for isosurface in self.getIsosurfaces():
self.removeIsosurface(isosurface)
def _updateIsosurfaces(self, level=None):
"""Handle updates of iso-surfaces level and add/remove"""
# Sorting using minus, this supposes data 'object' to be max values
sortedIso = sorted(self.getIsosurfaces(), key=lambda iso: -iso.getLevel())
self._isogroup.children = [iso._get3DPrimitive() for iso in sortedIso]