# Set general settings for markdown file
knitr::opts_chunk$set(
  message = FALSE,
  warning = FALSE,
  comment = "",
  results = "hold")


# Clear environment
rm(list = ls())


# Enable/disable caching of time-consuming code chunks
knitr_cache_enabled = TRUE


# Load packages
library(dplyr)      # for data wrangling
library(knitr)      # for integrating computing and reporting in markdown
library(kableExtra) # for customizing appearance of tables
library(MASS)       # for boxcox function and contrast definition
library(lme4)       # for (G)LMMs
library(lmerTest)   # for LMM p values (Satterthwaite's method for approximating dfs for t and F tests)
library(sjPlot)     # for tab_model function to display (G)LMM results
library(emmeans)    # for pairwise comparisons


# Load functions
source("./functions/summarySEwithinO.R")  # Function provided by R-cookbook: http://www.cookbook-r.com/Graphs/Plotting_means_and_error_bars_(ggplot2)/
source("./functions/my_table_template.R") # Function to create table template


# Turn off scientific notation
options(scipen = 999)


# Prepare labels for (G)LMM tables
labels <- c(
  "(Intercept)" = "Intercept",
  "gng_response_type2-1" = "FH – SH",
  "gng_response_type3-2" = "FA – FH",
  "gng_response_type4-3" = "IR – FA",
  "word_valence2-1" = "Valence (Pos – Neg)",
  "group2-1" = "Group (OCD – HC)",
  "gng_response_type2-1:word_valence2-1" = "FH – SH x Valence",
  "gng_response_type3-2:word_valence2-1" = "FA – FH x Valence",
  "gng_response_type4-3:word_valence2-1" = "IR – FA x Valence",
  "gng_response_type2-1:group2-1" = "FH – SH x Group",
  "gng_response_type3-2:group2-1" = "FA – FH x Group",
  "gng_response_type4-3:group2-1" = "IR – FA x Group",
  "word_valence2-1:group2-1" = "Valence x Group",
  "gng_response_type2-1:word_valence2-1:group2-1" = "FH – SH x Valence x Group",
  "gng_response_type3-2:word_valence2-1:group2-1" = "FA – FH x Valence x Group",
  "gng_response_type4-3:word_valence2-1:group2-1" = "IR – FA x Valence x Group",
  "MFN_standardized" = "ERP",
  "word_valence2-1:MFN_standardized" = "Valence x ERP",
  "group2-1:MFN_standardized" = "Group x ERP",
  "word_valence2-1:group2-1:MFN_standardized" = "Valence x Group x ERP",
  "FH_FH" = "FA – FH",
  "pos_neg" = "Valence (Pos – Neg)",
  "FA_FH:pos_neg" = "FA – FH x Valence",
  "oci_centered" = "OCI-R",
  "FA_FH:oci_centered" = "FA – FH x OCI-R",
  "pos_neg:oci_centered" = "Valence x OCI-R",
  "FA_FH:pos_neg:oci_centered" = "FA – FH x Valence x OCI-R",
  "stai_centered" = "STAI",
  "FA_FH:stai_centered" = "FA – FH x STAI",
  "pos_neg:stai_centered" = "Valence x STAI",
  "FA_FH:pos_neg:stai_centered" = "FA – FH x Valence x STAI",
  "pswq_centered" = "PSWQ",
  "FA_FH:pswq_centered" = "FA – FH x PSWQ",
  "pos_neg:pswq_centered" = "Valence x PSWQ",
  "FA_FH:pos_neg:pswq_centered" = "FA – FH x Valence x PSWQ",
  "session2-1" = "Session (2 – 1)",
  "gng_response_type2-1:session2-1" = "FH – SH x Session",
  "gng_response_type3-2:session2-1" = "FA – FH x Session",
  "gng_response_type4-3:session2-1" = "IR – FA x Session",
  "word_valence2-1:session2-1" = "Valence x Session",
  "group2-1:session2-1" = "Group x Session",
  "gng_response_type2-1:word_valence2-1:session2-1" = "FH – SH x Valence x Session",
  "gng_response_type3-2:word_valence2-1:session2-1" = "FA – FH x Valence x Session",
  "gng_response_type4-3:word_valence2-1:session2-1" = "IR – FA x Valence x Session",
  "gng_response_type2-1:group2-1:session2-1" = "FH – SH x Group x Session",
  "gng_response_type3-2:group2-1:session2-1" = "FA – FH x Group x Session",
  "gng_response_type4-3:group2-1:session2-1" = "IR – FA x Group x Session",
  "word_valence2-1:group2-1:session2-1" = "Valence x Group x Session",
  "gng_response_type2-1:word_valence2-1:group2-1:session2-1" = "FH – SH x Valence x Group x Session",
  "gng_response_type3-2:word_valence2-1:group2-1:session2-1" = "FA – FH x Valence x Group x Session",
  "gng_response_type4-3:word_valence2-1:group2-1:session2-1" = "IR – FA x Valence x Group x Session",
  "MFN_standardized:session2-1" = "ERP x Session",
  "word_valence2-1:MFN_standardized:session2-1" = "Valence x ERP x Session",
  "group2-1:MFN_standardized:session2-1" = "Group x ERP x Session",
  "word_valence2-1:group2-1:MFN_standardized:session2-1" = "Valence x Group x ERP x Session")


Data Cleaning


# Load data
load(file = "./data/Single_Trial_Data.rda")
load(file = "./data/Trait_Data.rda")


# Create numeric accuracy variable (1 = correct, 0 = incorrect)
single_trial_data <- single_trial_data %>%
  dplyr::mutate(word_accuracy_numeric = ifelse(word_accuracy == "correct", 1, 0))


# For RT analysis, exclude missing responses, responses with wrong key, RT outliers,
# incorrect word categorizations, and trials with ERP artifacts
single_trial_data_rt <- single_trial_data %>%
  dplyr::filter(
    gng_response_type != "miss" &
    gng_response_type != "wrong_key" &
    word_accuracy     == "correct" &
    (is.na(gng_rt_invalid)  | gng_rt_invalid  == FALSE) &
    (is.na(word_rt_outlier) | word_rt_outlier == FALSE) &
    stimulation == "sham" &
    MFN_artifact == FALSE)


# Calculate standardized MFN (based on trials with correct word categorization, as only
# these  will be used in the LMMs with ERP as predictor)
single_trial_data_rt <- single_trial_data_rt %>%
  dplyr::group_by(participant_id) %>%
  dplyr::mutate(MFN_standardized = scale(MFN_0_100_FCz, center = TRUE, scale = TRUE)) %>%
  dplyr::ungroup()


# Center trait variables within groups (will be used as predictors)
traits <- traits %>%
  dplyr::rename(oci = "OCI-R", pswq = "PSWQ", stai = "STAI trait") %>%  
  dplyr::mutate(group = as.factor(ifelse(substr(participant_id, 1, 1) == "C", "HC", "OCD"))) %>%
  dplyr::group_by(group) %>%
  dplyr::mutate(
    oci_centered  = scale(oci,  center = TRUE, scale = FALSE),
    pswq_centered = scale(pswq, center = TRUE, scale = FALSE),
    stai_centered = scale(stai, center = TRUE, scale = FALSE)) %>%
  dplyr::ungroup()


# Add trait variables to single trial data
single_trial_data_rt <- single_trial_data_rt %>%
  dplyr::left_join(., traits %>% dplyr::select(participant_id, oci_centered, pswq_centered,
    stai_centered), by = "participant_id")


# For accuracy analysis, exclude missing responses, responses with wrong key, RT outliers,
# and trials with ERP artifacts
single_trial_data_acc <- single_trial_data %>%
  dplyr::filter(
    gng_response_type != "miss" &
    gng_response_type != "wrong_key" &
    word_accuracy     != "miss" &
    word_accuracy     != "wrong_key" &
    (is.na(gng_rt_invalid)  | gng_rt_invalid  == FALSE) &
    (is.na(word_rt_outlier) | word_rt_outlier == FALSE) &
    stimulation == "sham" &
    MFN_artifact == FALSE)


# Calculate % of excluded trials (missing responses, responses with wrong key, RT outliers)
excluded_trials_per_participant <- single_trial_data %>%
  dplyr::filter(stimulation == "sham") %>%
  dplyr::group_by(participant_id) %>%
  dplyr::summarize(excluded = sum(
    gng_response_type == "miss" |
    gng_response_type == "wrong_key" |
    word_accuracy     == "miss" |
    word_accuracy     == "wrong_key" |
    (!is.na(gng_rt_invalid)  & gng_rt_invalid  != FALSE) |
    (!is.na(word_rt_outlier) & word_rt_outlier != FALSE)) / length(participant_id) * 100) %>%
  dplyr::ungroup() %>%
  dplyr::summarize(across(-c(participant_id), list(mean, sd)))


# Make categorical variables factors
single_trial_data_rt$gng_response_type  <- factor(single_trial_data_rt$gng_response_type,
  levels = c("SH", "FH", "FA", "IR"))
single_trial_data_rt$word_valence       <- factor(single_trial_data_rt$word_valence)
single_trial_data_rt$participant_id     <- factor(single_trial_data_rt$participant_id)
single_trial_data_rt$group              <- factor(single_trial_data_rt$group)
single_trial_data_rt$session            <- factor(single_trial_data_rt$session)
single_trial_data_acc$gng_response_type <- factor(single_trial_data_acc$gng_response_type,
  levels = c("SH", "FH", "FA", "IR"))
single_trial_data_acc$word_valence      <- factor(single_trial_data_acc$word_valence)
single_trial_data_acc$participant_id    <- factor(single_trial_data_acc$participant_id)
single_trial_data_acc$group             <- factor(single_trial_data_acc$group)
single_trial_data_acc$session           <- factor(single_trial_data_acc$session)


# Aggregate data per participant for plots
data_aggregated_rt <- single_trial_data_rt %>%
  dplyr::group_by(participant_id, group, gng_response_type, word_valence) %>%
  dplyr::summarize(word_rt = mean(word_rt, na.rm = TRUE)) %>%
  dplyr::ungroup()

data_aggregated_acc <- single_trial_data_acc %>%
  dplyr::group_by(participant_id, group, gng_response_type, word_valence) %>%
  dplyr::summarize(accuracy = mean(word_accuracy_numeric, na.rm = TRUE) * 100) %>%
  dplyr::ungroup()


# Save aggregated data for plots
save(data_aggregated_rt,  file = "./saved_objects_for_plots/data_aggregated_rt.Rda")
save(data_aggregated_acc, file = "./saved_objects_for_plots/data_aggregated_acc.Rda")

In the analysis of RT and accuracy in the word categorization task, trials were discarded (M = 8.72%, SD = 3.62) if RT in the go/no-go task was below 100 ms or above 800 ms, word categorization RT deviated more than three median absolute deviations from the individual condition-specific median, response was missing, or the response was made with a key not assigned to the current task. We further discarded trials containing artifacts in the EEG, i.e., a voltage difference exceeding 200 μV within an epoch or 50 μV between sample points. In the analysis of word categorization RT, we excluded incorrect categorizations.


Descriptive Statistics


This table corresponds to Table S2 in the supplemental material.

# Calculate descriptive statistics for RT
descriptive_statistics_rt <- summarySEwithinO(
  data          = single_trial_data_rt,
  measurevar    = "word_rt",
  withinvars    = c("gng_response_type", "word_valence"),
  betweenvars   = "group",
  idvar         = "participant_id",
  conf.interval = .95) %>%
  # Rename variable and select variables for subsequent merging
  dplyr::rename(mean = word_rt) %>%
  dplyr::select("group", "gng_response_type", "word_valence", "mean", "ci")


# Calculate descriptive statistics for accuracy
descriptive_statistics_acc <- summarySEwithinO(
  data          = single_trial_data_acc,
  measurevar    = "word_accuracy_numeric",
  withinvars    = c("gng_response_type", "word_valence"),
  betweenvars   = "group",
  idvar         = "participant_id",
  conf.interval = .95) %>%
  # Multiply numeric values by 100 to obtain values in percent
  dplyr::mutate_if(is.numeric, list(~ . * 100)) %>%
  # Rename variable and select variables for subsequent merging
  dplyr::rename(mean = word_accuracy_numeric) %>%
  dplyr::select("group", "gng_response_type", "word_valence", "mean", "ci")


# Combine measures
descriptive_statistics <- as.data.frame(rbind(descriptive_statistics_rt, descriptive_statistics_acc))


# Create column with format "M [CI]" for RT (round to 0 decimals)
descriptive_statistics$M_CI <- paste0(round(descriptive_statistics$mean, 0), " [",
  round(descriptive_statistics$mean - descriptive_statistics$ci, 0), ", ",
  round(descriptive_statistics$mean + descriptive_statistics$ci, 0), "]")


# Create column with format "M [CI]" for accuracy (round to 2 decimals)
descriptive_statistics[c(17:32), ]$M_CI <- paste0(
  format(round(descriptive_statistics[c(17:32), ]$mean, 2), nsmall = 2), " [",
  format(round(descriptive_statistics[c(17:32), ]$mean -
               descriptive_statistics[c(17:32), ]$ci,   2), nsmall = 2), ", ",
  format(round(descriptive_statistics[c(17:32), ]$mean +
               descriptive_statistics[c(17:32), ]$ci,   2), nsmall = 2), "]")


# Split and re-merge table to display both groups next to each other
descriptive_statistics <- split(descriptive_statistics, descriptive_statistics$group)
descriptive_statistics_display <- cbind(
  paste(descriptive_statistics$OCD$word_valence, "word after", descriptive_statistics$OCD$gng_response_type),
  descriptive_statistics$OCD[, "M_CI"], descriptive_statistics$HC[, "M_CI"])


# Display table (and rearrange rows)
my_table_template(descriptive_statistics_display[c(8, 7, 4, 3, 2, 1, 6, 5, 16, 15, 12, 11, 10, 9, 14, 13), ],
  caption = "Descriptive Statistics for Behavioral Performance in the Word Categorization Task",
  col_names = c("Measure", rep("M [95% CI]", 2)),
  header_above_config = c("", "Patients with OCD", "Healthy control participants")) %>%
  add_footnote("\n Note. CIs are adjusted for within-participant comparisons using the method described 
  by Morey (2008).", notation = "none") %>%
  pack_rows("Response time (ms)", 1, 8) %>%
  pack_rows("Accuracy (%)", 9, 16)
Descriptive Statistics for Behavioral Performance in the Word Categorization Task
Patients with OCD
Healthy control participants
Measure M [95% CI] M [95% CI]
Response time (ms)
pos word after SH 595 [590, 600] 573 [569, 578]
neg word after SH 621 [616, 625] 590 [585, 594]
pos word after FH 596 [589, 602] 584 [577, 592]
neg word after FH 635 [628, 642] 616 [608, 624]
pos word after FA 737 [713, 760] 763 [734, 792]
neg word after FA 675 [658, 692] 628 [613, 643]
pos word after IR 696 [688, 704] 702 [693, 711]
neg word after IR 689 [681, 698] 665 [657, 672]
Accuracy (%)
pos word after SH 94.86 [94.02, 95.70] 91.66 [90.63, 92.69]
neg word after SH 94.03 [93.12, 94.93] 90.64 [89.60, 91.68]
pos word after FH 94.18 [92.89, 95.47] 91.95 [90.40, 93.49]
neg word after FH 92.35 [90.86, 93.83] 91.23 [89.69, 92.77]
pos word after FA 79.21 [75.88, 82.55] 69.01 [65.23, 72.80]
neg word after FA 94.57 [92.59, 96.54] 96.71 [94.99, 98.44]
pos word after IR 96.43 [95.45, 97.42] 92.06 [90.68, 93.44]
neg word after IR 96.14 [95.10, 97.17] 96.52 [95.53, 97.51]

Note. CIs are adjusted for within-participant comparisons using the method described
by Morey (2008).



Analysis of Response Time


We performed a linear mixed model (LMM) on single-trial data. Word categorization RT was modeled using type of the preceding go/no-go response (slow hit, fast hit, false alarm, inhibited response), word valence, and group as fixed effects.

We used sliding difference contrasts for categorical fixed effects and started with the maximal random-effects structure for each model. In case of non-convergence, random effects were specified as uncorrelated. Using principal component analysis, we identified random effects explaining zero variance and removed these, as recommended by Bates, Kliegl, et al. (2015).

Main Model

Determine Data Transformation

RT was log-transformed to normalize residuals. This was determined using the Box–Cox procedure (Box & Cox, 1964).

# Arrange plots
par(mfrow = c(1, 5))


# Determine transformation by estimating optimal lambda using Box–Cox procedure
bc_rt <- boxcox(word_rt ~ 1, data = single_trial_data_rt)
optlambda_rt <- bc_rt$x[which.max(bc_rt$y)]


# Density plot for raw RT
plot(density(single_trial_data_rt$word_rt), main = "Raw RT: Density Plot")


# Q-q plot for raw RT
qqnorm(single_trial_data_rt$word_rt, main = "Raw RT: Q-Q Plot", pch = 1)


# Density plot for log RT
plot(density(single_trial_data_rt$word_rt_log), main = "Log RT: Density Plot")


# Q-q plot for log RT
qqnorm(single_trial_data_rt$word_rt_log, main = "Log RT: Q-Q Plot", pch = 1)

# Reset plot layout
par(mfrow = c(1, 1))

The optimal lambda is -0.3, suggesting that log transformation is appropriate.


LMM

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_rt$gng_response_type) <- contr.sdif(4)
contrasts(single_trial_data_rt$word_valence)      <- contr.sdif(2)
contrasts(single_trial_data_rt$group)             <- contr.sdif(2)


