Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ENH: API change in TransformChain - new composition convention #165

Merged
merged 5 commits into from
Jul 19, 2022
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 19 additions & 0 deletions nitransforms/linear.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
from scipy import ndimage as ndi

from nibabel.loadsave import load as _nbload
from nibabel.affines import from_matvec

from nitransforms.base import (
ImageGrid,
Expand Down Expand Up @@ -218,6 +219,24 @@ def from_filename(cls, filename, fmt=None, reference=None, moving=None):
f"Could not open <{filename}> (formats tried: {', '.join(fmtlist)})."
)

@classmethod
def from_matvec(cls, mat=None, vec=None, reference=None):
oesteban marked this conversation as resolved.
Show resolved Hide resolved
"""
Create an affine from a matrix and translation pair.

Example
-------
>>> Affine.from_matvec(vec=(4, 0, 0)) # doctest: +NORMALIZE_WHITESPACE
array([[1., 0., 0., 4.],
[0., 1., 0., 0.],
[0., 0., 1., 0.],
[0., 0., 0., 1.]])

"""
mat = mat if mat is not None else np.eye(3)
vec = vec if vec is not None else np.zeros((3,))
return cls(from_matvec(mat, vector=vec), reference=reference)

def __repr__(self):
"""
Change representation to the internal matrix.
Expand Down
42 changes: 32 additions & 10 deletions nitransforms/manip.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
### ### ### ### ### ### ### ### ### ### ### ### ### ### ### ### ### ### ### ##
"""Common interface for transforms."""
from collections.abc import Iterable
import numpy as np

from .base import (
TransformBase,
Expand Down Expand Up @@ -74,8 +75,8 @@ def transforms(self):
@transforms.setter
def transforms(self, value):
self._transforms = _as_chain(value)
if self.transforms[-1].reference:
self.reference = self.transforms[-1].reference
if self.transforms[0].reference:
self.reference = self.transforms[0].reference

def append(self, x):
"""
Expand Down Expand Up @@ -131,18 +132,39 @@ def map(self, x, inverse=False):
raise TransformError("Cannot apply an empty transforms chain.")

transforms = self.transforms
if not inverse:
transforms = self.transforms[::-1]
if inverse:
transforms = list(reversed(self.transforms))

for xfm in transforms:
x = xfm(x, inverse=inverse)

return x

def asaffine(self):
"""Combine a succession of linear transforms into one."""
retval = self.transforms[-1]
for xfm in self.transforms[:-1][::-1]:
def asaffine(self, indices=None):
"""
Combine a succession of linear transforms into one.

Example
------
>>> chain = TransformChain(transforms=[
... Affine.from_matvec(vec=(2, -10, 3)),
... Affine.from_matvec(vec=(-2, 10, -3)),
... ])
>>> chain.asaffine()
array([[1., 0., 0., 0.],
[0., 1., 0., 0.],
[0., 0., 1., 0.],
[0., 0., 0., 1.]])

Parameters
----------
indices : :obj:`numpy.array_like`
The indices of the values to extract.

"""
affines = self.transforms if indices is None else np.take(self.transforms, indices)
retval = affines[0]
for xfm in affines[1:]:
retval @= xfm
return retval

Expand All @@ -157,9 +179,9 @@ def from_filename(cls, filename, fmt="X5", reference=None, moving=None):
xforms = itk.ITKCompositeH5.from_filename(filename)
for xfmobj in xforms:
if isinstance(xfmobj, itk.ITKLinearTransform):
retval.append(Affine(xfmobj.to_ras(), reference=reference))
retval.insert(0, Affine(xfmobj.to_ras(), reference=reference))
else:
retval.append(DisplacementsFieldTransform(xfmobj))
retval.insert(0, DisplacementsFieldTransform(xfmobj))

return TransformChain(retval)

Expand Down
8 changes: 4 additions & 4 deletions nitransforms/tests/test_manip.py
Original file line number Diff line number Diff line change
Expand Up @@ -68,16 +68,16 @@ def test_collapse_affines(tmp_path, data_path, ext0, ext1, ext2):
"""Check whether affines are correctly collapsed."""
chain = TransformChain(
[
Affine.from_filename(
data_path / "regressions" / f"from-scanner_to-bold_mode-image.{ext1}",
fmt=f"{FMT[ext1]}",
),
Affine.from_filename(
data_path
/ "regressions"
/ f"from-fsnative_to-scanner_mode-image.{ext0}",
fmt=f"{FMT[ext0]}",
),
Affine.from_filename(
data_path / "regressions" / f"from-scanner_to-bold_mode-image.{ext1}",
fmt=f"{FMT[ext1]}",
),
]
)
assert np.allclose(
Expand Down