Sequential object (re-)labeling#
As mentioned above, depending on the use-case it might be important to label objects in an image subsequently. It could for example be that a post-processing algorithm for label images crashes in case we pass a label image with missing labels. Hence, we should know how to relabel an image sequentially.
import numpy as np
from skimage.io import imread
from skimage.segmentation import relabel_sequential
import pyclesperanto_prototype as cle
Our starting point is a label image with labels 1-8, where some labels are not present:
label_image = imread("../../data/label_map_with_index_gaps.tif")
cle.imshow(label_image, labels=True)
data:image/s3,"s3://crabby-images/82bf8/82bf8fa6e26389e10ec41cb42ca60807a0ca2074" alt="../_images/51805c923c35709f24d6f0350c13be42a832a6352c02b05b0d89eb7504a3320d.png"
When measuring the maximum intensity in the image, we can see that this label image containing 4 labels is obviously not sequentially labeled.
np.max(label_image)
8
We can use the unique
function to figure out which labels are present:
np.unique(label_image)
array([0, 1, 2, 4, 8], dtype=uint8)
Sequential labeling#
We can now relabel this image and remove these gaps using scikit-image’s relabel_sequential()
function. We’re entering the _
as additional return variables as we’re not interested in them. This is necessary because the relabel_sequential
function returns three things, but we only need the first.
relabeled, _, _ = relabel_sequential(label_image)
cle.imshow(relabeled, labels=True)
data:image/s3,"s3://crabby-images/b66c3/b66c3c9348fd62a07efc09b20ad0a8222935d4af" alt="../_images/3ba6457cc995726598ebdc81386238075079725fa050bfd766321b040c813966.png"
Afterwards, the unique labels should be sequential:
np.unique(relabeled)
array([0, 1, 2, 3, 4], dtype=uint8)
Also pyclesperanto has a function for relabeling label images sequentially. The result is supposed identical to the result in scikit-image. It just doesn’t return the additional values.
relabeled1 = cle.relabel_sequential(label_image)
cle.imshow(relabeled1, labels=True)
data:image/s3,"s3://crabby-images/b66c3/b66c3c9348fd62a07efc09b20ad0a8222935d4af" alt="../_images/3ba6457cc995726598ebdc81386238075079725fa050bfd766321b040c813966.png"
Reverting sequential labeling#
In some cases we apply an operation to a label image that returns a new label image with less labels that are sequentially labeled but the label-identity is lost. This happens for example when excluding labels from the label image that are too small.
large_labels = cle.exclude_small_labels(relabeled, maximum_size=260)
cle.imshow(large_labels, labels=True, max_display_intensity=4)
data:image/s3,"s3://crabby-images/9b3cf/9b3cfd178aa2d7e857943c469e4dc5b3dd1306b2" alt="../_images/36f7ec9698b614553fae93f11f9805517e0f0c5ff9395649d8a2f2a612d437b8.png"
np.unique(large_labels)
array([0, 1, 2], dtype=uint32)
To restore the original label identities, we need to multiply a binary image representing the remaining labels with the original label image.
binary_remaining_labels = large_labels > 0
cle.imshow(binary_remaining_labels)
data:image/s3,"s3://crabby-images/a1e99/a1e99bd47f52cb04c997ab295c96bada0885991d" alt="../_images/c1423a2d9cd701d92e76baaea3eea0337cb0843c7be9208511df9b36b77de9ce.png"
large_labels_with_original_identity = binary_remaining_labels * relabeled
cle.imshow(large_labels_with_original_identity, labels=True, max_display_intensity=4)
data:image/s3,"s3://crabby-images/7dc01/7dc012ad64f2509ff7b78a28bddecbd7092b18f2" alt="../_images/ba79843daab6429f7be86b46ff2fc6726f79e5eee587c077b33cea75b9e8f2fc.png"
np.unique(large_labels_with_original_identity)
array([0., 1., 3.], dtype=float32)
We can now conclude that labels with identities 2 and 4 were too small and thus, excluded.