# Add contrasts as numerical covariates via model matrix*
model_matrix <- model.matrix(~ gng_response_type * word_valence * group, single_trial_data_rt)


# Attach the model matrix (16 columns) to the dataframe
single_trial_data_rt[, (ncol(single_trial_data_rt) + 1):(ncol(single_trial_data_rt) + 16)] <- model_matrix


# Assign descriptive names to the contrasts
names(single_trial_data_rt)[(ncol(single_trial_data_rt) - 15):ncol(single_trial_data_rt)] <- c(
  "Grand Mean", "FH_SH", "FA_FH", "IR_FA", "pos_neg", "OCD_HC", "FH_SH:pos_neg", "FA_FH:pos_neg",
  "IR_FA:pos_neg", "FH_SH:OCD_HC", "FA_FH:OCD_HC", "IR_FA:OCD_HC", "pos_neg:OCD_HC",
  "FH_SH:pos_neg:OCD_HC", "FA_FH:pos_neg:OCD_HC", "IR_FA:pos_neg:OCD_HC")


# *Note: For the random effects, we needed to enter the separate random effect terms in the models to enable
# double-bar notation (||). This allows fitting a model that sets correlations of the random terms to zero.
# Run model with maximal random-effects structure
LMM_rt_max <- lmer(word_rt_log ~ gng_response_type * word_valence * group +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg | participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FH_SH:OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC | word),
  data = single_trial_data_rt,
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output
summary(LMM_rt_max)    # Model does not converge
isSingular(LMM_rt_max) # Singular fit


# Run zero-correlation parameter model by using || syntax to reduce model complexity
LMM_rt_red1 <- lmer(word_rt_log ~ gng_response_type * word_valence * group +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FH_SH:OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_rt,
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
summary(LMM_rt_red1)        # Model does converge
isSingular(LMM_rt_red1)     # Singular fit
summary(rePCA(LMM_rt_red1)) # For words there are 3 terms that do not explain variance (< 0.5%)
print(VarCorr(LMM_rt_red1), comp = "Variance") # It is IR_FA, FH_SH:OCD_HC, and FA_FH:OCD_HC for words

This table corresponds to Table 3 in the manuscript.

# Run final model without random terms explaining zero variance (use log(word_rt) not word_rt_log for plotting)
LMM_rt_final <- lmer(log(word_rt) ~ gng_response_type * word_valence * group + 
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_rt,
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_final)        # Model does converge
# isSingular(LMM_rt_final)     # No singular fit
# summary(rePCA(LMM_rt_final)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_final,
  dv.labels = "RT", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite",
  wrap.labels = 80, title = "LMM Results for Word Categorization RT")


# Display random effects
print("Random effects:")
print(VarCorr(LMM_rt_final), digits = 1, comp = "Std.Dev.")


# Save model output for plot
saveRDS(LMM_rt_final, file = "./saved_objects_for_plots/LMM_rt_final.rds")
LMM Results for Word Categorization RT
  RT
Fixed effects b 95 % CI t p
Intercept 6.44 6.39 – 6.49 272.84 <0.001
FH – SH -0.02 -0.03 – -0.01 -3.05 0.003
FA – FH 0.17 0.13 – 0.21 8.42 <0.001
IR – FA -0.02 -0.06 – 0.03 -0.75 0.457
Valence (Pos – Neg) 0.01 -0.02 – 0.04 0.83 0.410
Group (OCD – HC) 0.04 -0.05 – 0.13 0.85 0.402
FH – SH x Valence -0.02 -0.04 – -0.00 -2.13 0.038
FA – FH x Valence 0.18 0.13 – 0.22 8.15 <0.001
IR – FA x Valence -0.10 -0.14 – -0.06 -5.29 <0.001
FH – SH x Group 0.01 -0.01 – 0.03 1.11 0.272
FA – FH x Group -0.02 -0.09 – 0.06 -0.40 0.688
IR – FA x Group -0.03 -0.12 – 0.06 -0.57 0.569
Valence x Group -0.05 -0.09 – -0.00 -2.17 0.034
FH – SH x Valence x Group 0.00 -0.03 – 0.04 0.05 0.963
FA – FH x Valence x Group -0.08 -0.17 – 0.01 -1.87 0.067
IR – FA x Valence x Group 0.06 -0.02 – 0.13 1.52 0.134
Observations 24189
[1] "Random effects:"
 Groups           Name          Std.Dev.
 word             OCD_HC:IR_FA  0.010   
 word.1           OCD_HC        0.005   
 word.2           FA_FH         0.009   
 word.3           FH_SH         0.009   
 word.4           (Intercept)   0.030   
 participant_id   IR_FA:pos_neg 0.116   
 participant_id.1 FA_FH:pos_neg 0.140   
 participant_id.2 FH_SH:pos_neg 0.046   
 participant_id.3 pos_neg       0.078   
 participant_id.4 IR_FA         0.163   
 participant_id.5 FA_FH         0.141   
 participant_id.6 FH_SH         0.033   
 participant_id.7 (Intercept)   0.174   
 Residual                       0.176   


RT analysis yielded a main effect of response type. Participants across groups categorized words slower after false alarms than after fast hits, indicating post-error slowing. Moreover, word categorization after fast hits was faster than after slow hits. The analysis further revealed significant two-way interactions between group and word valence and between each response type contrast and word valence. Crucially, the significant two-way interactions were qualified by a statistical trend for a three-way interaction between the response type contrast false alarms compared to fast hits, word valence, and group.


Pairwise Comparisons

Significant and hypothesis-relevant interactions were followed up with post hoc comparisons using false discovery rate adjustment.

These tables correspond to Table 4 in the manuscript.

# Set emmeans options
emm_options(lmer.df = "Satterthwaite", lmerTest.limit = Inf)


#### 1) Get contrasts for effect of word valence within each response type
emms1 <- emmeans(LMM_rt_final, ~ word_valence | gng_response_type)
contrast_1 <- contrast(emms1, interaction = "revpairwise")


# Get CIs of estimates
contrast_1_ci <- confint(contrast_1, adjust = "none", level = 0.95)[c("lower.CL", "upper.CL")]


# Merge contrast and CI in one table
follow_up_1 <- cbind(contrast_1, contrast_1_ci)


# Add adjusted p values
follow_up_1$p.value_adj_fdr <- p.adjust(follow_up_1$p.value, method = "fdr")


# Display table
my_table_template(follow_up_1[, c(2, 1, 3, 8, 9, 6, 7, 10)],
  caption = "Follow-up effect of valence nested within response type",
  digits = c(0, 0, 2, 2, 2, 2, 3, 3))


#### 2) Get contrasts for resolving interaction group * valence nested within response type
emms2 <- emmeans(LMM_rt_final, ~ word_valence * group | gng_response_type)
contrast_2 <- contrast(emms2, interaction = "revpairwise")


# Get CIs of estimates
contrast_2_ci <- confint(contrast_2, adjust = "none", level = 0.95)[c("lower.CL", "upper.CL")]


# Put contrast and CI in one table (only for FH and FA)
follow_up_2 <- cbind(contrast_2[c(2, 3), ], contrast_2_ci[c(2, 3), ])


# Get adjusted p values
follow_up_2$p.value_adj_fdr <- p.adjust(follow_up_2$p.value, method = "fdr")


# Display table
my_table_template(follow_up_2[, c(3, 1, 2, 4, 9, 10, 7, 8, 11)],
  caption = "Follow-up interaction group * valence nested within response type",
  digits = c(0, 0, 0, 2, 2, 2, 2, 3, 3))


#### 3) Get contrasts for resolving effect of valence nested within group and response type
emms3 <- emmeans(LMM_rt_final, ~ word_valence | gng_response_type * group)
contrast_3 <- contrast(emms3, interaction = "revpairwise")


# Get CIs of estimates
contrast_3_ci <- confint(contrast_3, adjust = "none", level = 0.95)[c("lower.CL", "upper.CL")]


# Put contrast and CI in one table (only for FH and FA)
follow_up_3 <- cbind(contrast_3[c(2, 3, 6, 7), ], contrast_3_ci[c(2, 3, 6, 7), ])


# Get adjusted p values 
follow_up_3$p.value_adj_fdr <- p.adjust(follow_up_3$p.value, method = "fdr")


# Display table
my_table_template(follow_up_3[c(1, 3, 2, 4), c(2, 3, 1, 4, 9, 10, 7, 8, 11)],
  caption = "Follow-up effect of valence nested within group and response type",
  digits = c(0, 0, 0, 2, 2, 2, 2, 3, 3))
Follow-up effect of valence nested within response type
gng_response_type word_valence_revpairwise estimate lower.CL upper.CL t.ratio p.value p.value_adj_fdr
SH pos - neg -0.04 -0.07 0.00 -2.16 0.033 0.044
FH pos - neg -0.06 -0.09 -0.02 -3.30 0.001 0.002
FA pos - neg 0.12 0.08 0.16 6.29 0.000 0.000
IR pos - neg 0.02 -0.02 0.06 0.99 0.325 0.325
Follow-up interaction group * valence nested within response type
gng_response_type word_valence_revpairwise group_revpairwise estimate lower.CL upper.CL t.ratio p.value p.value_adj_fdr
FH pos - neg OCD - HC -0.02 -0.08 0.04 -0.67 0.502 0.502
FA pos - neg OCD - HC -0.10 -0.17 -0.03 -2.94 0.004 0.008
Follow-up effect of valence nested within group and response type
gng_response_type group word_valence_revpairwise estimate lower.CL upper.CL t.ratio p.value p.value_adj_fdr
FH HC pos - neg -0.05 -0.09 0.00 -2.00 0.047 0.047
FH OCD pos - neg -0.07 -0.11 -0.02 -2.92 0.004 0.008
FA HC pos - neg 0.17 0.12 0.22 6.61 0.000 0.000
FA OCD pos - neg 0.07 0.02 0.12 2.67 0.008 0.011


Post hoc comparisons indicated that after false alarms, participants across groups categorized negative words faster than positive words. After fast and slow hits, participants categorized positive words faster than negative words.

After false alarms, both groups categorized negative words faster than positive words, whereas after fast hits, they categorized positive words faster than negative words. Comparisons conducted to investigate hypothesized group differences yielded that this response facilitation to negative relative to positive words after false alarms was significantly smaller in patients with OCD than in the control group, indicating a reduced priming effect.


ERPs as Predictors

For testing brain–behavior relations, the single-trial within-participant z-standardized ERP was added as predictor. ERN amplitude and CRN amplitude were entered as z-standardized continuous predictors in the models including only false alarm and fast hit trials, respectively.

ERN

# Run model with maximal random-effects structure
LMM_rt_ern_max <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg + MFN_standardized + pos_neg:MFN_standardized | participant_id) +
  (1 + OCD_HC + MFN_standardized + OCD_HC:MFN_standardized | word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FA", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output
summary(LMM_rt_ern_max)    # Model does converge
isSingular(LMM_rt_ern_max) # Singular fit


# Run zero-correlation parameter model by using || syntax to reduce model complexity
LMM_rt_ern_red1 <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg + MFN_standardized + pos_neg:MFN_standardized || participant_id) +
  (1 + OCD_HC + MFN_standardized + OCD_HC:MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FA", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
summary(LMM_rt_ern_red1)        # Model does converge
isSingular(LMM_rt_ern_red1)     # Singular fit
summary(rePCA(LMM_rt_ern_red1)) # For words there is 1 term, for participants 2 terms that do not explain variance (< 0.5%)
print(VarCorr(LMM_rt_ern_red1), comp = "Variance") # It is OCD_HC:MFN for words; MFN and pos_neg:MFN for participants

This table corresponds to Table 6 in the manuscript.

# Run final model without random terms explaining zero variance
LMM_rt_ern_final <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg || participant_id) +
  (1 + OCD_HC + MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FA", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_ern_final)        # Model does converge
# isSingular(LMM_rt_ern_final)     # No singular fit
# summary(rePCA(LMM_rt_ern_final)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_ern_final,
  dv.labels = "RT", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", 
  wrap.labels = 80, title = "LMM Results for Word Categorization RT With ERN as Predictor")


# Display random effects
print("Random effects:")
print(VarCorr(LMM_rt_ern_final), digits = 1, comp = "Std.Dev.")
LMM Results for Word Categorization RT With ERN as Predictor
  RT
Fixed effects b 95 % CI t p
Intercept 6.52 6.46 – 6.59 197.70 <0.001
Valence (Pos – Neg) 0.12 0.07 – 0.18 4.50 <0.001
Group (OCD – HC) 0.05 -0.09 – 0.18 0.68 0.497
ERP 0.00 -0.01 – 0.01 0.46 0.646
Valence x Group -0.09 -0.20 – 0.01 -1.75 0.086
Valence x ERP 0.00 -0.02 – 0.02 0.25 0.807
Group x ERP 0.01 -0.01 – 0.03 1.17 0.243
Valence x Group x ERP 0.01 -0.03 – 0.05 0.57 0.571
Observations 1992
[1] "Random effects:"
 Groups           Name             Std.Dev.
 word             MFN_standardized 0.01    
 word.1           OCD_HC           0.02    
 word.2           (Intercept)      0.02    
 participant_id   pos_neg          0.17    
 participant_id.1 (Intercept)      0.24    
 Residual                          0.23    


Analysis of word categorization RT in false alarm trials with ERN as predictor yielded a main effect of word valence, again revealing the priming effect after false alarms, with negative words being categorized faster than positive words. Contrary to our prediction, we found no significant interaction between ERN and word valence. Hence, there was no evidence for a relation between error evaluation (indexed by the RT priming effect) and ERN. Further, there was no difference in this relation between groups.


CRN

# Run model with maximal random-effects structure
LMM_rt_crn_max <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg + MFN_standardized + pos_neg:MFN_standardized | participant_id) +
  (1 + OCD_HC + MFN_standardized + OCD_HC:MFN_standardized | word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FH", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output
summary(LMM_rt_crn_max)    # Model does converge
isSingular(LMM_rt_crn_max) # Singular fit


# Run zero-correlation parameter model by using || syntax to reduce model complexity
LMM_rt_crn_red1 <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg + MFN_standardized + pos_neg:MFN_standardized || participant_id) +
  (1 + OCD_HC + MFN_standardized + OCD_HC:MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FH", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
summary(LMM_rt_crn_red1)        # Model does converge
isSingular(LMM_rt_crn_red1)     # Singular fit
summary(rePCA(LMM_rt_crn_red1)) # For words and participants there are each 2 terms that do not explain variance (< 0.5%)
print(VarCorr(LMM_rt_crn_red1), comp = "Variance") # It is OCD_HC and OCD_HC:MFN for words and MFN and pos_neg:MFN for participants

This table corresponds to Table 6 in the manuscript.

# Run final model without random terms explaining zero variance
LMM_rt_crn_final <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg || participant_id) +
  (1 + MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FH", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_crn_final)        # Model does converge
# isSingular(LMM_rt_crn_final)     # No singular fit
# summary(rePCA(LMM_rt_crn_final)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_crn_final,
  dv.labels = "RT", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", 
  wrap.labels = 80, title = "LMM Results for Word Categorization RT With CRN as Predictor")


# Display random effects
print("Random effects:")
print(VarCorr(LMM_rt_crn_final), digits = 1, comp = "Std.Dev.")
LMM Results for Word Categorization RT With CRN as Predictor
  RT
Fixed effects b 95 % CI t p
Intercept 6.36 6.31 – 6.40 269.74 <0.001
Valence (Pos – Neg) -0.06 -0.08 – -0.03 -4.21 <0.001
Group (OCD – HC) 0.06 -0.04 – 0.15 1.25 0.216
ERP -0.01 -0.01 – -0.00 -2.04 0.046
Valence x Group -0.03 -0.07 – 0.02 -1.14 0.257
Valence x ERP -0.00 -0.02 – 0.01 -0.84 0.406
Group x ERP -0.01 -0.02 – 0.00 -1.58 0.115
Valence x Group x ERP 0.01 -0.01 – 0.03 1.10 0.269
Observations 5025
[1] "Random effects:"
 Groups           Name             Std.Dev.
 word             MFN_standardized 0.01    
 word.1           (Intercept)      0.03    
 participant_id   pos_neg          0.08    
 participant_id.1 (Intercept)      0.17    
 Residual                          0.17    


Analysis of word categorization RT in fast hit trials including CRN as predictor indicated a main effect of word valence. This again demonstrated the priming effect after fast hits, with positive words being categorized faster than negative words. Moreover, a main effect for CRN was found, with faster RT being related to smaller CRN amplitude. There were no significant interactions between CRN, word valence, and group. Thus, we found no evidence for a relation between action evaluation and CRN and no group difference in this relation.


