SimpleITK  2.0.0
SliceBySliceDecorator/SliceBySliceDecorator.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 from __future__ import print_function
21 
22 import SimpleITK as sitk
23 import sys
24 
25 import itertools
26 from functools import wraps
27 
28 
29 def slice_by_slice_decorator(func):
30  """
31  A function decorator which executes func on each 3D sub-volume and *in-place* pastes the results into the
32  input image. The input image type and the output image type are required to be the same type.
33 
34  :param func: A function which take a SimpleITK Image as it's first argument and returns an Image as results.
35  :return: A decorated function.
36  """
37 
38  iter_dim = 2
39 
40  @wraps(func)
41  def slice_by_slice(image, *args, **kwargs):
42 
43  dim = image.GetDimension()
44 
45  if dim <= iter_dim:
46  #
47  image = func(image, *args, **kwargs)
48  return image
49 
50  extract_size = list(image.GetSize())
51  extract_size[iter_dim:] = itertools.repeat(0, dim-iter_dim)
52 
53  extract_index = [0] * dim
54  paste_idx = [slice(None, None)] * dim
55 
56  extractor = sitk.ExtractImageFilter()
57  extractor.SetSize(extract_size)
58 
59  for high_idx in itertools.product(*[range(s) for s in image.GetSize()[iter_dim:]]):
60 
61  # The lower 2 elements of extract_index are always 0.
62  # The remaining indices are iterated through all indexes.
63  extract_index[iter_dim:] = high_idx
64  extractor.SetIndex(extract_index)
65 
66  # Sliced based indexing for setting image values internally uses the PasteImageFilter executed "inplace".
67  # The lower 2 elements are equivalent to ":". For a less general case the assignment could be written
68  # as image[:,:,z] = ...
69  paste_idx[iter_dim:] = high_idx
70  image[paste_idx] = func(extractor.Execute(image), *args, **kwargs)
71 
72  return image
73 
74  return slice_by_slice
75 
76 
77 if len(sys.argv) < 3:
78  print("Usage: SubDimensionProcess inputImage outputImage", file=sys.stderr)
79  sys.exit(1)
80 
81 inputImage = sitk.ReadImage(sys.argv[1])
82 
83 # Decorate the function
84 adaptive_histogram_equalization_2d = slice_by_slice_decorator(sitk.AdaptiveHistogramEqualization)
85 
86 adaptive_histogram_equalization_2d(inputImage, radius=[20]*2, alpha=0.3, beta=0.3)
87 
88 sitk.WriteImage(inputImage, sys.argv[2])
itk::simple::WriteImage
SITKIO_EXPORT void WriteImage(const Image &image, const std::vector< std::string > &fileNames, bool useCompression=false, int compressionLevel=-1)
WriteImage is a procedural interface to the ImageSeriesWriter. class which is convenient for many ima...
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::ExtractImageFilter
Decrease the image size by cropping the image to the selected region bounds.
Definition: sitkExtractImageFilter.h:72