Skip to content

Commit

Permalink
scale diagonal values correctly for ACR MRI
Browse files Browse the repository at this point in the history
  • Loading branch information
jrkerns committed Dec 30, 2024
1 parent 4ea5d46 commit 7da163e
Show file tree
Hide file tree
Showing 3 changed files with 46 additions and 3 deletions.
10 changes: 10 additions & 0 deletions docs/source/changelog.rst
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,16 @@ Legend
* :bdg-primary:`Refactor` denotes a code refactor; usually this means an efficiency boost or code cleanup.
* :bdg-danger:`Change` denotes a change that may break existing code.

v 3.31.0
--------

ACR MRI
^^^^^^^

* :bdg-warning:`Fixed` The positive and negative diagonal geometric distortion calculations (``results_data().geometric_distortion_module.profiles["negative diagonal"]['width (mm)']``) for the ACR Large MRI phantom were not scaled correctly.
Since the pixel distance is diagonal the physical spacing between pixels is actually :math:`\sqrt{2}` times the pixel spacing.
This is now fixed. Prior values can be scaled by :math:`\sqrt{2}` to get the correct values.

v 3.30.0
--------

Expand Down
9 changes: 6 additions & 3 deletions pylinac/acr.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
from __future__ import annotations

import io
import math
import textwrap
import warnings
import webbrowser
Expand Down Expand Up @@ -1079,7 +1080,6 @@ def _setup_rois(self) -> None:
ys = xs + b
coords = ndimage.map_coordinates(bin_image, [ys, xs], order=1, mode="mirror")
f_data = fill_middle_zeros(coords, cutoff_px=px_to_cut_off)
# pixels are now diagonal and thus spacing between pixels is now the hypotenuse
prof = FWXMProfile(values=f_data)
line = Line(
Point(
Expand All @@ -1091,8 +1091,11 @@ def _setup_rois(self) -> None:
ys[int(round(prof.field_edge_idx(side="right")))],
),
)
# pixels are now diagonal and thus spacing between pixels is now the hypotenuse
# We don't have to fix the line above because that's in pixels. The issue is the
# geometric distance and thus we only have to correct here.
self.profiles["negative diagonal"] = {
"width (mm)": prof.field_width_px * self.mm_per_pixel,
"width (mm)": prof.field_width_px * self.mm_per_pixel * math.sqrt(2),
"line": line,
}
# calculate positive diagonal
Expand All @@ -1114,7 +1117,7 @@ def _setup_rois(self) -> None:
),
)
self.profiles["positive diagonal"] = {
"width (mm)": prof.field_width_px * self.mm_per_pixel,
"width (mm)": prof.field_width_px * self.mm_per_pixel * math.sqrt(2),
"line": line,
}

Expand Down
30 changes: 30 additions & 0 deletions tests_basic/test_acr.py
Original file line number Diff line number Diff line change
Expand Up @@ -300,6 +300,36 @@ def test_load_from_list_of_streams(self):
paths = [io.BytesIO(open(p, "rb").read()) for p in paths]
ACRMRILarge(paths)

def test_geometric_distortion_profile_lengths(self):
path = get_file_from_cloud_test_repo([*TEST_DIR_MR, "GE 3T.zip"])
mri = ACRMRILarge.from_zip(path)
mri.analyze()
data = mri.results_data()
self.assertAlmostEqual(
data.geometric_distortion_module.profiles["negative diagonal"][
"width (mm)"
],
190.63,
delta=0.05,
)
self.assertAlmostEqual(
data.geometric_distortion_module.profiles["positive diagonal"][
"width (mm)"
],
190.59,
delta=0.05,
)
self.assertAlmostEqual(
data.geometric_distortion_module.profiles["vertical"]["width (mm)"],
190.44,
delta=0.05,
)
self.assertAlmostEqual(
data.geometric_distortion_module.profiles["horizontal"]["width (mm)"],
190.44,
delta=0.05,
)


class TestACRMRIResultData(TestCase, ResultsDataBase):
def construct_analyzed_instance(self):
Expand Down

0 comments on commit 7da163e

Please sign in to comment.