Traits as Predictors

To explore effects of trait anxiety, worry, and OCD symptom severity on action evaluation, we included these OCD-related characteristics as predictors. Models were calculated separately for each group since otherwise effects of trait measures are confounded by group differences. Trait measures were group-mean centered.

This table corresponds to Table S5 in the supplemental material.

# OCI in HC: Run reduced final model
LMM_rt_oci_hc <- lmer(log(word_rt) ~ FA_FH * pos_neg * oci_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "HC", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_oci_hc)        # Model does converge
# isSingular(LMM_rt_oci_hc)     # No singular fit
# summary(rePCA(LMM_rt_oci_hc)) # All terms explain variance


# OCI in OCD: Run reduced final model
LMM_rt_oci_ocd <- lmer(log(word_rt) ~ FA_FH * pos_neg * oci_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "OCD", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_oci_ocd)        # Model does converge
# isSingular(LMM_rt_oci_ocd)     # No singular fit
# summary(rePCA(LMM_rt_oci_ocd)) # All terms explain variance


# STAI in HC: Run reduced final model
LMM_rt_stai_hc <- lmer(log(word_rt) ~ FA_FH * pos_neg * stai_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "HC", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_stai_hc)        # Model does converge
# isSingular(LMM_rt_stai_hc)     # No singular fit
# summary(rePCA(LMM_rt_stai_hc)) # All terms explain variance


# STAI in OCD: Run reduced final model
LMM_rt_stai_ocd <- lmer(log(word_rt) ~ FA_FH * pos_neg * stai_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "OCD", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_stai_ocd)        # Model does converge
# isSingular(LMM_rt_stai_ocd)     # No singular fit
# summary(rePCA(LMM_rt_stai_ocd)) # All terms explain variance


# PSWQ in HC: Run reduced final model
LMM_rt_pswq_hc <- lmer(log(word_rt) ~ FA_FH * pos_neg * pswq_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "HC", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_pswq_hc)        # Model does converge
# isSingular(LMM_rt_pswq_hc)     # No singular fit
# summary(rePCA(LMM_rt_pswq_hc)) # All terms explain variance


# PSWQ in OCD: Run reduced final model
LMM_rt_pswq_ocd <- lmer(log(word_rt) ~ FA_FH * pos_neg * pswq_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "OCD", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_pswq_ocd)        # Model does converge
# isSingular(LMM_rt_pswq_ocd)     # No singular fit
# summary(rePCA(LMM_rt_pswq_ocd)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_oci_ocd, LMM_rt_oci_hc,
  dv.labels = c("RT: OCI-R as predictor in patients with OCD", "RT: OCI-R as predictor in control participants"),
  pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", wrap.labels = 80,
  title = "LMM Results for Word Categorization RT With OCD Symptom Severity (OCI-R) as Predictor")


tab_model(LMM_rt_stai_ocd, LMM_rt_stai_hc,
  dv.labels = c("RT: STAI as predictor in patients with OCD", "RT: STAI as predictor in control participants"),
  pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", wrap.labels = 80,
  title = "LMM Results for Word Categorization RT With Trait Anxiety (STAI) as Predictor")


tab_model(LMM_rt_pswq_ocd, LMM_rt_pswq_hc,
  dv.labels = c("RT: PSWQ as predictor in patients with OCD", "RT: PSWQ as predictor in control participants"),
  pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", wrap.labels = 80,
  title = "LMM Results for Word Categorization RT With Trait Worry (PSWQ) as Predictor")


# Save model output for plot
saveRDS(LMM_rt_oci_ocd,  file = "./saved_objects_for_plots/LMM_rt_oci_ocd.rds")
saveRDS(LMM_rt_stai_ocd, file = "./saved_objects_for_plots/LMM_rt_stai_ocd.rds")


# Note: For all models with traits as predictors, the reduced models were run
# (random effects were specified as uncorrelated due to non-convergence of the
# models with the maximal random-effects structure; random effects explaining
# zero variance [all random slopes of traits for words] were removed)
LMM Results for Word Categorization RT With OCD Symptom Severity (OCI-R) as Predictor
  RT: OCI-R as predictor in patients with OCD RT: OCI-R as predictor in control participants
Fixed effects b 95 % CI t p b 95 % CI t p
Intercept 6.46 6.39 – 6.53 193.95 <0.001 6.41 6.33 – 6.50 153.07 <0.001
FA_FH 0.16 0.10 – 0.21 6.06 <0.001 0.18 0.11 – 0.24 5.21 <0.001
Valence (Pos – Neg) 0.00 -0.04 – 0.04 0.01 0.996 0.06 0.01 – 0.11 2.50 0.018
OCI-R -0.00 -0.01 – 0.01 -0.49 0.628 -0.00 -0.02 – 0.01 -0.24 0.814
FA – FH x Valence 0.14 0.09 – 0.19 5.62 <0.001 0.22 0.13 – 0.30 5.19 <0.001
FA – FH x OCI-R -0.00 -0.01 – 0.00 -1.30 0.205 0.00 -0.01 – 0.02 0.52 0.607
Valence x OCI-R -0.00 -0.01 – 0.00 -0.87 0.391 -0.00 -0.01 – 0.01 -0.32 0.755
FA – FH x Valence x OCI-R -0.01 -0.01 – -0.00 -2.44 0.021 -0.00 -0.02 – 0.01 -0.08 0.934
Observations 3660 3357
LMM Results for Word Categorization RT With Trait Anxiety (STAI) as Predictor
  RT: STAI as predictor in patients with OCD RT: STAI as predictor in control participants
Fixed effects b 95 % CI t p b 95 % CI t p
Intercept 6.46 6.39 – 6.53 194.97 <0.001 6.41 6.33 – 6.49 163.54 <0.001
FA_FH 0.16 0.10 – 0.21 6.00 <0.001 0.18 0.11 – 0.24 5.27 <0.001
Valence (Pos – Neg) 0.00 -0.04 – 0.04 0.05 0.958 0.06 0.01 – 0.11 2.51 0.018
STAI 0.00 -0.00 – 0.01 0.68 0.505 0.01 -0.00 – 0.02 1.93 0.064
FA – FH x Valence 0.14 0.09 – 0.19 5.53 <0.001 0.22 0.13 – 0.30 5.20 <0.001
FA – FH x STAI -0.00 -0.01 – 0.00 -0.89 0.380 0.00 -0.01 – 0.01 0.86 0.399
Valence x STAI -0.00 -0.00 – 0.00 -0.62 0.539 0.00 -0.00 – 0.01 0.53 0.604
FA – FH x Valence x STAI -0.00 -0.01 – 0.00 -2.03 0.053 0.00 -0.01 – 0.01 0.46 0.653
Observations 3660 3357
LMM Results for Word Categorization RT With Trait Worry (PSWQ) as Predictor
  RT: PSWQ as predictor in patients with OCD RT: PSWQ as predictor in control participants
Fixed effects b 95 % CI t p b 95 % CI t p
Intercept 6.46 6.39 – 6.53 195.05 <0.001 6.41 6.33 – 6.50 153.40 <0.001
FA_FH 0.16 0.11 – 0.21 6.19 <0.001 0.18 0.11 – 0.24 5.22 <0.001
Valence (Pos – Neg) 0.00 -0.04 – 0.04 0.04 0.968 0.06 0.01 – 0.11 2.55 0.016
PSWQ -0.00 -0.01 – 0.00 -0.76 0.451 -0.00 -0.01 – 0.01 -0.35 0.731
FA – FH x Valence 0.14 0.09 – 0.19 5.25 <0.001 0.22 0.13 – 0.30 5.21 <0.001
FA – FH x PSWQ -0.00 -0.01 – 0.00 -1.51 0.142 -0.00 -0.01 – 0.01 -0.46 0.650
Valence x PSWQ 0.00 -0.00 – 0.00 0.38 0.704 0.00 -0.00 – 0.01 1.08 0.293
FA – FH x Valence x PSWQ -0.00 -0.01 – 0.00 -0.52 0.610 0.00 -0.01 – 0.01 0.09 0.930
Observations 3660 3357


Exploratory analyses on effects of trait anxiety, worry, and OCD symptom severity on action evaluation within both groups revealed an interaction between OCD symptom severity, word valence, and the response type contrast false alarms compared to fast hits in the patient group. Thus, higher symptom severity in patients was associated with a smaller overall priming effect after fast hits and false alarms. A trend for a similar relation was observed for trait anxiety. There was no significant effect of trait worry.


Correlations

The middle and lower tables correspond to Table S6 and S7 in the supplemental material.

# Calculate overall score for ERPs and priming
df_corr <- single_trial_data_rt %>%
  # Calculate mean per participant
  dplyr::group_by(participant_id) %>%
  dplyr::summarize(
    ERN              = mean(MFN_0_100_FCz[gng_response_type == "FA"], na.rm = TRUE),
    CRN              = mean(MFN_0_100_FCz[gng_response_type == "FH"], na.rm = TRUE),
    pos_after_FA     = mean(word_rt[gng_response_type == "FA" & word_valence == "pos"], na.rm = TRUE),
    neg_after_FA     = mean(word_rt[gng_response_type == "FA" & word_valence == "neg"], na.rm = TRUE),
    pos_after_FH     = mean(word_rt[gng_response_type == "FH" & word_valence == "pos"], na.rm = TRUE),
    neg_after_FH     = mean(word_rt[gng_response_type == "FH" & word_valence == "neg"], na.rm = TRUE),
    priming_after_FA = pos_after_FA - neg_after_FA,
    priming_after_FH = neg_after_FH - pos_after_FH,
    priming          = (pos_after_FA + neg_after_FH) - (neg_after_FA + pos_after_FH)) %>%
  dplyr::ungroup() %>%
  # Merge trait variables
  dplyr::left_join(., traits, by = "participant_id")


# Calculate correlations
corr_priming_FA_ERN   <- cor.test(df_corr$priming_after_FA, df_corr$ERN)
corr_priming_FH_CRN   <- cor.test(df_corr$priming_after_FH, df_corr$CRN)
corr_priming_oci_ocd  <- cor.test(df_corr[df_corr$group == "OCD",]$priming, df_corr[df_corr$group == "OCD",]$oci)
corr_priming_oci_hc   <- cor.test(df_corr[df_corr$group == "HC", ]$priming, df_corr[df_corr$group == "HC", ]$oci)
corr_priming_stai_ocd <- cor.test(df_corr[df_corr$group == "OCD",]$priming, df_corr[df_corr$group == "OCD",]$stai)
corr_priming_stai_hc  <- cor.test(df_corr[df_corr$group == "HC", ]$priming, df_corr[df_corr$group == "HC", ]$stai)
corr_priming_pswq_ocd <- cor.test(df_corr[df_corr$group == "OCD",]$priming, df_corr[df_corr$group == "OCD",]$pswq)
corr_priming_pswq_hc  <- cor.test(df_corr[df_corr$group == "HC", ]$priming, df_corr[df_corr$group == "HC", ]$pswq)
corr_oci_stai_ocd     <- cor.test(df_corr[df_corr$group == "OCD",]$oci,     df_corr[df_corr$group == "OCD",]$stai)
corr_oci_stai_hc      <- cor.test(df_corr[df_corr$group == "HC", ]$oci,     df_corr[df_corr$group == "HC", ]$stai)
corr_oci_pswq_ocd     <- cor.test(df_corr[df_corr$group == "OCD",]$oci,     df_corr[df_corr$group == "OCD",]$pswq)
corr_oci_pswq_hc      <- cor.test(df_corr[df_corr$group == "HC", ]$oci,     df_corr[df_corr$group == "HC", ]$pswq)
corr_stai_pswq_ocd    <- cor.test(df_corr[df_corr$group == "OCD",]$stai,    df_corr[df_corr$group == "OCD",]$pswq)
corr_stai_pswq_hc     <- cor.test(df_corr[df_corr$group == "HC", ]$stai,    df_corr[df_corr$group == "HC", ]$pswq)


# Write correlations priming & ERPs in data frame
corr_erp <- rbind(cbind(corr_priming_FA_ERN$estimate,    corr_priming_FA_ERN$conf.int[1], 
                        corr_priming_FA_ERN$conf.int[2], corr_priming_FA_ERN$p.value),
                  cbind(corr_priming_FH_CRN$estimate,    corr_priming_FH_CRN$conf.int[1], 
                        corr_priming_FH_CRN$conf.int[2], corr_priming_FH_CRN$p.value))


# Write correlations priming & traits in data frame
corr_trait <- rbind(cbind(corr_priming_oci_ocd$estimate,     corr_priming_oci_ocd$conf.int[1], 
                          corr_priming_oci_ocd$conf.int[2],  corr_priming_oci_ocd$p.value,
                          corr_priming_oci_hc$estimate,      corr_priming_oci_hc$conf.int[1], 
                          corr_priming_oci_hc$conf.int[2],   corr_priming_oci_hc$p.value),
                    cbind(corr_priming_stai_ocd$estimate,    corr_priming_stai_ocd$conf.int[1], 
                          corr_priming_stai_ocd$conf.int[2], corr_priming_stai_ocd$p.value,
                          corr_priming_stai_hc$estimate,     corr_priming_stai_hc$conf.int[1], 
                          corr_priming_stai_hc$conf.int[2],  corr_priming_stai_hc$p.value),
                    cbind(corr_priming_pswq_ocd$estimate,    corr_priming_pswq_ocd$conf.int[1], 
                          corr_priming_pswq_ocd$conf.int[2], corr_priming_pswq_ocd$p.value,
                          corr_priming_pswq_hc$estimate,     corr_priming_pswq_hc$conf.int[1], 
                          corr_priming_pswq_hc$conf.int[2],  corr_priming_pswq_hc$p.value))


# Write correlations between questionnaire scores in data frame
corr_quest <- rbind(cbind(corr_oci_stai_ocd$estimate,     corr_oci_stai_ocd$conf.int[1], 
                          corr_oci_stai_ocd$conf.int[2],  corr_oci_stai_ocd$p.value,
                          corr_oci_stai_hc$estimate,      corr_oci_stai_hc$conf.int[1], 
                          corr_oci_stai_hc$conf.int[2],   corr_oci_stai_hc$p.value),
                    cbind(corr_oci_pswq_ocd$estimate,     corr_oci_pswq_ocd$conf.int[1], 
                          corr_oci_pswq_ocd$conf.int[2],  corr_oci_pswq_ocd$p.value,
                          corr_oci_pswq_hc$estimate,      corr_oci_pswq_hc$conf.int[1], 
                          corr_oci_pswq_hc$conf.int[2],   corr_oci_pswq_hc$p.value),
                    cbind(corr_stai_pswq_ocd$estimate,    corr_stai_pswq_ocd$conf.int[1], 
                          corr_stai_pswq_ocd$conf.int[2], corr_stai_pswq_ocd$p.value,
                          corr_stai_pswq_hc$estimate,     corr_stai_pswq_hc$conf.int[1], 
                          corr_stai_pswq_hc$conf.int[2],  corr_stai_pswq_hc$p.value))


# Display correlations priming & ERPs
my_table_template(data.frame(c("ERN & priming after false alarms", "CRN & priming after fast hits"), corr_erp),
  col_names = c("", "r(54)", "LL", "UL", "p"),
  digits = c(0, 2, 2, 2, 3),
  header_above_config = c(" " = 2, "95% CI" = 2, " "),
  caption = "Correlations Between ERPs and the Priming Effect")


