Help with Blood Vessel Segmentation Module for My Engineering Thesis

Hello everyone,

I am currently working on my engineering thesis and I am developing a module for 3D Slicer that aims to detect and segment blood vessels from CT scan images. This module will be part of my research project, and my goal is to automate the process of extracting vascular structures from medical imaging data.

At the moment, I am in the early stages of the project, and I have written some code that allows me to segment the blood vessels from CT images. I would greatly appreciate any feedback or suggestions regarding my approach.

Below is the code I have written so far. If anyone has experience with blood vessel segmentation or working with medical image data in 3D Slicer, your help in reviewing and improving this code would be invaluable.

# Retrieve the volume node
volumeNode = slicer.util.getNode("CTChest")  # Replace with your volume node's actual name

if not volumeNode:
    raise ValueError("Volume node 'CTChest' not found. Check the name and ensure it's loaded into the scene.")

# Create a new segmentation node
segmentationNode = slicer.mrmlScene.AddNewNodeByClass("vtkMRMLSegmentationNode")
segmentationNode.CreateDefaultDisplayNodes()  # Ensure display nodes are initialized

# Initialize the Segment Editor
segmentEditorWidget = slicer.qMRMLSegmentEditorWidget()
segmentEditorWidget.setMRMLScene(slicer.mrmlScene)
segmentEditorWidget.setSegmentationNode(segmentationNode)

# Explicitly set the Master Volume Node
segmentEditorWidget.setMasterVolumeNode(volumeNode)

# Debugging: Ensure the Master Volume Node is set correctly
if not segmentEditorWidget.masterVolumeNode():
    raise RuntimeError("Master volume node is not set correctly in the Segment Editor Widget.")

# Activate the Threshold effect
segmentEditorWidget.setActiveEffectByName("Threshold")
effect = segmentEditorWidget.activeEffect()

if effect:
    print("Threshold effect activated successfully!")
    # Set threshold values
    effect.setParameter("MinimumThreshold", "100")  # Replace with your desired lower threshold
    effect.setParameter("MaximumThreshold", "300")  # Replace with your desired upper threshold
    effect.self().onApply()  # Apply the threshold effect
    print("Threshold segmentation applied successfully.")
else:
    raise RuntimeError("Failed to activate the Threshold effect. Check Slicer configuration and plugins.")

# Optional: Smooth the segmentation to refine it
segmentEditorWidget.setActiveEffectByName("Smoothing")
effect = segmentEditorWidget.activeEffect()

if effect:
    effect.setParameter("SmoothingMethod", "Gaussian")
    effect.setParameter("KernelSizeMm", "1.5")
    effect.self().onApply()
    print("Smoothing applied successfully.")
else:
    print("Smoothing effect could not be activated.")

# Finalize (clean up resources)
segmentEditorWidget = None

Thank you in advance for any help you can provide!

The qMRMLSegmentEditorWidget needs a vtkMRMLSegmentEditorNode.

An effet needs a segment that you must create and select. Identify the added lines below.

# Initialize the Segment Editor
segmentEditorWidget = slicer.qMRMLSegmentEditorWidget()
segmentEditorWidget.setMRMLScene(slicer.mrmlScene)
mrmlSegmentEditorNode = slicer.mrmlScene.AddNewNodeByClass("vtkMRMLSegmentEditorNode")
segmentEditorWidget.setMRMLSegmentEditorNode(mrmlSegmentEditorNode)
segmentEditorWidget.setSegmentationNode(segmentationNode)

# Explicitly set the Master Volume Node
segmentEditorWidget.setMasterVolumeNode(volumeNode)

# Debugging: Ensure the Master Volume Node is set correctly
if not segmentEditorWidget.masterVolumeNode():
    raise RuntimeError("Master volume node is not set correctly in the Segment Editor Widget.")

segmentID = segmentationNode.GetSegmentation().AddEmptySegment()
segmentEditorWidget.setCurrentSegmentID(segmentID)

Please note that bones have a wide intensity range, that includes the range in the contrast media.

Thank you for your feedback!

I really appreciate your suggestions and the time you took to help me with my code. Your insights are very helpful, and I’ll definitely implement your recommendations to improve my approach.

I also have a follow-up question: do you think it’s possible to segment only the blood vessels using the threshold function in 3D Slicer? If not, do you have any ideas or suggestions on alternative methods to achieve this? I’m open to exploring different techniques if thresholding alone isn’t sufficient.

