From 373e139fddb19e54af64310e93d4961ffc4145df Mon Sep 17 00:00:00 2001 From: Pablo Villacorta Aylagas Date: Wed, 11 Dec 2024 11:36:25 +0100 Subject: [PATCH 1/8] =?UTF-8?q?Improve=20test=20and=20add=20neccesary=20`?= =?UTF-8?q?=E2=89=88`=20methods?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- KomaMRIBase/src/motion/Action.jl | 1 + KomaMRIBase/src/motion/SpinSpan.jl | 3 +- KomaMRIBase/src/motion/TimeSpan.jl | 2 + KomaMRIBase/test/runtests.jl | 318 ++++++++++-------- .../test/GUI_PlotlyJS_backend_test.jl | 5 +- 5 files changed, 193 insertions(+), 136 deletions(-) diff --git a/KomaMRIBase/src/motion/Action.jl b/KomaMRIBase/src/motion/Action.jl index a47143e49..cdd8bc021 100644 --- a/KomaMRIBase/src/motion/Action.jl +++ b/KomaMRIBase/src/motion/Action.jl @@ -1,5 +1,6 @@ abstract type AbstractAction{T<:Real} end +Base.:(≈)(a1::AbstractAction, a2::AbstractAction) = (typeof(a1) == typeof(a2)) & reduce(&, [getfield(a1, field) ≈ getfield(a2, field) for field in fieldnames(typeof(a1))]) is_composable(m::AbstractAction) = false # Simple actions diff --git a/KomaMRIBase/src/motion/SpinSpan.jl b/KomaMRIBase/src/motion/SpinSpan.jl index fab1b9787..fa6060aa3 100644 --- a/KomaMRIBase/src/motion/SpinSpan.jl +++ b/KomaMRIBase/src/motion/SpinSpan.jl @@ -1,5 +1,7 @@ abstract type AbstractSpinSpan end +Base.:(≈)(s1::AbstractSpinSpan, s2::AbstractSpinSpan) = (typeof(s1) == typeof(s2)) & reduce(&, [getfield(s1, field) ≈ getfield(s2, field) for field in fieldnames(typeof(s1))]; init=true) + """ allspins = AllSpins() @@ -64,7 +66,6 @@ function Base.getindex(spins::SpinRange, p) return intersect, spin_range end Base.view(spins::SpinRange, p) = spins[p] -Base.:(==)(sr1::SpinRange, sr2::SpinRange) = sr1.range == sr2.range Base.length(sr::SpinRange) = length(sr.range) get_indexing_range(spins::SpinRange) = spins.range expand(sr::SpinRange, Ns::Int) = sr diff --git a/KomaMRIBase/src/motion/TimeSpan.jl b/KomaMRIBase/src/motion/TimeSpan.jl index 551e4965a..1175d4873 100644 --- a/KomaMRIBase/src/motion/TimeSpan.jl +++ b/KomaMRIBase/src/motion/TimeSpan.jl @@ -1,5 +1,7 @@ abstract type AbstractTimeSpan{T<:Real} end +Base.:(≈)(t1::AbstractTimeSpan, t2::AbstractTimeSpan) = (typeof(t1) == typeof(t2)) & reduce(&, [getfield(t1, field) ≈ getfield(t2, field) for field in fieldnames(typeof(t1))]) + """ timerange = TimeRange(t_start, t_end) diff --git a/KomaMRIBase/test/runtests.jl b/KomaMRIBase/test/runtests.jl index 2ed71555f..f5029fdf7 100644 --- a/KomaMRIBase/test/runtests.jl +++ b/KomaMRIBase/test/runtests.jl @@ -366,148 +366,40 @@ end end end -@testitem "Phantom" tags = [:base] begin - using Suppressor - # Test phantom struct creation - name = "Bulks" - x = [-2e-3; -1e-3; 0.0; 1e-3; 2e-3] - y = [-4e-3; -2e-3; 0.0; 2e-3; 4e-3] - z = [-6e-3; -3e-3; 0.0; 3e-3; 6e-3] - ρ = [0.2; 0.4; 0.6; 0.8; 1.0] - T1 = [0.9; 0.9; 0.5; 0.25; 0.4] - T2 = [0.09; 0.05; 0.04; 0.07; 0.005] - T2s = [0.1; 0.06; 0.05; 0.08; 0.015] - Δw = [-2e-6; -1e-6; 0.0; 1e-6; 2e-6] - Dλ1 = [-4e-6; -2e-6; 0.0; 2e-6; 4e-6] - Dλ2 = [-6e-6; -3e-6; 0.0; 3e-6; 6e-6] - Dθ = [-8e-6; -4e-6; 0.0; 4e-6; 8e-6] - obj1 = Phantom(name=name, x=x, y=y, z=z, ρ=ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ) - obj2 = Phantom(name=name, x=x, y=y, z=z, ρ=ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ) - @test obj1 == obj2 - - # Test size and length definitions of a phantom - @test size(obj1) == size(ρ) - @test length(obj1) == length(ρ) - - simplemotion = MotionList( - Translate(0.05, 0.05, 0.0, Periodic(period=0.5, asymmetry=0.5)), - Rotate(0.0, 0.0, 90.0, TimeRange(t_start=0.05, t_end=0.5), SpinRange(1:3)) - ) - - Ns = length(obj1) - Nt = 3 - t_start = 0.0 - t_end = 1.0 - arbitrarymotion = Path(0.01 .* rand(Ns, Nt), 0.01 .* rand(Ns, Nt), 0.01 .* rand(Ns, Nt), TimeRange(t_start, t_end), SpinRange(2:2:4)) - - obs1 = Phantom( - name, - x, - y, - z, - ρ, - T1, - T2, - T2s, - Δw, - Dλ1, - Dλ2, - Dθ, - simplemotion - ) - - # Test phantom subset (simple range) - rng = 1:2:5 - obs2 = Phantom( - name, - x[rng], - y[rng], - z[rng], - ρ[rng], - T1[rng], - T2[rng], - T2s[rng], - Δw[rng], - Dλ1[rng], - Dλ2[rng], - Dθ[rng], - simplemotion[rng], - ) - @test obs1[rng] == obs2 - @test @view(obs1[rng]) == obs2 - - obs1.motion = arbitrarymotion - obs2.motion = arbitrarymotion[rng] - @test obs1[rng] == obs2 - - # Test addition of phantoms - oba = Phantom( - name, - [x; x[rng]], - [y; y[rng]], - [z; z[rng]], - [ρ; ρ[rng]], - [T1; T1[rng]], - [T2; T2[rng]], - [T2s; T2s[rng]], - [Δw; Δw[rng]], - [Dλ1; Dλ1[rng]], - [Dλ2; Dλ2[rng]], - [Dθ; Dθ[rng]], - vcat(obs1.motion, obs2.motion, length(obs1), length(obs2)) - ) - @test obs1 + obs2 == oba - - # Test phantom subset (BitVector range) - obs3 = copy(obs1) - obs4 = copy(obs1) - rng = obs3.x .> 0 - obs3.motion = Translate(5e-4, 6e-4, 7e-4, TimeRange(0.0, 1.0), SpinRange(rng)) - obs4.motion = Translate(5e-4, 6e-4, 7e-4, TimeRange(0.0, 1.0), SpinRange(1:length(obs4))) - @test obs3[rng] == obs4[rng] - @test obs3[rng].motion == obs4.motion[rng] - - # Test scalar multiplication of a phantom - c = 7 - obc = Phantom(name=name, x=x, y=y, z=z, ρ=c*ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ) - @test c * obj1 == obc - - #Test brain phantom 2D - ph = brain_phantom2D() - @test ph.name == "brain2D_axial" - @test KomaMRIBase.get_dims(ph) == Bool[1, 1, 0] - - #Test brain phantom 3D - ph = brain_phantom3D() - @test ph.name == "brain3D" - @test KomaMRIBase.get_dims(ph) == Bool[1, 1, 1] - - #Test pelvis phantom 2D - ph = pelvis_phantom2D() - @test ph.name == "pelvis2D" - @test KomaMRIBase.get_dims(ph) == Bool[1, 1, 0] - - #Test heart phantom - ph = heart_phantom() - @test ph.name == "LeftVentricle" - @test KomaMRIBase.get_dims(ph) == Bool[1, 1, 0] -end - @testitem "Motion" tags=[:base] begin - # Test Motion constructors @testset "Constructors" begin action = Rotate(10.0, 20.0, 40.0) time = TimeRange(0.0, 0.0) spins = AllSpins() - + # Motion m = Motion(action, time, spins) - @test Motion(action) == m @test Motion(action, time) == m @test Motion(action, spins) == m + # MotionList + @test MotionList() == NoMotion() + @test MotionList(m) == m + end + @testset "Subset" begin + rotate = Rotate(10.0, 20.0, 40.0) + translate = Translate(0.1, 0.2, 0.3) + time = TimeRange(0.0, 0.0) + spins = AllSpins() + rng = rng = 1:2:5 + # NoMotion + nm = NoMotion() + @test nm[rng] == nm + # Motion + m = Motion(rotate, time, spins) + @test m[rng] == m + # MotionList + ml = MotionList( + Motion(rotate, time, spins), + Motion(translate, time, spins) + ) + @test ml[rng] == ml end - - # Tests obtaining spin positions + # Spin Positions @testset "NoMotion" begin ph = Phantom(x=[1.0, 2.0], y=[1.0, 2.0]) t_start=0.0; t_end=1.0 @@ -670,7 +562,6 @@ end @test yt == ph.y .+ dy @test zt == ph.z .+ dz end - @testset "FlowPath" begin # 1 spin ph = Phantom(x=[1.0], y=[1.0]) @@ -703,6 +594,167 @@ end @test yt == ph.y .+ dy @test zt == ph.z .+ dz end + @testset "Translate + Rotate" begin + ph = Phantom(x=[1.0], y=[1.0]) + t_start=0.0; t_end=1.0 + t = collect(range(t_start, t_end, 11)) + # Translate + dx, dy, dz = [1.0, 0.0, 0.0] + vx, vy, vz = [dx, dy, dz] ./ (t_end - t_start) + translation = Translate(dx, dy, dz, TimeRange(t_start, t_end)) + # Rotate + pitch, roll, yaw = [45.0, 0.0, 45.0] + rotation = Rotate(pitch, roll, yaw, TimeRange(t_start, t_end)) + r = vcat(ph.x, ph.y, ph.z) + R = rotz(π*yaw/180) * roty(π*roll/180) * rotx(π*pitch/180) + rot_x, rot_y, rot_z = R*r + # Combination into a MotionList + motion = MotionList(translation, rotation) + xt, yt, zt = get_spin_coords(motion, ph.x, ph.y, ph.z, t') + @test xt[end ,end] ≈ rot_x + vx*t[end] + @test yt[end ,end] ≈ rot_y + vy*t[end] + @test zt[end ,end] ≈ rot_z + vz*t[end] + end +end + +@testitem "Phantom" tags = [:base] begin + using Suppressor + # Phantom Struct Fields + name = "Bulks" + x = [-2e-3; -1e-3; 0.0; 1e-3; 2e-3] + y = [-4e-3; -2e-3; 0.0; 2e-3; 4e-3] + z = [-6e-3; -3e-3; 0.0; 3e-3; 6e-3] + ρ = [0.2; 0.4; 0.6; 0.8; 1.0] + T1 = [0.9; 0.9; 0.5; 0.25; 0.4] + T2 = [0.09; 0.05; 0.04; 0.07; 0.005] + T2s = [0.1; 0.06; 0.05; 0.08; 0.015] + Δw = [-2e-6; -1e-6; 0.0; 1e-6; 2e-6] + Dλ1 = [-4e-6; -2e-6; 0.0; 2e-6; 4e-6] + Dλ2 = [-6e-6; -3e-6; 0.0; 3e-6; 6e-6] + Dθ = [-8e-6; -4e-6; 0.0; 4e-6; 8e-6] + # Motion + Ns = length(x) + Nt = 3 + t_start = 0.0 + t_end = 1.0 + translate = Translate(0.05, 0.05, 0.0, Periodic(period=0.5, asymmetry=0.5)) + rotate = Rotate(0.0, 0.0, 90.0, TimeRange(t_start=0.05, t_end=0.5), SpinRange(1:3)) + path = Path(0.01 .* rand(Ns, Nt), 0.01 .* rand(Ns, Nt), 0.01 .* rand(Ns, Nt), TimeRange(t_start, t_end), SpinRange(2:2:4)) + @testset "Comparison" begin + obj1 = Phantom(name=name, x=x, y=y, z=z, ρ=ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ, motion=MotionList(translate, rotate)) + obj2 = Phantom(name=name, x=x, y=y, z=z, ρ=ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ, motion=MotionList(translate, rotate)) + @test obj1 == obj2 + obj2.x .+= 1e-10 + @test obj1 ≈ obj2 + end + @testset "Size and Length" begin + obj1 = Phantom(name=name, x=x, y=y, z=z, ρ=ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ, motion=MotionList(translate, rotate)) + @test size(obj1) == size(ρ) + @test length(obj1) == length(ρ) + end + @testset "Subset" begin + motion = MotionList(translate, rotate) + obj1 = Phantom(name, x, y, z, ρ, T1, T2, T2s, Δw, Dλ1, Dλ2, Dθ, motion) + rng = 1:2:5 + obj2 = Phantom( + name, x[rng], y[rng], z[rng], + ρ[rng], T1[rng], T2[rng], T2s[rng], + Δw[rng], Dλ1[rng], Dλ2[rng], Dθ[rng], + motion[rng] + ) + # Phantom subset + @test obj1[rng] == obj2 + @test @view(obj1[rng]) == obj2 + # Phantom view + obj_view = @view(obj1[rng]) + obj_view.ρ .= 0.0 + @test obj_view.ρ == obj1[rng].ρ + # BitVector range + obj3 = copy(obj1) + rng = obj1.x .> 0 + obj1.motion = Translate(5e-4, 6e-4, 7e-4, TimeRange(0.0, 1.0), SpinRange(rng)) + obj3.motion = Translate(5e-4, 6e-4, 7e-4, TimeRange(0.0, 1.0), SpinRange(1:length(obj3))) + @test obj1[rng] == obj3[rng] + @test obj1[rng].motion == obj3.motion[rng] + end + @testset "Addition" begin + obj1 = Phantom(name=name, x=x, y=y, z=z, ρ=ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ) + rng = 1:2:5 + obj2 = obj1[rng] + oba = Phantom( + name, [x; x[rng]], [y; y[rng]], [z; z[rng]], + [ρ; ρ[rng]], [T1; T1[rng]], [T2; T2[rng]], [T2s; T2s[rng]], + [Δw; Δw[rng]], [Dλ1; Dλ1[rng]], [Dλ2; Dλ2[rng]], [Dθ; Dθ[rng]], + vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + ) + # NoMotion + NoMotion + @test obj1 + obj2 == oba + # NoMotion + MotionList + obj2.motion = MotionList(translate, rotate) + oba.motion = vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + @test obj1 + obj2 == oba + # MotionList + NoMotion + obj1.motion = MotionList(translate, rotate) + obj2.motion = NoMotion() + oba.motion = vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + @test obj1 + obj2 == oba + # NoMotion + Motion + obj1.motion = NoMotion() + obj2.motion = translate + oba.motion = vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + @test obj1 + obj2 == oba + # Motion + NoMotion + obj1.motion = translate + obj2.motion = NoMotion() + oba.motion = vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + @test obj1 + obj2 == oba + # MotionList + MotionList + obj1.motion = MotionList(translate, rotate) + obj2.motion = MotionList(translate, rotate) + oba.motion = vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + @test obj1 + obj2 == oba + # Motion + Motion + obj1.motion = translate + obj2.motion = rotate + oba.motion = vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + @test obj1 + obj2 == oba + # Motion + MotionList + obj1.motion = translate + obj2.motion = MotionList(translate, rotate) + oba.motion = vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + @test obj1 + obj2 == oba + # MotionList + Motion + obj1.motion = MotionList(translate, rotate) + obj2.motion = translate + oba.motion = vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) + @test obj1 + obj2 == oba + end + @testset "Scalar multiplication" begin + obj1 = Phantom(name=name, x=x, y=y, z=z, ρ=ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ, motion=MotionList(translate, rotate)) + c = 7 + obc = Phantom(name=name, x=x, y=y, z=z, ρ=c*ρ, T1=T1, T2=T2, T2s=T2s, Δw=Δw, Dλ1=Dλ1, Dλ2=Dλ2, Dθ=Dθ, motion=MotionList(translate, rotate)) + @test c * obj1 == obc + end + @testset "Brain Phantom 2D" begin + ph = brain_phantom2D() + @test ph.name == "brain2D_axial" + @test KomaMRIBase.get_dims(ph) == Bool[1, 1, 0] + end + @testset "Brain Phantom 3D" begin + ph = brain_phantom3D() + @test ph.name == "brain3D" + @test KomaMRIBase.get_dims(ph) == Bool[1, 1, 1] + end + @testset "Pelvis Phantom" begin + ph = pelvis_phantom2D() + @test ph.name == "pelvis2D" + @test KomaMRIBase.get_dims(ph) == Bool[1, 1, 0] + end + @testset "Heart Phantom" begin + ph = heart_phantom() + @test ph.name == "LeftVentricle" + @test KomaMRIBase.get_dims(ph) == Bool[1, 1, 0] + end end @testitem "Scanner" tags=[:base] begin diff --git a/KomaMRIPlots/test/GUI_PlotlyJS_backend_test.jl b/KomaMRIPlots/test/GUI_PlotlyJS_backend_test.jl index 58ae36aa6..ed7800c5f 100644 --- a/KomaMRIPlots/test/GUI_PlotlyJS_backend_test.jl +++ b/KomaMRIPlots/test/GUI_PlotlyJS_backend_test.jl @@ -38,8 +38,9 @@ @testset "GUI_motion_phantom" begin ph = brain_phantom2D() #2D phantom - ph.motion = MotionList(Translate(0.1, 0.1, 0.1, TimeRange(0.0, 1.0), SpinRange(1:1000))) - + ph.motion = MotionList(Translate(0.1, 0.1, 0.1, TimeRange(0.0, 1.0), SpinRange(1:1000)), + Rotate(0.0, 0.0, 90.0, TimeRange(t_start=0.05, t_end=0.5), SpinRange(1:1000))) + @testset "plot_motion_phantom_map_rho" begin plot_phantom_map(ph, :ρ, width=800, height=600, max_spins=1_000) #Plotting the phantom's rho map (set max_spins=1_000) @test true #If the previous line fails the test will fail From 1ae5ee884c85b269dcdf364cb5f55f80d1a14bca Mon Sep 17 00:00:00 2001 From: Pablo Villacorta Aylagas Date: Wed, 11 Dec 2024 11:45:10 +0100 Subject: [PATCH 2/8] Add comment in KomaMRIBase/runtests.jl --- KomaMRIBase/test/runtests.jl | 2 ++ 1 file changed, 2 insertions(+) diff --git a/KomaMRIBase/test/runtests.jl b/KomaMRIBase/test/runtests.jl index f5029fdf7..ee2f18cb8 100644 --- a/KomaMRIBase/test/runtests.jl +++ b/KomaMRIBase/test/runtests.jl @@ -687,6 +687,8 @@ end [Δw; Δw[rng]], [Dλ1; Dλ1[rng]], [Dλ2; Dλ2[rng]], [Dθ; Dθ[rng]], vcat(obj1.motion, obj2.motion, length(obj1), length(obj2)) ) + # NOTE: these vcat methods must be simplified once the Vector{<:Motion} approach is accomplished: + # https://github.com/JuliaHealth/KomaMRI.jl/issues/480 # NoMotion + NoMotion @test obj1 + obj2 == oba # NoMotion + MotionList From faa285bc2335d698ab6a1934a4789de3fb9eaea7 Mon Sep 17 00:00:00 2001 From: Pablo Villacorta Aylagas Date: Sat, 14 Dec 2024 20:38:18 +0100 Subject: [PATCH 3/8] Add necessary `outflow_spin_reset!` method --- KomaMRICore/src/simulation/Flow.jl | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/KomaMRICore/src/simulation/Flow.jl b/KomaMRICore/src/simulation/Flow.jl index a9727b95d..970798569 100644 --- a/KomaMRICore/src/simulation/Flow.jl +++ b/KomaMRICore/src/simulation/Flow.jl @@ -2,19 +2,18 @@ function outflow_spin_reset!(args...; kwargs...) return nothing end -function outflow_spin_reset!( - spin_state_matrix, t, motion::MotionList; - replace_by=0, seq_t=0, add_t0=false -) - for m in motion.motions - outflow_spin_reset!( - spin_state_matrix, t, m.action, m.time, m.spins; - replace_by=replace_by, seq_t=seq_t, add_t0=add_t0 - ) +function outflow_spin_reset!(spin_state_matrix, t, ml::MotionList; replace_by=0, seq_t=0, add_t0=false) + for m in ml.motions + outflow_spin_reset!(spin_state_matrix, t, m; replace_by=replace_by, seq_t=seq_t, add_t0=add_t0) end return nothing end +function outflow_spin_reset!(spin_state_matrix, t, m::Motion; replace_by=0, seq_t=0, add_t0=false) + outflow_spin_reset!(spin_state_matrix, t, m.action, m.time, m.spins; replace_by=replace_by, seq_t=seq_t, add_t0=add_t0) + return nothing +end + function outflow_spin_reset!( spin_state_matrix::AbstractArray, t, From 3b6e79e716d7e7aa5fbf12530d14a7da6c9549f1 Mon Sep 17 00:00:00 2001 From: Pablo Villacorta Aylagas Date: Tue, 17 Dec 2024 19:05:52 +0100 Subject: [PATCH 4/8] Add digits to the time slider in `plot_phantom_map` --- KomaMRIPlots/src/ui/DisplayFunctions.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/KomaMRIPlots/src/ui/DisplayFunctions.jl b/KomaMRIPlots/src/ui/DisplayFunctions.jl index 5fce63a37..cec4f1e15 100644 --- a/KomaMRIPlots/src/ui/DisplayFunctions.jl +++ b/KomaMRIPlots/src/ui/DisplayFunctions.jl @@ -1258,7 +1258,7 @@ function plot_phantom_map( pad=attr(l=30, b=30), steps=[ attr( - label=round(t0*1e3), + label=round(t0*1e3; digits=2), method="update", args=[attr(visible=[fill(false, i-1); true; fill(false, length(t) - i)])] ) From 3d4019ec09681bd924fba04e3631dca6a5b2d4eb Mon Sep 17 00:00:00 2001 From: Pablo Villacorta Aylagas Date: Wed, 18 Dec 2024 12:14:35 +0100 Subject: [PATCH 5/8] Add auxiliary constructor for `FlowPath` --- KomaMRIBase/src/motion/actions/arbitraryactions/FlowPath.jl | 2 ++ KomaMRIBase/test/runtests.jl | 4 ++-- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/KomaMRIBase/src/motion/actions/arbitraryactions/FlowPath.jl b/KomaMRIBase/src/motion/actions/arbitraryactions/FlowPath.jl index 391a56839..12971f8ab 100644 --- a/KomaMRIBase/src/motion/actions/arbitraryactions/FlowPath.jl +++ b/KomaMRIBase/src/motion/actions/arbitraryactions/FlowPath.jl @@ -36,6 +36,8 @@ julia> flowpath = FlowPath( spin_reset::AbstractArray{Bool} end +FlowPath(dx::AbstractArray{T}, dy::AbstractArray{T}, dz::AbstractArray{T}, spin_reset::BitMatrix) where T<:Real = FlowPath(dx, dy, dz, collect(spin_reset)) + function add_jump_times!(t, a::FlowPath, time_span::AbstractTimeSpan) jump_times = (times(time_span)[end] - times(time_span)[1])/(size(a.spin_reset)[2]-1) * (getindex.(findall(a.spin_reset .== 1), 2) .- 1) .- 1e-6 append!(t, jump_times) diff --git a/KomaMRIBase/test/runtests.jl b/KomaMRIBase/test/runtests.jl index ee2f18cb8..9a1e1a415 100644 --- a/KomaMRIBase/test/runtests.jl +++ b/KomaMRIBase/test/runtests.jl @@ -572,7 +572,7 @@ end dx = rand(Ns, Nt) dy = rand(Ns, Nt) dz = rand(Ns, Nt) - arbitrarymotion = FlowPath(dx, dy, dz, collect(Bool.(zeros(Ns, Nt))), TimeRange(t_start, t_end)) + arbitrarymotion = FlowPath(dx, dy, dz, Bool.(zeros(Ns, Nt)), TimeRange(t_start, t_end)) t = range(t_start, t_end, Nt) xt, yt, zt = get_spin_coords(arbitrarymotion, ph.x, ph.y, ph.z, t') @test xt == ph.x .+ dx @@ -587,7 +587,7 @@ end dx = rand(Ns, Nt) dy = rand(Ns, Nt) dz = rand(Ns, Nt) - arbitrarymotion = FlowPath(dx, dy, dz, collect(Bool.(zeros(Ns, Nt))), TimeRange(t_start, t_end)) + arbitrarymotion = FlowPath(dx, dy, dz, Bool.(zeros(Ns, Nt)), TimeRange(t_start, t_end)) t = range(t_start, t_end, Nt) xt, yt, zt = get_spin_coords(arbitrarymotion, ph.x, ph.y, ph.z, t') @test xt == ph.x .+ dx From c92d9d4c592ee69ce9a67c6aff5d45cde989fd68 Mon Sep 17 00:00:00 2001 From: Pablo Villacorta Aylagas Date: Tue, 7 Jan 2025 00:16:17 +0100 Subject: [PATCH 6/8] Commit of the following: - Fix error in tests - Fix flow-related undetected errors from #531 --- KomaMRIBase/test/runtests.jl | 2 +- KomaMRICore/src/simulation/Flow.jl | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/KomaMRIBase/test/runtests.jl b/KomaMRIBase/test/runtests.jl index 87dfac72c..b2b029937 100644 --- a/KomaMRIBase/test/runtests.jl +++ b/KomaMRIBase/test/runtests.jl @@ -386,7 +386,7 @@ end @testset "Subset" begin rotate = Rotate(10.0, 20.0, 40.0) translate = Translate(0.1, 0.2, 0.3) - time = TimeRange(0.0, 0.0) + time = TimeRange(0.0, eps()) spins = AllSpins() rng = rng = 1:2:5 # NoMotion diff --git a/KomaMRICore/src/simulation/Flow.jl b/KomaMRICore/src/simulation/Flow.jl index 970798569..bec4e8387 100644 --- a/KomaMRICore/src/simulation/Flow.jl +++ b/KomaMRICore/src/simulation/Flow.jl @@ -18,14 +18,14 @@ function outflow_spin_reset!( spin_state_matrix::AbstractArray, t, action::FlowPath, - time_span, + time_curve, spin_span; replace_by=0, seq_t=0, add_t0=false, ) # Initialize time: add t0 and normalize - ts = KomaMRIBase.unit_time(init_time(t, seq_t, add_t0), time_span) + ts = KomaMRIBase.unit_time(init_time(t, seq_t, add_t0), time_curve.t, time_curve.t_unit, time_curve.periodic, time_curve.periods) # Get spin state range affected by the spin span idx = KomaMRIBase.get_indexing_range(spin_span) spin_state_matrix = @view(spin_state_matrix[idx, :]) @@ -42,14 +42,14 @@ function outflow_spin_reset!( M::Mag, t, action::FlowPath, - time_span, + time_curve, spin_span; replace_by=0, seq_t=0, add_t0=false, ) # Initialize time: add t0 and normalize - ts = KomaMRIBase.unit_time(init_time(t, seq_t, add_t0), time_span) + ts = KomaMRIBase.unit_time(init_time(t, seq_t, add_t0), time_curve.t, time_curve.t_unit, time_curve.periodic, time_curve.periods) # Get spin state range affected by the spin span idx = KomaMRIBase.get_indexing_range(spin_span) M = @view(M[idx]) From c79c2afaac5efd18e305dc10539150f6c58da456 Mon Sep 17 00:00:00 2001 From: pvillacorta Date: Mon, 13 Jan 2025 12:28:33 +0100 Subject: [PATCH 7/8] Store boolean fields as HDF5 attributes in KomaMRIFiles/Phantom.jl --- KomaMRIFiles/src/Phantom/Phantom.jl | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/KomaMRIFiles/src/Phantom/Phantom.jl b/KomaMRIFiles/src/Phantom/Phantom.jl index fe0c796aa..dd4173723 100644 --- a/KomaMRIFiles/src/Phantom/Phantom.jl +++ b/KomaMRIFiles/src/Phantom/Phantom.jl @@ -162,12 +162,12 @@ end function export_motion_subfield!(field_group::HDF5.Group, subfield::AbstractRange, subname::String) HDF5.attributes(field_group)[subname] = step(subfield) == 1 ? "$(first(subfield)):$(last(subfield))" : "$(first(subfield)):$(step(subfield)):$(last(subfield))" end +function export_motion_subfield!(field_group::HDF5.Group, subfield::Bool, subname::String) + HDF5.attributes(field_group)[subname] = string(subfield) +end function export_motion_subfield!(field_group::HDF5.Group, subfield::Array, subname::String) field_group[subname] = subfield end function export_motion_subfield!(field_group::HDF5.Group, subfield::BitMatrix, subname::String) field_group[subname] = Int.(subfield) -end -function export_motion_subfield!(field_group::HDF5.Group, subfield::Bool, subname::String) - field_group[subname] = string(subfield) end \ No newline at end of file From 85c01f8af8a10e26a68a3bc689408bba6ebc50c4 Mon Sep 17 00:00:00 2001 From: pvillacorta Date: Wed, 15 Jan 2025 10:06:59 +0100 Subject: [PATCH 8/8] Not import/export `t_start` and `t_end` fields in KomaMRIFiles/Phantom.jl --- KomaMRIFiles/src/Phantom/Phantom.jl | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/KomaMRIFiles/src/Phantom/Phantom.jl b/KomaMRIFiles/src/Phantom/Phantom.jl index dd4173723..efba27814 100644 --- a/KomaMRIFiles/src/Phantom/Phantom.jl +++ b/KomaMRIFiles/src/Phantom/Phantom.jl @@ -74,8 +74,10 @@ function import_motion_field!(motion_fields::Array, motion::HDF5.Group, name::St if type == subtype_string for subname in fieldnames(subtype_vector[i]) # dx, dy, dz, pitch, roll... key = string(subname) - subfield_value = key in keys(field_group) ? read(field_group, key) : read_attribute(field_group, key) - import_motion_subfield!(motion_subfields, subfield_value, key) + if !(key in ["t_start", "t_end"]) + subfield_value = key in keys(field_group) ? read(field_group, key) : read_attribute(field_group, key) + import_motion_subfield!(motion_subfields, subfield_value, key) + end end push!(motion_fields, (Symbol(name), subtype_vector[i](motion_subfields...))) end @@ -152,7 +154,7 @@ function export_motion_field!(field_group::HDF5.Group, field_value) HDF5.attributes(field_group)["type"] = string(typeof(field_value).name.name) for subname in fieldnames(typeof(field_value)) # dx, dy, dz, pitch, roll... subfield = getfield(field_value, subname) - export_motion_subfield!(field_group, subfield, string(subname)) + !(string(subname) in ["t_start", "t_end"]) ? export_motion_subfield!(field_group, subfield, string(subname)) : nothing end end