# Display correlations priming & traits
my_table_template(data.frame(c("Obsessive-compulsive symptoms (OCI-R)", "Trait anxiety (STAI trait)",
  "Trait worry (PSWQ)"), corr_trait),
  col_names = c("Characteristic", rep(c("r(26)", "LL", "UL", "p"), 2)),
  digits = c(0, 2, 2, 2, 3, 2, 2, 2, 3),
  header_above_config = c(" " = 2, "95% CI" = 2, " " = 2, "95% CI" = 2, " "),
  caption = "Correlations Between OCD-Related Characteristics and the Overall Priming Effect After False 
  Alarms and Fast Hits") %>%
  add_header_above(c("", "Patients with OCD" = 4, "Healthy control participants" = 4)) %>%
  add_footnote("\n Note. Pearson correlation coefficients are reported.", notation = "none")


# Display correlations between questionnaire scores
my_table_template(data.frame(c("OCI-R & STAI trait", "OCI-R & PSWQ", "STAI trait & PSWQ"), corr_quest),
  col_names = c("Questionnaire scores", rep(c("r(26)", "LL", "UL", "p"), 2)),
  digits = c(0, 2, 2, 2, 3, 2, 2, 2, 3),
  header_above_config = c(" " = 2, "95% CI" = 2, " " = 2, "95% CI" = 2, " "),
  caption = "Correlations Between Questionnaire Scores") %>%
  add_header_above(c("", "Patients with OCD" = 4, "Healthy control participants" = 4)) %>%
  add_footnote("\n Note. Pearson correlation coefficients are reported.", notation = "none")


# Save correlation dataframe for scatterplots
save(df_corr, file = "./saved_objects_for_plots/df_corr.Rda")
Correlations Between ERPs and the Priming Effect
95% CI
r(54) LL UL p
ERN & priming after false alarms -0.09 -0.34 0.18 0.525
CRN & priming after fast hits 0.15 -0.12 0.40 0.273
Correlations Between OCD-Related Characteristics and the Overall Priming Effect After False Alarms and Fast Hits
Patients with OCD
Healthy control participants
95% CI
95% CI
Characteristic r(26) LL UL p r(26) LL UL p
Obsessive-compulsive symptoms (OCI-R) -0.41 -0.68 -0.05 0.029 -0.05 -0.42 0.33 0.790
Trait anxiety (STAI trait) -0.34 -0.63 0.04 0.078 0.15 -0.24 0.49 0.459
Trait worry (PSWQ) -0.22 -0.55 0.17 0.264 0.02 -0.35 0.39 0.903

Note. Pearson correlation coefficients are reported.
Correlations Between Questionnaire Scores
Patients with OCD
Healthy control participants
95% CI
95% CI
Questionnaire scores r(26) LL UL p r(26) LL UL p
OCI-R & STAI trait 0.53 0.19 0.75 0.004 0.39 0.01 0.66 0.043
OCI-R & PSWQ 0.33 -0.05 0.62 0.091 0.40 0.03 0.67 0.034
STAI trait & PSWQ 0.62 0.32 0.81 0.000 0.42 0.06 0.69 0.025

Note. Pearson correlation coefficients are reported.


There was no correlation between the priming effect after false alarms and ERN across participants. Also no correlation was observed between the priming effect after fast hits and CRN across participants.

The correlation analysis revealed a negative correlation between the overall priming effect and symptom severity in patients. A trend for a similar relation was observed for trait anxiety.


Analysis of Accuracy


We performed a binomial generalized linear mixed model (GLMM) on single-trial data. Word categorization accuracy was modeled using type of the preceding go/no-go response (slow hit, fast hit, false alarm, inhibited response), word valence, and group as fixed effects.

We used sliding difference contrasts for categorical fixed effects and started with the maximal random-effects structure for each model. In case of non-convergence, random effects were specified as uncorrelated. Using principal component analysis, we identified random effects explaining zero variance and removed these, as recommended by Bates, Kliegl, et al. (2015).

GLMM

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_acc$gng_response_type) <- contr.sdif(4)
contrasts(single_trial_data_acc$word_valence)      <- contr.sdif(2)
contrasts(single_trial_data_acc$group)             <- contr.sdif(2)


# Add contrasts as numerical covariates via model matrix*
model_matrix_2 <- model.matrix(~ gng_response_type * word_valence * group, single_trial_data_acc)


# Attach the model matrix (16 columns) to the dataframe
single_trial_data_acc[, (ncol(single_trial_data_acc) + 1):(ncol(single_trial_data_acc) + 16)] <- model_matrix_2


# Assign descriptive names to the contrasts
names(single_trial_data_acc)[(ncol(single_trial_data_acc) - 15):ncol(single_trial_data_acc)] <- c(
  "Grand Mean", "FH_SH", "FA_FH", "IR_FA", "pos_neg", "OCD_HC", "FH_SH:pos_neg", "FA_FH:pos_neg",
  "IR_FA:pos_neg", "FH_SH:OCD_HC", "FA_FH:OCD_HC", "IR_FA:OCD_HC", "pos_neg:OCD_HC",
  "FH_SH:pos_neg:OCD_HC", "FA_FH:pos_neg:OCD_HC", "IR_FA:pos_neg:OCD_HC")


# *Note: For the random effects, we needed to enter the separate random effect terms in the models to enable
# double-bar notation (||). This allows fitting a model that sets correlations of the random terms to zero.
# Run model with maximal random-effects structure
GLMM_acc_max <- glmer(word_accuracy_numeric ~ gng_response_type * word_valence * group +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg | participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FH_SH:OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC | word),
  data = single_trial_data_acc,
  family = binomial,
  control = glmerControl(optimizer = "bobyqa"))


# Check model output
summary(GLMM_acc_max)    # Model does not converge
isSingular(GLMM_acc_max) # Singular fit


# Run zero-correlation parameter model by using || syntax to reduce model complexity
GLMM_acc_red1 <- glmer(word_accuracy_numeric ~ gng_response_type * word_valence * group +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FH_SH:OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_acc,
  family = binomial,
  control = glmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
summary(GLMM_acc_red1)        # Model does converge
isSingular(GLMM_acc_red1)     # Singular fit
summary(rePCA(GLMM_acc_red1)) # For words there is 1 term, for participants 2 terms that do not explain variance (< 0.5%)
print(VarCorr(GLMM_acc_red1), comp = "Variance") # It is FH_SH:OCD_HC for words and FH_SH, FH_SH:pos_neg for participants

This table corresponds to Table 3 in the manuscript.

# Run final model without random terms explaining zero variance
GLMM_acc_final <- glmer(word_accuracy_numeric ~ gng_response_type * word_valence * group +
  (1 + FA_FH + IR_FA + pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_acc,
  family = binomial,
  control = glmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(GLMM_acc_final)        # Model does converge
# isSingular(GLMM_acc_final)     # No singular fit
# summary(rePCA(GLMM_acc_final)) # All terms explain variance


# Display results (fixed effects)
tab_model(GLMM_acc_final,
  dv.labels = "Accuracy", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "Odds ratio", string.stat = "z", string.ci = "95 % CI", wrap.labels = 80,
  title = "GLMM Results for Word Categorization Accuracy")


# Display random effects
print("Random effects:")
print(VarCorr(GLMM_acc_final), digits = 1, comp = "Std.Dev.")


# Save model output for plot
saveRDS(GLMM_acc_final, file = "./saved_objects_for_plots/GLMM_acc_final.rds")
GLMM Results for Word Categorization Accuracy
  Accuracy
Fixed effects Odds ratio 95 % CI z p
Intercept 21.38 16.82 – 27.18 25.02 <0.001
FH – SH 0.87 0.76 – 1.00 -1.92 0.055
FA – FH 0.68 0.52 – 0.88 -2.89 0.004
IR – FA 2.92 2.26 – 3.78 8.17 <0.001
Valence (Pos – Neg) 0.55 0.37 – 0.80 -3.13 0.002
Group (OCD – HC) 1.30 0.86 – 1.96 1.25 0.213
FH – SH x Valence 1.07 0.81 – 1.42 0.46 0.646
FA – FH x Valence 0.10 0.06 – 0.18 -7.75 <0.001
IR – FA x Valence 5.44 3.20 – 9.26 6.26 <0.001
FH – SH x Group 0.82 0.63 – 1.07 -1.45 0.147
FA – FH x Group 0.92 0.56 – 1.52 -0.34 0.737
IR – FA x Group 1.18 0.71 – 1.96 0.65 0.517
Valence x Group 1.90 1.05 – 3.43 2.13 0.033
FH – SH x Valence x Group 1.05 0.62 – 1.79 0.20 0.844
FA – FH x Valence x Group 1.89 0.62 – 5.77 1.12 0.262
IR – FA x Valence x Group 0.75 0.26 – 2.17 -0.53 0.599
Observations 26123
[1] "Random effects:"
 Groups           Name          Std.Dev.
 word             IR_FA:OCD_HC  0.15    
 word.1           FA_FH:OCD_HC  0.17    
 word.2           OCD_HC        0.33    
 word.3           IR_FA         0.09    
 word.4           FA_FH         0.33    
 word.5           FH_SH         0.19    
 word.6           (Intercept)   0.50    
 participant_id   IR_FA:pos_neg 1.08    
 participant_id.1 FA_FH:pos_neg 1.39    
 participant_id.2 pos_neg       0.94    
 participant_id.3 IR_FA         0.44    
 participant_id.4 FA_FH         0.53    
 participant_id.5 (Intercept)   0.72    


Analysis of categorization accuracy revealed main effects of response type and word valence, which were qualified by significant interactions between response type and word valence and between group and word valence.


Pairwise Comparisons

Significant and hypothesis-relevant interactions were followed up with post hoc comparisons using false discovery rate adjustment.

These tables correspond to Table 4 in the manuscript.

#### 1) Get contrasts for effect of word valence within each response type
emms1 <- emmeans(GLMM_acc_final, ~ word_valence | gng_response_type)
contrast_1 <- contrast(emms1, interaction = "revpairwise")


# Get CIs of estimates
contrast_1_ci <- confint(contrast_1, adjust = "none", level = 0.95)[c("asymp.LCL", "asymp.UCL")]


# Merge contrast and CI in one table
follow_up_1 <- cbind(contrast_1, contrast_1_ci)


# Add adjusted p values
follow_up_1$p.value_adj_fdr <- p.adjust(follow_up_1$p.value, method = "fdr")


# Here, estimates and CI are log odds ratios, not like in the other tables odds ratios.
# To get the odds ratios from the log odds ratios, we have to transform the values by using the exp() function
follow_up_1$estimate  <- exp(follow_up_1$estimate)
follow_up_1$asymp.LCL <- exp(follow_up_1$asymp.LCL)
follow_up_1$asymp.UCL <- exp(follow_up_1$asymp.UCL)


# Display table
my_table_template(follow_up_1[, c(2, 1, 3, 8, 9, 6, 7, 10)],
  caption = "Follow-up effect of valence nested within response type",
  digits = c(0, 0, 2, 2, 2, 2, 3, 3))


#### 2) Get contrasts for resolving interaction group * valence nested within response type
emms2 <- emmeans(GLMM_acc_final, ~ word_valence * group | gng_response_type)
contrast_2 <- contrast(emms2, interaction = "revpairwise")


# Get CIs of estimates
contrast_2_ci <- confint(contrast_2, adjust = "none", level = 0.95)[c("asymp.LCL", "asymp.UCL")]


# Put contrast and CI in one table (only for FH and FA)
follow_up_2 <- cbind(contrast_2[c(2, 3), ], contrast_2_ci[c(2, 3), ])


# Get adjusted p values
follow_up_2$p.value_adj_fdr <- p.adjust(follow_up_2$p.value, method = "fdr")


# Here, estimates and CI are log odds ratios, not like in the other tables odds ratios.
# To get the odds ratios from the log odds ratios, we have to transform the values by using the exp() function
follow_up_2$estimate  <- exp(follow_up_2$estimate)
follow_up_2$asymp.LCL <- exp(follow_up_2$asymp.LCL)
follow_up_2$asymp.UCL <- exp(follow_up_2$asymp.UCL)


# Display table
my_table_template(follow_up_2[, c(3, 1, 2, 4, 9, 10, 7, 8, 11)],
  caption = "Follow-up interaction group * valence nested within response type",
  digits = c(0, 0, 0, 2, 2, 2, 2, 3, 3))


#### 3) Get contrasts for resolving effect of valence nested within group and response type
emms3 <- emmeans(GLMM_acc_final, ~ word_valence | gng_response_type * group)
contrast_3 <- contrast(emms3, interaction = "revpairwise")


# Get CIs of estimates
contrast_3_ci <- confint(contrast_3, adjust = "none", level = 0.95)[c("asymp.LCL", "asymp.UCL")]


# Put contrast and CI in one table (only for FH and FA)
follow_up_3 <- cbind(contrast_3[c(2, 3, 6, 7), ], contrast_3_ci[c(2, 3, 6, 7), ])


# Get adjusted p values
follow_up_3$p.value_adj_fdr <- p.adjust(follow_up_3$p.value, method = "fdr")


# Here, estimates and CI are log odds ratios, not like in the other tables odds ratios.
# To get the odds ratios from the log odds ratios, we have to transform the values by using the exp() function
follow_up_3$estimate  <- exp(follow_up_3$estimate)
follow_up_3$asymp.LCL <- exp(follow_up_3$asymp.LCL)
follow_up_3$asymp.UCL <- exp(follow_up_3$asymp.UCL)


# Display table
my_table_template(follow_up_3[c(1, 3, 2, 4), c(2, 3, 1, 4, 9, 10, 7, 8, 11)],
  caption = "Follow-up effect of valence nested within group and response type",
  digits = c(0, 0, 0, 2, 2, 2, 2, 3, 3))
Follow-up effect of valence nested within response type
gng_response_type word_valence_revpairwise estimate asymp.LCL asymp.UCL z.ratio p.value p.value_adj_fdr
SH pos - neg 1.08 0.68 1.69 0.32 0.752 0.752
FH pos - neg 1.15 0.72 1.84 0.58 0.563 0.750
FA pos - neg 0.11 0.07 0.20 -7.79 0.000 0.000
IR pos - neg 0.63 0.36 1.08 -1.69 0.091 0.182
Follow-up interaction group * valence nested within response type
gng_response_type word_valence_revpairwise group_revpairwise estimate asymp.LCL asymp.UCL z.ratio p.value p.value_adj_fdr
FH pos - neg OCD - HC 1.50 0.68 3.33 1.01 0.315 0.315
FA pos - neg OCD - HC 2.84 1.08 7.46 2.12 0.034 0.067
Follow-up effect of valence nested within group and response type
gng_response_type group word_valence_revpairwise estimate asymp.LCL asymp.UCL z.ratio p.value p.value_adj_fdr
FH HC pos - neg 0.94 0.51 1.73 -0.21 0.836 0.836
FH OCD pos - neg 1.41 0.76 2.62 1.08 0.278 0.371
FA HC pos - neg 0.07 0.03 0.14 -7.09 0.000 0.000
FA OCD pos - neg 0.19 0.10 0.39 -4.52 0.000 0.000


Post hoc comparisons indicated that after false alarms, participants across and within groups categorized negative words more accurately than positive words. This response facilitation to negative relative to positive words after false alarms was smaller in patients with OCD compared to control participants at trend level.


Control Analysis


The study was part of a project comprising one session with active and one with placebo transcranial direct current stimulation in counterbalanced order, which preceded task performance. In the present study, data from the placebo session were analyzed. Thus, measurements of participants who completed the task for the first or second time were included. Importantly, control analyses indicated that task familiarity (i.e., session number) did not impact hypothesis-relevant results.

RT: Session as Cov.

This table corresponds to Table S10 in the supplemental material.

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_rt$session) <- contr.sdif(2)


# Run reduced final model (we get same model when starting with all random effects & correlation parameters)
LMM_rt_session <- lmer(word_rt_log ~ gng_response_type * word_valence * group * session +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_rt,
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_session)        # Model does converge
# isSingular(LMM_rt_session)     # No singular fit
# summary(rePCA(LMM_rt_session)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_session,
  dv.labels = "RT", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", wrap.labels = 80,
  title = "LMM Results for Word Categorization RT With Session Number as Covariate")
LMM Results for Word Categorization RT With Session Number as Covariate
  RT
Fixed effects b 95 % CI t p
Intercept 6.44 6.39 – 6.49 272.93 <0.001
FH – SH -0.02 -0.03 – -0.01 -3.09 0.003
FA – FH 0.17 0.13 – 0.21 8.31 <0.001
IR – FA -0.02 -0.06 – 0.03 -0.71 0.482
Valence (Pos – Neg) 0.01 -0.02 – 0.04 0.77 0.445
Group (OCD – HC) 0.03 -0.06 – 0.13 0.74 0.466
Session (2 – 1) -0.07 -0.16 – 0.02 -1.53 0.132
FH – SH x Valence -0.02 -0.04 – -0.00 -2.29 0.027
FA – FH x Valence 0.18 0.14 – 0.22 8.31 <0.001
IR – FA x Valence -0.10 -0.14 – -0.06 -5.41 <0.001
FH – SH x Group 0.01 -0.01 – 0.03 1.04 0.305
FA – FH x Group -0.02 -0.10 – 0.06 -0.42 0.675
IR – FA x Group -0.02 -0.12 – 0.07 -0.54 0.591
Valence x Group -0.04 -0.09 – -0.00 -2.10 0.041
FH – SH x Session -0.01 -0.03 – 0.02 -0.45 0.652
FA – FH x Session -0.02 -0.10 – 0.06 -0.60 0.551
IR – FA x Session 0.02 -0.07 – 0.11 0.45 0.654
Valence x Session 0.03 -0.01 – 0.08 1.60 0.116
Group x Session -0.01 -0.20 – 0.17 -0.16 0.875
FH – SH x Valence x Group -0.00 -0.04 – 0.03 -0.16 0.872
FA – FH x Valence x Group -0.08 -0.17 – 0.00 -1.94 0.058
IR – FA x Valence x Group 0.06 -0.01 – 0.14 1.68 0.100
FH – SH x Valence x Session -0.04 -0.08 – -0.01 -2.53 0.015
FA – FH x Valence x Session -0.03 -0.12 – 0.05 -0.77 0.443
IR – FA x Valence x Session 0.06 -0.02 – 0.13 1.58 0.120
FH – SH x Group x Session -0.02 -0.06 – 0.03 -0.87 0.389
FA – FH x Group x Session 0.01 -0.15 – 0.18 0.19 0.854
IR – FA x Group x Session 0.05 -0.13 – 0.24 0.60 0.552
Valence x Group x Session -0.03 -0.11 – 0.06 -0.60 0.550
FH – SH x Valence x Group x Session -0.02 -0.08 – 0.05 -0.46 0.647
FA – FH x Valence x Group x Session 0.15 -0.03 – 0.32 1.71 0.093
IR – FA x Valence x Group x Session -0.10 -0.25 – 0.05 -1.32 0.192
Observations 24189




RT with ERN & CRN: Session as Cov.

This table corresponds to Table S11 in the supplemental material.

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_rt$session) <- contr.sdif(2)


