In this post, I will document how I conduct planned contrasts analyses to test whether experimental groups differ from each other. Planned contrasts are similar to t tests, but provide more power when you have several groups:
Statistical power is lower with the standard t test compared than it is with the planned contrast version for two reasons: a) the sample size is smaller with the t test, because only the cases in the two groups are selected; and b) in the planned contrast the error term is smaller than it is with the standard t test because it is based on all the cases (source)
I will first demonstrate a simple example for conducting the analysis, exporting the table of results to Microsoft Word, and producing the figure. After demonstrating it for a single variable, we will then look at a full workflow, i.e., how we can automatize the process for several variables simultaneously.
For the minimal example, we will use the iris
dataset,
which is installed with R
by default.
nice_contrasts()
Load the rempsyc
package:
library(rempsyc)
Note: If you haven’t installed this package yet, you will need to install it via the following command:
install.packages("rempsyc")
.
Let’s test out the function on existing data
<- nice_contrasts(response = "Sepal.Length",
table.stats group = "Species",
data = iris)
table.stats
## Dependent Variable Comparison df t p
## 1 Sepal.Length setosa - virginica 147 -15.365506 2.214821e-32
## 2 Sepal.Length versicolor - virginica 147 -6.332686 2.765638e-09
## 3 Sepal.Length setosa - versicolor 147 -9.032819 8.770194e-16
## dR CI_lower CI_upper
## 1 -2.929466 -3.532345 -2.2135086
## 2 -1.205879 -1.747363 -0.6614399
## 3 -1.723587 -2.169764 -1.2079653
This will give us the ‘robust’ version of Cohen’s d (i.e., robust to deviations from parametric assumptions). It will also provide us with bootstrapped 95% confidence intervals of that effect size.
nice_table()
.<- nice_table(table.stats)) (my_table
Dependent Variable | Comparison | df | t | p | dR | 95% CI |
Sepal.Length | setosa - virginica | 147 | -15.37 | < .001 | -2.93 | [-3.53, -2.21] |
Sepal.Length | versicolor - virginica | 147 | -6.33 | < .001 | -1.21 | [-1.75, -0.66] |
Sepal.Length | setosa - versicolor | 147 | -9.03 | < .001 | -1.72 | [-2.17, -1.21] |
Save the table to Word
save_as_docx(my_table, path = "contrasts.docx")
nice_violin()
.<- nice_violin(group = "Species",
(figure response = "Sepal.Length",
data = iris,
ytitle = "Length of Sepal",
signif_annotation = c("***", "***", "***"),
signif_yposition = c(8.7, 7.3, 8.2),
signif_xmin = c("setosa", "setosa", "versicolor"),
signif_xmax = c("virginica", "versicolor", "virginica")))
Let’s save a high-resolution/vector version of this figure in PDF (it could be .png or any other format too if necessary).
::ggsave('Figure 1.pdf', figure, width = 7, height = 7,
ggplot2unit = 'in', dpi = 300)
Congratulations for making it that far! You’ve done it! From A to Z, you now know how to make things work! If you would like to integrate this with a broader workflow, read on for the next part!
For the full workflow, we will use the dataset from one of my studies published at the Quarterly Journal of Experimental Psychology and available on the Open Science Framework.
Load the data file from the Open Science Framework
<- read.csv("https://osf.io/qkmnp//?action=download", header=TRUE) data
Specify the order of factor levels for “Group” (otherwise R will alphabetize them)
$Group <- factor(data$Group, levels = c("Embodied", "Mental", "Control"))) (data
## [1] Embodied Embodied Control Control Mental Embodied Control Control
## [9] Control Embodied Mental Mental Control Control Mental Control
## [17] Mental Mental Embodied Embodied Control Embodied Mental Control
## [25] Embodied Embodied Mental Control Mental Embodied Embodied Mental
## [33] Mental Mental Embodied Mental Embodied Mental Control Embodied
## [41] Mental Mental Embodied Mental Mental Embodied Control Control
## [49] Control Mental Embodied Control Embodied Control Mental Embodied
## [57] Embodied Embodied Control Control Control Control Mental Control
## [65] Mental Embodied Embodied Mental Embodied Control Control Mental
## [73] Mental Control Control Embodied Mental Control Embodied Embodied
## [81] Embodied Mental Control Control Embodied Mental Mental Mental
## [89] Embodied Control
## Levels: Embodied Mental Control
Define our dependent variables
library(dplyr)
<- data %>% select(QCAEPR:IOS) %>% names) (DV
## [1] "QCAEPR" "IRIPT" "IRIFS" "IRIEC" "IRIPD" "IOS"
<- nice_contrasts(response = DV,
table.stats group = "Group",
data = data)
table.stats
## Dependent Variable Comparison df t p
## 1 QCAEPR Embodied - Control 87 2.19599050 3.075313e-02
## 2 QCAEPR Mental - Control 87 0.23740438 8.129013e-01
## 3 QCAEPR Embodied - Mental 87 1.95858612 5.336471e-02
## 4 IRIPT Embodied - Control 86 0.12249596 9.027921e-01
## 5 IRIPT Mental - Control 86 0.01932946 9.846231e-01
## 6 IRIPT Embodied - Mental 86 0.10212396 9.188960e-01
## 7 IRIFS Embodied - Control 86 0.15139155 8.800214e-01
## 8 IRIFS Mental - Control 86 -1.38629687 1.692400e-01
## 9 IRIFS Embodied - Mental 86 1.53639996 1.281114e-01
## 10 IRIEC Embodied - Control 86 2.30118373 2.379970e-02
## 11 IRIEC Mental - Control 86 1.44763512 1.513548e-01
## 12 IRIEC Embodied - Mental 86 0.83396371 4.066119e-01
## 13 IRIPD Embodied - Control 86 2.13034255 3.599933e-02
## 14 IRIPD Mental - Control 86 0.85841114 3.930517e-01
## 15 IRIPD Embodied - Mental 86 1.25380051 2.133121e-01
## 16 IOS Embodied - Control 87 4.70778837 9.385982e-06
## 17 IOS Mental - Control 87 2.12312024 3.658408e-02
## 18 IOS Embodied - Mental 87 2.58466812 1.141344e-02
## dR CI_lower CI_upper
## 1 0.52518512 -4.742695e-02 1.1547655
## 2 0.03643629 -5.027216e-01 0.5938155
## 3 0.48874884 4.381093e-03 1.0323659
## 4 -0.06831227 -6.440638e-01 0.5023384
## 5 -0.17448179 -7.425397e-01 0.2860143
## 6 0.10616953 -4.590608e-01 0.6829533
## 7 0.09489742 -4.041887e-01 0.6836456
## 8 -0.35461669 -8.623792e-01 0.1856774
## 9 0.44951412 -1.325340e-01 1.1282090
## 10 0.58908525 -2.452403e-16 1.1869188
## 11 0.42094553 -9.351874e-02 0.9914871
## 12 0.16813972 -4.117783e-01 0.7279966
## 13 0.47999336 -5.416202e-02 1.0859253
## 14 0.18562672 -2.620938e-01 0.6725473
## 15 0.29436664 -2.962480e-01 0.9615316
## 16 1.08839542 4.482234e-01 1.8962878
## 17 0.49918888 -1.210932e-02 1.0376966
## 18 0.58920655 2.198890e-02 1.2444935
Add more meaningful names for measures
1] <- rep(c("Peripheral Responsivity (QCAE)",
table.stats["Perspective-Taking (IRI)",
"Fantasy (IRI)", "Empathic Concern (IRI)",
"Personal Distress (IRI)",
"Inclusion of Other in the Self (IOS)"), each = 3)
Note that with option highlight = TRUE
, it will
automatically highlight significant results.
<- nice_table(table.stats, highlight = TRUE)) (my_table
Dependent Variable | Comparison | df | t | p | dR | 95% CI |
Peripheral Responsivity (QCAE) | Embodied - Control | 87 | 2.20 | .031 | 0.53 | [-0.05, 1.15] |
Peripheral Responsivity (QCAE) | Mental - Control | 87 | 0.24 | .813 | 0.04 | [-0.50, 0.59] |
Peripheral Responsivity (QCAE) | Embodied - Mental | 87 | 1.96 | .053 | 0.49 | [0.00, 1.03] |
Perspective-Taking (IRI) | Embodied - Control | 86 | 0.12 | .903 | -0.07 | [-0.64, 0.50] |
Perspective-Taking (IRI) | Mental - Control | 86 | 0.02 | .985 | -0.17 | [-0.74, 0.29] |
Perspective-Taking (IRI) | Embodied - Mental | 86 | 0.10 | .919 | 0.11 | [-0.46, 0.68] |
Fantasy (IRI) | Embodied - Control | 86 | 0.15 | .880 | 0.09 | [-0.40, 0.68] |
Fantasy (IRI) | Mental - Control | 86 | -1.39 | .169 | -0.35 | [-0.86, 0.19] |
Fantasy (IRI) | Embodied - Mental | 86 | 1.54 | .128 | 0.45 | [-0.13, 1.13] |
Empathic Concern (IRI) | Embodied - Control | 86 | 2.30 | .024 | 0.59 | [-0.00, 1.19] |
Empathic Concern (IRI) | Mental - Control | 86 | 1.45 | .151 | 0.42 | [-0.09, 0.99] |
Empathic Concern (IRI) | Embodied - Mental | 86 | 0.83 | .407 | 0.17 | [-0.41, 0.73] |
Personal Distress (IRI) | Embodied - Control | 86 | 2.13 | .036 | 0.48 | [-0.05, 1.09] |
Personal Distress (IRI) | Mental - Control | 86 | 0.86 | .393 | 0.19 | [-0.26, 0.67] |
Personal Distress (IRI) | Embodied - Mental | 86 | 1.25 | .213 | 0.29 | [-0.30, 0.96] |
Inclusion of Other in the Self (IOS) | Embodied - Control | 87 | 4.71 | < .001 | 1.09 | [0.45, 1.90] |
Inclusion of Other in the Self (IOS) | Mental - Control | 87 | 2.12 | .037 | 0.50 | [-0.01, 1.04] |
Inclusion of Other in the Self (IOS) | Embodied - Mental | 87 | 2.58 | .011 | 0.59 | [0.02, 1.24] |
Save the table to Word
save_as_docx(my_table, path = "contrasts.docx")
nice_violin
.We will have to make the four figures for which comparisons were significant and then combine them in one plot.
This function will throw an error if your dataset contains missing data. We will omit those rows missing data for now.
<- na.omit(data) Data
Make figure panel 1
<- nice_violin(group = "Group",
(EC response = "IRIEC",
data = Data,
colours = c("#00BA38", "#619CFF", "#F8766D"),
ytitle = "Empathic Concern (from IRI)",
signif_annotation = "*",
signif_yposition = 5.2,
signif_xmin = 1,
signif_xmax = 3))
Make figure panel 2
<- nice_violin(group = "Group",
(PD response = "IRIPD",
data = Data,
colours = c("#00BA38", "#619CFF", "#F8766D"),
ytitle = "Personal Distress (from IRI)",
signif_annotation = "*",
signif_yposition = 5,
signif_xmin = 1,
signif_xmax = 3))
Make figure panel 3
<- nice_violin(group = "Group",
(PR response = "QCAEPR",
data = Data,
colours = c("#00BA38", "#619CFF", "#F8766D"),
ytitle = "Peripheral Responsivity (from QCAE)",
signif_annotation="*",
signif_yposition=4.2,
signif_xmin=1,
signif_xmax=3))
Make figure panel 4
<- nice_violin(group = "Group",
(IOS response = "IOS",
data = Data,
colours = c("#00BA38", "#619CFF", "#F8766D"),
ytitle = "Self-Other Merging (from IOS)",
signif_annotation=c("***", "*", "*"),
signif_yposition=c(8.25, 7.5, 6.75),
signif_xmin=c(1,1,2),
signif_xmax=c(3,2,3)))
It’s now time to combine our four plots into one figure! Yeah!
library(ggpubr)
<- ggarrange(EC, PD, PR, IOS,
(figure labels = "AUTO",
ncol = 2, nrow = 2))
Let’s save a high-resolution/vector version of this figure in PDF (it could be .png or any other format too if necessary).
::ggsave('Figure 1.pdf', figure, width = 14, height = 14,
ggplot2unit = 'in', dpi = 300)
Make sure to check out this page again if you use the code after a time or if you encounter errors, as I periodically update or improve the code. Feel free to contact me for comments, questions, or requests to improve this function at https://github.com/rempsyc/rempsyc/issues. See all tutorials here: https://remi-theriault.com/tutorials.