Thank you again for your assistance!

You may investigate the “Flood filling” effect of the “Segment editor” with high values of “Intensity tolerance”. Other effects allow an efficient segmentation also, but this one is probably the fastest. No segmentation technique is magic, you’ll have to play a lot and a lot to grasp their offerings and limits.

A few modules of the SlicerVMTK installable extension use the “Flood filling” effect, this time on targeted regions that you define.

In all cases, you need good quality CT scans with sufficiently contrasted lumen.

To put it simply, the module I want to create looks as follows in my mind. First, a script would perform segmentation of regions containing blood vessels from a specific CT image. Due to the similarity in grayscale values to bone, the module would need to prompt the user to select the regions of interest.

Next, I need my module to calculate values such as the volume and other related parameters of the blood vessels, and then display the obtained results. Do you have any advice that could help me achieve this?

Currently, the segmentation part works, but it also segments parts of the bone. The image I’m using to test my program is from Slicer and is named CTACArdio. I would be very grateful for any suggestions you might have.

To do that, you must work on segmented blood vessels and segmented blood vessels only. That’s why you want your users to select regions of interest.

With a volume like CTA-cardio and most real-life volumes, thresholding will give you a lumen that will be attached to bones and soft tissues often. Even the ‘Islands’ segment editor effect won’t be of much help, it many remove some detached parts of the segment, called Islands, but you won’t get just nice blood vessels.

You may still use the ‘Scissors’ effect to cut anything that is not an artery but it’s obviously impractical in this use case.

Consider using the ‘Flood filling’ effect with an “Intensity tolerance” of 90 and a “Neighbourhood size” of 2.0. Just click in the abdominal aorta for example and you will see the difference. Your module can ask the user to do that.

Another option is to crop CTA-cardio to a ROI and threshold in that targeted volume.

Can u tell me what is wrong with activing flood filling effect in this script?

Step 2: Refine with Flood Filling

segmentEditorWidget.setActiveEffectByName(“FloodFilling”)
effect = segmentEditorWidget.activeEffect()

if effect:
… print(“Flood Filling effect activated successfully!”)
… # Set flood filling parameters
… effect.setParameter(“IntensityTolerance”, “90”) # Adjust intensity tolerance
… effect.setParameter(“NeighborhoodSize”, “2.0”) # Adjust neighborhood size
… print(“Click on a vessel in the volume to apply the flood filling effect.”)
… input(“Press Enter after applying flood filling…”) # Wait for user interaction
… else:
… raise RuntimeError(“Failed to activate the Flood Filling effect. Check Slicer configuration and plugins.”)

The user must type something or an error will be raised.

I’m not sure if the values can be passed as string; try with int and float respectively.

You could also call

effect.updateGUIFromMRML()

after setting the parameters.

I have changed the code like this but I still have errors, any ideas? # Step 2: Refine with Flood Filling in a new segment

floodFillSegmentID = segmentationNode.GetSegmentation().AddEmptySegment(“FloodFillVessels”)
segmentEditorWidget.setCurrentSegmentID(floodFillSegmentID)

segmentEditorWidget.setActiveEffectByName(“FloodFilling”)
effect = segmentEditorWidget.activeEffect()

if effect:
… print(“Flood Filling effect activated successfully!”)

… # Set flood filling parameters with int and float values
… intensity_tolerance = 90 # Intensity tolerance as integer
… neighborhood_size = 2.0 # Neighborhood size as float

… effect.setParameter(“IntensityTolerance”, intensity_tolerance) # Use int for IntensityTolerance
… effect.setParameter(“NeighborhoodSize”, neighborhood_size) # Use float for NeighborhoodSize

… effect.updateGUIFromMRML() # Synchronize GUI with parameters
… print(“Click on a vessel in the volume to apply the flood filling effect.”)

… # Pause for user interaction
… input(“Press Enter after completing the flood filling interaction…”)
… else:
… raise RuntimeError(“Failed to activate the Flood Filling effect. Check Slicer configuration and plugins.”)

Replace “FloodFilling” by “Flood filling”.

Also, ask the user to type something before pressing enter, else an EOL error will be raised. Just after this ‘input’ line, use setActiveEffect(None).

BTW, you seem to be targeting power users since your code should run in the Python console so that they get a chance to press enter. I don’t know what is the design of your research but it should not imply the console. You could add a button in your UI to restore the active effect to None rather. That part is however totally your design and target audience.