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

Save output with the same format as brainreg #66

Merged
merged 40 commits into from
Jan 10, 2025
Merged
Show file tree
Hide file tree
Changes from 34 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
d02d294
Added output directory field
IgorTatarnikov Dec 11, 2024
15bf1ea
WIP use output directory as output for elastix
IgorTatarnikov Dec 11, 2024
807e034
Save the deformation fields in the brainreg format to the output dire…
IgorTatarnikov Dec 11, 2024
bc1b8cd
Save deformation fields as tiff not tif
IgorTatarnikov Dec 11, 2024
b7f79eb
Add function to save output from registration
IgorTatarnikov Dec 11, 2024
b2733e7
Update elastix default to save tiff not nii
IgorTatarnikov Dec 11, 2024
189f52b
WIP writing brainreg output directory
IgorTatarnikov Dec 11, 2024
84f010f
WIP calculate area of annotations
IgorTatarnikov Dec 13, 2024
e8f745a
Remap labels outside of float precision to allow transforming of anno…
IgorTatarnikov Dec 13, 2024
2b96d46
Save all files as per brainreg, save the forward and inverse transfor…
IgorTatarnikov Dec 13, 2024
75f70f5
Make ara-tools default options with 2 registration types
IgorTatarnikov Dec 16, 2024
d42ebc6
Changed convert_atlas_labels function to avoid weird edge cases
IgorTatarnikov Dec 17, 2024
e1a5c79
Merge branch 'main' into output-deformation-fields
IgorTatarnikov Dec 17, 2024
ec01d54
Update docstrings
IgorTatarnikov Dec 17, 2024
eea29a6
Add tests for dconverting atlas labels
IgorTatarnikov Dec 17, 2024
f766fdb
Updated docstring for convert_atlas_labels
IgorTatarnikov Dec 17, 2024
c713371
Fixed tests to account for updates for changes to transform select view
IgorTatarnikov Dec 17, 2024
afa7e51
Add pandas as explicit dependency
IgorTatarnikov Dec 17, 2024
a91f176
Update docstring for calculate_areas
IgorTatarnikov Dec 17, 2024
574187b
Added tests for calculating area
IgorTatarnikov Dec 17, 2024
b23bd8f
Remove superfluous function
IgorTatarnikov Dec 17, 2024
d46f771
Remove rigid transformation option
IgorTatarnikov Dec 18, 2024
862b616
Fixed tests to account for removing rigid transformation option
IgorTatarnikov Dec 18, 2024
a3db01b
Add new test data
IgorTatarnikov Dec 20, 2024
47e8683
Update register.py testing such that affine only transforms are run o…
IgorTatarnikov Dec 20, 2024
116c14c
Added tests for register
IgorTatarnikov Dec 20, 2024
87f7c58
Tests for calculate_deformation_field, transform_image
IgorTatarnikov Dec 20, 2024
5f80694
Add tolerance when comparing transform parameter objects
IgorTatarnikov Dec 20, 2024
f53e02c
Add slack for all floating point parameters
IgorTatarnikov Dec 20, 2024
b073fed
Ensure example_mouse_100um is loaded for testing the registration_widget
IgorTatarnikov Dec 20, 2024
b4ceda3
GetImage not View from annotation array
IgorTatarnikov Dec 20, 2024
b10f636
Fixed key error for transforming annotation
IgorTatarnikov Jan 2, 2025
1f18fbb
Removed non-linear regression test files
IgorTatarnikov Jan 2, 2025
01f4ce2
merge main
adamltyson Jan 3, 2025
35f021b
Serialise registration widget to brainglobe_registration.json
IgorTatarnikov Jan 6, 2025
155c6b8
Serialise registration widget to brainglobe_registration.json
IgorTatarnikov Jan 6, 2025
498ce1a
Add debug flag to calculating deformation field
IgorTatarnikov Jan 6, 2025
668bd80
Add indentation to pretify brainglobe-registration.json
IgorTatarnikov Jan 6, 2025
de2714c
Make output_directory mandatory before running registration from the …
IgorTatarnikov Jan 6, 2025
4e276f3
Merge branch 'main' into output-deformation-fields
IgorTatarnikov Jan 7, 2025
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
2 changes: 1 addition & 1 deletion .github/workflows/test_and_deploy.yml
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ name: tests
on:
push:
branches:
- '*'
- 'main'
tags:
- '*'
pull_request:
Expand Down
231 changes: 191 additions & 40 deletions brainglobe_registration/elastix/register.py
Original file line number Diff line number Diff line change
@@ -1,36 +1,22 @@
from typing import List, Tuple
from pathlib import Path
from typing import List, Optional, Tuple

import itk
import numpy as np
import numpy.typing as npt
from brainglobe_atlasapi import BrainGlobeAtlas


def get_atlas_by_name(atlas_name: str) -> BrainGlobeAtlas:
"""
Get a BrainGlobeAtlas object by its name.

Parameters
----------
atlas_name : str
The name of the atlas.

Returns
-------
BrainGlobeAtlas
The BrainGlobeAtlas object.
"""
atlas = BrainGlobeAtlas(atlas_name)

