-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhow_to_interact_with_mitk_images_from_python.txt
234 lines (170 loc) · 6.85 KB
/
how_to_interact_with_mitk_images_from_python.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
Title: How to interact with MITK images from Python
Date: 2015-10-09
modified: 2015-10-09
Category: MITK, python
Tags: MITK, python
Author: Luis Javier Salvatierra
Email: [email protected]
Summary: A manual on how to interact with images loaded in MITK and processed in Python.
[TOC]
## Extending your MITK plugin
* I recommend that you build MITK with Boost enabled, also adding `filesystem` to the variable `MITK_USE_Boost_LIBRARIES`. It is not strictly necessary, but recommended for this example.
* Following the previous recommendation, you will need to modify your plugin `CMakeLists.txt`.
```cmake
project(com_company_example_python)
mitk_create_plugin(
EXPORT_DIRECTIVE EXAMPLE_PYTHON_EXPORT
EXPORTED_INCLUDE_SUFFIXES src
MODULE_DEPENDS MitkQtWidgetsExt MitkPython
PACKAGE_DEPENDS Boost #Add this line
)
```
* Boost in your plugin view file `MyView.cpp`.
```cpp
// Boost
#include <boost/filesystem.hpp>
namespace fs = boost::filesystem;
using fs::path;
```
* In order to follow this tutorial you will probably need the plugin source code. You can found it in my repository @ljsalvatierra.
- <a target="_blank" href="https://github.com/ljsalvatierra/mitk-projects/blob/master/LJSProject/Plugins/com.ljsalvatierra.example_python/src/internal/LJSView.cpp">LJSView.cpp</a>
- <a target="_blank" href="https://github.com/ljsalvatierra/mitk-projects/blob/master/python/mitk_python_images/mitk_python_images/raw_images.py">Python module</a>
### Method 1: SimpleITK
* Process a SimpleItk image and load it in the MITK Viewer.
```cpp
//DoImageProcessing()
mitk::Image::Pointer postProcessedImage = this->ProcessSimpleItkImageInPython(image);
this->LoadImage(node, postProcessedImage, name);
```
Where `node` is a `mitk::DataNode` that contains the image. And `name` is one of the properties of the `node`.
I have successfully tested the plugin with Raw `.nrrd` and Dicom `.dcm` images. The Dicom images must be loaded with the `Open Dicom tool`, once that is loaded in the `DataStorage` you can get the image from the `node`.
data:image/s3,"s3://crabby-images/eac91/eac9112ee052c29d5872742b443d36223bda9518" alt="Mitk Dicom Plugin"
data:image/s3,"s3://crabby-images/9544e/9544e60ec021e782c42d147e6e256be10e4414d7" alt="Mitk Dicom Plugin"
data:image/s3,"s3://crabby-images/f0289/f02894c41d9d4c11e3d87d45b1d2288a9ff907fa" alt="Mitk Dicom Plugin"
How do we send the image to our Python module?
- Import the module.
```cpp
StringList code = QStringList()
<< "import sys"
<< "try:"
<< " sys.path.append('" + pythonScriptsPath + "')"
<< " import " + pythonPluginName + " as " + pythonPluginNameAlias
<< " plugin_available = True"
<< "except ImportError:"
<< " plugin_available = False"
<< " raise ImportError('No module named " + pythonPluginName + "')"
```
- If we successfully import the module, call an example function.
```cpp
<< "if plugin_available:"
<< " try:"
<< " " + pythonOutputImage + " = "
+ pythonPluginNameAlias + "." + pythonPluginFunction
+ "(" + pythonInputImage + ")"
<< " except TypeError:"
<< " raise TypeError('Image Type Error')";
```
- Run the script.
```cpp
const std::string command = code.join("\n").toStdString();
m_PythonService->Execute(command, mitk::PythonService::MULTI_LINE_COMMAND);
```
Ok, but where is `pythonOutputImage`?
- Thanks to `PythonService` we can get all the variables saved in the stack.
```cpp
std::vector<mitk::PythonVariable> list = m_PythonService->GetVariableStack();
mitk::Image::Pointer outputImage;
if ( m_PythonService->DoesVariableExist(pythonOutputImage.toStdString()) )
{
outputImage = m_PythonService->CopySimpleItkImageFromPython(pythonOutputImage.toStdString());
}
```
Now that we have the image, how do we load it in the Workbench Viewer?
- Create a new node and populate it.
```cpp
mitk::DataNode::Pointer ds = mitk::DataNode::New();
ds->SetData(image);
ds->SetProperty("name", mitk::StringProperty::New(name + "_postProcessedImage"));
```
- Add the new node to the `DataStorage` with the original image as `Parent node`.
```cpp
this->GetDataStorage()->Add(ds,originalImageNode);
```
- Update the `RenderingManager`.
```cpp
mitk::RenderingManager::GetInstance()->RequestUpdateAll();
```
And that's it, you should see the processed image like this.
data:image/s3,"s3://crabby-images/80dff/80dff42e535d88ec0cdc9253c746d885b7fa0e49" alt="Mitk Processed image"
### Method 2: Other (process an image from path)
`This example doesn't work for dicom images.`
* Process an image from path.
```cpp
std::string imagePath = this->GetDataPath(data);
path p = path(imagePath);
std::string filename = p.filename().string();
std::string postProcessedImagePath = "/tmp/postProcessed_" + filename;
this->ProcessImageInPython(imagePath, postProcessedImagePath);
this->LoadImageFromPath(node, postProcessedImagePath, filename);
```
Where `GetDataPath()` is a function to get the image path inside `mitk::BaseData`.
```cpp
std::string imagePath = (data->GetProperty("path"))->GetValueAsString();
```
How do we send the image path to our Python module?
- Import the module, as in the first method (SimpleITK).
```cpp
QStringList code = QStringList()
<< "import sys"
<< "try:"
<< " sys.path.append('" + pythonScriptsPath + "')"
<< " import " + pythonPluginName + " as " + pythonPluginNameAlias
<< " plugin_available = True"
<< "except ImportError:"
<< " plugin_available = False"
<< " raise ImportError('No module named " + pythonPluginName + "')"
```
- If we successfully import the module, call and example function.
```cpp
<< "if plugin_available:"
<< " try:"
<< " " + pythonPluginNameAlias + "." + pythonPluginFunction
+ "('" + pythonInputImage + "','" + pythonOutputImage + "')"
<< " except TypeError:"
<< " raise TypeError('Image Type Error')";
```
- Run the script.
```cpp
const std::string command = code.join("\n").toStdString();
m_PythonService->Execute(command, mitk::PythonService::MULTI_LINE_COMMAND);
```
Now we need to get the output image path.
- Get the path from the variable stack.
```cpp
std::string outputImage;
std::vector<mitk::PythonVariable> list = m_PythonService->GetVariableStack();
for (auto &i : list){
if ( i.m_Name == pythonOutputImage.toStdString() )
{
outputImage = i.m_Value;
break;
}
}
```
Load the image in the Workbench Viewer.
- Create a new node and load the image from path.
```cpp
mitk::StandaloneDataStorage::Pointer ds = mitk::StandaloneDataStorage::New();
mitk::StandaloneDataStorage::SetOfObjects::Pointer dataNodes = mitk::IOUtil::Load(processedImage, *ds);
mitk::DataNode::Pointer node = dataNodes->at(0);
```
- Add the new node to the `DataStorage` with the original image as `Parent node`.
```cpp
this->GetDataStorage()->Add(node,originalImageNode);
```
- Update the `RenderingManager`.
```cpp
mitk::RenderingManager::GetInstance()->RequestUpdateAll();
```
The result, as before, should look like this.
data:image/s3,"s3://crabby-images/968d3/968d39d1e31e5386be5dc5b0edd79ea4a83e3cc7" alt="Mitk Processed image"