SimpleITK  2.0.0
DicomSeriesFromArray/DicomSeriesFromArray.py
1 # =========================================================================
2 #
3 # Copyright NumFOCUS
4 #
5 # Licensed under the Apache License, Version 2.0 (the "License");
6 # you may not use this file except in compliance with the License.
7 # You may obtain a copy of the License at
8 #
9 # http://www.apache.org/licenses/LICENSE-2.0.txt
10 #
11 # Unless required by applicable law or agreed to in writing, software
12 # distributed under the License is distributed on an "AS IS" BASIS,
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 # See the License for the specific language governing permissions and
15 # limitations under the License.
16 #
17 # =========================================================================
18 
19 import SimpleITK as sitk
20 
21 import sys
22 import time
23 import os
24 import numpy as np
25 
26 pixel_dtypes = {"int16": np.int16,
27  "float64": np.float64}
28 
29 
30 def writeSlices(series_tag_values, new_img, out_dir, i):
31  image_slice = new_img[:, :, i]
32 
33  # Tags shared by the series.
34  list(map(lambda tag_value: image_slice.SetMetaData(tag_value[0],
35  tag_value[1]),
36  series_tag_values))
37 
38  # Slice specific tags.
39  # Instance Creation Date
40  image_slice.SetMetaData("0008|0012", time.strftime("%Y%m%d"))
41  # Instance Creation Time
42  image_slice.SetMetaData("0008|0013", time.strftime("%H%M%S"))
43 
44  # Setting the type to CT so that the slice location is preserved and
45  # the thickness is carried over.
46  image_slice.SetMetaData("0008|0060", "CT")
47 
48  # (0020, 0032) image position patient determines the 3D spacing between
49  # slices.
50  # Image Position (Patient)
51  image_slice.SetMetaData("0020|0032", '\\'.join(
52  map(str, new_img.TransformIndexToPhysicalPoint((0, 0, i)))))
53  # Instance Number
54  image_slice.SetMetaData("0020,0013", str(i))
55 
56  # Write to the output directory and add the extension dcm, to force
57  # writing in DICOM format.
58  writer.SetFileName(os.path.join(out_dir, str(i) + '.dcm'))
59  writer.Execute(image_slice)
60 
61 
62 if len(sys.argv) < 3:
63  print("Usage: python " + __file__ + " <output_directory> [" + ", "
64  .join(pixel_dtypes) + "]")
65  sys.exit(1)
66 
67 # Create a new series from a numpy array
68 try:
69  pixel_dtype = pixel_dtypes[sys.argv[2]]
70 except KeyError:
71  pixel_dtype = pixel_dtypes["int16"]
72 
73 new_arr = np.random.uniform(-10, 10, size=(3, 4, 5)).astype(pixel_dtype)
74 new_img = sitk.GetImageFromArray(new_arr)
75 new_img.SetSpacing([2.5, 3.5, 4.5])
76 
77 # Write the 3D image as a series
78 # IMPORTANT: There are many DICOM tags that need to be updated when you modify
79 # an original image. This is a delicate opration and requires
80 # knowledge of the DICOM standard. This example only modifies some.
81 # For a more complete list of tags that need to be modified see:
82 # http://gdcm.sourceforge.net/wiki/index.php/Writing_DICOM
83 # If it is critical for your work to generate valid DICOM files,
84 # It is recommended to use David Clunie's Dicom3tools to validate
85 # the files:
86 # http://www.dclunie.com/dicom3tools.html
87 
88 writer = sitk.ImageFileWriter()
89 # Use the study/series/frame of reference information given in the meta-data
90 # dictionary and not the automatically generated information from the file IO
91 writer.KeepOriginalImageUIDOn()
92 
93 modification_time = time.strftime("%H%M%S")
94 modification_date = time.strftime("%Y%m%d")
95 
96 # Copy some of the tags and add the relevant tags indicating the change.
97 # For the series instance UID (0020|000e), each of the components is a number,
98 # cannot start with zero, and separated by a '.' We create a unique series ID
99 # using the date and time. Tags of interest:
100 direction = new_img.GetDirection()
101 series_tag_values = [
102  ("0008|0031", modification_time), # Series Time
103  ("0008|0021", modification_date), # Series Date
104  ("0008|0008", "DERIVED\\SECONDARY"), # Image Type
105  ("0020|000e", "1.2.826.0.1.3680043.2.1125."
106  + modification_date + ".1" + modification_time), # Series Instance UID
107  ("0020|0037", '\\'.join(map(str, (direction[0], direction[3], direction[6],
108  direction[1], direction[4],
109  direction[7])))), # Image Orientation
110  # (Patient)
111  ("0008|103e", "Created-SimpleITK") # Series Description
112 ]
113 
114 if pixel_dtype == np.float64:
115  # If we want to write floating point values, we need to use the rescale
116  # slope, "0028|1053", to select the number of digits we want to keep. We
117  # also need to specify additional pixel storage and representation
118  # information.
119  rescale_slope = 0.001 # keep three digits after the decimal point
120  series_tag_values = series_tag_values + [
121  ('0028|1053', str(rescale_slope)), # rescale slope
122  ('0028|1052', '0'), # rescale intercept
123  ('0028|0100', '16'), # bits allocated
124  ('0028|0101', '16'), # bits stored
125  ('0028|0102', '15'), # high bit
126  ('0028|0103', '1')] # pixel representation
127 
128 # Write slices to output directory
129 list(map(lambda i: writeSlices(series_tag_values, new_img, sys.argv[1], i),
130  range(new_img.GetDepth())))
131 
132 # Re-read the series
133 # Read the original series. First obtain the series file names using the
134 # image series reader.
135 data_directory = sys.argv[1]
136 series_IDs = sitk.ImageSeriesReader.GetGDCMSeriesIDs(data_directory)
137 if not series_IDs:
138  print("ERROR: given directory \"" + data_directory +
139  "\" does not contain a DICOM series.")
140  sys.exit(1)
141 series_file_names = sitk.ImageSeriesReader.GetGDCMSeriesFileNames(
142  data_directory, series_IDs[0])
143 
144 series_reader = sitk.ImageSeriesReader()
145 series_reader.SetFileNames(series_file_names)
146 
147 # Configure the reader to load all of the DICOM tags (public+private):
148 # By default tags are not loaded (saves time).
149 # By default if tags are loaded, the private tags are not loaded.
150 # We explicitly configure the reader to load tags, including the
151 # private ones.
152 series_reader.LoadPrivateTagsOn()
153 image3D = series_reader.Execute()
154 print(image3D.GetSpacing(), 'vs', new_img.GetSpacing())
155 sys.exit(0)
itk::simple::ImageSeriesReader
Read series of image files into a SimpleITK image.
Definition: sitkImageSeriesReader.h:45
itk::simple::ImageFileWriter
Write out a SimpleITK image to the specified file location.
Definition: sitkImageFileWriter.h:48