return atlas
from brainglobe_registration.utils.utils import (
convert_atlas_labels,
restore_atlas_labels,
)


def run_registration(
atlas_image,
moving_image,
annotation_image,
atlas_image: npt.NDArray,
moving_image: npt.NDArray,
parameter_lists: List[Tuple[str, dict]],
) -> Tuple[npt.NDArray, itk.ParameterObject, npt.NDArray]:
output_directory: Optional[Path] = None,
) -> Tuple[npt.NDArray, itk.ParameterObject]:
"""
Run the registration process on the given images.

Expand All @@ -40,19 +26,17 @@
The atlas image.
moving_image : npt.NDArray
The moving image.
annotation_image : npt.NDArray
The annotation image.
parameter_lists : List[tuple[str, dict]], optional
The list of parameter lists, by default None
parameter_lists : List[tuple[str, dict]]
The list of registration parameters, one for each transform.
output_directory : Optional[Path], optional
The output directory for the registration results, by default None

Returns
-------
npt.NDArray
The result image.
itk.ParameterObject
The result transform parameters.
npt.NDArray
The transformed annotation image.
"""
# convert to ITK, view only
atlas_image = itk.GetImageViewFromArray(atlas_image).astype(itk.F)
Expand All @@ -66,33 +50,200 @@
parameter_object = setup_parameter_object(parameter_lists=parameter_lists)

elastix_object.SetParameterObject(parameter_object)

# update filter object
elastix_object.UpdateLargestPossibleRegion()

# get results
result_image = elastix_object.GetOutput()
result_transform_parameters = elastix_object.GetTransformParameterObject()
temp_interp_order = result_transform_parameters.GetParameter(

if output_directory:
file_names = [

Check warning on line 60 in brainglobe_registration/elastix/register.py

View check run for this annotation

Codecov / codecov/patch

brainglobe_registration/elastix/register.py#L60

Added line #L60 was not covered by tests
f"{output_directory}/TransformParameters.{i}.txt"
for i in range(len(parameter_lists))
]

itk.ParameterObject.WriteParameterFile(

Check warning on line 65 in brainglobe_registration/elastix/register.py

View check run for this annotation

Codecov / codecov/patch

brainglobe_registration/elastix/register.py#L65

Added line #L65 was not covered by tests
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does it make sense to use pathlib here?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tried using pathlib but ITK wants a string, and crashes otherwise. It felt a bit convoluted to create a bunch of Path objects just to convert them back to strings before using them.

result_transform_parameters, file_names
)

return (
np.asarray(result_image),
result_transform_parameters,
)


def transform_annotation_image(
annotation_image: npt.NDArray[np.uint32],
transform_parameters: itk.ParameterObject,
) -> npt.NDArray[np.uint32]:
"""
Transform the annotation image using the given transform parameters.
Sets the FinalBSplineInterpolationOrder to 0 to avoid interpolation.
Resets the FinalBSplineInterpolationOrder to its original value after
transforming the annotation image.

Parameters
----------
annotation_image : npt.NDArray
The annotation image.
transform_parameters : itk.ParameterObject
The transform parameters.

Returns
-------
npt.NDArray
The transformed annotation image.
"""
adjusted_annotation_image, mapping = convert_atlas_labels(annotation_image)

annotation_image = itk.GetImageFromArray(adjusted_annotation_image).astype(
itk.F
)
temp_interp_order = transform_parameters.GetParameter(
0, "FinalBSplineInterpolationOrder"
)
result_transform_parameters.SetParameter(
"FinalBSplineInterpolationOrder", "0"
transform_parameters.SetParameter("FinalBSplineInterpolationOrder", "0")

transformix_object = itk.TransformixFilter.New(annotation_image)
transformix_object.SetTransformParameterObject(transform_parameters)
transformix_object.UpdateLargestPossibleRegion()

transformed_annotation = transformix_object.GetOutput()

transform_parameters.SetParameter(
"FinalBSplineInterpolationOrder", temp_interp_order
)
transformed_annotation_array = np.asarray(transformed_annotation).astype(
np.uint32
)

annotation_image_transformix = itk.transformix_filter(
annotation_image.astype(np.float32, copy=False),
result_transform_parameters,
transformed_annotation_array = restore_atlas_labels(
transformed_annotation_array, mapping
)

return transformed_annotation_array


def transform_image(
image: npt.NDArray,
transform_parameters: itk.ParameterObject,
) -> npt.NDArray:
"""
Transform the image using the given transform parameters.

Parameters
----------
image: npt.NDArray
The image to transform.
transform_parameters: itk.ParameterObject
The transform parameters.

Returns
-------
npt.NDArray
The transformed image.
"""
image = itk.GetImageViewFromArray(image).astype(itk.F)

transformix_object = itk.TransformixFilter.New(image)
transformix_object.SetTransformParameterObject(transform_parameters)
transformix_object.UpdateLargestPossibleRegion()

transformed_image = transformix_object.GetOutput()

return np.asarray(transformed_image)


def calculate_deformation_field(
moving_image: npt.NDArray,
transform_parameters: itk.ParameterObject,
) -> npt.NDArray:
"""
Calculate the deformation field for the moving image using the given
transform parameters.

Parameters
----------
moving_image : npt.NDArray
The moving image.
transform_parameters : itk.ParameterObject
The transform parameters.

Returns
-------
npt.NDArray
The deformation field.
"""
transformix_object = itk.TransformixFilter.New(
itk.GetImageViewFromArray(moving_image).astype(itk.F),
transform_parameters,
)
transformix_object.SetComputeDeformationField(True)

transformix_object.UpdateLargestPossibleRegion()

# Change from ITK to numpy axes ordering
deformation_field = itk.GetArrayFromImage(
transformix_object.GetOutputDeformationField()
)[..., ::-1]

# Cleanup files generated by elastix
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there any conceivable reason to keep these files, e.g. in a "Debug" mode that could be chosen by the user?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure, the generated file is the deformation field saved in (x, y, z) rather than (z, y, x). I'll add a debug flag to the Python API but won't expose it via the widget at this time.

(Path.cwd() / "DeformationField.tiff").unlink(missing_ok=True)

return deformation_field


def invert_transformation(
fixed_image: npt.NDArray,
parameter_list: List[Tuple[str, dict]],
transform_parameters: itk.ParameterObject,
output_directory: Optional[Path] = None,
) -> itk.ParameterObject:

fixed_image = itk.GetImageFromArray(fixed_image).astype(itk.F)

elastix_object = itk.ElastixRegistrationMethod.New(
fixed_image, fixed_image
)

parameter_object_inverse = setup_parameter_object(parameter_list)

elastix_object.SetInitialTransformParameterObject(transform_parameters)

elastix_object.SetParameterObject(parameter_object_inverse)

elastix_object.UpdateLargestPossibleRegion()

num_initial_transforms = transform_parameters.GetNumberOfParameterMaps()

result_image = elastix_object.GetOutput()
out_parameters = elastix_object.GetTransformParameterObject()
result_transform_parameters = itk.ParameterObject.New()

for i in range(
num_initial_transforms, out_parameters.GetNumberOfParameterMaps()
):
result_transform_parameters.AddParameterMap(
out_parameters.GetParameterMap(i)
)

result_transform_parameters.SetParameter(
"FinalBSplineInterpolationOrder", temp_interp_order
0, "InitialTransformParameterFileName", "NoInitialTransform"
)

if output_directory:
file_names = [

Check warning on line 235 in brainglobe_registration/elastix/register.py

View check run for this annotation

Codecov / codecov/patch

brainglobe_registration/elastix/register.py#L235

Added line #L235 was not covered by tests
f"{output_directory}/InverseTransformParameters.{i}.txt"
for i in range(len(parameter_list))
]

itk.ParameterObject.WriteParameterFiles(

Check warning on line 240 in brainglobe_registration/elastix/register.py

View check run for this annotation

Codecov / codecov/patch

brainglobe_registration/elastix/register.py#L240

Added line #L240 was not covered by tests
result_transform_parameters, file_names
)

return (
np.asarray(result_image),
result_transform_parameters,
np.asarray(annotation_image_transformix),
)


Expand Down
4 changes: 2 additions & 2 deletions brainglobe_registration/parameters/ara_tools/affine.txt
Original file line number Diff line number Diff line change
Expand Up @@ -4,9 +4,9 @@

//ImageTypes
(FixedInternalImagePixelType "float")
(FixedImageDimension 3)
(FixedImageDimension 2)
(MovingInternalImagePixelType "float")
(MovingImageDimension 3)
(MovingImageDimension 2)

//Components
(Registration "MultiResolutionRegistration")
Expand Down
4 changes: 2 additions & 2 deletions brainglobe_registration/parameters/ara_tools/bspline.txt
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,9 @@

//ImageTypes
(FixedInternalImagePixelType "float")
(FixedImageDimension 3)
(FixedImageDimension 2)
(MovingInternalImagePixelType "float")
(MovingImageDimension 3)
(MovingImageDimension 2)

//Components
(Registration "MultiResolutionRegistration")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
(Registration "MultiResolutionRegistration")
(ResampleInterpolator "FinalBSplineInterpolator")
(Resampler "DefaultResampler")
(ResultImageFormat "nii")
(ResultImageFormat "tiff")
(Transform "AffineTransform")
(WriteIterationInfo "false")
(WriteResultImage "true")
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@
(Registration "MultiMetricMultiResolutionRegistration")
(ResampleInterpolator "FinalBSplineInterpolator")
(Resampler "DefaultResampler")
(ResultImageFormat "nii")
(ResultImageFormat "tiff")
(Transform "BSplineTransform")
(WriteIterationInfo "false")
(WriteResultImage "true")
24 changes: 0 additions & 24 deletions brainglobe_registration/parameters/elastix_default/rigid.txt

This file was deleted.

Loading
Loading