5 GLM 2
In the held-out validation data, hexadirectional signal modulations based on the direction of trajectories relative to the estimated orientation \(\varphi\) are tested in a new GLM (GLM2, see figure below). These modulations can be tested for in two slightly different ways: In a continuous approach using parametric regression or in an approach based on binning the data. Both approaches are described below. In either case, you will typically use the result of GLM2 for your group-level statistics.
Parametric regression looking for sinusoidal modulation
In this approach, a parametric regressor testing for a six-fold sinusoidal modulation \((cos(6[\theta(t)-\varphi]))\) will be fitted in GLM2 (see top of the figure above).
From Doeller et al. (Nature, 2010):
“We looked for sinusoidal modulation of activation with six-fold rotational symmetry aligned with each participant’s mean grid orientation across the whole brain (using the regressor \((cos(6[\theta(t)-\varphi]))\)) where \(\theta(t)\) is running direction…”
Contrast of aligned vs. misaligned trajectories
An alternative approach is to use separate regressors for trajectories that are “aligned” (within ± 15° of the nearest axis of the hexadirectional orientation) and “misaligned” (more than ± 15° from the hexadirectional orientation \(\varphi\)) in GLM2 (see bottom of the figure above).
From the methods of Kunz et al. (Science, 2015):
“…a second GLM was applied to the a priori defined second half of data (sessions 2, 4, and 6) containing regressors for”aligned” (within ± 15° of the nearest axis of the grid) and “misaligned” (more than ± 15° from a grid axis) movements in the virtual arena (Fig. 2A, right panel; Table S5). Contrast values (aligned > misaligned) were extracted from the right EC ROI and averaged across voxels within participants.”
For now, the code below implements the parametric modulations approach.
Let’s start by loading the estimated orientation to use as reference
<- here("data", "glm1", "glm1_firsthalf", "avg_orient_ec.txt")
fn
<- read_delim(file = fn, delim = "\t",
ref_orient col_names = TRUE, col_types = c("fffdddf"))
head(ref_orient)
## # A tibble: 1 × 7
## subID run periodicity r avg_ori_deg avg_ori_rad weighted
## <fct> <fct> <fct> <dbl> <dbl> <dbl> <fct>
## 1 sub01 average 6 0.220 29.3 0.512 weighted
We need to run GLM 2 on a data partition that is independent from the one used to estimate GLM 1. In this example case, this is the second half of each run.
5.1 Parametric modulation with cosine regressor
5.1.1 EV files for FEAT
For each regressor, FSL FEAT expects an EV file with 3-column format for each regressor. Let’s load the modified event file and write out the regressors we will want to use. For the parametric modulation approach these are:
- navigation EV for all timepoints of translational movement in the entire run
- feedback EV modulating times participants received feedback (entire run)
- Cosine of the angular difference between movement direction and estimated orientation to be used as a parametric modulator for the second half of onsets of the navigation EV. To look at an n-fold periodic signal, the angular difference between estimated orientation and the trajectory angle is multiplied by the desired periodicity (e.g. 6 for hexadirectional analysis) before calculating the cosine.
# we want to look for six-fold signals, other periodicities are controls
<- c(6)
periodicities
for (i_sub in subjects){
for (i_run in runs){
# create directory for EV files
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", "feat_ev", i_sub, i_run)
ev_dir if(!dir.exists(ev_dir)){dir.create(ev_dir, recursive = TRUE)}
# read the event table file
<- here("data", "EventTables", i_sub, paste0("eventTable_", i_run, "_jb.txt"))
fn <- read_delim(fn, delim = "\t", col_names = TRUE, col_types = c("fddd"))
event_table
# write the navigation (translation) regressor to file in 3-column format for FSL
<- file.path(ev_dir, "translation.txt")
fn %>% filter(name=="translation") %>%
event_table select(c(onset, duration)) %>%
mutate(weight=1) %>%
write_delim(file = fn, delim = "\t", col_names = FALSE)
# write the feedback regressor to file in 3-column format for FSL
<- file.path(ev_dir, "feedback.txt")
fn %>% filter(name=="feedback") %>%
event_table select(c(onset, duration)) %>%
mutate(weight=1) %>%
write_delim(file = fn, delim = "\t", col_names = FALSE)
for (i_period in periodicities){
# get the relevant reference orientation
<- ref_orient %>%
curr_ref_orient filter(subID==i_sub, periodicity==i_period) %>%
pull(avg_ori_deg)
# add cosine of angular difference of running angle and reference orientation
<- event_table %>%
event_table mutate(
ref_orient = curr_ref_orient,
# angular difference of running direction and reference orientation (max 180 degrees)
ang_diff2ref = if_else(angle - curr_ref_orient < 180, # condition
- curr_ref_orient, # if condition met
angle 360 - (angle - curr_ref_orient)), # if condition not met
# CAVE: transform angles to radians before calculating cosine!
cosine = cos(i_period * (ang_diff2ref*pi/180)))
head(event_table, n=10)
# write the cosine of translation angle regressor to file in 3-column format for 2nd half of trials
<- file.path(ev_dir, paste0("translation_cosine_", i_period, "fold_half2.txt"))
fn %>% filter(name=="translation") %>%
event_table slice_tail(n=round(nrow(.)/2)) %>% # second half (round up for uneven numbers)
select(c(onset, duration, cosine)) %>%
write_delim(file = fn, delim = "\t", col_names = FALSE)
}
} }
5.1.2 Set up the GLM using FSL FEAT
The text below is based on clicking through the FEAT GUI for one run and copying the content of the resulting fsf-file to the chunk below. In this text, we replace lines/variables that need to be altered for each subject with a placeholder. These can then easily be filled in for each run of each subject. The chunk below could be shortened by removing the lines that are irrelevant for GLM 1, e.g. preprocessing, which is not carried out here.
<- '
dsgn
# FEAT version number
set fmri(version) 6.00
# Are we in MELODIC?
set fmri(inmelodic) 0
# Analysis level
# 1 : First-level analysis
# 2 : Higher-level analysis
set fmri(level) 1
# Which stages to run
# 0 : No first-level analysis (registration and/or group stats only)
# 7 : Full first-level analysis
# 1 : Pre-processing
# 2 : Statistics
set fmri(analysis) 2
# Use relative filenames
set fmri(relative_yn) 0
# Balloon help
set fmri(help_yn) 1
# Run Featwatcher
set fmri(featwatcher_yn) 1
# Cleanup first-level standard-space images
set fmri(sscleanup_yn) 0
# Output directory
set fmri(outputdir) "@out_dir@"
# TR(s)
set fmri(tr) @tr@
# Total volumes
set fmri(npts) @n_vols@
# Delete volumes
set fmri(ndelete) 0
# Perfusion tag/control order
set fmri(tagfirst) 1
# Number of first-level analyses
set fmri(multiple) 1
# Higher-level input type
# 1 : Inputs are lower-level FEAT directories
# 2 : Inputs are cope images from FEAT directories
set fmri(inputtype) 2
# Carry out pre-stats processing?
set fmri(filtering_yn) 0
# Brain/background threshold, %
set fmri(brain_thresh) 10
# Critical z for design efficiency calculation
set fmri(critical_z) 5.3
# Noise level
set fmri(noise) 0.66
# Noise AR(1)
set fmri(noisear) 0.34
# Motion correction
# 0 : None
# 1 : MCFLIRT
set fmri(mc) 1
# Spin-history (currently obsolete)
set fmri(sh_yn) 0
# B0 fieldmap unwarping?
set fmri(regunwarp_yn) 0
# GDC Test
set fmri(gdc) ""
# EPI dwell time (ms)
set fmri(dwell) 0.7
# EPI TE (ms)
set fmri(te) 35
# % Signal loss threshold
set fmri(signallossthresh) 10
# Unwarp direction
set fmri(unwarp_dir) y-
# Slice timing correction
# 0 : None
# 1 : Regular up (0, 1, 2, 3, ...)
# 2 : Regular down
# 3 : Use slice order file
# 4 : Use slice timings file
# 5 : Interleaved (0, 2, 4 ... 1, 3, 5 ... )
set fmri(st) 0
# Slice timings file
set fmri(st_file) ""
# BET brain extraction
set fmri(bet_yn) 1
# Spatial smoothing FWHM (mm)
set fmri(smooth) 5
# Intensity normalization
set fmri(norm_yn) 0
# Perfusion subtraction
set fmri(perfsub_yn) 0
# Highpass temporal filtering
set fmri(temphp_yn) 1
# Lowpass temporal filtering
set fmri(templp_yn) 0
# MELODIC ICA data exploration
set fmri(melodic_yn) 0
# Carry out main stats?
set fmri(stats_yn) 1
# Carry out prewhitening?
set fmri(prewhiten_yn) 1
# Add motion parameters to model
# 0 : No
# 1 : Yes
set fmri(motionevs) 0
set fmri(motionevsbeta) ""
set fmri(scriptevsbeta) ""
# Robust outlier detection in FLAME?
set fmri(robust_yn) 0
# Higher-level modelling
# 3 : Fixed effects
# 0 : Mixed Effects: Simple OLS
# 2 : Mixed Effects: FLAME 1
# 1 : Mixed Effects: FLAME 1+2
set fmri(mixed_yn) 2
# Higher-level permutations
set fmri(randomisePermutations) 5000
# Number of EVs
set fmri(evs_orig) 3
set fmri(evs_real) 6
set fmri(evs_vox) 0
# Number of contrasts
set fmri(ncon_orig) 1
set fmri(ncon_real) 1
# Number of F-tests
set fmri(nftests_orig) 0
set fmri(nftests_real) 0
# Add constant column to design matrix? (obsolete)
set fmri(constcol) 0
# Carry out post-stats steps?
set fmri(poststats_yn) 0
# Pre-threshold masking?
set fmri(threshmask) ""
# Thresholding
# 0 : None
# 1 : Uncorrected
# 2 : Voxel
# 3 : Cluster
set fmri(thresh) 3
# P threshold
set fmri(prob_thresh) 0.05
# Z threshold
set fmri(z_thresh) 3.1
# Z min/max for colour rendering
# 0 : Use actual Z min/max
# 1 : Use preset Z min/max
set fmri(zdisplay) 0
# Z min in colour rendering
set fmri(zmin) 2
# Z max in colour rendering
set fmri(zmax) 8
# Colour rendering type
# 0 : Solid blobs
# 1 : Transparent blobs
set fmri(rendertype) 1
# Background image for higher-level stats overlays
# 1 : Mean highres
# 2 : First highres
# 3 : Mean functional
# 4 : First functional
# 5 : Standard space template
set fmri(bgimage) 1
# Create time series plots
set fmri(tsplot_yn) 1
# Registration to initial structural
set fmri(reginitial_highres_yn) 0
# Search space for registration to initial structural
# 0 : No search
# 90 : Normal search
# 180 : Full search
set fmri(reginitial_highres_search) 90
# Degrees of Freedom for registration to initial structural
set fmri(reginitial_highres_dof) 3
# Registration to main structural
set fmri(reghighres_yn) 0
# Search space for registration to main structural
# 0 : No search
# 90 : Normal search
# 180 : Full search
set fmri(reghighres_search) 90
# Degrees of Freedom for registration to main structural
set fmri(reghighres_dof) BBR
# Registration to standard image?
set fmri(regstandard_yn) 1
# Use alternate reference images?
set fmri(alternateReference_yn) 0
# Standard image
set fmri(regstandard) "/usr/local/fsl/data/standard/MNI152_T1_2mm_brain"
# Search space for registration to standard space
# 0 : No search
# 90 : Normal search
# 180 : Full search
set fmri(regstandard_search) 90
# Degrees of Freedom for registration to standard space
set fmri(regstandard_dof) 3
# Do nonlinear registration from structural to standard space?
set fmri(regstandard_nonlinear_yn) 0
# Control nonlinear warp field resolution
set fmri(regstandard_nonlinear_warpres) 10
# High pass filter cutoff
set fmri(paradigm_hp) 100
# Total voxels
set fmri(totalVoxels) 179159040
# Number of lower-level copes feeding into higher-level analysis
set fmri(ncopeinputs) 0
# 4D AVW data or FEAT directory (1)
set feat_files(1) "@func_fn@"
# Add confound EVs text file
set fmri(confoundevs) 1
# Confound EVs text file for analysis 1
set confoundev_files(1) "@conf_ev_fn@"
# EV 1 title
set fmri(evtitle1) "navigation"
# Basic waveform shape (EV 1)
# 0 : Square
# 1 : Sinusoid
# 2 : Custom (1 entry per volume)
# 3 : Custom (3 column format)
# 4 : Interaction
# 10 : Empty (all zeros)
set fmri(shape1) 3
# Convolution (EV 1)
# 0 : None
# 1 : Gaussian
# 2 : Gamma
# 3 : Double-Gamma HRF
# 4 : Gamma basis functions
# 5 : Sine basis functions
# 6 : FIR basis functions
# 8 : Alternate Double-Gamma
set fmri(convolve1) 2
# Convolve phase (EV 1)
set fmri(convolve_phase1) 0
# Apply temporal filtering (EV 1)
set fmri(tempfilt_yn1) 1
# Add temporal derivative (EV 1)
set fmri(deriv_yn1) 1
# Custom EV file (EV 1)
set fmri(custom1) "@navi_ev_fn@"
# Gamma sigma (EV 1)
set fmri(gammasigma1) 3
# Gamma delay (EV 1)
set fmri(gammadelay1) 6
# Orthogonalise EV 1 wrt EV 0
set fmri(ortho1.0) 0
# Orthogonalise EV 1 wrt EV 1
set fmri(ortho1.1) 0
# Orthogonalise EV 1 wrt EV 2
set fmri(ortho1.2) 0
# Orthogonalise EV 1 wrt EV 3
set fmri(ortho1.3) 0
# EV 2 title
set fmri(evtitle2) "cosine_pmod"
# Basic waveform shape (EV 2)
# 0 : Square
# 1 : Sinusoid
# 2 : Custom (1 entry per volume)
# 3 : Custom (3 column format)
# 4 : Interaction
# 10 : Empty (all zeros)
set fmri(shape2) 3
# Convolution (EV 2)
# 0 : None
# 1 : Gaussian
# 2 : Gamma
# 3 : Double-Gamma HRF
# 4 : Gamma basis functions
# 5 : Sine basis functions
# 6 : FIR basis functions
# 8 : Alternate Double-Gamma
set fmri(convolve2) 2
# Convolve phase (EV 2)
set fmri(convolve_phase2) 0
# Apply temporal filtering (EV 2)
set fmri(tempfilt_yn2) 1
# Add temporal derivative (EV 2)
set fmri(deriv_yn2) 1
# Custom EV file (EV 2)
set fmri(custom2) "@cosine_ev_fn@"
# Gamma sigma (EV 2)
set fmri(gammasigma2) 3
# Gamma delay (EV 2)
set fmri(gammadelay2) 6
# Orthogonalise EV 2 wrt EV 0
set fmri(ortho2.0) 0
# Orthogonalise EV 2 wrt EV 1
set fmri(ortho2.1) 0
# Orthogonalise EV 2 wrt EV 2
set fmri(ortho2.2) 0
# Orthogonalise EV 2 wrt EV 3
set fmri(ortho2.3) 0
# EV 3 title
set fmri(evtitle3) "feedback"
# Basic waveform shape (EV 3)
# 0 : Square
# 1 : Sinusoid
# 2 : Custom (1 entry per volume)
# 3 : Custom (3 column format)
# 4 : Interaction
# 10 : Empty (all zeros)
set fmri(shape3) 3
# Convolution (EV 3)
# 0 : None
# 1 : Gaussian
# 2 : Gamma
# 3 : Double-Gamma HRF
# 4 : Gamma basis functions
# 5 : Sine basis functions
# 6 : FIR basis functions
# 8 : Alternate Double-Gamma
set fmri(convolve3) 2
# Convolve phase (EV 3)
set fmri(convolve_phase3) 0
# Apply temporal filtering (EV 3)
set fmri(tempfilt_yn3) 1
# Add temporal derivative (EV 3)
set fmri(deriv_yn3) 1
# Custom EV file (EV 3)
set fmri(custom3) "@feedback_ev_fn@"
# Gamma sigma (EV 3)
set fmri(gammasigma3) 3
# Gamma delay (EV 3)
set fmri(gammadelay3) 6
# Orthogonalise EV 3 wrt EV 0
set fmri(ortho3.0) 0
# Orthogonalise EV 3 wrt EV 1
set fmri(ortho3.1) 0
# Orthogonalise EV 3 wrt EV 2
set fmri(ortho3.2) 0
# Orthogonalise EV 3 wrt EV 3
set fmri(ortho3.3) 0
# Contrast & F-tests mode
# real : control real EVs
# orig : control original EVs
set fmri(con_mode_old) orig
set fmri(con_mode) orig
# Display images for contrast_real 1
set fmri(conpic_real.1) 1
# Title for contrast_real 1
set fmri(conname_real.1) ""
# Real contrast_real vector 1 element 1
set fmri(con_real1.1) 0.0
# Real contrast_real vector 1 element 2
set fmri(con_real1.2) 0
# Real contrast_real vector 1 element 3
set fmri(con_real1.3) 1.0
# Real contrast_real vector 1 element 4
set fmri(con_real1.4) 0
# Real contrast_real vector 1 element 5
set fmri(con_real1.5) 0
# Real contrast_real vector 1 element 6
set fmri(con_real1.6) 0
# Display images for contrast_orig 1
set fmri(conpic_orig.1) 1
# Title for contrast_orig 1
set fmri(conname_orig.1) ""
# Real contrast_orig vector 1 element 1
set fmri(con_orig1.1) 0.0
# Real contrast_orig vector 1 element 2
set fmri(con_orig1.2) 1.0
# Real contrast_orig vector 1 element 3
set fmri(con_orig1.3) 0
# Contrast masking - use >0 instead of thresholding?
set fmri(conmask_zerothresh_yn) 0
# Do contrast masking at all?
set fmri(conmask1_1) 0
##########################################################
# Now options that do not appear in the GUI
# Alternative (to BETting) mask image
set fmri(alternative_mask) ""
# Initial structural space registration initialisation transform
set fmri(init_initial_highres) ""
# Structural space registration initialisation transform
set fmri(init_highres) ""
# Standard space registration initialisation transform
set fmri(init_standard) ""
# For full FEAT analysis: overwrite existing .feat output dir?
set fmri(overwrite_yn) 0
'
Now, we define the function that creates the fsf-file by filling the placeholders and writing to file.
<- function(fsf = dsgn, placeholder = c("@n_vols@"), replace_with = c("666"), out_fn = NA){
create_fsf ::are_equal(length(placeholder), length(replace_with))
assertthat
# loop over the placeholders and fill them with info
for (i in 1:length(placeholder)){
<- gsub(pattern=placeholder[i], replacement = replace_with[i], x=fsf)
fsf
}
# write fsf file
<-file(out_fn)
conwriteLines(fsf, con)
close(con)
}
Next, for each run and periodicity, let’s replace the placeholders with the required variables.
# these are the placeholders in the fsf
<- c("@out_dir@",
to_replace "@func_fn@",
"@tr@",
"@n_vols@",
"@navi_ev_fn@",
"@cosine_ev_fn@",
"@feedback_ev_fn@",
"@conf_ev_fn@")
for (i_sub in subjects){
for (i_run in runs){
# current functional data and find number of volumes in case they differ
<- here("data", "FunctionalScans", "4D", i_sub, i_run, paste0(i_run, "_masked.nii.gz"))
func_fn <- fslhd(func_fn, verbose =FALSE)
hdr <- hdr[grep("\\bdim4\\b", hdr)]
hdr <- regmatches(hdr, gregexpr("[[:digit:]]+", hdr))[[1]][2]
n_vols
for (i_period in periodicities){
# folder to write the fsf-files to
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", "feat_design",
fsf_dir paste0(i_period, "fold"), i_sub, i_run)
if(!dir.exists(fsf_dir)){dir.create(fsf_dir, recursive = TRUE)}
# collect the info we need
<- file.path(fsf_dir, sprintf("%s_%s_%dfold.fsf", i_sub, i_run, i_period))
fsf_fn <- here("data", "glm2", "glm2_secondhalf", "cosine_pmod",
out_dir paste0(i_period, "fold"), i_sub, i_run)
<- 1.5
tr <- here("data", "glm2", "glm2_secondhalf", "cosine_pmod",
navi_ev_fn "feat_ev", i_sub, i_run, "translation.txt")
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod",
cosine_ev_fn "feat_ev", i_sub, i_run,
paste0("translation_cosine_", i_period, "fold_half2.txt"))
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod",
feedback_ev_fn "feat_ev", i_sub, i_run, "feedback.txt")
<- here("data", "AdditionalRegressors", i_sub,
conf_ev_fn paste0("realignmentParameters_", i_run ,".txt"))
# collect the filler items. CAVE: Must be in same order as to_replace
<- c(out_dir,
fillers
func_fn,
tr,
n_vols,
navi_ev_fn,
cosine_ev_fn,
feedback_ev_fn,
conf_ev_fn)
# create the fsf file
create_fsf(fsf = dsgn, placeholder = to_replace, replace_with = fillers, out_fn = fsf_fn)
# create GLM design files to use with film_gls
system(sprintf("%s/bin/feat_model %s %s",
fsldir(), tools::file_path_sans_ext(fsf_fn), conf_ev_fn))
}
} }
5.1.3 Run GLM2
We are ready to run GLM2. We will do so using FSL FEAT and the bare film_gls
command.
5.1.3.1 FEAT
Let’s start by running the GLM using FEAT.
for (i_sub in subjects){
for (i_run in runs){
for (i_period in periodicities){
# check if the t-stats image exists, this should be one of the last files written
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", paste0(i_period, "fold"),
fn paste0(i_run, ".feat"), "stats", "tstat1.nii.gz")
i_sub, if (!file.exists(fn)){
# build the command and run
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", "feat_design",
fsf_fn paste0(i_period, "fold"), i_sub, i_run,
sprintf("%s_%s_%dfold.fsf", i_sub, i_run, i_period))
<- sprintf("%s/bin/feat %s", fsldir(), fsf_fn)
cmd system(cmd)
# remove the residuals file to clear some space
system(sprintf("rm %s",
here("data", "glm2", "glm2_secondhalf", "cosine_pmod",
paste0(i_period, "fold"), i_sub, paste0(i_run, ".feat"),
"stats", "res4d.nii.gz")))
}
}
} }
Here is the design matrix as visualized by FEAT:
Covariance matrix & design efficiency:
5.1.3.2 FILM_GLS
The command here is based on the command used by FEAT. The results should thus be identical.
NB: For this to work, make sure FSL is correctly configured. On MacOS that might mean starting RStudio from the terminal (open -na Rstudio
).
for (i_sub in subjects){
if (!dir.exists(here("data", "glm2", "glm2_secondhalf", "cosine_pmod",paste0(i_period, "fold"), i_sub)))
dir.create(here("data", "glm2", "glm2_secondhalf", "cosine_pmod",paste0(i_period, "fold"), i_sub))}
{
for (i_run in runs){
for (i_period in periodicities){
# files and folders to be used as input for film_gls
<- here("data", "FunctionalScans", "4D", i_sub, i_run, paste0(i_run, "_masked.nii.gz"))
func_fn <- here("data", "glm2", "glm2_secondhalf", "cosine_pmod",
out_dir paste0(i_period, "fold"), i_sub, i_run)
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", "feat_design",
pd_fn paste0(i_period, "fold"), i_sub, i_run,
sprintf("%s_%s_%dfold.mat", i_sub, i_run, i_period))
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", "feat_design",
con_fn paste0(i_period, "fold"), i_sub, i_run,
sprintf("%s_%s_%dfold.con", i_sub, i_run, i_period))
# check if the zstat of the t-contrast exists, this should be one of the last files written
<- file.path(out_dir, "tstat1.nii.gz")
fn if (!file.exists(fn)){
# build the command and run
<- Sys.time()
t1 <- sprintf("%s/bin/film_gls --in=%s --rn=%s --pd=%s --thr=1 --sa --ms=5 --con=%s",
cmd fsldir(), func_fn, out_dir, pd_fn, con_fn)
system(cmd, intern = TRUE)
Sys.time() - t1
# remove the residuals file to clear some space
system(sprintf("rm %s", file.path(out_dir, "res4d.nii.gz")))
}
}
} }
5.1.4 GLM 2 results
5.1.4.1 Check if results are the same
After running the analysis using FEAT and via film_gls directly, let’s check if the results are comparable.
for (i_sub in subjects){
for (i_run in runs){
for (i_period in periodicities){
# load the t-Stat images
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", paste0(i_period, "fold"),
fn1 paste0(i_run, ".feat"), "stats", "tstat1.nii.gz")
i_sub, <- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", paste0(i_period, "fold"),
fn2 "tstat1.nii.gz")
i_sub, i_run, <- readNIfTI2(fn1)
feat_nii <- readNIfTI2(fn2)
film_nii
# store in data frame
<- tibble(feat = c(feat_nii), film = c(film_nii))
stat_df
# diagnostic plot
<- ggplot(stat_df, aes(x=feat, y=film)) +
p1 geom_point() +
theme_cowplot() +
theme(aspect.ratio = 1) +
labs(title = "T-Statistics from FEAT and FILM",
subtitle = sprintf("r=%.3f", cor(stat_df$feat, stat_df$film)))
print(p1)
}
} }
5.1.4.2 Brain plot
On the whole-brain level, we can plot the t-values of the parametric modulator across voxels. The plots below are arbitrarily thresholded at t > 1.5.
for (i_sub in subjects){
for (i_run in runs){
for (i_period in periodicities){
# load mean functional and t-values
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", paste0(i_period, "fold"),
fn paste0(i_run, ".feat"), "mean_func.nii.gz")
i_sub, <- readNIfTI2(fn)
mean_func_nii <- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", paste0(i_period, "fold"),
fn "tstat1.nii.gz")
i_sub, i_run, <- readNIfTI2(fn)
t_nii < 1.5] <- 0
f_nii[f_nii
# overlay t-values on mean func
ortho2(mean_func_nii, f_nii, xyz = c(46,46,15),
ycolorbar = TRUE,
col.y = oro.nifti::hotmetal(10),
ybreaks = seq(1.5,6.5,0.5))
}
} }
5.1.4.3 EC results
Let’s load the EC results into a dataframe.
# tibble to store output
<- tibble()
stat_df
for (i_sub in subjects){
# load the right EC mask and linearize it
<- readNIfTI2(here("data", "masks", "ROI_masks",
ec_mask_nii "ROImask_entorhinalCortex_RH.nii"))
<- c(ec_mask_nii)
ec_mask_lin
# load the combined brain mask and linearize it
<- readNIfTI2(here("data", "masks", "brain_mask", i_sub,
brain_mask_nii "comb_brain_mask_tMean100.nii.gz"))
<- as.logical(c(brain_mask_nii))
brain_mask_lin
for (i_run in runs){
for (i_period in periodicities){
# load the t-stat image
#fn <- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", paste0(i_period, "fold"),
# i_sub, paste0(i_run, ".feat"), "stats", "tstat1.nii.gz")
<- here("data", "glm2", "glm2_secondhalf", "cosine_pmod", paste0(i_period, "fold"),
fn "tstat1.nii.gz")
i_sub, i_run, <- readNIfTI2(fn)
stat_nii
# store in data frame
<- tibble(subID = i_sub, periodicity = i_period, run = i_run,
temp_stat_df pmod_t = c(stat_nii)[brain_mask_lin==1],
ec = as.logical(ec_mask_lin[brain_mask_lin==1]),
vox = which(brain_mask_lin==1))
<- rbind(stat_df, temp_stat_df)
stat_df
}
} }
Plot the results for the entorhinal cortex ROI. The boxplot shows the distribution of t-values across entorhinal voxels.
# boxplot of t-values of cosine regressor
ggplot(stat_df %>% filter(ec == TRUE), aes(x=run, y=pmod_t)) +
geom_boxplot(width=0.3, fill="darkgrey") +
geom_point(position = position_nudge(x = +0.2)) +
theme_cowplot()
The scatter plot shows the relationship of t-values from the same voxels across the two runs.
# scatter plot with one dot per voxel
ggplot(stat_df %>% filter(ec == TRUE) %>%
pivot_wider(names_from = run, values_from = pmod_t),
aes(x=run01, y=run02)) +
geom_point() +
geom_smooth(formula = "y ~ x", method="lm", color="darkred") +
theme_cowplot() +
theme(aspect.ratio = 1)