Skip to content

Commit

Permalink
Merge pull request #12 from JuliaImageRecon/pj/vesselPhantoms
Browse files Browse the repository at this point in the history
Modify vessel generator
  • Loading branch information
pauljuerss authored Mar 7, 2024
2 parents 90d0192 + 9f8a86b commit 99ff7c4
Show file tree
Hide file tree
Showing 2 changed files with 226 additions and 123 deletions.
264 changes: 157 additions & 107 deletions src/Vessel.jl
Original file line number Diff line number Diff line change
@@ -1,146 +1,196 @@
# This file is originally based on Matlab code written by Christine Droigk
"""
appendRoute!(route, stepsize, angles)
Append a new point to the route of the vessel.
"""
function appendRoute!(route, stepsize, angles::NTuple{2,Float64})
angle_xy, angle_xz = angles
push!(route, route[end] .+ stepsize .* (cos(angle_xy)*cos(angle_xz),
sin(angle_xy)*cos(angle_xz),
sin(angle_xz)))
end

function appendRoute!(route, stepsize, angles::NTuple{1,Float64})
(angle_xy,) = angles
push!(route, route[end] .+ stepsize .* (cos(angle_xy),
sin(angle_xy)))
end

"""
vesselPath(N::NTuple{3,Int}; start, angle_xy, angle_xz, diameter, split_prob, change_prob, max_change, splitnr)
changeDirection!(route, N, stepsize, angles, change_prob, max_change, rng)
Input parameters:
* start: starting point given as a 3x1 vector
* angle_xy: angle in radians describing the starting angle with which the
* vessel runs.
* angle_xz: same, but angle from xy-plane to vessel's z
Simulate if and how the vessel changes its direction.
"""
function changeDirection!(N::NTuple{D,Int}, route, stepsize, angles, change_prob, max_change, steps_each_change, steps_no_change, rng) where D
if (rand(rng, Float64) < change_prob) && (length(route) > 2)# if directional change of the vessel
# randomly select new angles
change_angles = ntuple(_ -> 2*max_change*rand(rng,Float64), D-1)
# create range such that change is gradually applied
step_angles = zip(ntuple(d -> range(0, change_angles[d], length=steps_each_change), D-1)...) |> collect
for i = 1:steps_each_change
if all( 1 .<= route[end] .<= N) # check whether image boundaries are reached
appendRoute!(route, stepsize, angles .+ step_angles[i])
end
end
# set current angles to new angles
angles = angles .+ change_angles
else
# if no directional change
for _=1:steps_no_change
if all( 1 .<= route[end] .<= N) # check whether image boundaries are reached
appendRoute!(route, stepsize, angles)
end
end
end
return angles
end

"""
getDiameterRoute(route, diameter, change_diameter_splitting, splitnr)
Compute the diameter anlong the route of the vessel.
"""
function getDiameterRoute(route, diameter, change_diameter_splitting, splitnr)
if splitnr>1
# for the case that the vessel leaves the image during a split the change is set to 1
# otherwise `range` throws an error since start and end do not match but length is 1
change_diameter_splitting_ = length(route) > 1 ? change_diameter_splitting : 1
# gradually change diameter from old value to current one
diameter_route = collect(range((1/change_diameter_splitting_)*diameter, diameter, length=length(route)))
else
# no change if vessel did not already split once
diameter_route = diameter*ones(length(route))
end
return diameter_route
end