# Run reduced final model on ERN (we get same model when starting with all random effects & correlation parameters)
LMM_rt_ern_session <- lmer(word_rt_log ~ word_valence * group * MFN_standardized * session +
  (1 + pos_neg || participant_id) +
  (1 + OCD_HC + MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FA", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_ern_session)        # Model does converge
# isSingular(LMM_rt_ern_session)     # No singular fit
# summary(rePCA(LMM_rt_ern_session)) # All terms explain variance


# Run reduced final model on CRN (we get same model when starting with all random effects & correlation parameters)
LMM_rt_crn_session <- lmer(word_rt_log ~ word_valence * group * MFN_standardized * session +
  (1 + pos_neg || participant_id) +
  (1 + MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FH", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_crn_session)        # Model does converge
# isSingular(LMM_rt_crn_session)     # No singular fit
# summary(rePCA(LMM_rt_crn_session)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_ern_session, LMM_rt_crn_session,
  dv.labels = c("With ERN as predictor", "With CRN as predictor"), pred.labels = labels, show.stat = TRUE, 
  show.icc = FALSE, show.r2 = FALSE, show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", 
  string.pred = "Fixed effects", string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", 
  wrap.labels = 80, title = "LMM Results for Word Categorization RT With ERN and CRN as Predictor and Session Number 
  as Covariate")
LMM Results for Word Categorization RT With ERN and CRN as Predictor and Session Number as Covariate
  With ERN as predictor With CRN as predictor
Fixed effects b 95 % CI t p b 95 % CI t p
Intercept 6.52 6.46 – 6.59 197.06 <0.001 6.35 6.31 – 6.40 269.82 <0.001
Valence (Pos – Neg) 0.12 0.07 – 0.18 4.44 <0.001 -0.06 -0.08 – -0.03 -4.27 <0.001
Group (OCD – HC) 0.04 -0.09 – 0.17 0.59 0.555 0.05 -0.04 – 0.15 1.16 0.253
ERP 0.00 -0.01 – 0.01 0.43 0.665 -0.01 -0.01 – -0.00 -2.25 0.027
Session (2 – 1) -0.09 -0.22 – 0.04 -1.33 0.189 -0.06 -0.16 – 0.03 -1.35 0.182
Valence x Group -0.09 -0.20 – 0.01 -1.76 0.084 -0.03 -0.07 – 0.02 -1.10 0.274
Valence x ERP 0.00 -0.02 – 0.02 0.21 0.833 -0.01 -0.02 – 0.01 -0.93 0.357
Group x ERP 0.01 -0.01 – 0.03 1.20 0.229 -0.01 -0.02 – 0.00 -1.39 0.166
Valence x Session -0.01 -0.12 – 0.10 -0.19 0.851 0.02 -0.02 – 0.07 1.06 0.295
Group x Session -0.02 -0.29 – 0.24 -0.16 0.872 -0.05 -0.23 – 0.14 -0.49 0.627
ERP x Session 0.00 -0.02 – 0.02 0.30 0.764 -0.01 -0.02 – 0.00 -1.33 0.185
Valence x Group x ERP 0.01 -0.03 – 0.05 0.52 0.605 0.01 -0.01 – 0.03 0.61 0.541
Valence x Group x Session 0.06 -0.15 – 0.28 0.60 0.554 -0.07 -0.16 – 0.03 -1.44 0.156
Valence x ERP x Session 0.00 -0.03 – 0.04 0.20 0.840 -0.00 -0.02 – 0.02 -0.36 0.716
Group x ERP x Session -0.01 -0.04 – 0.03 -0.29 0.770 0.02 -0.01 – 0.04 1.43 0.153
Valence x Group x ERP x Session 0.01 -0.07 – 0.08 0.18 0.856 -0.04 -0.08 – 0.01 -1.71 0.088
Observations 1992 5025




Accuracy: Session as Cov.

This table corresponds to Table S10 in the supplemental material.

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_acc$session) <- contr.sdif(2)


# Run reduced final model (we get same model when starting with all random effects & correlation parameters)
GLMM_acc_session <- glmer(word_accuracy_numeric ~ gng_response_type * word_valence * group * session +
  (1 + FA_FH + IR_FA + pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_acc,
  family = binomial,
  control = glmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(GLMM_acc_session)        # Model does converge
# isSingular(GLMM_acc_session)     # No singular fit
# summary(rePCA(GLMM_acc_session)) # All terms explain variance


# Display results (fixed effects)
tab_model(GLMM_acc_session,
  dv.labels = "Accuracy", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "Odds ratio", string.stat = "z", string.ci = "95 % CI", wrap.labels = 80,
  title = "GLMM Results for Word Categorization Accuracy With Session Number as Covariate")
GLMM Results for Word Categorization Accuracy With Session Number as Covariate
  Accuracy
Fixed effects Odds ratio 95 % CI z p
Intercept 21.31 16.79 – 27.05 25.14 <0.001
FH – SH 0.86 0.75 – 1.00 -2.01 0.045
FA – FH 0.70 0.54 – 0.91 -2.66 0.008
IR – FA 2.83 2.19 – 3.65 7.95 <0.001
Valence (Pos – Neg) 0.55 0.38 – 0.79 -3.18 0.001
Group (OCD – HC) 1.27 0.85 – 1.91 1.15 0.248
Session (2 – 1) 0.88 0.59 – 1.31 -0.61 0.539
FH – SH x Valence 1.07 0.80 – 1.42 0.47 0.639
FA – FH x Valence 0.09 0.05 – 0.17 -8.12 <0.001
IR – FA x Valence 5.56 3.29 – 9.38 6.42 <0.001
FH – SH x Group 0.83 0.63 – 1.08 -1.38 0.167
FA – FH x Group 0.90 0.54 – 1.49 -0.41 0.683
IR – FA x Group 1.17 0.71 – 1.94 0.62 0.537
Valence x Group 1.87 1.06 – 3.30 2.15 0.031
FH – SH x Session 0.96 0.73 – 1.25 -0.32 0.752
FA – FH x Session 1.97 1.19 – 3.25 2.65 0.008
IR – FA x Session 0.49 0.30 – 0.81 -2.79 0.005
Valence x Session 0.49 0.29 – 0.84 -2.58 0.010
Group x Session 0.80 0.36 – 1.77 -0.56 0.578
FH – SH x Valence x Group 1.03 0.61 – 1.76 0.12 0.907
FA – FH x Valence x Group 2.29 0.77 – 6.80 1.49 0.136
IR – FA x Valence x Group 0.68 0.24 – 1.93 -0.73 0.466
FH – SH x Valence x Session 1.07 0.63 – 1.81 0.24 0.812
FA – FH x Valence x Session 0.48 0.16 – 1.41 -1.34 0.181
IR – FA x Valence x Session 2.58 0.91 – 7.34 1.78 0.075
FH – SH x Group x Session 1.44 0.84 – 2.45 1.34 0.181
FA – FH x Group x Session 0.48 0.18 – 1.30 -1.44 0.150
IR – FA x Group x Session 2.42 0.90 – 6.52 1.74 0.081
Valence x Group x Session 0.69 0.23 – 2.03 -0.68 0.498
FH – SH x Valence x Group x Session 0.58 0.20 – 1.66 -1.02 0.308
FA – FH x Valence x Group x Session 0.17 0.02 – 1.43 -1.63 0.103
IR – FA x Valence x Group x Session 2.27 0.29 – 17.67 0.78 0.435
Observations 26123




References


Bates, D., Kliegl, R., Vasishth, S., & Baayen, H. (2015). Parsimonious mixed models. arXiv. https://arxiv.org/abs/1506.04967v2

Box, G. E., & Cox, D. R. (1964). An analysis of transformations. Journal of the Royal Statistical Society: Series B (Methodological), 26(2), 211–243. https://doi.org/10.1111/j.2517-6161.1964.tb00553.x

Morey, R. (2008). Confidence intervals from normalized data: A correction to Cousineau (2005). Tutorials in Quantitative Methods for Psychology, 4(2), 61–64. https://doi.org/10.20982/tqmp.04.2.p061


Session Info


sessionInfo()
R version 3.6.1 (2019-07-05)
Platform: x86_64-w64-mingw32/x64 (64-bit)
Running under: Windows 10 x64 (build 19044)

Matrix products: default

locale:
[1] LC_COLLATE=German_Germany.1252  LC_CTYPE=German_Germany.1252   
[3] LC_MONETARY=German_Germany.1252 LC_NUMERIC=C                   
[5] LC_TIME=German_Germany.1252    

attached base packages:
[1] stats     graphics  grDevices datasets  utils     methods   base     

other attached packages:
 [1] plyr_1.8.7       emmeans_1.6.0    sjPlot_2.8.11    lmerTest_3.1-3  
 [5] lme4_1.1-25      Matrix_1.5-1     MASS_7.3-51.4    kableExtra_1.3.4
 [9] knitr_1.40       dplyr_1.0.10    

loaded via a namespace (and not attached):
 [1] httr_1.4.4          sass_0.4.0          tidyr_1.2.1        
 [4] jsonlite_1.8.3      viridisLite_0.4.1   splines_3.6.1      
 [7] modelr_0.1.9        bslib_0.4.0         datawizard_0.6.2   
[10] assertthat_0.2.1    statmod_1.4.37      highr_0.9          
[13] renv_0.12.0         yaml_2.2.1          bayestestR_0.13.0  
[16] numDeriv_2016.8-1.1 pillar_1.8.1        backports_1.4.1    
[19] lattice_0.20-45     glue_1.6.2          digest_0.6.30      
[22] rvest_1.0.3         minqa_1.2.4         colorspace_2.0-3   
[25] htmltools_0.5.3     pkgconfig_2.0.3     broom_1.0.1        
[28] purrr_0.3.5         xtable_1.8-4        mvtnorm_1.1-3      
[31] scales_1.2.1        webshot_0.5.4       svglite_2.1.0      
[34] tibble_3.1.8        generics_0.1.3      ggplot2_3.3.6      
[37] sjlabelled_1.2.0    cachem_1.0.4        withr_2.5.0        
[40] cli_3.4.1           magrittr_2.0.3      effectsize_0.7.0.5 
[43] estimability_1.4.1  evaluate_0.16       fansi_1.0.3        
[46] nlme_3.1-140        xml2_1.3.3          tools_3.6.1        
[49] lifecycle_1.0.3     stringr_1.4.1       munsell_0.5.0      
[52] ggeffects_1.1.3     compiler_3.6.1      jquerylib_0.1.4    
[55] systemfonts_1.0.4   rlang_1.0.6         grid_3.6.1         
[58] nloptr_1.2.2.2      parameters_0.19.0   rstudioapi_0.14    
[61] rmarkdown_2.16      boot_1.3-22         gtable_0.3.1       
[64] sjstats_0.18.1      DBI_1.1.3           sjmisc_2.8.9       
[67] R6_2.5.1            performance_0.10.0  fastmap_1.1.0      
[70] utf8_1.2.2          insight_0.18.4      stringi_1.6.1      
[73] Rcpp_1.0.9          vctrs_0.5.0         tidyselect_1.2.0   
[76] xfun_0.33          
---
title: "Word Categorization Data (& ERPs)"
output: 
  html_document
---

<!-- Set general settings -->

```{r setup, message = FALSE, warning = FALSE}

# Set general settings for markdown file
knitr::opts_chunk$set(
  message = FALSE,
  warning = FALSE,
  comment = "",
  results = "hold")


# Clear environment
rm(list = ls())


# Enable/disable caching of time-consuming code chunks
knitr_cache_enabled = TRUE


# Load packages
library(dplyr)      # for data wrangling
library(knitr)      # for integrating computing and reporting in markdown
library(kableExtra) # for customizing appearance of tables
library(MASS)       # for boxcox function and contrast definition
library(lme4)       # for (G)LMMs
library(lmerTest)   # for LMM p values (Satterthwaite's method for approximating dfs for t and F tests)
library(sjPlot)     # for tab_model function to display (G)LMM results
library(emmeans)    # for pairwise comparisons


# Load functions
source("./functions/summarySEwithinO.R")  # Function provided by R-cookbook: http://www.cookbook-r.com/Graphs/Plotting_means_and_error_bars_(ggplot2)/
source("./functions/my_table_template.R") # Function to create table template


# Turn off scientific notation
options(scipen = 999)


# Prepare labels for (G)LMM tables
labels <- c(
  "(Intercept)" = "Intercept",
  "gng_response_type2-1" = "FH – SH",
  "gng_response_type3-2" = "FA – FH",
  "gng_response_type4-3" = "IR – FA",
  "word_valence2-1" = "Valence (Pos – Neg)",
  "group2-1" = "Group (OCD – HC)",
  "gng_response_type2-1:word_valence2-1" = "FH – SH x Valence",
  "gng_response_type3-2:word_valence2-1" = "FA – FH x Valence",
  "gng_response_type4-3:word_valence2-1" = "IR – FA x Valence",
  "gng_response_type2-1:group2-1" = "FH – SH x Group",
  "gng_response_type3-2:group2-1" = "FA – FH x Group",
  "gng_response_type4-3:group2-1" = "IR – FA x Group",
  "word_valence2-1:group2-1" = "Valence x Group",
  "gng_response_type2-1:word_valence2-1:group2-1" = "FH – SH x Valence x Group",
  "gng_response_type3-2:word_valence2-1:group2-1" = "FA – FH x Valence x Group",
  "gng_response_type4-3:word_valence2-1:group2-1" = "IR – FA x Valence x Group",
  "MFN_standardized" = "ERP",
  "word_valence2-1:MFN_standardized" = "Valence x ERP",
  "group2-1:MFN_standardized" = "Group x ERP",
  "word_valence2-1:group2-1:MFN_standardized" = "Valence x Group x ERP",
  "FH_FH" = "FA – FH",
  "pos_neg" = "Valence (Pos – Neg)",
  "FA_FH:pos_neg" = "FA – FH x Valence",
  "oci_centered" = "OCI-R",
  "FA_FH:oci_centered" = "FA – FH x OCI-R",
  "pos_neg:oci_centered" = "Valence x OCI-R",
  "FA_FH:pos_neg:oci_centered" = "FA – FH x Valence x OCI-R",
  "stai_centered" = "STAI",
  "FA_FH:stai_centered" = "FA – FH x STAI",
  "pos_neg:stai_centered" = "Valence x STAI",
  "FA_FH:pos_neg:stai_centered" = "FA – FH x Valence x STAI",
  "pswq_centered" = "PSWQ",
  "FA_FH:pswq_centered" = "FA – FH x PSWQ",
  "pos_neg:pswq_centered" = "Valence x PSWQ",
  "FA_FH:pos_neg:pswq_centered" = "FA – FH x Valence x PSWQ",
  "session2-1" = "Session (2 – 1)",
  "gng_response_type2-1:session2-1" = "FH – SH x Session",
  "gng_response_type3-2:session2-1" = "FA – FH x Session",
  "gng_response_type4-3:session2-1" = "IR – FA x Session",
  "word_valence2-1:session2-1" = "Valence x Session",
  "group2-1:session2-1" = "Group x Session",
  "gng_response_type2-1:word_valence2-1:session2-1" = "FH – SH x Valence x Session",
  "gng_response_type3-2:word_valence2-1:session2-1" = "FA – FH x Valence x Session",
  "gng_response_type4-3:word_valence2-1:session2-1" = "IR – FA x Valence x Session",
  "gng_response_type2-1:group2-1:session2-1" = "FH – SH x Group x Session",
  "gng_response_type3-2:group2-1:session2-1" = "FA – FH x Group x Session",
  "gng_response_type4-3:group2-1:session2-1" = "IR – FA x Group x Session",
  "word_valence2-1:group2-1:session2-1" = "Valence x Group x Session",
  "gng_response_type2-1:word_valence2-1:group2-1:session2-1" = "FH – SH x Valence x Group x Session",
  "gng_response_type3-2:word_valence2-1:group2-1:session2-1" = "FA – FH x Valence x Group x Session",
  "gng_response_type4-3:word_valence2-1:group2-1:session2-1" = "IR – FA x Valence x Group x Session",
  "MFN_standardized:session2-1" = "ERP x Session",
  "word_valence2-1:MFN_standardized:session2-1" = "Valence x ERP x Session",
  "group2-1:MFN_standardized:session2-1" = "Group x ERP x Session",
  "word_valence2-1:group2-1:MFN_standardized:session2-1" = "Valence x Group x ERP x Session")
```
<br>

## Data Cleaning
***

```{r load-and-clean-data}

# Load data
load(file = "./data/Single_Trial_Data.rda")
load(file = "./data/Trait_Data.rda")


# Create numeric accuracy variable (1 = correct, 0 = incorrect)
single_trial_data <- single_trial_data %>%
  dplyr::mutate(word_accuracy_numeric = ifelse(word_accuracy == "correct", 1, 0))


# For RT analysis, exclude missing responses, responses with wrong key, RT outliers,
# incorrect word categorizations, and trials with ERP artifacts
single_trial_data_rt <- single_trial_data %>%
  dplyr::filter(
    gng_response_type != "miss" &
    gng_response_type != "wrong_key" &
    word_accuracy     == "correct" &
    (is.na(gng_rt_invalid)  | gng_rt_invalid  == FALSE) &
    (is.na(word_rt_outlier) | word_rt_outlier == FALSE) &
    stimulation == "sham" &
    MFN_artifact == FALSE)


# Calculate standardized MFN (based on trials with correct word categorization, as only
# these  will be used in the LMMs with ERP as predictor)
single_trial_data_rt <- single_trial_data_rt %>%
  dplyr::group_by(participant_id) %>%
  dplyr::mutate(MFN_standardized = scale(MFN_0_100_FCz, center = TRUE, scale = TRUE)) %>%
  dplyr::ungroup()


# Center trait variables within groups (will be used as predictors)
traits <- traits %>%
  dplyr::rename(oci = "OCI-R", pswq = "PSWQ", stai = "STAI trait") %>%  
  dplyr::mutate(group = as.factor(ifelse(substr(participant_id, 1, 1) == "C", "HC", "OCD"))) %>%
  dplyr::group_by(group) %>%
  dplyr::mutate(
    oci_centered  = scale(oci,  center = TRUE, scale = FALSE),
    pswq_centered = scale(pswq, center = TRUE, scale = FALSE),
    stai_centered = scale(stai, center = TRUE, scale = FALSE)) %>%
  dplyr::ungroup()


# Add trait variables to single trial data
single_trial_data_rt <- single_trial_data_rt %>%
  dplyr::left_join(., traits %>% dplyr::select(participant_id, oci_centered, pswq_centered,
    stai_centered), by = "participant_id")


# For accuracy analysis, exclude missing responses, responses with wrong key, RT outliers,
# and trials with ERP artifacts
single_trial_data_acc <- single_trial_data %>%
  dplyr::filter(
    gng_response_type != "miss" &
    gng_response_type != "wrong_key" &
    word_accuracy     != "miss" &
    word_accuracy     != "wrong_key" &
    (is.na(gng_rt_invalid)  | gng_rt_invalid  == FALSE) &
    (is.na(word_rt_outlier) | word_rt_outlier == FALSE) &
    stimulation == "sham" &
    MFN_artifact == FALSE)


# Calculate % of excluded trials (missing responses, responses with wrong key, RT outliers)
excluded_trials_per_participant <- single_trial_data %>%
  dplyr::filter(stimulation == "sham") %>%
  dplyr::group_by(participant_id) %>%
  dplyr::summarize(excluded = sum(
    gng_response_type == "miss" |
    gng_response_type == "wrong_key" |
    word_accuracy     == "miss" |
    word_accuracy     == "wrong_key" |
    (!is.na(gng_rt_invalid)  & gng_rt_invalid  != FALSE) |
    (!is.na(word_rt_outlier) & word_rt_outlier != FALSE)) / length(participant_id) * 100) %>%
  dplyr::ungroup() %>%
  dplyr::summarize(across(-c(participant_id), list(mean, sd)))


# Make categorical variables factors
single_trial_data_rt$gng_response_type  <- factor(single_trial_data_rt$gng_response_type,
  levels = c("SH", "FH", "FA", "IR"))
single_trial_data_rt$word_valence       <- factor(single_trial_data_rt$word_valence)
single_trial_data_rt$participant_id     <- factor(single_trial_data_rt$participant_id)
single_trial_data_rt$group              <- factor(single_trial_data_rt$group)
single_trial_data_rt$session            <- factor(single_trial_data_rt$session)
single_trial_data_acc$gng_response_type <- factor(single_trial_data_acc$gng_response_type,
  levels = c("SH", "FH", "FA", "IR"))
single_trial_data_acc$word_valence      <- factor(single_trial_data_acc$word_valence)
single_trial_data_acc$participant_id    <- factor(single_trial_data_acc$participant_id)
single_trial_data_acc$group             <- factor(single_trial_data_acc$group)
single_trial_data_acc$session           <- factor(single_trial_data_acc$session)


# Aggregate data per participant for plots
data_aggregated_rt <- single_trial_data_rt %>%
  dplyr::group_by(participant_id, group, gng_response_type, word_valence) %>%
  dplyr::summarize(word_rt = mean(word_rt, na.rm = TRUE)) %>%
  dplyr::ungroup()

data_aggregated_acc <- single_trial_data_acc %>%
  dplyr::group_by(participant_id, group, gng_response_type, word_valence) %>%
  dplyr::summarize(accuracy = mean(word_accuracy_numeric, na.rm = TRUE) * 100) %>%
  dplyr::ungroup()


# Save aggregated data for plots
save(data_aggregated_rt,  file = "./saved_objects_for_plots/data_aggregated_rt.Rda")
save(data_aggregated_acc, file = "./saved_objects_for_plots/data_aggregated_acc.Rda")
```

In the analysis of RT and accuracy in the word categorization task, trials were discarded (*M* = `r round(excluded_trials_per_participant$excluded_1, digits = 2)`%, *SD* = `r round(excluded_trials_per_participant$excluded_2, digits = 2)`) if RT in the go/no-go task was below 100 ms or above 800 ms, word categorization RT deviated more than three median absolute deviations from the individual condition-specific median, response was missing, or the response was made with a key not assigned to the current task. We further discarded trials containing artifacts in the EEG, i.e., a voltage difference exceeding 200 μV within an epoch or 50 μV between sample points. In the analysis of word categorization RT, we excluded incorrect categorizations.
<br><br><br>

## Descriptive Statistics
***

This table corresponds to Table S2 in the supplemental material.  

```{r descriptive-statistics-table}

# Calculate descriptive statistics for RT
descriptive_statistics_rt <- summarySEwithinO(
  data          = single_trial_data_rt,
  measurevar    = "word_rt",
  withinvars    = c("gng_response_type", "word_valence"),
  betweenvars   = "group",
  idvar         = "participant_id",
  conf.interval = .95) %>%
  # Rename variable and select variables for subsequent merging
  dplyr::rename(mean = word_rt) %>%
  dplyr::select("group", "gng_response_type", "word_valence", "mean", "ci")


# Calculate descriptive statistics for accuracy
descriptive_statistics_acc <- summarySEwithinO(
  data          = single_trial_data_acc,
  measurevar    = "word_accuracy_numeric",
  withinvars    = c("gng_response_type", "word_valence"),
  betweenvars   = "group",
  idvar         = "participant_id",
  conf.interval = .95) %>%
  # Multiply numeric values by 100 to obtain values in percent
  dplyr::mutate_if(is.numeric, list(~ . * 100)) %>%
  # Rename variable and select variables for subsequent merging
  dplyr::rename(mean = word_accuracy_numeric) %>%
  dplyr::select("group", "gng_response_type", "word_valence", "mean", "ci")


# Combine measures
descriptive_statistics <- as.data.frame(rbind(descriptive_statistics_rt, descriptive_statistics_acc))


# Create column with format "M [CI]" for RT (round to 0 decimals)
descriptive_statistics$M_CI <- paste0(round(descriptive_statistics$mean, 0), " [",
  round(descriptive_statistics$mean - descriptive_statistics$ci, 0), ", ",
  round(descriptive_statistics$mean + descriptive_statistics$ci, 0), "]")


# Create column with format "M [CI]" for accuracy (round to 2 decimals)
descriptive_statistics[c(17:32), ]$M_CI <- paste0(
  format(round(descriptive_statistics[c(17:32), ]$mean, 2), nsmall = 2), " [",
  format(round(descriptive_statistics[c(17:32), ]$mean -
               descriptive_statistics[c(17:32), ]$ci,   2), nsmall = 2), ", ",
  format(round(descriptive_statistics[c(17:32), ]$mean +
               descriptive_statistics[c(17:32), ]$ci,   2), nsmall = 2), "]")


# Split and re-merge table to display both groups next to each other
descriptive_statistics <- split(descriptive_statistics, descriptive_statistics$group)
descriptive_statistics_display <- cbind(
  paste(descriptive_statistics$OCD$word_valence, "word after", descriptive_statistics$OCD$gng_response_type),
  descriptive_statistics$OCD[, "M_CI"], descriptive_statistics$HC[, "M_CI"])


# Display table (and rearrange rows)
my_table_template(descriptive_statistics_display[c(8, 7, 4, 3, 2, 1, 6, 5, 16, 15, 12, 11, 10, 9, 14, 13), ],
  caption = "Descriptive Statistics for Behavioral Performance in the Word Categorization Task",
  col_names = c("Measure", rep("M [95% CI]", 2)),
  header_above_config = c("", "Patients with OCD", "Healthy control participants")) %>%
  add_footnote("\n Note. CIs are adjusted for within-participant comparisons using the method described 
  by Morey (2008).", notation = "none") %>%
  pack_rows("Response time (ms)", 1, 8) %>%
  pack_rows("Accuracy (%)", 9, 16)
```
<br><br>

## Analysis of Response Time
***

We performed a linear mixed model (LMM) on single-trial data. Word categorization RT was modeled using type of the preceding go/no-go response (slow hit, fast hit, false alarm, inhibited response), word valence, and group as fixed effects. 

We used sliding difference contrasts for categorical fixed effects and started with the maximal random-effects structure for each model. In case of non-convergence, random effects were specified as uncorrelated. Using principal component analysis, we identified random effects explaining zero variance and removed these, as recommended by Bates, Kliegl, et al. (2015). 
<br><br>

### Main Model {.tabset}

#### Determine Data Transformation

RT was log-transformed to normalize residuals. This was determined using the Box–Cox procedure (Box & Cox, 1964).

```{r rt-determine-transformation, fig.width = 12, fig.height = 3}

# Arrange plots
par(mfrow = c(1, 5))


# Determine transformation by estimating optimal lambda using Box–Cox procedure
bc_rt <- boxcox(word_rt ~ 1, data = single_trial_data_rt)
optlambda_rt <- bc_rt$x[which.max(bc_rt$y)]


# Density plot for raw RT
plot(density(single_trial_data_rt$word_rt), main = "Raw RT: Density Plot")


# Q-q plot for raw RT
qqnorm(single_trial_data_rt$word_rt, main = "Raw RT: Q-Q Plot", pch = 1)


# Density plot for log RT
plot(density(single_trial_data_rt$word_rt_log), main = "Log RT: Density Plot")


# Q-q plot for log RT
qqnorm(single_trial_data_rt$word_rt_log, main = "Log RT: Q-Q Plot", pch = 1)


# Reset plot layout
par(mfrow = c(1, 1))
```
The optimal lambda is `r round(optlambda_rt, digits = 2)`, suggesting that log transformation is appropriate. 
<br><br><br>

#### LMM {.active}

<!-- Contrast coding -->

```{r lmm-rt-contrast-coding}

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_rt$gng_response_type) <- contr.sdif(4)
contrasts(single_trial_data_rt$word_valence)      <- contr.sdif(2)
contrasts(single_trial_data_rt$group)             <- contr.sdif(2)


# Add contrasts as numerical covariates via model matrix*
model_matrix <- model.matrix(~ gng_response_type * word_valence * group, single_trial_data_rt)


# Attach the model matrix (16 columns) to the dataframe
single_trial_data_rt[, (ncol(single_trial_data_rt) + 1):(ncol(single_trial_data_rt) + 16)] <- model_matrix


# Assign descriptive names to the contrasts
names(single_trial_data_rt)[(ncol(single_trial_data_rt) - 15):ncol(single_trial_data_rt)] <- c(
  "Grand Mean", "FH_SH", "FA_FH", "IR_FA", "pos_neg", "OCD_HC", "FH_SH:pos_neg", "FA_FH:pos_neg",
  "IR_FA:pos_neg", "FH_SH:OCD_HC", "FA_FH:OCD_HC", "IR_FA:OCD_HC", "pos_neg:OCD_HC",
  "FH_SH:pos_neg:OCD_HC", "FA_FH:pos_neg:OCD_HC", "IR_FA:pos_neg:OCD_HC")


# *Note: For the random effects, we needed to enter the separate random effect terms in the models to enable
# double-bar notation (||). This allows fitting a model that sets correlations of the random terms to zero.
```

<!-- Model specification -->

```{r lmm-rt-model-specification, eval = FALSE}

# Run model with maximal random-effects structure
LMM_rt_max <- lmer(word_rt_log ~ gng_response_type * word_valence * group +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg | participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FH_SH:OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC | word),
  data = single_trial_data_rt,
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output
summary(LMM_rt_max)    # Model does not converge
isSingular(LMM_rt_max) # Singular fit


# Run zero-correlation parameter model by using || syntax to reduce model complexity
LMM_rt_red1 <- lmer(word_rt_log ~ gng_response_type * word_valence * group +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FH_SH:OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_rt,
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
summary(LMM_rt_red1)        # Model does converge
isSingular(LMM_rt_red1)     # Singular fit
summary(rePCA(LMM_rt_red1)) # For words there are 3 terms that do not explain variance (< 0.5%)
print(VarCorr(LMM_rt_red1), comp = "Variance") # It is IR_FA, FH_SH:OCD_HC, and FA_FH:OCD_HC for words
```

<!-- Final model -->

This table corresponds to Table 3 in the manuscript. 

```{r lmm-rt-final-model, cache = knitr_cache_enabled}

# Run final model without random terms explaining zero variance (use log(word_rt) not word_rt_log for plotting)
LMM_rt_final <- lmer(log(word_rt) ~ gng_response_type * word_valence * group + 
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_rt,
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_final)        # Model does converge
# isSingular(LMM_rt_final)     # No singular fit
# summary(rePCA(LMM_rt_final)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_final,
  dv.labels = "RT", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite",
  wrap.labels = 80, title = "LMM Results for Word Categorization RT")


# Display random effects
print("Random effects:")
print(VarCorr(LMM_rt_final), digits = 1, comp = "Std.Dev.")


# Save model output for plot
saveRDS(LMM_rt_final, file = "./saved_objects_for_plots/LMM_rt_final.rds")
```
<br>
RT analysis yielded a main effect of response type. Participants across groups categorized words slower after false alarms than after fast hits, indicating post-error slowing. Moreover, word categorization after fast hits was faster than after slow hits. The analysis further revealed significant two-way interactions between group and word valence and between each response type contrast and word valence. Crucially, the significant two-way interactions were qualified by a statistical trend for a three-way interaction between the response type contrast false alarms compared to fast hits, word valence, and group. 
<br><br><br>

#### Pairwise Comparisons

Significant and hypothesis-relevant interactions were followed up with post hoc comparisons using false discovery rate adjustment. <br><br>
These tables correspond to Table 4 in the manuscript. 

```{r lmm-rt-pairwise-comparisons}

# Set emmeans options
emm_options(lmer.df = "Satterthwaite", lmerTest.limit = Inf)


#### 1) Get contrasts for effect of word valence within each response type
emms1 <- emmeans(LMM_rt_final, ~ word_valence | gng_response_type)
contrast_1 <- contrast(emms1, interaction = "revpairwise")


# Get CIs of estimates
contrast_1_ci <- confint(contrast_1, adjust = "none", level = 0.95)[c("lower.CL", "upper.CL")]


# Merge contrast and CI in one table
follow_up_1 <- cbind(contrast_1, contrast_1_ci)


# Add adjusted p values
follow_up_1$p.value_adj_fdr <- p.adjust(follow_up_1$p.value, method = "fdr")


# Display table
my_table_template(follow_up_1[, c(2, 1, 3, 8, 9, 6, 7, 10)],
  caption = "Follow-up effect of valence nested within response type",
  digits = c(0, 0, 2, 2, 2, 2, 3, 3))


#### 2) Get contrasts for resolving interaction group * valence nested within response type
emms2 <- emmeans(LMM_rt_final, ~ word_valence * group | gng_response_type)
contrast_2 <- contrast(emms2, interaction = "revpairwise")


# Get CIs of estimates
contrast_2_ci <- confint(contrast_2, adjust = "none", level = 0.95)[c("lower.CL", "upper.CL")]


# Put contrast and CI in one table (only for FH and FA)
follow_up_2 <- cbind(contrast_2[c(2, 3), ], contrast_2_ci[c(2, 3), ])


# Get adjusted p values
follow_up_2$p.value_adj_fdr <- p.adjust(follow_up_2$p.value, method = "fdr")


# Display table
my_table_template(follow_up_2[, c(3, 1, 2, 4, 9, 10, 7, 8, 11)],
  caption = "Follow-up interaction group * valence nested within response type",
  digits = c(0, 0, 0, 2, 2, 2, 2, 3, 3))


#### 3) Get contrasts for resolving effect of valence nested within group and response type
emms3 <- emmeans(LMM_rt_final, ~ word_valence | gng_response_type * group)
contrast_3 <- contrast(emms3, interaction = "revpairwise")


# Get CIs of estimates
contrast_3_ci <- confint(contrast_3, adjust = "none", level = 0.95)[c("lower.CL", "upper.CL")]


# Put contrast and CI in one table (only for FH and FA)
follow_up_3 <- cbind(contrast_3[c(2, 3, 6, 7), ], contrast_3_ci[c(2, 3, 6, 7), ])


# Get adjusted p values 
follow_up_3$p.value_adj_fdr <- p.adjust(follow_up_3$p.value, method = "fdr")


# Display table
my_table_template(follow_up_3[c(1, 3, 2, 4), c(2, 3, 1, 4, 9, 10, 7, 8, 11)],
  caption = "Follow-up effect of valence nested within group and response type",
  digits = c(0, 0, 0, 2, 2, 2, 2, 3, 3))
```
<br>
Post hoc comparisons indicated that after false alarms, participants across groups categorized negative words faster than positive words. After fast and slow hits, participants categorized positive words faster than negative words. <br><br>
After false alarms, both groups categorized negative words faster than positive words, whereas after fast hits, they categorized positive words faster than negative words. Comparisons conducted to investigate hypothesized group differences yielded that this response facilitation to negative relative to positive words after false alarms was significantly smaller in patients with OCD than in the control group, indicating a reduced priming effect. 
<br><br><br>

### ERPs as Predictors {.tabset}

For testing brain–behavior relations, the single-trial within-participant *z*-standardized ERP was added as predictor. ERN amplitude and CRN amplitude were entered as *z*-standardized continuous predictors in the models including only false alarm and fast hit trials, respectively. 
<br><br>

#### ERN 

<!-- Model specification -->

```{r lmm-rt-ern-model-specification, eval = FALSE}

# Run model with maximal random-effects structure
LMM_rt_ern_max <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg + MFN_standardized + pos_neg:MFN_standardized | participant_id) +
  (1 + OCD_HC + MFN_standardized + OCD_HC:MFN_standardized | word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FA", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output
summary(LMM_rt_ern_max)    # Model does converge
isSingular(LMM_rt_ern_max) # Singular fit


# Run zero-correlation parameter model by using || syntax to reduce model complexity
LMM_rt_ern_red1 <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg + MFN_standardized + pos_neg:MFN_standardized || participant_id) +
  (1 + OCD_HC + MFN_standardized + OCD_HC:MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FA", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
summary(LMM_rt_ern_red1)        # Model does converge
isSingular(LMM_rt_ern_red1)     # Singular fit
summary(rePCA(LMM_rt_ern_red1)) # For words there is 1 term, for participants 2 terms that do not explain variance (< 0.5%)
print(VarCorr(LMM_rt_ern_red1), comp = "Variance") # It is OCD_HC:MFN for words; MFN and pos_neg:MFN for participants
```

<!-- Final model -->

This table corresponds to Table 6 in the manuscript. 

```{r lmm-rt-ern-final-model}

# Run final model without random terms explaining zero variance
LMM_rt_ern_final <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg || participant_id) +
  (1 + OCD_HC + MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FA", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_ern_final)        # Model does converge
# isSingular(LMM_rt_ern_final)     # No singular fit
# summary(rePCA(LMM_rt_ern_final)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_ern_final,
  dv.labels = "RT", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", 
  wrap.labels = 80, title = "LMM Results for Word Categorization RT With ERN as Predictor")


# Display random effects
print("Random effects:")
print(VarCorr(LMM_rt_ern_final), digits = 1, comp = "Std.Dev.")
```
<br>
Analysis of word categorization RT in false alarm trials with ERN as predictor yielded a main effect of word valence, again revealing the priming effect after false alarms, with negative words being categorized faster than positive words. Contrary to our prediction, we found no significant interaction between ERN and word valence. Hence, there was no evidence for a relation between error evaluation (indexed by the RT priming effect) and ERN. Further, there was no difference in this relation between groups. 
<br><br><br>

#### CRN 

<!-- Model specification -->

```{r lmm-rt-crn-model-specification, eval = FALSE}

# Run model with maximal random-effects structure
LMM_rt_crn_max <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg + MFN_standardized + pos_neg:MFN_standardized | participant_id) +
  (1 + OCD_HC + MFN_standardized + OCD_HC:MFN_standardized | word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FH", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output
summary(LMM_rt_crn_max)    # Model does converge
isSingular(LMM_rt_crn_max) # Singular fit


# Run zero-correlation parameter model by using || syntax to reduce model complexity
LMM_rt_crn_red1 <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg + MFN_standardized + pos_neg:MFN_standardized || participant_id) +
  (1 + OCD_HC + MFN_standardized + OCD_HC:MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FH", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
summary(LMM_rt_crn_red1)        # Model does converge
isSingular(LMM_rt_crn_red1)     # Singular fit
summary(rePCA(LMM_rt_crn_red1)) # For words and participants there are each 2 terms that do not explain variance (< 0.5%)
print(VarCorr(LMM_rt_crn_red1), comp = "Variance") # It is OCD_HC and OCD_HC:MFN for words and MFN and pos_neg:MFN for participants
```

<!-- Final model -->

This table corresponds to Table 6 in the manuscript. 

```{r lmm-rt-crn-final-model}

# Run final model without random terms explaining zero variance
LMM_rt_crn_final <- lmer(word_rt_log ~ word_valence * group * MFN_standardized +
  (1 + pos_neg || participant_id) +
  (1 + MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FH", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_crn_final)        # Model does converge
# isSingular(LMM_rt_crn_final)     # No singular fit
# summary(rePCA(LMM_rt_crn_final)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_crn_final,
  dv.labels = "RT", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", 
  wrap.labels = 80, title = "LMM Results for Word Categorization RT With CRN as Predictor")


# Display random effects
print("Random effects:")
print(VarCorr(LMM_rt_crn_final), digits = 1, comp = "Std.Dev.")
```
<br>
Analysis of word categorization RT in fast hit trials including CRN as predictor indicated a main effect of word valence. This again demonstrated the priming effect after fast hits, with positive words being categorized faster than negative words. Moreover, a main effect for CRN was found, with faster RT being related to smaller CRN amplitude. There were no significant interactions between CRN, word valence, and group. Thus, we found no evidence for a relation between action evaluation and CRN and no group difference in this relation. 
<br><br><br>

### Traits as Predictors

To explore effects of trait anxiety, worry, and OCD symptom severity on action evaluation, we included these OCD-related characteristics as predictors. Models were calculated separately for each group since otherwise effects of trait measures are confounded by group differences. Trait measures were group-mean centered. <br><br>
This table corresponds to Table S5 in the supplemental material. 

```{r lmm-rt-traits, cache = knitr_cache_enabled}

# OCI in HC: Run reduced final model
LMM_rt_oci_hc <- lmer(log(word_rt) ~ FA_FH * pos_neg * oci_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "HC", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_oci_hc)        # Model does converge
# isSingular(LMM_rt_oci_hc)     # No singular fit
# summary(rePCA(LMM_rt_oci_hc)) # All terms explain variance


# OCI in OCD: Run reduced final model
LMM_rt_oci_ocd <- lmer(log(word_rt) ~ FA_FH * pos_neg * oci_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "OCD", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_oci_ocd)        # Model does converge
# isSingular(LMM_rt_oci_ocd)     # No singular fit
# summary(rePCA(LMM_rt_oci_ocd)) # All terms explain variance


# STAI in HC: Run reduced final model
LMM_rt_stai_hc <- lmer(log(word_rt) ~ FA_FH * pos_neg * stai_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "HC", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_stai_hc)        # Model does converge
# isSingular(LMM_rt_stai_hc)     # No singular fit
# summary(rePCA(LMM_rt_stai_hc)) # All terms explain variance


# STAI in OCD: Run reduced final model
LMM_rt_stai_ocd <- lmer(log(word_rt) ~ FA_FH * pos_neg * stai_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "OCD", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_stai_ocd)        # Model does converge
# isSingular(LMM_rt_stai_ocd)     # No singular fit
# summary(rePCA(LMM_rt_stai_ocd)) # All terms explain variance


# PSWQ in HC: Run reduced final model
LMM_rt_pswq_hc <- lmer(log(word_rt) ~ FA_FH * pos_neg * pswq_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "HC", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_pswq_hc)        # Model does converge
# isSingular(LMM_rt_pswq_hc)     # No singular fit
# summary(rePCA(LMM_rt_pswq_hc)) # All terms explain variance


# PSWQ in OCD: Run reduced final model
LMM_rt_pswq_ocd <- lmer(log(word_rt) ~ FA_FH * pos_neg * pswq_centered +
  (1 + FA_FH + pos_neg + FA_FH:pos_neg || participant_id) +
  (1 + FA_FH || word),
  data = single_trial_data_rt[(single_trial_data_rt$gng_response_type == "FA" |
                               single_trial_data_rt$gng_response_type == "FH") &
                               single_trial_data_rt$group == "OCD", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_pswq_ocd)        # Model does converge
# isSingular(LMM_rt_pswq_ocd)     # No singular fit
# summary(rePCA(LMM_rt_pswq_ocd)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_oci_ocd, LMM_rt_oci_hc,
  dv.labels = c("RT: OCI-R as predictor in patients with OCD", "RT: OCI-R as predictor in control participants"),
  pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", wrap.labels = 80,
  title = "LMM Results for Word Categorization RT With OCD Symptom Severity (OCI-R) as Predictor")


tab_model(LMM_rt_stai_ocd, LMM_rt_stai_hc,
  dv.labels = c("RT: STAI as predictor in patients with OCD", "RT: STAI as predictor in control participants"),
  pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", wrap.labels = 80,
  title = "LMM Results for Word Categorization RT With Trait Anxiety (STAI) as Predictor")


tab_model(LMM_rt_pswq_ocd, LMM_rt_pswq_hc,
  dv.labels = c("RT: PSWQ as predictor in patients with OCD", "RT: PSWQ as predictor in control participants"),
  pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", wrap.labels = 80,
  title = "LMM Results for Word Categorization RT With Trait Worry (PSWQ) as Predictor")


# Save model output for plot
saveRDS(LMM_rt_oci_ocd,  file = "./saved_objects_for_plots/LMM_rt_oci_ocd.rds")
saveRDS(LMM_rt_stai_ocd, file = "./saved_objects_for_plots/LMM_rt_stai_ocd.rds")


# Note: For all models with traits as predictors, the reduced models were run
# (random effects were specified as uncorrelated due to non-convergence of the
# models with the maximal random-effects structure; random effects explaining
# zero variance [all random slopes of traits for words] were removed)
```
<br>
Exploratory analyses on effects of trait anxiety, worry, and OCD symptom severity on action evaluation within both groups revealed an interaction between OCD symptom severity, word valence, and the response type contrast false alarms compared to fast hits in the patient group. Thus, higher symptom severity in patients was associated with a smaller overall priming effect after fast hits and false alarms. A trend for a similar relation was observed for trait anxiety. There was no significant effect of trait worry.
<br><br><br>

### Correlations

The middle and lower tables correspond to Table S6 and S7 in the supplemental material.  

```{r correlations}

# Calculate overall score for ERPs and priming
df_corr <- single_trial_data_rt %>%
  # Calculate mean per participant
  dplyr::group_by(participant_id) %>%
  dplyr::summarize(
    ERN              = mean(MFN_0_100_FCz[gng_response_type == "FA"], na.rm = TRUE),
    CRN              = mean(MFN_0_100_FCz[gng_response_type == "FH"], na.rm = TRUE),
    pos_after_FA     = mean(word_rt[gng_response_type == "FA" & word_valence == "pos"], na.rm = TRUE),
    neg_after_FA     = mean(word_rt[gng_response_type == "FA" & word_valence == "neg"], na.rm = TRUE),
    pos_after_FH     = mean(word_rt[gng_response_type == "FH" & word_valence == "pos"], na.rm = TRUE),
    neg_after_FH     = mean(word_rt[gng_response_type == "FH" & word_valence == "neg"], na.rm = TRUE),
    priming_after_FA = pos_after_FA - neg_after_FA,
    priming_after_FH = neg_after_FH - pos_after_FH,
    priming          = (pos_after_FA + neg_after_FH) - (neg_after_FA + pos_after_FH)) %>%
  dplyr::ungroup() %>%
  # Merge trait variables
  dplyr::left_join(., traits, by = "participant_id")


# Calculate correlations
corr_priming_FA_ERN   <- cor.test(df_corr$priming_after_FA, df_corr$ERN)
corr_priming_FH_CRN   <- cor.test(df_corr$priming_after_FH, df_corr$CRN)
corr_priming_oci_ocd  <- cor.test(df_corr[df_corr$group == "OCD",]$priming, df_corr[df_corr$group == "OCD",]$oci)
corr_priming_oci_hc   <- cor.test(df_corr[df_corr$group == "HC", ]$priming, df_corr[df_corr$group == "HC", ]$oci)
corr_priming_stai_ocd <- cor.test(df_corr[df_corr$group == "OCD",]$priming, df_corr[df_corr$group == "OCD",]$stai)
corr_priming_stai_hc  <- cor.test(df_corr[df_corr$group == "HC", ]$priming, df_corr[df_corr$group == "HC", ]$stai)
corr_priming_pswq_ocd <- cor.test(df_corr[df_corr$group == "OCD",]$priming, df_corr[df_corr$group == "OCD",]$pswq)
corr_priming_pswq_hc  <- cor.test(df_corr[df_corr$group == "HC", ]$priming, df_corr[df_corr$group == "HC", ]$pswq)
corr_oci_stai_ocd     <- cor.test(df_corr[df_corr$group == "OCD",]$oci,     df_corr[df_corr$group == "OCD",]$stai)
corr_oci_stai_hc      <- cor.test(df_corr[df_corr$group == "HC", ]$oci,     df_corr[df_corr$group == "HC", ]$stai)
corr_oci_pswq_ocd     <- cor.test(df_corr[df_corr$group == "OCD",]$oci,     df_corr[df_corr$group == "OCD",]$pswq)
corr_oci_pswq_hc      <- cor.test(df_corr[df_corr$group == "HC", ]$oci,     df_corr[df_corr$group == "HC", ]$pswq)
corr_stai_pswq_ocd    <- cor.test(df_corr[df_corr$group == "OCD",]$stai,    df_corr[df_corr$group == "OCD",]$pswq)
corr_stai_pswq_hc     <- cor.test(df_corr[df_corr$group == "HC", ]$stai,    df_corr[df_corr$group == "HC", ]$pswq)


# Write correlations priming & ERPs in data frame
corr_erp <- rbind(cbind(corr_priming_FA_ERN$estimate,    corr_priming_FA_ERN$conf.int[1], 
                        corr_priming_FA_ERN$conf.int[2], corr_priming_FA_ERN$p.value),
                  cbind(corr_priming_FH_CRN$estimate,    corr_priming_FH_CRN$conf.int[1], 
                        corr_priming_FH_CRN$conf.int[2], corr_priming_FH_CRN$p.value))


# Write correlations priming & traits in data frame
corr_trait <- rbind(cbind(corr_priming_oci_ocd$estimate,     corr_priming_oci_ocd$conf.int[1], 
                          corr_priming_oci_ocd$conf.int[2],  corr_priming_oci_ocd$p.value,
                          corr_priming_oci_hc$estimate,      corr_priming_oci_hc$conf.int[1], 
                          corr_priming_oci_hc$conf.int[2],   corr_priming_oci_hc$p.value),
                    cbind(corr_priming_stai_ocd$estimate,    corr_priming_stai_ocd$conf.int[1], 
                          corr_priming_stai_ocd$conf.int[2], corr_priming_stai_ocd$p.value,
                          corr_priming_stai_hc$estimate,     corr_priming_stai_hc$conf.int[1], 
                          corr_priming_stai_hc$conf.int[2],  corr_priming_stai_hc$p.value),
                    cbind(corr_priming_pswq_ocd$estimate,    corr_priming_pswq_ocd$conf.int[1], 
                          corr_priming_pswq_ocd$conf.int[2], corr_priming_pswq_ocd$p.value,
                          corr_priming_pswq_hc$estimate,     corr_priming_pswq_hc$conf.int[1], 
                          corr_priming_pswq_hc$conf.int[2],  corr_priming_pswq_hc$p.value))


# Write correlations between questionnaire scores in data frame
corr_quest <- rbind(cbind(corr_oci_stai_ocd$estimate,     corr_oci_stai_ocd$conf.int[1], 
                          corr_oci_stai_ocd$conf.int[2],  corr_oci_stai_ocd$p.value,
                          corr_oci_stai_hc$estimate,      corr_oci_stai_hc$conf.int[1], 
                          corr_oci_stai_hc$conf.int[2],   corr_oci_stai_hc$p.value),
                    cbind(corr_oci_pswq_ocd$estimate,     corr_oci_pswq_ocd$conf.int[1], 
                          corr_oci_pswq_ocd$conf.int[2],  corr_oci_pswq_ocd$p.value,
                          corr_oci_pswq_hc$estimate,      corr_oci_pswq_hc$conf.int[1], 
                          corr_oci_pswq_hc$conf.int[2],   corr_oci_pswq_hc$p.value),
                    cbind(corr_stai_pswq_ocd$estimate,    corr_stai_pswq_ocd$conf.int[1], 
                          corr_stai_pswq_ocd$conf.int[2], corr_stai_pswq_ocd$p.value,
                          corr_stai_pswq_hc$estimate,     corr_stai_pswq_hc$conf.int[1], 
                          corr_stai_pswq_hc$conf.int[2],  corr_stai_pswq_hc$p.value))


# Display correlations priming & ERPs
my_table_template(data.frame(c("ERN & priming after false alarms", "CRN & priming after fast hits"), corr_erp),
  col_names = c("", "r(54)", "LL", "UL", "p"),
  digits = c(0, 2, 2, 2, 3),
  header_above_config = c(" " = 2, "95% CI" = 2, " "),
  caption = "Correlations Between ERPs and the Priming Effect")


# Display correlations priming & traits
my_table_template(data.frame(c("Obsessive-compulsive symptoms (OCI-R)", "Trait anxiety (STAI trait)",
  "Trait worry (PSWQ)"), corr_trait),
  col_names = c("Characteristic", rep(c("r(26)", "LL", "UL", "p"), 2)),
  digits = c(0, 2, 2, 2, 3, 2, 2, 2, 3),
  header_above_config = c(" " = 2, "95% CI" = 2, " " = 2, "95% CI" = 2, " "),
  caption = "Correlations Between OCD-Related Characteristics and the Overall Priming Effect After False 
  Alarms and Fast Hits") %>%
  add_header_above(c("", "Patients with OCD" = 4, "Healthy control participants" = 4)) %>%
  add_footnote("\n Note. Pearson correlation coefficients are reported.", notation = "none")


# Display correlations between questionnaire scores
my_table_template(data.frame(c("OCI-R & STAI trait", "OCI-R & PSWQ", "STAI trait & PSWQ"), corr_quest),
  col_names = c("Questionnaire scores", rep(c("r(26)", "LL", "UL", "p"), 2)),
  digits = c(0, 2, 2, 2, 3, 2, 2, 2, 3),
  header_above_config = c(" " = 2, "95% CI" = 2, " " = 2, "95% CI" = 2, " "),
  caption = "Correlations Between Questionnaire Scores") %>%
  add_header_above(c("", "Patients with OCD" = 4, "Healthy control participants" = 4)) %>%
  add_footnote("\n Note. Pearson correlation coefficients are reported.", notation = "none")


# Save correlation dataframe for scatterplots
save(df_corr, file = "./saved_objects_for_plots/df_corr.Rda")
```
<br>
There was no correlation between the priming effect after false alarms and ERN across participants. Also no correlation was observed between the priming effect after fast hits and CRN across participants.

The correlation analysis revealed a negative correlation between the overall priming effect and symptom severity in patients. A trend for a similar relation was observed for trait anxiety. 
<br><br><br>

## Analysis of Accuracy {.tabset}
***

We performed a binomial generalized linear mixed model (GLMM) on single-trial data. Word categorization accuracy was modeled using type of the preceding go/no-go response (slow hit, fast hit, false alarm, inhibited response), word valence, and group as fixed effects. 

We used sliding difference contrasts for categorical fixed effects and started with the maximal random-effects structure for each model. In case of non-convergence, random effects were specified as uncorrelated. Using principal component analysis, we identified random effects explaining zero variance and removed these, as recommended by Bates, Kliegl, et al. (2015). 
<br><br>

### GLMM

<!-- Contrast coding -->

```{r glmm-accuracy-contrast-coding}

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_acc$gng_response_type) <- contr.sdif(4)
contrasts(single_trial_data_acc$word_valence)      <- contr.sdif(2)
contrasts(single_trial_data_acc$group)             <- contr.sdif(2)


# Add contrasts as numerical covariates via model matrix*
model_matrix_2 <- model.matrix(~ gng_response_type * word_valence * group, single_trial_data_acc)


# Attach the model matrix (16 columns) to the dataframe
single_trial_data_acc[, (ncol(single_trial_data_acc) + 1):(ncol(single_trial_data_acc) + 16)] <- model_matrix_2


# Assign descriptive names to the contrasts
names(single_trial_data_acc)[(ncol(single_trial_data_acc) - 15):ncol(single_trial_data_acc)] <- c(
  "Grand Mean", "FH_SH", "FA_FH", "IR_FA", "pos_neg", "OCD_HC", "FH_SH:pos_neg", "FA_FH:pos_neg",
  "IR_FA:pos_neg", "FH_SH:OCD_HC", "FA_FH:OCD_HC", "IR_FA:OCD_HC", "pos_neg:OCD_HC",
  "FH_SH:pos_neg:OCD_HC", "FA_FH:pos_neg:OCD_HC", "IR_FA:pos_neg:OCD_HC")


# *Note: For the random effects, we needed to enter the separate random effect terms in the models to enable
# double-bar notation (||). This allows fitting a model that sets correlations of the random terms to zero.
```

<!-- Model specification -->

```{r glmm-accuracy-model-specification, eval = FALSE}

# Run model with maximal random-effects structure
GLMM_acc_max <- glmer(word_accuracy_numeric ~ gng_response_type * word_valence * group +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg | participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FH_SH:OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC | word),
  data = single_trial_data_acc,
  family = binomial,
  control = glmerControl(optimizer = "bobyqa"))


# Check model output
summary(GLMM_acc_max)    # Model does not converge
isSingular(GLMM_acc_max) # Singular fit


# Run zero-correlation parameter model by using || syntax to reduce model complexity
GLMM_acc_red1 <- glmer(word_accuracy_numeric ~ gng_response_type * word_valence * group +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FH_SH:OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_acc,
  family = binomial,
  control = glmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
summary(GLMM_acc_red1)        # Model does converge
isSingular(GLMM_acc_red1)     # Singular fit
summary(rePCA(GLMM_acc_red1)) # For words there is 1 term, for participants 2 terms that do not explain variance (< 0.5%)
print(VarCorr(GLMM_acc_red1), comp = "Variance") # It is FH_SH:OCD_HC for words and FH_SH, FH_SH:pos_neg for participants
```

<!-- Final model -->

This table corresponds to Table 3 in the manuscript.

```{r glmm-accuracy-final-model, cache = knitr_cache_enabled}

# Run final model without random terms explaining zero variance
GLMM_acc_final <- glmer(word_accuracy_numeric ~ gng_response_type * word_valence * group +
  (1 + FA_FH + IR_FA + pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_acc,
  family = binomial,
  control = glmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(GLMM_acc_final)        # Model does converge
# isSingular(GLMM_acc_final)     # No singular fit
# summary(rePCA(GLMM_acc_final)) # All terms explain variance


# Display results (fixed effects)
tab_model(GLMM_acc_final,
  dv.labels = "Accuracy", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "Odds ratio", string.stat = "z", string.ci = "95 % CI", wrap.labels = 80,
  title = "GLMM Results for Word Categorization Accuracy")


# Display random effects
print("Random effects:")
print(VarCorr(GLMM_acc_final), digits = 1, comp = "Std.Dev.")


# Save model output for plot
saveRDS(GLMM_acc_final, file = "./saved_objects_for_plots/GLMM_acc_final.rds")
```
<br>
Analysis of categorization accuracy revealed main effects of response type and word valence, which were qualified by significant interactions between response type and word valence and between group and word valence.
<br><br><br>

### Pairwise Comparisons

Significant and hypothesis-relevant interactions were followed up with post hoc comparisons using false discovery rate adjustment. <br><br>
These tables correspond to Table 4 in the manuscript.  

```{r glmm-accuracy-pairwise-comparisons}

#### 1) Get contrasts for effect of word valence within each response type
emms1 <- emmeans(GLMM_acc_final, ~ word_valence | gng_response_type)
contrast_1 <- contrast(emms1, interaction = "revpairwise")


# Get CIs of estimates
contrast_1_ci <- confint(contrast_1, adjust = "none", level = 0.95)[c("asymp.LCL", "asymp.UCL")]


# Merge contrast and CI in one table
follow_up_1 <- cbind(contrast_1, contrast_1_ci)


# Add adjusted p values
follow_up_1$p.value_adj_fdr <- p.adjust(follow_up_1$p.value, method = "fdr")


# Here, estimates and CI are log odds ratios, not like in the other tables odds ratios.
# To get the odds ratios from the log odds ratios, we have to transform the values by using the exp() function
follow_up_1$estimate  <- exp(follow_up_1$estimate)
follow_up_1$asymp.LCL <- exp(follow_up_1$asymp.LCL)
follow_up_1$asymp.UCL <- exp(follow_up_1$asymp.UCL)


# Display table
my_table_template(follow_up_1[, c(2, 1, 3, 8, 9, 6, 7, 10)],
  caption = "Follow-up effect of valence nested within response type",
  digits = c(0, 0, 2, 2, 2, 2, 3, 3))


#### 2) Get contrasts for resolving interaction group * valence nested within response type
emms2 <- emmeans(GLMM_acc_final, ~ word_valence * group | gng_response_type)
contrast_2 <- contrast(emms2, interaction = "revpairwise")


# Get CIs of estimates
contrast_2_ci <- confint(contrast_2, adjust = "none", level = 0.95)[c("asymp.LCL", "asymp.UCL")]


# Put contrast and CI in one table (only for FH and FA)
follow_up_2 <- cbind(contrast_2[c(2, 3), ], contrast_2_ci[c(2, 3), ])


# Get adjusted p values
follow_up_2$p.value_adj_fdr <- p.adjust(follow_up_2$p.value, method = "fdr")


# Here, estimates and CI are log odds ratios, not like in the other tables odds ratios.
# To get the odds ratios from the log odds ratios, we have to transform the values by using the exp() function
follow_up_2$estimate  <- exp(follow_up_2$estimate)
follow_up_2$asymp.LCL <- exp(follow_up_2$asymp.LCL)
follow_up_2$asymp.UCL <- exp(follow_up_2$asymp.UCL)


# Display table
my_table_template(follow_up_2[, c(3, 1, 2, 4, 9, 10, 7, 8, 11)],
  caption = "Follow-up interaction group * valence nested within response type",
  digits = c(0, 0, 0, 2, 2, 2, 2, 3, 3))


#### 3) Get contrasts for resolving effect of valence nested within group and response type
emms3 <- emmeans(GLMM_acc_final, ~ word_valence | gng_response_type * group)
contrast_3 <- contrast(emms3, interaction = "revpairwise")


# Get CIs of estimates
contrast_3_ci <- confint(contrast_3, adjust = "none", level = 0.95)[c("asymp.LCL", "asymp.UCL")]


# Put contrast and CI in one table (only for FH and FA)
follow_up_3 <- cbind(contrast_3[c(2, 3, 6, 7), ], contrast_3_ci[c(2, 3, 6, 7), ])


# Get adjusted p values
follow_up_3$p.value_adj_fdr <- p.adjust(follow_up_3$p.value, method = "fdr")


# Here, estimates and CI are log odds ratios, not like in the other tables odds ratios.
# To get the odds ratios from the log odds ratios, we have to transform the values by using the exp() function
follow_up_3$estimate  <- exp(follow_up_3$estimate)
follow_up_3$asymp.LCL <- exp(follow_up_3$asymp.LCL)
follow_up_3$asymp.UCL <- exp(follow_up_3$asymp.UCL)


# Display table
my_table_template(follow_up_3[c(1, 3, 2, 4), c(2, 3, 1, 4, 9, 10, 7, 8, 11)],
  caption = "Follow-up effect of valence nested within group and response type",
  digits = c(0, 0, 0, 2, 2, 2, 2, 3, 3))
```
<br>
Post hoc comparisons indicated that after false alarms, participants across and within groups categorized negative words more accurately than positive words. This response facilitation to negative relative to positive words after false alarms was smaller in patients with OCD compared to control participants at trend level. 
<br><br><br>

## Control Analysis {.tabset}
***

The study was part of a project comprising one session with active and one with placebo transcranial direct current stimulation in counterbalanced order, which preceded task performance. In the present study, data from the placebo session were analyzed. Thus, measurements of participants who completed the task for the first or second time were included. Importantly, control analyses indicated that task familiarity (i.e., session number) did not impact hypothesis-relevant results.
<br><br>

### RT: Session as Cov. 

This table corresponds to Table S10 in the supplemental material. 

```{r lmm-rt-session, cache = knitr_cache_enabled}

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_rt$session) <- contr.sdif(2)


# Run reduced final model (we get same model when starting with all random effects & correlation parameters)
LMM_rt_session <- lmer(word_rt_log ~ gng_response_type * word_valence * group * session +
  (1 + FH_SH + FA_FH + IR_FA + pos_neg + FH_SH:pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_rt,
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_session)        # Model does converge
# isSingular(LMM_rt_session)     # No singular fit
# summary(rePCA(LMM_rt_session)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_session,
  dv.labels = "RT", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", wrap.labels = 80,
  title = "LMM Results for Word Categorization RT With Session Number as Covariate")
```
<br><br><br>

### RT with ERN & CRN: Session as Cov. 

This table corresponds to Table S11 in the supplemental material. 

``` {r lmm-rt-ern-crn-session}

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_rt$session) <- contr.sdif(2)


# Run reduced final model on ERN (we get same model when starting with all random effects & correlation parameters)
LMM_rt_ern_session <- lmer(word_rt_log ~ word_valence * group * MFN_standardized * session +
  (1 + pos_neg || participant_id) +
  (1 + OCD_HC + MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FA", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_ern_session)        # Model does converge
# isSingular(LMM_rt_ern_session)     # No singular fit
# summary(rePCA(LMM_rt_ern_session)) # All terms explain variance


# Run reduced final model on CRN (we get same model when starting with all random effects & correlation parameters)
LMM_rt_crn_session <- lmer(word_rt_log ~ word_valence * group * MFN_standardized * session +
  (1 + pos_neg || participant_id) +
  (1 + MFN_standardized || word),
  data = single_trial_data_rt[single_trial_data_rt$gng_response_type == "FH", ],
  REML = TRUE,
  control = lmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(LMM_rt_crn_session)        # Model does converge
# isSingular(LMM_rt_crn_session)     # No singular fit
# summary(rePCA(LMM_rt_crn_session)) # All terms explain variance


# Display results (fixed effects)
tab_model(LMM_rt_ern_session, LMM_rt_crn_session,
  dv.labels = c("With ERN as predictor", "With CRN as predictor"), pred.labels = labels, show.stat = TRUE, 
  show.icc = FALSE, show.r2 = FALSE, show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", 
  string.pred = "Fixed effects", string.est = "b", string.stat = "t", string.ci = "95 % CI", p.val = "satterthwaite", 
  wrap.labels = 80, title = "LMM Results for Word Categorization RT With ERN and CRN as Predictor and Session Number 
  as Covariate")
```
<br><br><br>

### Accuracy: Session as Cov. 

This table corresponds to Table S10 in the supplemental material.

```{r glmm-accuracy-session, cache = knitr_cache_enabled}

# Define contrasts (sliding difference contrasts = effect coding for factors with 2 levels)
contrasts(single_trial_data_acc$session) <- contr.sdif(2)


# Run reduced final model (we get same model when starting with all random effects & correlation parameters)
GLMM_acc_session <- glmer(word_accuracy_numeric ~ gng_response_type * word_valence * group * session +
  (1 + FA_FH + IR_FA + pos_neg + FA_FH:pos_neg + IR_FA:pos_neg || participant_id) +
  (1 + FH_SH + FA_FH + IR_FA + OCD_HC + FA_FH:OCD_HC + IR_FA:OCD_HC || word),
  data = single_trial_data_acc,
  family = binomial,
  control = glmerControl(optimizer = "bobyqa"))


# Check model output and PCA of random-effects variance-covariance estimates
# summary(GLMM_acc_session)        # Model does converge
# isSingular(GLMM_acc_session)     # No singular fit
# summary(rePCA(GLMM_acc_session)) # All terms explain variance


# Display results (fixed effects)
tab_model(GLMM_acc_session,
  dv.labels = "Accuracy", pred.labels = labels, show.stat = TRUE, show.icc = FALSE, show.r2 = FALSE,
  show.re.var = FALSE, show.ngroups = FALSE, minus.sign = "-", string.pred = "Fixed effects",
  string.est = "Odds ratio", string.stat = "z", string.ci = "95 % CI", wrap.labels = 80,
  title = "GLMM Results for Word Categorization Accuracy With Session Number as Covariate")
```
<br><br><br>

## References
***
Bates, D., Kliegl, R., Vasishth, S., & Baayen, H. (2015). *Parsimonious mixed models.* arXiv. https://arxiv.org/abs/1506.04967v2 <br><br>
Box, G. E., & Cox, D. R. (1964). An analysis of transformations. *Journal of the Royal Statistical Society: Series B (Methodological), 26*(2), 211–243. https://doi.org/10.1111/j.2517-6161.1964.tb00553.x <br><br>
Morey, R. (2008). Confidence intervals from normalized data: A correction to Cousineau (2005). *Tutorials in Quantitative Methods for Psychology, 4*(2), 61–64. https://doi.org/10.20982/tqmp.04.2.p061 
<br><br><br>

## Session Info
***
```{r session-info}

sessionInfo()
```
