Something went wrong on our end
Forked from
FSL / fslpy
4071 commits behind the upstream repository.
-
Paul McCarthy authored
vertex shader. 2. Bounds test used in gl21/[tensor/volume]_frag moved to separate file. 3. Transformation matrix offset by 0.5, in Display class, removed, as I've decided that it should be unnecessary.
Paul McCarthy authoredvertex shader. 2. Bounds test used in gl21/[tensor/volume]_frag moved to separate file. 3. Transformation matrix offset by 0.5, in Display class, removed, as I've decided that it should be unnecessary.
display.py 10.65 KiB
#!/usr/bin/env python
#
# display.py - Definitions of the Display and DisplayOpts classes.
#
# Author: Paul McCarthy <pauldmccarthy@gmail.com>
#
"""This module provides definitions of an important class - the
:class:`Display` class.
A ``Display`` contains a specification for the way in which an
:class:`~fsl.data.image.Image` instance is to be displayed.
..note:: Put a description of the three coordinate systems which
exist in the system.
"""
import logging
import numpy as np
import props
import fsl.data.image as fslimage
import fsl.data.strings as strings
import fsl.utils.transform as transform
log = logging.getLogger(__name__)
class DisplayOpts(props.SyncableHasProperties):
def __init__(self, image, display, imageList, displayCtx, parent=None):
props.SyncableHasProperties.__init__(self, parent)
self.image = image
self.display = display
self.imageList = imageList
self.displayCtx = displayCtx
self.imageType = image.imageType
self.name = '{}_{}'.format(type(self).__name__, id(self))
class Display(props.SyncableHasProperties):
"""
"""
name = fslimage.Image.name
"""The image name. This property is bound to the
:attr:`~fsl.data.image.Image.name` property.
"""
imageType = fslimage.Image.imageType
"""The image data type. This property is bound to the
:attr:`~fsl.data.image.Image.imageType` property.
"""
enabled = props.Boolean(default=True)
"""Should this image be displayed at all?"""
alpha = props.Real(minval=0.0, maxval=1.0, default=1.0)
"""Transparency - 1.0 is fully opaque, and 0.0 is fully transparent."""
resolution = props.Real(maxval=10,
default=1,
clamped=True,
editLimits=False)
"""Data resolution in world space. The minimum value is set in __init__."""
volume = props.Int(minval=0, maxval=0, default=0, clamped=True)
"""If a 4D image, the current volume to display."""
syncVolume = props.Boolean(default=True)
transform = props.Choice(
('affine', 'pixdim', 'id'),
labels=[strings.choices['Display.transform.affine'],
strings.choices['Display.transform.pixdim'],
strings.choices['Display.transform.id']],
default='pixdim')
"""This property defines how the image should be transformd into the display
coordinate system.
- ``affine``: Use the affine transformation matrix stored in the image
(the ``qform``/``sform`` fields in NIFTI1 headers).
- ``pixdim``: Scale voxel sizes by the ``pixdim`` fields in the image
header.
- ``id``: Perform no scaling or transformation - voxels will be
interpreted as :math:`1mm^3` isotropic, with the origin at voxel
(0,0,0).
"""
interpolation = props.Choice(
('none', 'linear', 'spline'),
labels=[strings.choices['Display.interpolation.none'],
strings.choices['Display.interpolation.linear'],
strings.choices['Display.interpolation.spline']])
"""How the value shown at a real world location is derived from the
corresponding voxel value(s). 'No interpolation' is equivalent to nearest
neighbour interpolation.
"""
def is4DImage(self):
"""Returns ``True`` if this image is 4D, ``False`` otherwise.
"""
return self.image.is4DImage()
def __init__(self, image, imageList, displayCtx, parent=None):
"""Create a :class:`Display` for the specified image.
:arg image: A :class:`~fsl.data.image.Image` object.
:arg parent:
"""
self.image = image
self.imageList = imageList
self.displayCtx = displayCtx
# bind self.name to image.name, so changes
# in one are propagated to the other
self.bindProps('name', image)
self.bindProps('imageType', image)
# The display<->* transformation matrices
# are created in the _transformChanged method
self.voxToWorldMat = image.voxToWorldMat.transpose()
self.worldToVoxMat = image.worldToVoxMat.transpose()
self.voxToDisplayMat = None
self.displayToVoxMat = None
self.worldToDisplayMat = None
self.displayToWorldMat = None
# is this a 4D volume?
if image.is4DImage():
self.setConstraint('volume', 'maxval', image.shape[3] - 1)
# Update transformation matrices when
# the transform property changes
self.addListener(
'transform',
'Display_{}'.format(id(self)),
self.__transformChanged)
self.addListener(
'imageType',
'Display_{}'.format(id(self)),
self.__imageTypeChanged)
self.__transformChanged()
# When the transform property changes,
# the display<->* transformation matrices
# are recalculated. References to the
# previous matrices are stored here, just
# in case anything (hint the DisplayContext
# object) needs them for any particular
# reason (hint: so the DisplayContext can
# preserve the current display location,
# in terms of image world space, when the
# transform of the selected changes)
self._oldVoxToDisplayMat = self.voxToDisplayMat
self._oldDisplayToVoxMat = self.displayToVoxMat
self._oldWorldToDisplayMat = self.worldToDisplayMat
self._oldDisplayToWorldMat = self.displayToWorldMat
# limit resolution to the image dimensions
self.resolution = min(image.pixdim[:3])
self.setConstraint('resolution', 'minval', self.resolution)
# Call the super constructor after our own
# initialisation, in case the provided parent
# has different property values to our own,
# and our values need to be updated
props.SyncableHasProperties.__init__(
self,
parent,
# The name property is implicitly bound
# through the image object so it doesn't
# need to be linked between ImageDisplays
nobind=['name'],
# These properties cannot be unbound, as
# they affect the OpenGL representation
nounbind=['interpolation',
'volume',
'resolution',
'transform',
'imageType'])
self.__displayOpts = None
self.__imageTypeChanged()
def getDisplayBounds(self):
"""Calculates and returns the min/max values of a 3D bounding box,
in the display coordinate system, which is big enough to contain
the image associated with this :class:`ImageDisplay` instance.
The coordinate system in which the bounding box is defined is
determined by the current value of the :attr:`transform` property.
A tuple containing two values is returned, with the first value
a sequence of three low bounds, and the second value a sequence
of three high bounds.
"""
return transform.axisBounds(self.image.shape[:3], self.voxToDisplayMat)
def getDisplayOpts(self):
"""
"""
if (self.__displayOpts is None) or \
(self.__displayOpts.imageType != self.imageType):
self.__displayOpts = self.__makeDisplayOpts()
return self.__displayOpts
def __makeDisplayOpts(self):
"""
"""
import volumeopts
import tensoropts
import maskopts
if self.getParent() is None:
oParent = None
else:
oParent = self.getParent().getDisplayOpts()
optsMap = {
'volume' : volumeopts.VolumeOpts,
'tensor' : tensoropts.TensorOpts,
'mask' : maskopts. MaskOpts
}
optType = optsMap[self.imageType]
log.debug('Creating DisplayOpts for image {}: {}'.format(
self.name,
optType.__name__))
return optType(self.image,
self,
self.imageList,
self.displayCtx,
oParent)
def __imageTypeChanged(self, *a):
"""
"""
# make sure that the display
# options instance is up to date
self.getDisplayOpts()
def __transformChanged(self, *a):
"""Called when the :attr:`transform` property is changed.
Generates transformation matrices for transforming between voxel and
display coordinate space.
If :attr:`transform` is set to ``affine``, the :attr:`interpolation`
property is changed to ``spline. Otherwise, it is set to ``none``.
"""
# Store references to the previous display related
# transformation matrices (see comments in __init__)
self._oldVoxToDisplayMat = self.voxToDisplayMat
self._oldDisplayToVoxMat = self.displayToVoxMat
self._oldWorldToDisplayMat = self.worldToDisplayMat
self._oldDisplayToWorldMat = self.displayToWorldMat
# The transform property defines the way
# in which image voxel coordinates map
# to the display coordinate system
if self.transform == 'id':
pixdim = [1.0, 1.0, 1.0]
voxToDisplayMat = np.eye(4)
elif self.transform == 'pixdim':
pixdim = self.image.pixdim
voxToDisplayMat = np.diag([pixdim[0], pixdim[1], pixdim[2], 1.0])
elif self.transform == 'affine':
voxToDisplayMat = self.voxToWorldMat
# Transformation matrices for moving between the voxel
# coordinate system and the display coordinate system
self.voxToDisplayMat = np.array(voxToDisplayMat, dtype=np.float32)
self.displayToVoxMat = transform.invert(self.voxToDisplayMat)
# Matrices for moving between the display coordinate
# system, and the image world coordinate system
self.displayToWorldMat = transform.concat(self.displayToVoxMat,
self.voxToWorldMat)
self.worldToDisplayMat = transform.invert(self.displayToWorldMat)
# When transform is changed to 'affine', enable interpolation
# and, when changed to 'pixdim' or 'id', disable interpolation
if self.transform == 'affine': self.interpolation = 'spline'
else: self.interpolation = 'none'