"""
vesselPath(N::NTuple{D,Int}; start, angles, diameter, split_prob, change_prob,
max_change, splitnr, max_number_splits, stepsize, change_diameter_splitting, split_prob_factor,
change_prob_increase, steps_each_change, steps_no_change, rng)
### Input parameters:
* N: Image size, given as a D tuple
* start: starting point given as a D tuple
* angles: D-1 tuple of angles for the vessel's direction (xy in 2D, xy and xz in 3D)
* diameter: starting diameter of vessel
* split_prob: probability for a splitting of the vessel into two vessel segments. Values between 0 and 1.
* change_prob: probability for directional change of the vessel route. Values between 0 and 1.
* max_change: max_change * pi specifies the maximum direction-change angle.
* splitnr: used for recursive call of the function. For the first call set it to 1.
* N: Image size, given as a 3x1 vector
* max_number_splits: maximum number of splits of the vessel.
* stepsize: stepsize of the vessel.
* change_diameter_splitting: Indicates by how much the diameter decreases when the vessel splits
* split_prob_factor: Factor by which the split probability `split_prob` is multiplied when the vessel splits
* change_prob_increase: Increase of the change probability `change_prob` when the vessel splits
* steps_each_change: Number of steps for the change of the vessel direction
* steps_no_change: Number of steps for the case that the vessel does not change its direction
* rng: Random number generator
Output:
* route: A length N vector containing the points 3D of the route of the vessel. The
* length N depends on the random route.
* route: A length N vector containing the D dimensional points of the route of the vessel. The
length N depends on the random route.
* diameter_route: A length N vector containing the diameter of the vessel at the positions of the route.
"""
function vesselPath(N::NTuple{3,Int};
start,
angle_xy,
angle_xz,
diameter,
split_prob,
change_prob,
max_change,
splitnr,
rng::AbstractRNG = GLOBAL_RNG)

stepsize = 0.25
change_diameter_splitting = 4/5 # Indicates by how much the diameter decreases when the vessel is divided
route = NTuple{3,Float64}[]
function vesselPath(N::NTuple{D,Int};
start,
angles::NTuple{Da,Real},
diameter::Real,
split_prob::Real,
change_prob::Real,
max_change::Real,
splitnr::Int=1,
max_number_splits::Real = Inf,
stepsize::Real = max(N...)/200,
change_diameter_splitting::Real = 0.6,
split_prob_factor::Real = 0.5,
change_prob_increase::Real = 0.01,
steps_each_change::Int = 20,
steps_no_change::Int = 15,
rng::AbstractRNG = GLOBAL_RNG) where {D, Da}
@assert (D-1) == Da "The length of the angles tuple must be $(D-1) but is $Da"
route = NTuple{D,Float64}[]
push!(route, start)

counter = 1;
while all( 1 .<= route[end] .< N) # while the route of the vessel is inside the image area
while all( 1 .<= route[end] .<= N) # while the route of the vessel is inside the image area
angles = changeDirection!(N, route, stepsize, angles, change_prob,
max_change, steps_each_change, steps_no_change, rng)

if rand(rng, Float64) < change_prob # if directional change of the vessel
change_angle_xy = pi*(max_change-2*max_change*rand(rng,Float64)) # throw dice for the angles of directional change. For more or less variations replace (0.1-0.2*rand)
change_angle_xz = pi*(max_change-2*max_change*rand(rng,Float64)) # same for other angle
step_angle_xy = 0:(sign(change_angle_xy)*0.1*stepsize):change_angle_xy # to obtain a piecewise change of angle
step_angle_xz = 0:(sign(change_angle_xz)*0.1*stepsize):change_angle_xz

for i = 1:max(length(step_angle_xy),length(step_angle_xz)) # until the largest change of the two angles is reached
if all( 1 .<= route[end] .< N) # check whether image boundaries are reached
if i<=length(step_angle_xy) && i<=length(step_angle_xz) # both angles are still changing
push!(route, route[end] .+
stepsize .* (cos(angle_xy + step_angle_xy[i])*cos(angle_xz + step_angle_xz[i]),
sin(angle_xy + step_angle_xy[i])*cos(angle_xz + step_angle_xz[i]),
sin(angle_xz + step_angle_xz[i])) )
elseif i>length(step_angle_xy) && i<=length(step_angle_xz) # only xz-angle changes
push!(route, route[end] .+
stepsize .* (cos(angle_xy + change_angle_xy)*cos(angle_xz + step_angle_xz[i]),
sin(angle_xy + change_angle_xy)*cos(angle_xz + step_angle_xz[i]),
sin(angle_xz + step_angle_xz[i])) )
elseif i<=length(step_angle_xy) && i>length(step_angle_xz) # only xy-angle changes
push!(route, route[end] .+
stepsize .* (cos(angle_xy+step_angle_xy[i])*cos(angle_xz + change_angle_xz),
sin(angle_xy+step_angle_xy[i])*cos(angle_xz + change_angle_xz),
sin(angle_xz + change_angle_xz)) )
end
end
end
# set current angles to new angles
angle_xy = angle_xy + change_angle_xy # set current angles to new angles
angle_xz = angle_xz + change_angle_xz
else
# if no directional change
push!(route, route[end] .+ stepsize .* (cos(angle_xy)*cos(angle_xz),
sin(angle_xy)*cos(angle_xz),
sin(angle_xz))) # use old angle
end


