Seeded watershed for membrane-based cell segmentation
Contents
Seeded watershed for membrane-based cell segmentation#
In this section we will use a seeded watershed approach to cell segmentation. This approach is very common when cell segmentation based on images of membrane markers are given. Therefore, we use the napari plugin napari-segment-blobs-and-things-with-membranes. Under the hood, this plugins uses functions from scikit-image.
See also
from napari_segment_blobs_and_things_with_membranes import voronoi_otsu_labeling, \
seeded_watershed, \
local_minima_seeded_watershed
from skimage.io import imread
from skimage.filters import gaussian
from skimage import data
from pyclesperanto_prototype import imshow
We load the Cells3d example image from scikit-image which is a two-channel image showing nuclei and membranes.
cells = data.cells3d()
cells.shape
(60, 2, 256, 256)
nuclei_channel = cells[30, 1]
imshow(nuclei_channel)
data:image/s3,"s3://crabby-images/c38f5/c38f5cf4c9326d2f8e3a32b7ac349a370f9fc89e" alt="../_images/14_segmentation_2d_membranes_4_0.png"
membrane_channel = cells[30, 0]
imshow(membrane_channel, max_display_intensity=6000)
data:image/s3,"s3://crabby-images/9eb08/9eb0869ed6471c9a015178ef9d833b130467f979" alt="../_images/14_segmentation_2d_membranes_5_0.png"
Voronoi-Otsu-Labeling for nuclei segmentation#
First, we start with segmenting the nuclei using the Voronoi-Otsu-Labeling algorithm.
labeled_nuclei = voronoi_otsu_labeling(nuclei_channel, spot_sigma=10, outline_sigma=2)
imshow(labeled_nuclei, labels=True)
data:image/s3,"s3://crabby-images/31337/31337f4aff92c4ef56b4c37cb974775b1bd670cf" alt="../_images/14_segmentation_2d_membranes_7_0.png"
Seeded watershed#
We can use the image of labeled nuclei as starting point for flooding the low-intensity areas in the membrane image. This allows us to determine a cell segmentation.
labeled_cells = seeded_watershed(membrane_channel, labeled_nuclei)
imshow(labeled_cells, labels=True)
data:image/s3,"s3://crabby-images/9065b/9065b38f1df8abc69cc52e68e5d0f00a82c82386" alt="../_images/14_segmentation_2d_membranes_9_0.png"
If the outlines of the cells are not 100% accurate, it may make sense to blur the membrane image a bit before segmenting the cells.
blurred = gaussian(membrane_channel, sigma=3)
labeled_cells = seeded_watershed(blurred, labeled_nuclei)
imshow(labeled_cells, labels=True)
data:image/s3,"s3://crabby-images/ddd80/ddd807afb9e827ccd3a8b919208213a2be26614c" alt="../_images/14_segmentation_2d_membranes_11_0.png"
Seeded watershed using automatic seed detection#
In case we didn’t image a separate nuclei channel and only have the membrane channel available for segmentation, we can use the membrane image to search for local minima (dark areas).
labeles_cells2 = local_minima_seeded_watershed(membrane_channel)
imshow(labeles_cells2, labels=True)
data:image/s3,"s3://crabby-images/3c7c8/3c7c8926b637723c006f091cbe857b568c24af7b" alt="../_images/14_segmentation_2d_membranes_13_0.png"
This function also has some parameters to allow fine tuning the segmentation. The parameter outline_sigma
allows to control a Gaussian blur filter that allows fine-tuning the outlines of the segmented cells as shown above.
labeles_cells3 = local_minima_seeded_watershed(membrane_channel, outline_sigma=3)
imshow(labeles_cells3, labels=True)
data:image/s3,"s3://crabby-images/66387/66387d4895090b99a75fc795cf246a862846441c" alt="../_images/14_segmentation_2d_membranes_15_0.png"
If there multiple cells sticking together, it may make sense to specify spot_sigma
. This parameter allows to configure how close / large cells are.
labeles_cells4 = local_minima_seeded_watershed(membrane_channel, spot_sigma=9, outline_sigma=3)
imshow(labeles_cells4, labels=True)
data:image/s3,"s3://crabby-images/f3277/f3277bd7a9982ad6681a516b7baef54e9abf0f36" alt="../_images/14_segmentation_2d_membranes_17_0.png"
Exercise#
Load the following dataset and find good parameters for processing it using a seeded watershed approach. This example image data is a courtesy of Sascha M. Kuhn, Nadler Lab, MPI-CBG Dresden.
image_slice = imread("../../data/membrane_2d_timelapse.tif")[2]
imshow(image_slice)
data:image/s3,"s3://crabby-images/7745e/7745e08f66dcaf6059f68110b8ecc55b9f59bff5" alt="../_images/14_segmentation_2d_membranes_19_0.png"