Head Models and Coordinate Reference Systems (CRS)
[1]:
# This cells setups the environment when executed in Google Colab.
try:
    import google.colab
    !curl -s https://raw.githubusercontent.com/ibs-lab/cedalion/dev/scripts/colab_setup.py -o colab_setup.py
    # Select branch with --branch "branch name" (default is "dev")
    %run colab_setup.py
except ImportError:
    pass
[2]:
import pyvista as pv
#pv.set_jupyter_backend('server')
pv.set_jupyter_backend('static')
import os
import xarray as xr
import cedalion
import cedalion.io
import cedalion.plots
import cedalion.datasets
import cedalion.imagereco.forward_model as fw
xr.set_options(display_expand_data=False);
Loading the ICBM-152 head model
- the - TwoSurfaceHeadModelholds the segmented MRT image and derived cortex and scalp surfaces
- we provide functionality to derive these surfaces from the masks or to load them from files 
[3]:
# load pathes to segmentation data for the icbm-152 atlas
SEG_DATADIR, mask_files, landmarks_file = cedalion.datasets.get_icbm152_segmentation()
# create forward model class for icbm152 atlas
head_icbm152 = fw.TwoSurfaceHeadModel.from_surfaces(
    segmentation_dir=SEG_DATADIR,
    mask_files=mask_files,
    brain_surface_file=os.path.join(SEG_DATADIR, "mask_brain.obj"),
    landmarks_ras_file=landmarks_file,
    brain_face_count=None,
    scalp_face_count=None,
)
Visualization
[4]:
plt = pv.Plotter()
cedalion.plots.plot_surface(plt, head_icbm152.brain, color="#d3a6a1")
cedalion.plots.plot_surface(plt, head_icbm152.scalp, opacity=.1)
cedalion.plots.plot_labeled_points(plt, head_icbm152.landmarks, show_labels=True)
plt.show()
 
Segmentation masks
The head model comprises masks for different tissue types: CSF, Gray Matter, White Matter, Scalp and Skull
[5]:
head_icbm152.segmentation_masks
[5]:
<xarray.DataArray (segmentation_type: 5, i: 193, j: 239, k: 263)> Size: 61MB 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 ... 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 Coordinates: * segmentation_type (segmentation_type) <U5 100B 'csf' 'gm' ... 'skull' 'wm' Dimensions without coordinates: i, j, k
Coordinate System
- we need to distinguish several coordinate systems: voxel space, scanner space, subject space, … 
- geometric data types carry information about which crs they use 
- transformations between coordinate systems through affine transformations 
The head model is loaded in voxel space (‘ijk’)
[6]:
head_icbm152.crs
[6]:
'ijk'
The head model contains initial landmarks (‘Nz’, ‘Iz’, ‘LPA’ and ‘RPA’) stored as a LabeledPointCloud. The crs is stored as the name of the second dimension, easily retrievable through the .points-accessor
[7]:
display(head_icbm152.landmarks)
display(head_icbm152.landmarks.points.crs)
<xarray.DataArray (label: 4, ijk: 3)> Size: 96B
[] 96.0 218.1 108.0 96.0 18.9 101.0 15.9 120.0 102.0 178.1 120.0 103.0
Coordinates:
  * label    (label) <U3 48B 'Nz' 'Iz' 'LPA' 'RPA'
    type     (label) object 32B PointType.LANDMARK ... PointType.LANDMARK
Dimensions without coordinates: ijk
'ijk'
Triangulated surface meshes of the scalp and brain:
[8]:
display(head_icbm152.brain)
display(head_icbm152.scalp)
TrimeshSurface(mesh=<trimesh.Trimesh(vertices.shape=(15002, 3), faces.shape=(29978, 3))>, crs='ijk', units=<Unit('dimensionless')>, vertex_coords={})
TrimeshSurface(mesh=<trimesh.Trimesh(vertices.shape=(255950, 3), faces.shape=(511242, 3))>, crs='ijk', units=<Unit('dimensionless')>, vertex_coords={})
[9]:
head_icbm152.t_ijk2ras # transformation from voxel to subject (RAS) space
[9]:
<xarray.DataArray (aligned: 4, ijk: 4)> Size: 128B [mm] 1.0 0.0 0.0 -96.0 0.0 1.0 0.0 -132.0 0.0 0.0 1.0 -148.0 0.0 0.0 0.0 1.0 Dimensions without coordinates: aligned, ijk
Change to subject (RAS) space by applying an affine transformation on the head model. This transforms all components.
Here, the subject space is called ‘aligned’ (the label is derived from information in the mask’s nifti file)
The scanner space also uses physical units whereas coordinates in voxel space are dimensionless.
[10]:
trafo = head_icbm152.t_ijk2ras
head_icbm152_ras = head_icbm152.apply_transform(trafo)
display(head_icbm152_ras.crs)
display(head_icbm152_ras.landmarks.points.crs)
display(head_icbm152_ras.brain.crs)
display(head_icbm152.landmarks.pint.units)
display(head_icbm152_ras.landmarks.pint.units)
'aligned'
'aligned'
'aligned'