if rand(rng, Float64) < split_prob && counter > 5 # if vessel is splitting
angle_diff = rand(rng, Float64)*pi/2 # throw dice for angle between the resulting two vessel parts
angle_diff_z = rand(rng, Float64)*pi/2 # same for z-angle

part_a = rand(rng, Float64) # Part of the angle related to the first division
part_a_z = rand(rng, Float64) # Same for z-angle
# Splitting part
if rand(rng, Float64) < split_prob && (length(route) > 3) && (splitnr max_number_splits) # if vessel is splitting
min_angle_diff = 0.15*pi
# Randomly select the angle between the resulting two vessel parts
angle_diffs = ntuple(_ -> rand(rng, Float64)*(pi/2 - min_angle_diff) + min_angle_diff, D-1)
# Randomly select the part of the angle related to the first division
parts_a = ntuple(_ -> rand(rng, Float64), D-1)

# Call recursively this function for each of the two vessel
# segments with adjusted angles and diameter. Here, an adjustment
# of the splitting probabilities and directional change
# probabilities is made.
routeA, diameterA = vesselPath(N; start=route[end], angle_xy=angle_xy-part_a*angle_diff,
angle_xz=angle_xz-part_a_z*angle_diff_z, diameter=diameter*change_diameter_splitting,
split_prob=split_prob/2, change_prob=change_prob+0.01, max_change, splitnr=splitnr+1, rng)
routeB, diameterB = vesselPath(N; start=route[end], angle_xy=angle_xy+(1-part_a)*angle_diff,
angle_xz=angle_xy+(1-part_a_z)*angle_diff_z, diameter=diameter*change_diameter_splitting,
split_prob=split_prob/2, change_prob=change_prob+0.01, max_change, splitnr=splitnr+1, rng)
# probabilities is made.
routeA, diameterA = vesselPath(N; start=route[end], angles=angles .- parts_a .* angle_diffs,
diameter=diameter*change_diameter_splitting,
split_prob=split_prob*split_prob_factor, change_prob=change_prob+change_prob_increase,
max_change, splitnr=splitnr+1, max_number_splits, stepsize, change_diameter_splitting,
split_prob_factor, change_prob_increase, steps_each_change, steps_no_change, rng)
routeB, diameterB = vesselPath(N; start=route[end], angles=angles .+ (1 .- parts_a) .* angle_diffs,
diameter=diameter*change_diameter_splitting,
split_prob=split_prob*split_prob_factor, change_prob=change_prob+change_prob_increase,
max_change, splitnr=splitnr+1, max_number_splits, stepsize, change_diameter_splitting,
split_prob_factor, change_prob_increase, steps_each_change, steps_no_change, rng)

if splitnr>1
# set diameter for the splitting parts
change_diameter_splitting_ = length(route) > 1 ? change_diameter_splitting : 1
diameter_route = collect(range((1/change_diameter_splitting_)*diameter, diameter, length=length(route)))
else
diameter_route = diameter*ones(length(route))
end
diameter_route = getDiameterRoute(route, diameter, change_diameter_splitting, splitnr)
append!(route, routeA, routeB)
append!(diameter_route, diameterA, diameterB)
return route, diameter_route
end

