/
ImageData.java
386 lines (304 loc) · 11.3 KB
/
ImageData.java
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
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
/*-
* #%L
* This file is part of QuPath.
* %%
* Copyright (C) 2014 - 2016 The Queen's University of Belfast, Northern Ireland
* Contact: IP Management (ipmanagement@qub.ac.uk)
* %%
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public
* License along with this program. If not, see
* <http://www.gnu.org/licenses/gpl-3.0.html>.
* #L%
*/
package qupath.lib.images;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import qupath.lib.color.ColorDeconvolutionStains;
import qupath.lib.color.ColorDeconvolutionStains.DEFAULT_CD_STAINS;
import qupath.lib.common.GeneralTools;
import qupath.lib.images.servers.ImageServer;
import qupath.lib.objects.hierarchy.PathObjectHierarchy;
import qupath.lib.objects.hierarchy.events.PathObjectHierarchyEvent;
import qupath.lib.objects.hierarchy.events.PathObjectHierarchyListener;
import qupath.lib.plugins.workflow.DefaultScriptableWorkflowStep;
import qupath.lib.plugins.workflow.Workflow;
import qupath.lib.plugins.workflow.WorkflowListener;
import qupath.lib.plugins.workflow.WorkflowStep;
/**
* Class that brings together the main data in connection with the analysis of a single image.
*
* Currently, this is really the server (to access the image & its pixels) and the object hierarchy that represents detections.
* In addition, there is an ImageType - as some options may change depending on this.
* One particularly significant example is that of Brightfield images in pathology, for which stain vectors are often required for
* effective stain separation.
*
* @author Pete Bankhead
*
*/
public class ImageData<T> implements WorkflowListener, PathObjectHierarchyListener {
public enum ImageType {
BRIGHTFIELD_H_DAB("Brightfield (H-DAB)"),
BRIGHTFIELD_H_E("Brightfield (H&E)"),
BRIGHTFIELD_OTHER("Brightfield (other)"),
FLUORESCENCE("Fluorescence"),
OTHER("Other"),
UNSET("Not set");
private final String text;
ImageType(String text) {
this.text = text;
}
@Override
public String toString() {
return text;
}
}
final private static Logger logger = LoggerFactory.getLogger(ImageData.class);
transient private PropertyChangeSupport pcs;
transient private ImageServer<T> server;
private String lastSavedPath = null;
private String serverPath;
private PathObjectHierarchy hierarchy;
private ImageType type;
// A log of steps that have been applied
private Workflow workflow = new Workflow();
// Really just one set of stains will be used, as there will only be one ImageType -
// but here we store stains by ImageType anyway so that they are not instantly forgotten if the ImageType changes
// Will return null for unsupported ImageTypes
private Map<ImageType, ColorDeconvolutionStains> stainMap = new HashMap<>();
private Map<String, Object> propertiesMap = new HashMap<>();
protected boolean changes = false; // Indicating changes since this ImageData was last saved
public ImageData(ImageServer<T> server, PathObjectHierarchy hierarchy, ImageType type) {
pcs = new PropertyChangeSupport(this);
this.server = server;
this.hierarchy = hierarchy == null ? new PathObjectHierarchy() : hierarchy;
this.serverPath = server == null ? null : server.getPath(); // TODO: Deal with sub image servers
initializeStainMap();
if (type == null)
type = ImageType.UNSET;
setImageType(type);
// Add listeners for changes
this.hierarchy.addPathObjectListener(this);
workflow.addWorkflowListener(this);
// Discard any changes during construction
changes = false;
}
public ImageData(ImageServer<T> server, ImageType type) {
this(server, new PathObjectHierarchy(), type);
}
/**
* Get a workflow representing a history of the processing steps applied to the ImageData.
*
* @return
*/
public Workflow getHistoryWorkflow() {
return workflow;
}
/**
* Create a new ImageData with ImageType.UNKNOWN.
*
* @param server
* @param hierarchy
*/
public ImageData(ImageServer<T> server, PathObjectHierarchy hierarchy) {
this(server, hierarchy, null);
}
/**
* Create a new ImageData with ImageType.UNKNOWN and a new PathObjectHierarchy.
*
* @param server
* @param hierarchy
*/
public ImageData(ImageServer<T> server) {
this(server, new PathObjectHierarchy());
}
private void initializeStainMap() {
stainMap.put(ImageType.BRIGHTFIELD_H_DAB, ColorDeconvolutionStains.makeDefaultColorDeconvolutionStains(DEFAULT_CD_STAINS.H_DAB));
stainMap.put(ImageType.BRIGHTFIELD_H_E, ColorDeconvolutionStains.makeDefaultColorDeconvolutionStains(DEFAULT_CD_STAINS.H_E));
stainMap.put(ImageType.BRIGHTFIELD_OTHER, ColorDeconvolutionStains.makeDefaultColorDeconvolutionStains(DEFAULT_CD_STAINS.H_DAB));
}
/**
* Set the color deconvolution stain vectors for the current image type.
* If the type is not brightfield, an IllegalArgumentException is thrown.
*
* @param stains
*/
public void setColorDeconvolutionStains(ColorDeconvolutionStains stains) {
if (!isBrightfield())
throw new IllegalArgumentException("Cannot set color deconvolution stains for image type " + type);
ColorDeconvolutionStains stainsOld = stainMap.put(type, stains);
pcs.firePropertyChange("stains", stainsOld, stains);
addColorDeconvolutionStainsToWorkflow(this);
// logger.error("WARNING: Setting color deconvolution stains is not yet scriptable!!!!");
changes = true;
}
public void setColorDeconvolutionStains(final String stainsString) {
setColorDeconvolutionStains(ColorDeconvolutionStains.parseColorDeconvolutionStainsArg(stainsString));
}
public boolean isBrightfield() {
return getImageType().toString().toLowerCase().startsWith("brightfield");
}
public boolean isFluorescence() {
return getImageType() == ImageType.FLUORESCENCE;
}
public void setImageType(final String type) {
setImageType(ImageType.valueOf(type));
}
public void setImageType(final ImageType type) {
if (this.type == type)
return;
ImageType oldType = this.type;
this.type = type;
// Log the step
getHistoryWorkflow().addStep(
new DefaultScriptableWorkflowStep("Set image type",
Collections.singletonMap("Image type", type),
"setImageType(\'" + type.name() + "');")
);
if (isBrightfield())
addColorDeconvolutionStainsToWorkflow(this);
// TODO: REINTRODUCE LOGGING!
// // Log the step
// getWorkflow().addStep(
// new DefaultScriptableWorkflowStep("Set image type",
// Collections.singletonMap("Image type", type),
// QP.class.getSimpleName() + ".setImageType(\'" + type.toString() + "');")
// );
// if (isBrightfield())
// addColorDeconvolutionStainsToWorkflow(this);
pcs.firePropertyChange("imageType", oldType, type);
changes = true;
}
// TODO: REINTRODUCE LOGGING!
static void addColorDeconvolutionStainsToWorkflow(ImageData<?> imageData) {
// logger.warn("Color deconvolution stain logging not currently enabled!");
ColorDeconvolutionStains stains = imageData.getColorDeconvolutionStains();
if (stains == null) {
return;
}
String arg = ColorDeconvolutionStains.getColorDeconvolutionStainsAsString(imageData.getColorDeconvolutionStains(), 5);
Map<String, String> map = GeneralTools.parseArgStringValues(arg);
WorkflowStep lastStep = imageData.getHistoryWorkflow().getLastStep();
String commandName = "Set color deconvolution stains";
WorkflowStep newStep = new DefaultScriptableWorkflowStep(commandName,
map,
"setColorDeconvolutionStains(\'" + arg + "');");
if (lastStep != null && commandName.equals(lastStep.getName()))
imageData.getHistoryWorkflow().replaceLastStep(newStep);
else
imageData.getHistoryWorkflow().addStep(newStep);
// ColorDeconvolutionStains stains = imageData.getColorDeconvolutionStains();
// if (stains == null)
// return;
//
// String arg = ColorDeconvolutionStains.getColorDeconvolutionStainsAsString(imageData.getColorDeconvolutionStains(), 5);
// Map<String, String> map = GeneralTools.parseArgStringValues(arg);
// WorkflowStep lastStep = imageData.getWorkflow().getLastStep();
// String commandName = "Set color deconvolution stains";
// WorkflowStep newStep = new DefaultScriptableWorkflowStep(commandName,
// map,
// QP.class.getSimpleName() + ".setColorDeconvolutionStains(\'" + arg + "');");
//
// if (lastStep != null && commandName.equals(lastStep.getName()))
// imageData.getWorkflow().replaceLastStep(newStep);
// else
// imageData.getWorkflow().addStep(newStep);
}
public ImageServer<T> getServer() {
return server;
}
public String getServerPath() {
return serverPath;
}
public PathObjectHierarchy getHierarchy() {
return hierarchy;
}
public ImageType getImageType() {
return type;
}
public ColorDeconvolutionStains getColorDeconvolutionStains() {
return stainMap.get(getImageType());
}
public void addPropertyChangeListener(PropertyChangeListener listener) {
if (pcs == null)
pcs = new PropertyChangeSupport(this);
this.pcs.addPropertyChangeListener(listener);
}
public void removePropertyChangeListener(PropertyChangeListener listener) {
this.pcs.removePropertyChangeListener(listener);
}
public Object getProperty(String key) {
return propertiesMap.get(key);
}
public Object setProperty(String key, Object value) {
changes = true;
return propertiesMap.put(key, value);
}
public Object removeProperty(String key) {
if (propertiesMap.containsKey(key)) {
changes = true;
return propertiesMap.remove(key);
}
return null;
}
public Map<String, Object> getProperties() {
return Collections.unmodifiableMap(propertiesMap);
}
/**
* Get the last path used to save this object;
* @return
*/
public String getLastSavedPath() {
return lastSavedPath;
}
/**
* Returns true if changes have been recorded since the last time this object was notified that it was saved.
*
* @return
*
* @see setLastSavedPath
*/
public boolean isChanged() {
return changes;
}
/**
* Set the last path used to save this object;
*
* @param path
* @param resetChanged If true, then the isChanged() flag will be reset to false;
*/
public void setLastSavedPath(final String path, final boolean resetChanged) {
this.lastSavedPath = path;
if (resetChanged)
this.changes = false;
}
@Override
public void hierarchyChanged(PathObjectHierarchyEvent event) {
changes = true;
}
@Override
public void workflowUpdated(Workflow workflow) {
changes = true;
}
@Override
public String toString() {
if (getServer() == null)
return "ImageData: " + getImageType() + ", no server";
else
return "ImageData: " + getImageType() + ", " + getServer().getShortServerName();
}
}