@@ -121,74 +121,56 @@ def opening(
121
121
background_only :bool = True ,
122
122
parallel :int = 0 ,
123
123
mode :Mode = Mode .multilabel ,
124
- iterations :int = 1 ,
125
124
erode_border :bool = True ,
126
125
) -> np .ndarray :
127
126
"""Performs morphological opening of labels.
128
127
128
+ This operator is idempotent with the exception
129
+ of boundary effects.
130
+
129
131
background_only is passed through to dilate.
130
132
True: Only evaluate background voxels for dilation.
131
133
False: Allow labels to erode each other as they grow.
132
134
parallel: how many pthreads to use in a threadpool
133
135
mode:
134
136
Mode.multilabel: are all surrounding pixels the same?
135
137
Mode.grey: use grayscale image dilation (min value)
136
-
137
- iterations: number of times to iterate the result
138
-
139
138
erode_border: if True, the border is treated as background,
140
139
else it is regarded as a value that would preserve the
141
140
current value. Only has an effect for multilabel erosion.
142
141
"""
143
- if iterations < 0 :
144
- raise ValueError (f"iterations ({ iterations } ) must be a positive integer." )
145
- elif iterations == 0 :
146
- return np .copy (labels , order = "F" )
147
-
148
- output = labels
149
- for i in range (iterations ):
150
- output = dilate (
151
- erode (output , parallel , mode , iterations = 1 , erode_border = erode_border ),
152
- background_only , parallel , mode , iterations = 1
153
- )
154
- return output
142
+ return dilate (
143
+ erode (labels , parallel , mode , iterations = 1 , erode_border = erode_border ),
144
+ background_only , parallel , mode , iterations = 1
145
+ )
155
146
156
147
def closing (
157
148
labels :np .ndarray ,
158
149
background_only :bool = True ,
159
150
parallel :int = 0 ,
160
151
mode :Mode = Mode .multilabel ,
161
- iterations :int = 1 ,
162
152
erode_border :bool = True ,
163
153
) -> np .ndarray :
164
154
"""Performs morphological closing of labels.
165
155
156
+ This operator is idempotent with the exception
157
+ of boundary effects.
158
+
166
159
background_only is passed through to dilate.
167
160
True: Only evaluate background voxels for dilation.
168
161
False: Allow labels to erode each other as they grow.
169
162
parallel: how many pthreads to use in a threadpool
170
163
mode:
171
164
Mode.multilabel: are all surrounding pixels the same?
172
165
Mode.grey: use grayscale image dilation (min value)
173
-
174
- iterations: number of times to iterate the result
175
-
176
166
erode_border: if True, the border is treated as background,
177
167
else it is regarded as a value that would preserve the
178
168
current value. Only has an effect for multilabel erosion.
179
169
"""
180
- if iterations < 0 :
181
- raise ValueError (f"iterations ({ iterations } ) must be a positive integer." )
182
- elif iterations == 0 :
183
- return np .copy (labels , order = "F" )
184
-
185
- output = labels
186
- for i in range (iterations ):
187
- output = erode (
188
- dilate (output , background_only , parallel , mode , iterations = 1 ),
189
- parallel , mode , iterations = 1 , erode_border = erode_border ,
190
- )
191
- return output
170
+ return erode (
171
+ dilate (labels , background_only , parallel , mode , iterations = 1 ),
172
+ parallel , mode , iterations = 1 , erode_border = erode_border ,
173
+ )
192
174
193
175
def spherical_dilate (
194
176
labels :np .ndarray ,
@@ -283,6 +265,8 @@ def fill_holes(
283
265
return_fill_count :bool = False ,
284
266
remove_enclosed :bool = False ,
285
267
return_removed :bool = False ,
268
+ fix_borders :bool = False ,
269
+ morphological_closing :bool = False ,
286
270
) -> np .ndarray :
287
271
"""
288
272
For fill holes in toplogically closed objects.
@@ -298,7 +282,7 @@ def fill_holes(
298
282
Return value: (filled_labels, fill_count (if specified), removed_set (if specified))
299
283
"""
300
284
assert np .issubdtype (labels .dtype , np .integer ) or np .issubdtype (labels .dtype , bool ), "fill_holes is currently only supported for integer or binary images."
301
- if np .issubdtype (labels .dtype , bool ):
285
+ if np .issubdtype (labels .dtype , bool ) and not fix_borders and not morphological_closing :
302
286
filled_labels , filled_ct = fill_voids .fill (labels , return_fill_count = True )
303
287
ret = [ filled_labels ]
304
288
if return_fill_count :
@@ -327,10 +311,36 @@ def fill_holes(
327
311
continue
328
312
329
313
binary_image = (cc_labels [slices ] == label )
330
- binary_image , pixels_filled = fill_voids .fill (
314
+
315
+ pixels_filled = 0
316
+
317
+ if morphological_closing :
318
+ dilated_binary_image = dilate (binary_image )
319
+ pixels_filled += np .sum (dilated_binary_image != binary_image )
320
+ binary_image = dilated_binary_image
321
+ del dilated_binary_image
322
+
323
+ if fix_borders :
324
+ binary_image [:,:,0 ], pf1 = fill_voids .fill (binary_image [:,:,0 ], return_fill_count = True )
325
+ binary_image [:,:,- 1 ], pf2 = fill_voids .fill (binary_image [:,:,- 1 ], return_fill_count = True )
326
+ binary_image [:,0 ,:], pf3 = fill_voids .fill (binary_image [:,0 ,:], return_fill_count = True )
327
+ binary_image [:,- 1 ,:], pf4 = fill_voids .fill (binary_image [:,- 1 ,:], return_fill_count = True )
328
+ binary_image [0 ,:,:], pf5 = fill_voids .fill (binary_image [0 ,:,:], return_fill_count = True )
329
+ binary_image [- 1 ,:,:], pf6 = fill_voids .fill (binary_image [- 1 ,:,:], return_fill_count = True )
330
+ pixels_filled += pf1 + pf2 + pf3 + pf4 + pf5 + pf6
331
+
332
+ binary_image , pf7 = fill_voids .fill (
331
333
binary_image , in_place = True ,
332
334
return_fill_count = True
333
335
)
336
+ pixels_filled += pf7
337
+
338
+ if morphological_closing :
339
+ eroded_binary_image = erode (binary_image , erode_border = False )
340
+ pixels_filled -= np .sum (eroded_binary_image != binary_image )
341
+ binary_image = eroded_binary_image
342
+ del eroded_binary_image
343
+
334
344
fill_counts [label ] = pixels_filled
335
345
output [slices ][binary_image ] = mapping [label ]
336
346
@@ -339,7 +349,7 @@ def fill_holes(
339
349
340
350
sub_labels = fastremap .unique (cc_labels [slices ][binary_image ])
341
351
sub_labels = set (sub_labels )
342
- sub_labels .remove (label )
352
+ sub_labels .discard (label )
343
353
sub_labels .discard (0 )
344
354
if not remove_enclosed and sub_labels :
345
355
sub_labels = [ int (l ) for l in sub_labels ]
0 commit comments