Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
F
fslpy
Manage
Activity
Members
Code
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Build
Pipelines
Jobs
Pipeline schedules
Artifacts
Deploy
Releases
Package registry
Model registry
Operate
Environments
Terraform modules
Analyze
Contributor analytics
CI/CD analytics
Repository analytics
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
Evan Edmond
fslpy
Commits
fc2a1e61
Commit
fc2a1e61
authored
5 years ago
by
Paul McCarthy
Browse files
Options
Downloads
Patches
Plain Diff
ENH: Finish coefficientFieldToDisplacementField implementation. Add some docs
parent
98e9ddeb
No related branches found
Branches containing commit
No related tags found
Tags containing commit
No related merge requests found
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
fsl/transform/affine.py
+1
-1
1 addition, 1 deletion
fsl/transform/affine.py
fsl/transform/fnirt.py
+11
-5
11 additions, 5 deletions
fsl/transform/fnirt.py
fsl/transform/nonlinear.py
+158
-34
158 additions, 34 deletions
fsl/transform/nonlinear.py
with
170 additions
and
40 deletions
fsl/transform/affine.py
+
1
−
1
View file @
fc2a1e61
...
@@ -5,7 +5,7 @@
...
@@ -5,7 +5,7 @@
# Author: Paul McCarthy <pauldmccarthy@gmail.com>
# Author: Paul McCarthy <pauldmccarthy@gmail.com>
#
#
"""
This module contains utility functions for working with affine
"""
This module contains utility functions for working with affine
transformations. The following func
y
ions are available:
transformations. The following func
t
ions are available:
.. autosummary::
.. autosummary::
:nosignatures:
:nosignatures:
...
...
This diff is collapsed.
Click to expand it.
fsl/transform/fnirt.py
+
11
−
5
View file @
fc2a1e61
...
@@ -88,13 +88,19 @@ def _readFnirtCoefficientField(fname, img, src, ref):
...
@@ -88,13 +88,19 @@ def _readFnirtCoefficientField(fname, img, src, ref):
# The sform contains an initial
# The sform contains an initial
# global src-to-ref affine
# global src-to-ref affine
# (the starting point for the
# (the starting point for the
# non-linear registration)
# non-linear registration). This
# is encoded as a flirt matrix,
# i.e. it transforms from
# source-scaled-voxels to
# ref-scaled-voxels
srcToRefMat
=
img
.
header
.
get_sform
()
srcToRefMat
=
img
.
header
.
get_sform
()
# The fieldToRefMat affine allows us
# The fieldToRefMat affine tells
# to transform coefficient field voxel
# the CoefficientField class how
# coordinates into displacement field/
# to transform coefficient field
# reference image voxel coordinates.
# voxel coordinates into
# displacement field/reference
# image voxel coordinates.
fieldToRefMat
=
affine
.
scaleOffsetXform
(
knotSpacing
,
0
)
fieldToRefMat
=
affine
.
scaleOffsetXform
(
knotSpacing
,
0
)
return
nonlinear
.
CoefficientField
(
fname
,
return
nonlinear
.
CoefficientField
(
fname
,
...
...
This diff is collapsed.
Click to expand it.
fsl/transform/nonlinear.py
+
158
−
34
View file @
fc2a1e61
#!/usr/bin/env python
#!/usr/bin/env python
#
#
# nonlinear.py -
# nonlinear.py -
Functions/classes for non-linear transformations.
#
#
# Author: Paul McCarthy <pauldmccarthy@gmail.com>
# Author: Paul McCarthy <pauldmccarthy@gmail.com>
#
#
...
@@ -216,7 +216,7 @@ class DisplacementField(NonLinearTransform):
...
@@ -216,7 +216,7 @@ class DisplacementField(NonLinearTransform):
reference space.
reference space.
:arg from_: Reference image space that ``coords`` are defined in
:arg from_: Reference image space that ``coords`` are defined in
:arg to: Source image space to transform ``coords`` into
:arg to: Source image space to transform ``coords`` into
:returns ``coords``, transformed into the source image space
:returns
``coords``, transformed into the source image space
"""
"""
if
from_
is
None
:
from_
=
self
.
refSpace
if
from_
is
None
:
from_
=
self
.
refSpace
...
@@ -270,10 +270,24 @@ class DisplacementField(NonLinearTransform):
...
@@ -270,10 +270,24 @@ class DisplacementField(NonLinearTransform):
class
CoefficientField
(
NonLinearTransform
):
class
CoefficientField
(
NonLinearTransform
):
"""
Class which represents a quadratic or cubic B-spline coefficient field
"""
Class which represents a cubic B-spline coefficient field generated by
generated by FNIRT.
FNIRT.
The :meth:`displacements` method can be used to calculate relative
displacements for a set of reference space voxel coordinates.
A FNIRT coefficient field typically contains a *premat*, a global affine
transformation from the source space to the reference space, which was
used as the starting point for the non-linear optimisation performed by
FNIRT.
This affine must be provided when creating a ``CoefficientField``, and is
subsequently accessed via the :meth:`srcToRefMat` or :meth:`premat`
attributes.
"""
"""
def
__init__
(
self
,
def
__init__
(
self
,
image
,
image
,
src
,
src
,
...
@@ -287,13 +301,27 @@ class CoefficientField(NonLinearTransform):
...
@@ -287,13 +301,27 @@ class CoefficientField(NonLinearTransform):
**
kwargs
):
**
kwargs
):
"""
Create a ``CoefficientField``.
"""
Create a ``CoefficientField``.
:arg fieldType:
:arg fieldType: Must be ``
'
cubic
'
``
:arg knotSpacing:
:arg srcToRefMat:
:arg knotSpacing: A tuple containing the spline knot spacings along
:arg fieldToRefMat:
each axis.
:arg srcToRefMat: Initial global affine transformation from the
source image to the reference image. This is
assumed to be a FLIRT-style matrix, i.e. it
transforms from source image FSL coordinates
into reference image FSL coordinates (scaled
voxels).
:arg fieldToRefMat: Affine transformation which can transform reference
image voxel coordinates into coefficient field
voxel coordinates.
See the :class:`NonLinearTransform` class for details on the other
arguments.
"""
"""
if
fieldType
not
in
(
'
quadratic
'
,
'
cubic
'
):
if
fieldType
not
in
(
'
cubic
'
,
):
raise
ValueError
(
'
Unsupported field type: {}
'
.
format
(
fieldType
))
raise
ValueError
(
'
Unsupported field type: {}
'
.
format
(
fieldType
))
NonLinearTransform
.
__init__
(
self
,
NonLinearTransform
.
__init__
(
self
,
...
@@ -302,7 +330,6 @@ class CoefficientField(NonLinearTransform):
...
@@ -302,7 +330,6 @@ class CoefficientField(NonLinearTransform):
ref
,
ref
,
srcSpace
,
srcSpace
,
refSpace
,
refSpace
,
refSpace
=
'
voxel
'
,
**
kwargs
)
**
kwargs
)
self
.
__fieldType
=
fieldType
self
.
__fieldType
=
fieldType
...
@@ -314,8 +341,8 @@ class CoefficientField(NonLinearTransform):
...
@@ -314,8 +341,8 @@ class CoefficientField(NonLinearTransform):
@property
@property
def
fieldType
(
self
):
def
fieldType
(
self
):
"""
Return the type of the coefficient field,
either ``
'
cubic
'
`` or
"""
Return the type of the coefficient field,
currently always
``
'
quadrat
ic
'
``.
``
'
cub
ic
'
``.
"""
"""
return
self
.
__fieldType
return
self
.
__fieldType
...
@@ -352,7 +379,20 @@ class CoefficientField(NonLinearTransform):
...
@@ -352,7 +379,20 @@ class CoefficientField(NonLinearTransform):
return
np
.
copy
(
self
.
__refToFieldMat
)
return
np
.
copy
(
self
.
__refToFieldMat
)
def
transform
(
self
,
coords
,
from_
=
None
,
to
=
None
):
def
transform
(
self
,
coords
,
from_
=
None
,
to
=
None
,
premat
=
True
):
"""
Overrides :meth:`NonLinearTransform.transform`. Transforms the
given ``coords`` from the reference image space into the source image
space.
:arg coords: A sequence of XYZ coordinates, or ``numpy`` array of shape
``(n, 3)`` containing ``n`` sets of coordinates in the
reference space.
:arg from_: Reference image space that ``coords`` are defined in
:arg to: Source image space to transform ``coords`` into
:returns ``coords``, transformed into the source image space
:arg premat: If ``True``, the inverse :meth:`srcToRefMat` is applied
to
"""
raise
NotImplementedError
()
raise
NotImplementedError
()
...
@@ -368,7 +408,8 @@ class CoefficientField(NonLinearTransform):
...
@@ -368,7 +408,8 @@ class CoefficientField(NonLinearTransform):
raise
NotImplementedError
()
raise
NotImplementedError
()
# See
# See
# https://www.cs.jhu.edu/~cis/cista/746/papers/RueckertFreeFormBreastMRI.pdf
# https://www.cs.jhu.edu/~cis/cista/746/papers/\
# RueckertFreeFormBreastMRI.pdf
# https://www.fmrib.ox.ac.uk/datasets/techrep/tr07ja2/tr07ja2.pdf
# https://www.fmrib.ox.ac.uk/datasets/techrep/tr07ja2/tr07ja2.pdf
# Cubic b-spline basis functions
# Cubic b-spline basis functions
...
@@ -412,12 +453,12 @@ class CoefficientField(NonLinearTransform):
...
@@ -412,12 +453,12 @@ class CoefficientField(NonLinearTransform):
il
=
i
+
l
il
=
i
+
l
jm
=
j
+
m
jm
=
j
+
m
kn
=
k
+
n
kn
=
k
+
n
mask
=
(
(
il
>=
0
)
&
mask
=
(
il
>=
0
)
&
\
(
il
<
nx
)
&
(
il
<
nx
)
&
\
(
jm
>=
0
)
&
(
jm
>=
0
)
&
\
(
jm
<
ny
)
&
(
jm
<
ny
)
&
\
(
kn
>=
0
)
&
(
kn
>=
0
)
&
\
(
kn
<
nz
)
)
(
kn
<
nz
)
il
=
il
[
mask
]
il
=
il
[
mask
]
jm
=
jm
[
mask
]
jm
=
jm
[
mask
]
...
@@ -567,14 +608,24 @@ def convertDisplacementSpace(field, from_, to):
...
@@ -567,14 +608,24 @@ def convertDisplacementSpace(field, from_, to):
dispType
=
field
.
displacementType
)
dispType
=
field
.
displacementType
)
def
coefficientFieldToDisplacementField
(
field
):
def
coefficientFieldToDisplacementField
(
field
,
"""
Convert a :class:`CoefficientField` into a relative
dispType
=
'
relative
'
,
:class:`DisplacementField`.
premat
=
True
):
"""
Convert a :class:`CoefficientField` into a :class:`DisplacementField`.
:arg field: :class:`CoefficientField` to convert
:arg dispType: The type of displcaement field - either ``
'
relative
'
`` (the
default) or ``
'
absolute
'
``.
:arg field: :class:`CoefficientField` to convert
:arg premat: If ``True`` (the default), the :meth:`srcToRefMat` is
:return: :class:`DisplacementField` calculated from ``field``
encoded into the displacements.
:return: :class:`DisplacementField` calculated from ``field``.
"""
"""
# Generate coordinates for every
# voxel in the reference image
ix
,
iy
,
iz
=
field
.
ref
.
shape
[:
3
]
ix
,
iy
,
iz
=
field
.
ref
.
shape
[:
3
]
x
,
y
,
z
=
np
.
meshgrid
(
np
.
arange
(
ix
),
x
,
y
,
z
=
np
.
meshgrid
(
np
.
arange
(
ix
),
np
.
arange
(
iy
),
np
.
arange
(
iy
),
...
@@ -583,12 +634,85 @@ def coefficientFieldToDisplacementField(field):
...
@@ -583,12 +634,85 @@ def coefficientFieldToDisplacementField(field):
y
=
y
.
flatten
()
y
=
y
.
flatten
()
z
=
z
.
flatten
()
z
=
z
.
flatten
()
xyz
=
np
.
vstack
((
x
,
y
,
z
)).
T
xyz
=
np
.
vstack
((
x
,
y
,
z
)).
T
disps
=
field
.
displacements
(
xyz
).
reshape
((
ix
,
iy
,
iz
,
3
))
# There are three spaces to consider here:
return
DisplacementField
(
disps
,
#
src
=
field
.
src
,
# - ref space: Reference image scaled voxels ("fsl" space)
ref
=
field
.
ref
,
#
srcSpace
=
field
.
srcSpace
,
# - aligned-src space: Source image scaled voxels, after the
refSpace
=
field
.
refSpace
,
# source image has been linearly aligned to
header
=
field
.
ref
.
header
,
# the reference via the field.srcToRefMat
dispType
=
'
relative
'
)
# This will typically be equivalent to ref
# space
#
# - orig-src space: Source image scaled voxels, in the coordinate
# system of the original source image, without
# linear alignment to the reference image
# The displacements method will
# return relative displacements
# from ref space to aligned-src
# space.
disps
=
field
.
displacements
(
xyz
).
reshape
((
ix
,
iy
,
iz
,
3
))
rdfield
=
DisplacementField
(
disps
,
src
=
field
.
src
,
ref
=
field
.
ref
,
srcSpace
=
field
.
srcSpace
,
refSpace
=
field
.
refSpace
,
header
=
field
.
ref
.
header
,
dispType
=
'
relative
'
)
if
(
dispType
==
'
relative
'
)
and
(
not
premat
):
return
rdfield
# Convert to absolute - the
# displacements will now be
# absolute coordinates in
# aligned-src space
disps
=
convertDisplacementType
(
rdfield
)
# Apply the premat if requested -
# this will transform the coordinates
# from aligned-src to orig-src space.
if
premat
:
# We apply the premat in the same way
# that fnirtfileutils does - applying
# the inverse affine to every ref space
# voxel coordinate, then adding it to
# the existing displacements.
shape
=
disps
.
shape
disps
=
disps
.
reshape
(
-
1
,
3
)
refToSrc
=
affine
.
invert
(
field
.
srcToRefMat
)
premat
=
affine
.
concat
(
refToSrc
-
np
.
eye
(
4
),
field
.
ref
.
getAffine
(
'
voxel
'
,
'
fsl
'
))
disps
=
disps
+
affine
.
transform
(
xyz
,
premat
)
disps
=
disps
.
reshape
(
shape
)
# note that convertwarp applies a premat
# differently - its method is equivalent
# to directly transforming the existing
# absolute displacements, i.e.:
#
# disps = affine.transform(disps, refToSrc)
adfield
=
DisplacementField
(
disps
,
src
=
field
.
src
,
ref
=
field
.
ref
,
srcSpace
=
field
.
srcSpace
,
refSpace
=
field
.
refSpace
,
header
=
field
.
ref
.
header
,
dispType
=
'
absolute
'
)
# Not either return absolute displacements,
# or convert back to relative displacements
if
dispType
==
'
absolute
'
:
return
adfield
else
:
return
DisplacementField
(
convertDisplacementType
(
adfield
),
src
=
field
.
src
,
ref
=
field
.
ref
,
srcSpace
=
field
.
srcSpace
,
refSpace
=
field
.
refSpace
,
header
=
field
.
ref
.
header
,
dispType
=
'
relative
'
)
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment