SimpleITK  
ImageRegistrationMethodExhaustive/ImageRegistrationMethodExhaustive.py
1 #!/usr/bin/env python
2 # =========================================================================
3 #
4 # Copyright NumFOCUS
5 #
6 # Licensed under the Apache License, Version 2.0 (the "License");
7 # you may not use this file except in compliance with the License.
8 # You may obtain a copy of the License at
9 #
10 # http://www.apache.org/licenses/LICENSE-2.0.txt
11 #
12 # Unless required by applicable law or agreed to in writing, software
13 # distributed under the License is distributed on an "AS IS" BASIS,
14 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 # See the License for the specific language governing permissions and
16 # limitations under the License.
17 #
18 # =========================================================================
19 
20 """
21 This script demonstrates the use of the Exhaustive optimizer in the
22 ImageRegistrationMethod to estimate a good initial rotation position.
23 
24 Because gradient descent base optimization can get stuck in local
25 minima, a good initial transform is critical for reasonable
26 results. Search a reasonable space on a grid with brute force may be a
27 reliable way to get a starting location for further optimization.
28 
29 The initial translation and center of rotation for the transform is
30 initialized based on the first principle moments of the intensities of
31 the image. Then in either 2D or 3D a Euler transform is used to
32 exhaustively search a grid of the rotation space at a certain step
33 size. The resulting transform is a reasonable guess where to start
34 further registration.
35 """
36 
37 import SimpleITK as sitk
38 import sys
39 import os
40 from math import pi
41 
42 
43 def command_iteration(method):
44  if method.GetOptimizerIteration() == 0:
45  print("Scales: ", method.GetOptimizerScales())
46  print(
47  f"{method.GetOptimizerIteration():3} "
48  + f"= {method.GetMetricValue():7.5f} "
49  + f": {method.GetOptimizerPosition()}"
50  )
51 
52 
53 if len(sys.argv) < 4:
54  print(
55  "Usage:",
56  sys.argv[0],
57  "<fixedImageFilter> <movingImageFile>",
58  "<outputTransformFile>",
59  )
60  sys.exit(1)
61 
62 fixed = sitk.ReadImage(sys.argv[1], sitk.sitkFloat32)
63 
64 moving = sitk.ReadImage(sys.argv[2], sitk.sitkFloat32)
65 
67 
68 R.SetMetricAsMattesMutualInformation(numberOfHistogramBins=50)
69 
70 sample_per_axis = 12
71 if fixed.GetDimension() == 2:
73  # Set the number of samples (radius) in each dimension, with a
74  # default step size of 1.0
75  R.SetOptimizerAsExhaustive([sample_per_axis // 2, 0, 0])
76  # Utilize the scale to set the step size for each dimension
77  R.SetOptimizerScales([2.0 * pi / sample_per_axis, 1.0, 1.0])
78 elif fixed.GetDimension() == 3:
80  R.SetOptimizerAsExhaustive(
81  [
82  sample_per_axis // 2,
83  sample_per_axis // 2,
84  sample_per_axis // 4,
85  0,
86  0,
87  0,
88  ]
89  )
90  R.SetOptimizerScales(
91  [
92  2.0 * pi / sample_per_axis,
93  2.0 * pi / sample_per_axis,
94  2.0 * pi / sample_per_axis,
95  1.0,
96  1.0,
97  1.0,
98  ]
99  )
100 
101 # Initialize the transform with a translation and the center of
102 # rotation from the moments of intensity.
103 tx = sitk.CenteredTransformInitializer(fixed, moving, tx)
104 
105 R.SetInitialTransform(tx)
106 
107 R.SetInterpolator(sitk.sitkLinear)
108 
109 R.AddCommand(sitk.sitkIterationEvent, lambda: command_iteration(R))
110 
111 outTx = R.Execute(fixed, moving)
112 
113 print("-------")
114 print(outTx)
115 print(f"Optimizer stop condition: {R.GetOptimizerStopConditionDescription()}")
116 print(f" Iteration: {R.GetOptimizerIteration()}")
117 print(f" Metric value: {R.GetMetricValue()}")
118 
119 sitk.WriteTransform(outTx, sys.argv[3])
120 
121 if "SITK_NOSHOW" not in os.environ:
122  resampler = sitk.ResampleImageFilter()
123  resampler.SetReferenceImage(fixed)
124  resampler.SetInterpolator(sitk.sitkLinear)
125  resampler.SetDefaultPixelValue(1)
126  resampler.SetTransform(outTx)
127 
128  out = resampler.Execute(moving)
129 
130  simg1 = sitk.Cast(sitk.RescaleIntensity(fixed), sitk.sitkUInt8)
131  simg2 = sitk.Cast(sitk.RescaleIntensity(out), sitk.sitkUInt8)
132  cimg = sitk.Compose(simg1, simg2, simg1 // 2.0 + simg2 // 2.0)
133  sitk.Show(cimg, "ImageRegistrationExhaustive Composition")
itk::simple::WriteTransform
SITKCommon_EXPORT void WriteTransform(const Transform &transform, const std::string &filename)
itk::simple::RescaleIntensity
Image RescaleIntensity(const Image &image1, double outputMinimum=0, double outputMaximum=255)
Applies a linear transformation to the intensity levels of the input Image .
itk::simple::Show
void SITKIO_EXPORT Show(const Image &image, const std::string &title="", const bool debugOn=ProcessObject::GetGlobalDefaultDebug())
itk::simple::ReadImage
SITKIO_EXPORT Image ReadImage(const std::vector< std::string > &fileNames, PixelIDValueEnum outputPixelType=sitkUnknown, const std::string &imageIO="")
ReadImage is a procedural interface to the ImageSeriesReader class which is convenient for most image...
itk::simple::Cast
Image Cast(const Image &image, PixelIDValueEnum pixelID)
itk::simple::Euler2DTransform
A rigid 2D transform with rotation in radians around a fixed center with translation.
Definition: sitkEuler2DTransform.h:35
itk::simple::ResampleImageFilter
Resample an image via a coordinate transform.
Definition: sitkResampleImageFilter.h:53
itk::simple::Compose
Image Compose(const Image &image1, const Image &image2, const Image &image3, const Image &image4, const Image &image5)
ComposeImageFilter combine several scalar images into a multicomponent image.
itk::simple::ImageRegistrationMethod
An interface method to the modular ITKv4 registration framework.
Definition: sitkImageRegistrationMethod.h:89
itk::simple::Euler3DTransform
A rigid 3D transform with rotation in radians around a fixed center with translation.
Definition: sitkEuler3DTransform.h:35
itk::simple::CenteredTransformInitializer
Transform CenteredTransformInitializer(const Image &fixedImage, const Image &movingImage, const Transform &transform, CenteredTransformInitializerFilter::OperationModeType operationMode=itk::simple::CenteredTransformInitializerFilter::MOMENTS)
CenteredTransformInitializer is a helper class intended to initialize the center of rotation and the ...