counter = counter+1
end

if splitnr>1
change_diameter_splitting_ = length(route) > 1 ? change_diameter_splitting : 1
diameter_route = collect(range((1/change_diameter_splitting_)*diameter, diameter, length=length(route)))
else
diameter_route = diameter*ones(length(route))
end

diameter_route = getDiameterRoute(route, diameter, change_diameter_splitting, splitnr)
return route, diameter_route
end

sphereFunction(::NTuple{D, Int}) where D = throw(ArgumentError("Vessel phantoms are currently not implemented for $D dimensions."))
sphereFunction(::NTuple{2, Int}) = circle
sphereFunction(::NTuple{3, Int}) = sphere

"""
vesselPhantom(N::NTuple{3,Int}; oversampling=2, kargs...)
vesselPhantom(N::NTuple{D,Int}; oversampling=2, kargs...)
Example usage:
### Input parameters:
* N: Image size, given as a D tuple
* oversampling: Oversampling factor for the phantom. Default is 2.
* rng: Random number generator
* kernelWidth: Width (standard deviation) of the Gaussian kernel (in pixel) used for smoothing the phantom. If nothing is given, a random value is chosen.
* kargs...: remaining keyword arguments for `vesselPath`
using GR, TrainingPhantoms, StableRNGs
im = vesselPhantom((40,40,40); start=(1, 20, 20), angle_xy=0.0, angle_xz=0.0,
diameter=2, split_prob=0.5, change_prob=0.5, max_change=0.2, splitnr=1,
rng = StableRNG(1));
isosurface(im, isovalue=0.2, rotation=110, tilt=40)
### Example usage:
"""
function vesselPhantom(N::NTuple{3,Int}; oversampling=2, kargs...)
Q = zeros(N);
using GLMakie, TrainingPhantoms, StableRNGs
route, diameter_route = vesselPath(N; kargs...)
im = vesselPhantom((51,51,51);
start=(1, 25, 25), angles=(0.0, 0.0),
diameter=2.5, split_prob=0.4, change_prob=0.3,
max_change=0.3, splitnr=1, max_number_splits=1, rng StableRNG(123));
obs = [ ellipsoid( Float32.(route[i]), Float32.(ntuple(_->diameter_route[i],3)),
(0,0,0), 1.0f0) for i=1:length(route) ]
ranges = ntuple(d-> 1:N[d], 3)
f = Figure(size=(300,300))
ax = Axis3(f[1,1], aspect=:data)
volume!(ax, im, algorithm=:iso, isorange=0.13, isovalue=0.3, colormap=:viridis, colorrange=[0.0,0.2])
"""
function vesselPhantom(N::NTuple{D,Int}; oversampling=2, rng = GLOBAL_RNG, kernelWidth=nothing, kargs...) where D
objectFunction = sphereFunction(N)
route, diameter_route = vesselPath(N; rng, kargs...)
# add small sphere for every entry in the route
obs = [ objectFunction( Float32.(route[i]), Float32(diameter_route[i]), 1.0f0) for i=eachindex(route) ]
ranges = ntuple(d-> 1:N[d], D)
img = phantom(ranges..., obs, oversampling)
if isnothing(kernelWidth)
# filter for smoothing, offset to ensure minimal filter width
filterWidth = (1.0-0.3)*rand(rng) + 0.3
kernelWidth = ntuple(_ -> filterWidth*N[1] / 20, D)
end
img = imfilter(img, Kernel.gaussian(kernelWidth))
img[img .> 1] .= 1
img[img .< 0] .= 0
return img
end
85 changes: 69 additions & 16 deletions test/runtests.jl
Original file line number Diff line number Diff line change
Expand Up @@ -4,26 +4,79 @@ using Test

@testset "TrainingPhantoms.jl" begin
N = (40,40,40)
@testset "Vessel" begin
@testset "getDiameterRoute" begin
route = [1, 2, 3, 4, 5]
diameter = 2.0
change_diameter_splitting = 0.5
splitnr = 2

@testset "Normal operation" begin
result = TrainingPhantoms.getDiameterRoute(route, diameter, change_diameter_splitting, splitnr)
@test length(result) == length(route)
@test result[1] == (1/change_diameter_splitting)*diameter
@test result[end] == diameter
end

@testset "No split" begin
result = TrainingPhantoms.getDiameterRoute(route, diameter, change_diameter_splitting, 1)
@test all(result .== diameter)
end
end
end

# Vessel Phantom
im = vesselPhantom(N; start=(1, 20, 20), angle_xy=0.0, angle_xz=0.0,
diameter=2, split_prob=0.5, change_prob=0.5, max_change=0.2, splitnr=1,
rng = StableRNG(1));
@testset "vesselPath Tests" begin
rng = StableRNG(1)
@testset "Normal operation" begin
route, diameter_route = TrainingPhantoms.vesselPath(N; start=(1,20,20), angles=(0.0,0.0), diameter=2,
split_prob=0.5, change_prob=0.5, max_change=0.2,
splitnr=1, max_number_splits=2, stepsize=0.25,
change_diameter_splitting=4/5, split_prob_factor=0.5,
change_prob_increase=0.01, rng=rng)
@test length(route) == length(diameter_route)
end
@testset "Start outside volume" begin
route, diameter_route = TrainingPhantoms.vesselPath(N; start=(1,50,20), angles=(0.0,0.0), diameter=2,
split_prob=0.5, change_prob=0.5, max_change=0.2,
splitnr=1, max_number_splits=2, stepsize=0.25,
change_diameter_splitting=4/5, split_prob_factor=0.5,
change_prob_increase=0.01, rng=rng)
@test length(route) == 1
@test length(diameter_route) == 1
end
end
@testset "vesselPhantom" begin
im = vesselPhantom(N; start=(1, 20, 20), angles = (0.0, 0.0),
diameter=2, split_prob=0.5, change_prob=0.5, max_change=0.2, splitnr=1,
rng = StableRNG(1));

im2 = vesselPhantom(N; start=(1, 20, 20), angle_xy=0.0, angle_xz=0.0,
diameter=2, split_prob=0.5, change_prob=0.5, max_change=0.2, splitnr=1,
rng = StableRNG(1));
@test im im2
im2 = vesselPhantom(N; start=(1, 20, 20), angles = (0.0, 0.0),
diameter=2, split_prob=0.5, change_prob=0.5, max_change=0.2, splitnr=1,
rng = StableRNG(1));
@test im im2
@test size(im) == N
@test maximum(im) == 1.0
@test minimum(im) == 0.0

# Ellipsoid Phantom
im = ellipsoidPhantom(N; rng=StableRNG(1))
im2 = ellipsoidPhantom(N; rng=StableRNG(1))
@test im im2
im = vesselPhantom((40,40); start=(1, 20), angles = (0.0,),
diameter=2, split_prob=0.5, change_prob=0.5, max_change=0.2, splitnr=1,
rng = StableRNG(1));
@test size(im) == (40,40)

im = ellipsoidPhantom((20,20); allowOcclusion=true)
@test maximum(im) <= 1
@test_throws ArgumentError vesselPhantom((20,20,20,20))

@test_throws ArgumentError ellipsoidPhantom((20,20,20,20))
end

#isosurface(im, isovalue=0.2, rotation=110)
@testset "Ellipsoid" begin
# Ellipsoid Phantom
im = ellipsoidPhantom(N; rng=StableRNG(1))
im2 = ellipsoidPhantom(N; rng=StableRNG(1))
@test im im2
@test size(im) == N

im = ellipsoidPhantom((20,20); allowOcclusion=true)
@test maximum(im) <= 1

@test_throws ArgumentError ellipsoidPhantom((20,20,20,20))
end
end

0 comments on commit 99ff7c4

Please sign in to comment.