SimpleITK  2.0.0
DicomSeriesReadModifyWrite/DicomSeriesReadModifySeriesWrite.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 from __future__ import print_function
20 
21 import SimpleITK as sitk
22 
23 import sys
24 import time
25 import os
26 
27 if len(sys.argv) < 3:
28  print("Usage: python " + __file__ +
29  " <input_directory_with_DICOM_series> <output_directory>")
30  sys.exit(1)
31 
32 # Read the original series. First obtain the series file names using the
33 # image series reader.
34 data_directory = sys.argv[1]
35 series_IDs = sitk.ImageSeriesReader.GetGDCMSeriesIDs(data_directory)
36 if not series_IDs:
37  print("ERROR: given directory \"" + data_directory +
38  "\" does not contain a DICOM series.")
39  sys.exit(1)
40 series_file_names = sitk.ImageSeriesReader.GetGDCMSeriesFileNames(
41  data_directory, series_IDs[0])
42 
43 series_reader = sitk.ImageSeriesReader()
44 series_reader.SetFileNames(series_file_names)
45 
46 # Configure the reader to load all of the DICOM tags (public+private):
47 # By default tags are not loaded (saves time).
48 # By default if tags are loaded, the private tags are not loaded.
49 # We explicitly configure the reader to load tags, including the
50 # private ones.
51 series_reader.MetaDataDictionaryArrayUpdateOn()
52 series_reader.LoadPrivateTagsOn()
53 image3D = series_reader.Execute()
54 
55 # Modify the image (blurring)
56 filtered_image = sitk.DiscreteGaussian(image3D)
57 
58 # Write the 3D image as a series
59 # IMPORTANT: There are many DICOM tags that need to be updated when you modify
60 # an original image. This is a delicate opration and requires
61 # knowledge of the DICOM standard. This example only modifies some.
62 # For a more complete list of tags that need to be modified see:
63 # http://gdcm.sourceforge.net/wiki/index.php/Writing_DICOM
64 
65 writer = sitk.ImageFileWriter()
66 # Use the study/series/frame of reference information given in the meta-data
67 # dictionary and not the automatically generated information from the file IO
68 writer.KeepOriginalImageUIDOn()
69 
70 # Copy relevant tags from the original meta-data dictionary (private tags are
71 # also accessible).
72 tags_to_copy = ["0010|0010", # Patient Name
73  "0010|0020", # Patient ID
74  "0010|0030", # Patient Birth Date
75  "0020|000D", # Study Instance UID, for machine consumption
76  "0020|0010", # Study ID, for human consumption
77  "0008|0020", # Study Date
78  "0008|0030", # Study Time
79  "0008|0050", # Accession Number
80  "0008|0060" # Modality
81  ]
82 
83 modification_time = time.strftime("%H%M%S")
84 modification_date = time.strftime("%Y%m%d")
85 
86 # Copy some of the tags and add the relevant tags indicating the change.
87 # For the series instance UID (0020|000e), each of the components is a number,
88 # cannot start with zero, and separated by a '.' We create a unique series ID
89 # using the date and time.
90 # Tags of interest:
91 direction = filtered_image.GetDirection()
92 series_tag_values = [
93  (k, series_reader.GetMetaData(0, k))
94  for k in tags_to_copy
95  if series_reader.HasMetaDataKey(0, k)] + \
96  [("0008|0031", modification_time), # Series Time
97  ("0008|0021", modification_date), # Series Date
98  ("0008|0008", "DERIVED\\SECONDARY"), # Image Type
99  ("0020|000e", "1.2.826.0.1.3680043.2.1125." +
100  modification_date + ".1" + modification_time),
101  # Series Instance UID
102  ("0020|0037",
103  '\\'.join(map(str, (direction[0], direction[3],
104  direction[6],
105  # Image Orientation (Patient)
106  direction[1], direction[4],
107  direction[7])))),
108  ("0008|103e",
109  series_reader.GetMetaData(0, "0008|103e")
110  + " Processed-SimpleITK")] # Series Description
111 
112 for i in range(filtered_image.GetDepth()):
113  image_slice = filtered_image[:, :, i]
114  # Tags shared by the series.
115  for tag, value in series_tag_values:
116  image_slice.SetMetaData(tag, value)
117  # Slice specific tags.
118  # Instance Creation Date
119  image_slice.SetMetaData("0008|0012", time.strftime("%Y%m%d"))
120  # Instance Creation Time
121  image_slice.SetMetaData("0008|0013", time.strftime("%H%M%S"))
122  # Image Position (Patient)
123  image_slice.SetMetaData("0020|0032", '\\'.join(
124  map(str, filtered_image.TransformIndexToPhysicalPoint((0, 0, i)))))
125  # Instace Number
126  image_slice.SetMetaData("0020|0013", str(i))
127 
128  # Write to the output directory and add the extension dcm, to force writing
129  # in DICOM format.
130  writer.SetFileName(os.path.join(sys.argv[2], str(i) + '.dcm'))
131  writer.Execute(image_slice)
132 sys.exit(0)
itk::simple::DiscreteGaussian
Image DiscreteGaussian(const Image &image1, std::vector< double > variance=std::vector< double >(3, 1.0), unsigned int maximumKernelWidth=32u, std::vector< double > maximumError=std::vector< double >(3, 0.01), bool useImageSpacing=true)
Blurs an image by separable convolution with discrete gaussian kernels. This filter performs Gaussian...
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