Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Scripts to project 3D and 4D maps to streamline points and perform math operations in space of streamline points #810

Merged
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
Show all changes
79 commits
Select commit Hold shift + click to select a range
2ff0002
project maps to streamline points and do math
Nov 16, 2023
7eef429
fixed pep8
Nov 23, 2023
4d64eb6
fixed logging line
Nov 23, 2023
2fb63cd
fixed trailing whitespace
Nov 23, 2023
ba5a516
removed print statements
Nov 23, 2023
797bd95
removed useless assert resolution
Nov 23, 2023
cf12dc7
changed to voxel world coords for interpolation
Nov 24, 2023
5d014a8
fixed dimension error on endpoints only single val
Nov 24, 2023
6de542c
remove deepcopy
Nov 24, 2023
ba839fb
fixed too long lines
Nov 24, 2023
a4af4ac
fixed indent
Nov 24, 2023
fb0d9af
fixexed indents
Nov 24, 2023
76a78c4
update output dpp name
Nov 29, 2023
0e6fb5a
moved project_metrics func to module
grahamlittlephd Dec 14, 2023
53ec9cc
moved streamline ops to module
grahamlittlephd Dec 14, 2023
061f399
added blank unit tests
grahamlittlephd Dec 14, 2023
c8b3889
added script test
grahamlittlephd Dec 14, 2023
1f0ac45
renamed scripts
grahamlittlephd Dec 14, 2023
44fe697
update naming in tests
grahamlittlephd Dec 14, 2023
24d193e
update master
Jan 10, 2024
16e3084
add blank unit tests to streamline operations
Jan 10, 2024
b5660fd
pep errors
grahamlittlephd Jan 11, 2024
a0fc992
pep errors 2
grahamlittlephd Jan 11, 2024
f378999
Merge branch 'master' into Project-endpoint-data-to-streamlines
Jan 12, 2024
e1795fb
tests passed
Jan 15, 2024
dded388
added test for point math
Jan 15, 2024
7592d60
changed arg name for point math output
Jan 15, 2024
8a6973d
pep 8 long lines
Jan 15, 2024
62c103c
updated function names in stream_ops
Jan 15, 2024
ffc2c88
Fixed for loops and added multi input
Jan 25, 2024
89f40d2
updated projection tests with new required args
Jan 25, 2024
ed059af
Merge remote-tracking branch 'origin/master' into Project-endpoint-da…
Jan 25, 2024
e8bcadd
updated point math test with req args
Jan 25, 2024
212975b
added test for point math correlation
Jan 25, 2024
bdcf750
pep8 fix
Jan 26, 2024
4c2d844
Updated doc and logging in point_math
grahamlittlephd Jan 29, 2024
67873b9
added dpp and dpp mode argument
Jan 29, 2024
154f8ed
added overwrite functionality
Jan 29, 2024
e7b5f0b
simplify overwrite
Jan 30, 2024
39f65ef
remove append option
Jan 30, 2024
6b33f7c
arg multiinput just narg+
Jan 30, 2024
776e04d
Merge branch 'master' into Project-endpoint-data-to-streamlines
Jan 30, 2024
e679fa1
fixed project map tests
Jan 30, 2024
4046cc1
rm doc reference to endpoints
Jan 30, 2024
0c5cdd3
update test tractogram point math new args
Jan 30, 2024
0963198
updated name of to dpp_math
Jan 30, 2024
ff8c377
update doc for clearer on what happens in 4D case
Jan 30, 2024
d3fd65a
remove print
Jan 30, 2024
0778e96
moved project func to new file
Feb 5, 2024
918d4c2
changed metric to map
Feb 5, 2024
ce94ab0
changed metric to map
Feb 5, 2024
75a54bf
added dummy test for dpp management
Feb 5, 2024
74d6a87
4D dps math - now outputs array per streamline
Feb 5, 2024
507df9f
pep8 issues
Feb 5, 2024
0c19111
Merge branch 'master' into Project-endpoint-data-to-streamlines
Feb 7, 2024
3a59f6a
pep8
Feb 7, 2024
1f69f94
remove py version file
Feb 7, 2024
47d4a21
added utf header
Feb 7, 2024
8e8bb69
removed py version file
grahamlittlephd Feb 8, 2024
bc52040
Merge branch 'master' into Project-endpoint-data-to-streamlines
grahamlittlephd Feb 8, 2024
9ea5ec0
added back py ver file in root
grahamlittlephd Feb 8, 2024
03efce2
Merge branch 'master' into Project-endpoint-data-to-streamlines
Feb 19, 2024
d0410dd
remove unused StatefulTractogram
Feb 19, 2024
ced0217
removed extra spaces
Feb 19, 2024
0c8e07b
removed default false
Feb 19, 2024
e401f55
changed to nib.load
Feb 19, 2024
0f0abcc
added dual keep_dpp and ovewrite args
Feb 19, 2024
3fed0f1
changed map to map_volume
Feb 19, 2024
691c620
updated logging in project_map
Feb 19, 2024
172e821
remove if statement
Feb 19, 2024
d4c4f57
fixed comments in dpp_math
Feb 19, 2024
0a4966a
updated to mode required argument
Feb 19, 2024
efa733a
added keep_all_dpp_dps option
Feb 19, 2024
f4b4e39
moved streamline ops to dps_and_dpp
Feb 19, 2024
e0e70a3
blank tests to dps_dpp module
Feb 19, 2024
6d752e5
pep8
Feb 19, 2024
a21aa16
4D data stored as shape (#,) fixed checks
Feb 19, 2024
8db77a9
fixed dimension on tuple
Feb 19, 2024
47f287b
updated dimension on data_shape
Feb 19, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
164 changes: 164 additions & 0 deletions scripts/scil_project_map_to_streamlines.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,164 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-

"""
Projects metrics extracted from a map onto the endpoints of streamlines.

The default options will take data from a nifti image (3D or ND) and
project it onto the endpoints of streamlines.
"""

import argparse
import logging
from copy import deepcopy

from numpy import asarray, reshape, repeat

from scilpy.io.image import load_img
from scilpy.io.streamlines import load_tractogram_with_reference
from scilpy.io.utils import (add_overwrite_arg,
assert_inputs_exist,
assert_outputs_exist,
add_reference_arg,
add_verbose_arg)

from scilpy.image.volume_space_management import DataVolume

from dipy.io.streamline import save_tractogram, StatefulTractogram


def project_metric_to_streamlines(sft, metric, endpoints_only=False):
grahamlittlephd marked this conversation as resolved.
Show resolved Hide resolved
"""
Projects a metric onto the points of streamlines.

Parameters
----------
sft: StatefulTractogram
Input tractogram.
metric: DataVolume
Input metric.

Optional:
---------
endpoints_only: bool
If True, will only project the metric onto the endpoints of the
streamlines (all values along streamlines set to zero). If False,
will project the metric onto all points of the streamlines.

Returns
-------
streamline_data:
metric projected to each point of the streamlines.
"""
if len(metric.data.shape) == 4:
dimension = metric.data.shape[3]
else:
dimension = 1

streamline_data = []
if endpoints_only:
for s in sft.streamlines:
p1_data = metric.get_value_at_coordinate(
s[0][0], s[0][1], s[0][2], space=sft.space, origin=sft.origin)
p2_data = metric.get_value_at_coordinate(
s[-1][0], s[-1][1], s[-1][2], space=sft.space, origin=sft.origin)
thisstreamline_data = []
for p in s:
thisstreamline_data.append(
asarray(repeat(0, p1_data.shape[0])))
grahamlittlephd marked this conversation as resolved.
Show resolved Hide resolved

thisstreamline_data[0] = p1_data
thisstreamline_data[-1] = p2_data
thisstreamline_data = asarray(thisstreamline_data)

streamline_data.append(
reshape(thisstreamline_data, (len(thisstreamline_data), dimension)))
else:
for s in sft.streamlines:
thisstreamline_data = []
for p in s:
thisstreamline_data.append(metric.get_value_at_coordinate(
p[0], p[1], p[2], space=sft.space, origin=sft.origin))

streamline_data.append(
reshape(thisstreamline_data, (len(thisstreamline_data), dimension)))

return streamline_data


def _build_arg_parser():
p = argparse.ArgumentParser(
description=__doc__,
formatter_class=argparse.RawTextHelpFormatter)

# Mandatory arguments input and output tractogram must be in trk format
p.add_argument('in_tractogram',
help='Fiber bundle file.')
p.add_argument('in_metric',
help='Nifti metric to project onto streamlines.')
p.add_argument('out_tractogram',
help='Output file.')

# Optional arguments
p.add_argument('--trilinear', action='store_true',
help='If set, will use trilinear interpolation \n'
'else will use nearest neighbor interpolation \n'
'by default.')

p.add_argument('--endpoints_only', action='store_true',
help='If set, will only project the metric onto the endpoints \n'
'of the streamlines (all other values along streamlines set to zero). \n'
'If not set, will project the metric onto all points of the streamlines.')

p.add_argument('--dpp_name', default='metric',
help='Name of the data_per_point to be saved in the output tractogram. \n'
'(Default: %(default)s)')
add_reference_arg(p)
add_overwrite_arg(p)
add_verbose_arg(p)
return p


def main():
parser = _build_arg_parser()
args = parser.parse_args()

assert_inputs_exist(parser, [args.in_tractogram, args.in_metric])

assert_outputs_exist(parser, args, [args.out_tractogram])

logging.debug("Loading the tractogram...")
sft = load_tractogram_with_reference(parser, args, args.in_tractogram)
sft.to_vox()
sft.to_corner()

if len(sft.streamlines) == 0:
logging.warning('Empty bundle file {}. Skipping'.format(args.bundle))
return

logging.debug("Loading the metric...")
metric_img, metric_dtype = load_img(args.in_metric)
metric_data = metric_img.get_fdata(caching='unchanged', dtype=float)
metric_res = metric_img.header.get_zooms()[:3]

if args.trilinear:
interp = "trilinear"
else:
interp = "nearest"

metric = DataVolume(metric_data, metric_res, interp)

logging.debug("Projecting metric onto streamlines")
streamline_data = project_metric_to_streamlines(sft, metric,
endpoints_only=args.endpoints_only)

logging.debug("Saving the tractogram...")
data_per_point = {}
data_per_point[args.dpp_name] = streamline_data
out_sft = StatefulTractogram(sft.streamlines, metric_img,
sft.space, sft.origin, data_per_point=data_per_point)
save_tractogram(out_sft, args.out_tractogram)


if __name__ == '__main__':
main()
Loading