Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
F
fslpy
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Wiki
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Deploy
Releases
Container Registry
Model registry
Monitor
Incidents
Analyze
Value stream analytics
Contributor analytics
Repository analytics
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
Michiel Cottaar
fslpy
Commits
64862743
Commit
64862743
authored
8 years ago
by
Paul McCarthy
Browse files
Options
Downloads
Patches
Plain Diff
Refactorings/documentations/fixes to image wrapper. More to come.
parent
944c6390
No related branches found
No related tags found
No related merge requests found
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
fsl/data/imagewrapper.py
+82
-42
82 additions, 42 deletions
fsl/data/imagewrapper.py
with
82 additions
and
42 deletions
fsl/data/imagewrapper.py
+
82
−
42
View file @
64862743
...
...
@@ -9,7 +9,8 @@ to manage data access to ``nibabel`` NIFTI images.
"""
import
logging
import
logging
import
collections
import
numpy
as
np
import
nibabel
as
nib
...
...
@@ -38,9 +39,21 @@ class ImageWrapper(notifier.Notifier):
property.
.. todo:: Figure out if NIFTI2 can be supported as well.
The ``ImageWrapper`` class uses the following functions (also defined in
this module) to keep track of the portion of the image that has currently
been included in the data range calculation:
.. autosummary::
:nosignatures:
sliceObjToSliceTuple
sliceTupleToSliceObj
sliceCovered
calcSliceExpansion
adjustCoverage
"""
def
__init__
(
self
,
image
,
name
=
None
,
loadData
=
False
):
"""
Create an ``ImageWrapper``.
...
...
@@ -85,7 +98,11 @@ class ImageWrapper(notifier.Notifier):
# been included in the data range calculation, so
# we do not unnecessarily re-calculate ranges on
# the same part of the image.
self
.
__sliceCoverage
=
[]
self
.
__coverage
=
[]
# TODO Use a numpy array of size
# (2, [numRealDims - 1], [shape[numRealDims - 1]])
# instead of a list of lists
# This is a list of lists of (low, high) pairs,
# one list for each entry in the last dimension
...
...
@@ -96,7 +113,7 @@ class ImageWrapper(notifier.Notifier):
for
i
in
range
(
image
.
shape
[
self
.
__numRealDims
-
1
]):
cov
=
[[
None
,
None
]
for
i
in
range
(
self
.
__numRealDims
-
1
)]
self
.
__
sliceC
overage
.
append
(
cov
)
self
.
__
c
overage
.
append
(
cov
)
if
loadData
:
self
.
loadData
()
...
...
@@ -128,7 +145,7 @@ class ImageWrapper(notifier.Notifier):
"""
if
isTuple
:
sliceobj
=
sliceTuple
t
oSliceObj
(
sliceobj
)
sliceobj
=
sliceTuple
T
oSliceObj
(
sliceobj
)
# If the image has not been loaded
# into memory, we can use the nibabel
...
...
@@ -168,10 +185,10 @@ class ImageWrapper(notifier.Notifier):
self
.
__name
,
self
.
__range
[
0
],
self
.
__range
[
1
],
self
.
__
sliceC
overage
))
self
.
__
c
overage
))
volumes
,
expansions
=
calcSliceExpansion
(
slices
,
self
.
__
sliceC
overage
,
self
.
__
c
overage
,
self
.
__numRealDims
,
self
.
__numPadDims
)
...
...
@@ -190,10 +207,7 @@ class ImageWrapper(notifier.Notifier):
self
.
__range
=
(
newmin
,
newmax
)
for
vol
,
exp
in
zip
(
volumes
,
expansions
):
self
.
__sliceCoverage
[
vol
]
=
adjustSliceCoverage
(
self
.
__sliceCoverage
[
vol
],
exp
,
self
.
__numRealDims
)
self
.
__coverage
[
vol
]
=
adjustCoverage
(
self
.
__coverage
[
vol
],
exp
)
# TODO floating point error
if
newmin
!=
oldmin
or
newmax
!=
oldmax
:
...
...
@@ -234,7 +248,7 @@ class ImageWrapper(notifier.Notifier):
slices
=
sliceObjToSliceTuple
(
sliceobj
,
self
.
__image
.
shape
)
if
not
sliceCovered
(
slices
,
self
.
__
sliceC
overage
,
self
.
__
c
overage
,
self
.
__image
.
shape
,
self
.
__numRealDims
):
...
...
@@ -243,14 +257,25 @@ class ImageWrapper(notifier.Notifier):
return
data
def
sliceObjToSliceTuple
(
sliceobj
,
shape
):
"""
Turns an array slice object into a tuple of (low, high) index
pairs, one pair for each dimension in the given shape
:arg sliceobj: Something which can be used to slice an array of shape
``shape``.
:arg shape: Shape of the array being sliced.
"""
indices
=
[]
if
not
isinstance
(
sliceobj
,
collections
.
Sequence
):
sliceobj
=
[
sliceobj
]
if
len
(
sliceobj
)
!=
len
(
shape
):
missing
=
len
(
shape
)
-
len
(
sliceobj
)
sliceobj
=
list
(
sliceobj
)
+
[
slice
(
None
)
for
i
in
range
(
missing
)]
for
dim
,
s
in
enumerate
(
sliceobj
):
# each element in the slices tuple should
...
...
@@ -266,8 +291,11 @@ def sliceObjToSliceTuple(sliceobj, shape):
return
tuple
(
indices
)
def
sliceTupletoSliceObj
(
slices
):
"""
def
sliceTupleToSliceObj
(
slices
):
"""
Turns a sequence of (low, high) index pairs into a tuple of array
slice objects.
:arg slices: A sequence of (low, high) index pairs.
"""
sliceobj
=
[]
...
...
@@ -278,10 +306,41 @@ def sliceTupletoSliceObj(slices):
return
tuple
(
sliceobj
)
def
adjustCoverage
(
oldCoverage
,
slices
):
"""
Adjusts/expands the given ``oldCoverage`` so that it covers the
given set of ``slices``.
:arg oldCoverage: A sequence of (low, high) index pairs
:arg slices: A sequence of (low, high) index pairs. If ``slices``
contains more dimensions than are specified in
``oldCoverage``, the trailing dimensions are ignored.
:return: A list of (low, high) tuples containing the adjusted coverage.
"""
newCoverage
=
[]
for
dim
in
range
(
len
(
oldCoverage
)):
low
,
high
=
slices
[
dim
]
lowCover
,
highCover
=
oldCoverage
[
dim
]
if
lowCover
is
None
or
low
<
lowCover
:
lowCover
=
low
if
highCover
is
None
or
high
>
highCover
:
highCover
=
high
newCoverage
.
append
((
lowCover
,
highCover
))
return
newCoverage
def
sliceCovered
(
slices
,
sliceC
overage
,
shape
,
realDims
):
def
sliceCovered
(
slices
,
c
overage
,
shape
,
realDims
):
"""
Returns ``True`` if the portion of the image data calculated by
the given ``slices` has already been calculated, ``False`` otherwise.
:arg slices:
:arg coverage:
:arg shape:
:arg volDim:
"""
lowVol
,
highVol
=
slices
[
realDims
-
1
]
...
...
@@ -289,11 +348,11 @@ def sliceCovered(slices, sliceCoverage, shape, realDims):
for
vol
in
range
(
lowVol
,
highVol
):
c
overage
=
sliceC
overage
[
vol
]
volC
overage
=
c
overage
[
vol
]
for
dim
,
size
in
enumerate
(
shape
):
lowCover
,
highCover
=
c
overage
[
dim
]
lowCover
,
highCover
=
volC
overage
[
dim
]
if
lowCover
is
None
or
highCover
is
None
:
return
False
...
...
@@ -309,7 +368,7 @@ def sliceCovered(slices, sliceCoverage, shape, realDims):
return
True
def
calcSliceExpansion
(
slices
,
sliceC
overage
,
realDims
,
padDims
):
def
calcSliceExpansion
(
slices
,
c
overage
,
realDims
,
padDims
):
"""
"""
...
...
@@ -324,13 +383,13 @@ def calcSliceExpansion(slices, sliceCoverage, realDims, padDims):
for
vol
in
volumes
:
c
overage
=
sliceC
overage
[
vol
]
expansion
=
[]
volC
overage
=
c
overage
[
vol
]
expansion
=
[]
for
dim
in
range
(
realDims
-
1
):
lowCover
,
highCover
=
c
overage
[
dim
]
lowSlice
,
highSlice
=
slices
[
dim
]
lowCover
,
highCover
=
volC
overage
[
dim
]
lowSlice
,
highSlice
=
slices
[
dim
]
if
lowCover
is
None
:
lowCover
=
lowSlice
if
highCover
is
None
:
highCover
=
highSlice
...
...
@@ -345,22 +404,3 @@ def calcSliceExpansion(slices, sliceCoverage, realDims, padDims):
expansions
.
append
(
expansion
)
return
volumes
,
expansions
def
adjustSliceCoverage
(
oldCoverage
,
slices
,
realDims
):
"""
"""
newCoverage
=
[]
for
dim
in
range
(
realDims
-
1
):
low
,
high
=
slices
[
dim
]
lowCover
,
highCover
=
oldCoverage
[
dim
]
if
lowCover
is
None
or
low
<
lowCover
:
lowCover
=
low
if
highCover
is
None
or
high
<
highCover
:
highCover
=
high
newCoverage
.
append
((
lowCover
,
highCover
))
return
newCoverage
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment