library(miloR)
library(SingleCellExperiment)
library(scater)
library(scran)
library(dplyr)
library(patchwork)
library(MouseThymusAgeing)
library(scuttle)

1 Introduction

We have seen how Milo uses graph neighbourhoods to model cell state abundance differences in an experiment, when comparing 2 groups. However, we are often interested in testing between 2 specific groups in our analysis when our experiment has collected data from \(\gt\) 2 groups. We can focus our analysis to a 2 group comparison and still make use of all of the data for things like dispersion estimation, by using contrasts. For an in-depth use of contrasts we recommend users refer to the limma and edgeR Biostars and Bioconductor community forum threads on the subject. Here I will give an overview of how to use contrasts in the context of a Milo analysis.

2 Load data

We will use the MouseThymusAgeing data package as there are multiple groups that we can compare.

thy.sce <- MouseSMARTseqData() # this function downloads the full SCE object
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## field not found in version - adding
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
## see ?MouseThymusAgeing and browseVignettes('MouseThymusAgeing') for documentation
## loading from cache
thy.sce <- logNormCounts(thy.sce)
thy.sce
## class: SingleCellExperiment 
## dim: 48801 2327 
## metadata(0):
## assays(2): counts logcounts
## rownames(48801): ERCC-00002 ERCC-00003 ... ENSMUSG00000064371
##   ENSMUSG00000064372
## rowData names(6): Geneid Chr ... Strand Length
## colnames(2327): B13.B002229.1_52.1.32.1_S109
##   B13.B002297.1_32.4.52.1_S73 ... P9.B002345.5_52.1.32.1_S93
##   P9.B002450.5_4.52.16.1_S261
## colData names(11): CellID ClusterID ... SubType sizeFactor
## reducedDimNames(1): PCA
## mainExpName: NULL
## altExpNames(0):

3 Define cell neighbourhoods

thy.sce <- runUMAP(thy.sce) # add a UMAP for plotting results later

thy.milo <- Milo(thy.sce) # from the SCE object
reducedDim(thy.milo, "UMAP") <- reducedDim(thy.sce, "UMAP")

plotUMAP(thy.milo, colour_by="SubType") + plotUMAP(thy.milo, colour_by="Age")

These UMAPs shows how the different thymic epithelial cell subtypes and cells from different aged mice are distributed across our single-cell data set. Next we build the KNN graph and define neighbourhoods to quantify cell abundance across our experimental samples.

# we build KNN graph
thy.milo <- buildGraph(thy.milo, k = 11, d = 40)
## Constructing kNN graph with k:11
thy.milo <- makeNhoods(thy.milo, prop = 0.2, k = 11, d=40, refined = TRUE, refinement_scheme="graph") # make nhoods using graph-only as this is faster
## Checking valid object
## Running refined sampling with graph
colData(thy.milo)$Sample <- paste(colData(thy.milo)$SortDay, colData(thy.milo)$Age, sep="_")
thy.milo <- countCells(thy.milo, meta.data = data.frame(colData(thy.milo)), samples="Sample") # make the nhood X sample counts matrix
## Checking meta.data validity
## Counting cells in neighbourhoods
plotNhoodSizeHist(thy.milo)

4 Differential abundance testing with contrasts

Now we have the pieces in place for DA testing to demonstrate how to use contrasts. We will use these contrasts to explicitly define which groups will be compared to each other.

thy.design <- data.frame(colData(thy.milo))[,c("Sample", "SortDay", "Age")]
thy.design <- distinct(thy.design)
rownames(thy.design) <- thy.design$Sample
## Reorder rownames to match columns of nhoodCounts(milo)
thy.design <- thy.design[colnames(nhoodCounts(thy.milo)), , drop=FALSE]
table(thy.design$Age)
## 
## 16wk  1wk 32wk  4wk 52wk 
##    5    5    5    5    5

To demonstrate the use of contrasts we will fit the whole model to the whole data set, but we will compare sequential pairs of time points. I’ll start with week 1 vs.  week 4 to illustrate the syntax.

rownames(thy.design) <- thy.design$Sample
contrast.1 <- c("Age1wk - Age4wk") # the syntax is <VariableName><ConditionLevel> - <VariableName><ControlLevel>

# we need to use the ~ 0 + Variable expression here so that we have all of the levels of our variable as separate columns in our model matrix
da_results <- testNhoods(thy.milo, design = ~ 0 + Age, design.df = thy.design, model.contrasts = contrast.1,
                         fdr.weighting="graph-overlap", norm.method="TMM")
## Using TMM normalisation
## Performing spatial FDR correction with graph-overlap weighting
table(da_results$SpatialFDR < 0.1)
## 
## FALSE  TRUE 
##   331    29

This calculates a Fold-change and corrected P-value for each neighbourhood, which indicates whether there is significant differential abundance between conditions for 29 neighbourhoods.

You will notice that the syntax for the contrasts is quite specific. It starts with the name of the column variable that contains the different group levels; in this case it is the Age variable. We then define the comparison levels as level1 - level2. To understand this syntax we need to consider what we are concretely comparing. In this case we are asking what is the ratio of the average cell count at week1 compared to the average cell count at week 4, where the averaging is across the replicates. The reason we express this as a difference rather than a ratio is because we are dealing with the log fold change.

We can also pass multiple comparisons at the same time, for instance if we wished to compare each sequential pair of time points. This will give us a better intuition behind how to use contrasts to compare multiple groups.

contrast.all <- c("Age1wk - Age4wk", "Age4wk - Age16wk", "Age16wk - Age32wk", "Age32wk - Age52wk")

# this is the edgeR code called by `testNhoods`
model <- model.matrix(~ 0 + Age, data=thy.design)
mod.constrast <- makeContrasts(contrasts=contrast.all, levels=model)

mod.constrast
##          Contrasts
## Levels    Age1wk - Age4wk Age4wk - Age16wk Age16wk - Age32wk Age32wk - Age52wk
##   Age16wk               0               -1                 1                 0
##   Age1wk                1                0                 0                 0
##   Age32wk               0                0                -1                 1
##   Age4wk               -1                1                 0                 0
##   Age52wk               0                0                 0                -1

This shows the contrast matrix. If we want to test each of these comparisons then we need to pass them sequentially to testNhoods, then apply an additional multiple testing correction to the spatial FDR values.

contrast1.res <- testNhoods(thy.milo, design=~ Age, design.df=thy.design, fdr.weighting="graph-overlap", model.contrasts = mod.constrast)
## Using TMM normalisation
## Warning in makeContrasts(contrasts = model.contrasts, levels = x.model):
## Renaming (Intercept) to Intercept
## Performing spatial FDR correction with graph-overlap weighting
head(contrast1.res)
##   logFC.0    logFC.1 logFC.0.1   logFC..1 logFC.0.2 logFC..1.1 logFC.0.3
## 1       0 -9.0296732         0  9.0296732         0  9.0296732         0
## 2       0 -4.4656683         0  4.4656683         0  4.4656683         0
## 3       0 -5.9611863         0  5.9611863         0  5.9611863         0
## 4       0  0.1300642         0 -0.1300642         0 -0.1300642         0
## 5       0 -1.3998244         0  1.3998244         0  1.3998244         0
## 6       0 -6.2382609         0  6.2382609         0  6.2382609         0
##   logFC.0.4  logFC.1.1 logFC.0.5  logFC.1.2 logFC.0.6 logFC..1.2 logFC.0.7
## 1         0 -9.0296732         0 -9.0296732         0  9.0296732         0
## 2         0 -4.4656683         0 -4.4656683         0  4.4656683         0
## 3         0 -5.9611863         0 -5.9611863         0  5.9611863         0
## 4         0  0.1300642         0  0.1300642         0 -0.1300642         0
## 5         0 -1.3998244         0 -1.3998244         0  1.3998244         0
## 6         0 -6.2382609         0 -6.2382609         0  6.2382609         0
##   logFC.0.8 logFC.0.9 logFC.0.10  logFC.1.3 logFC.0.11 logFC..1.3   logCPM
## 1         0         0          0 -9.0296732          0  9.0296732 13.53824
## 2         0         0          0 -4.4656683          0  4.4656683 13.57337
## 3         0         0          0 -5.9611863          0  5.9611863 13.50392
## 4         0         0          0  0.1300642          0 -0.1300642 13.48151
## 5         0         0          0 -1.3998244          0  1.3998244 13.38044
## 6         0         0          0 -6.2382609          0  6.2382609 13.54314
##           F     PValue        FDR Nhood SpatialFDR
## 1 6.3469634 0.01177971 0.02918182     1 0.03140201
## 2 0.4831396 0.48702605 0.60667605     2 0.59419524
## 3 1.6116229 0.21323765 0.30462521     3 0.34989819
## 4 0.9680044 0.32521144 0.43201520     4 0.50625733
## 5 0.7055009 0.42133339 0.53787241     5 0.52361343
## 6 1.9728793 0.16018571 0.24643955     6 0.28370909

This matrix of contrasts will perform a quasi-likelihood F-test over all contrasts, hence a single p-value and spatial FDR are returned. Log fold changes are returned for all contrasts on each level of the Age variable, which gives 20 log-fold change columns for each - this is the default behaviour of glmQLFTest in the edgeR package which is what Milo uses for hypothesis testing. In general, and to avoid confusion, we recommend testing each pair of contrasts separately if these are the comparisons of interest, as shown below.

# compare weeks 4 and 16, with week 4 as the reference.
cont.4vs16.res <- testNhoods(thy.milo, design=~ Age, design.df=thy.design, fdr.weighting="graph-overlap", model.contrasts = mod.constrast[c("Age4wk"), c("Age4wk - Age16wk")])
## Using TMM normalisation
## Warning in makeContrasts(contrasts = model.contrasts, levels = x.model):
## Renaming (Intercept) to Intercept
## Performing spatial FDR correction with graph-overlap weighting
head(cont.4vs16.res)
##        logFC   logCPM         F     PValue        FDR Nhood SpatialFDR
## 1 -9.0296732 13.53824 6.3469634 0.01177971 0.02918182     1 0.03140201
## 2 -4.4656683 13.57337 0.4831396 0.48702605 0.60667605     2 0.59419524
## 3 -5.9611863 13.50392 1.6116229 0.21323765 0.30462521     3 0.34989819
## 4  0.1300642 13.48151 0.9680044 0.32521144 0.43201520     4 0.50625733
## 5 -1.3998244 13.38044 0.7055009 0.42133339 0.53787241     5 0.52361343
## 6 -6.2382609 13.54314 1.9728793 0.16018571 0.24643955     6 0.28370909

Now we have a single logFC which compares nhood abundance between week 4 and week 16.

Contrasts are not limited to these simple pair-wise comparisons, we can also group levels together for comparisons. For instance, imagine we want to know what the effect of the cell counts in the week 1 mice is compared to all other time points.

model <- model.matrix(~ 0 + Age, data=thy.design)
ave.contrast <- c("Age1wk - (Age4wk + Age16wk + Age32wk + Age52wk)/4")
mod.constrast <- makeContrasts(contrasts=ave.contrast, levels=model)

mod.constrast
##          Contrasts
## Levels    Age1wk - (Age4wk + Age16wk + Age32wk + Age52wk)/4
##   Age16wk                                             -0.25
##   Age1wk                                               1.00
##   Age32wk                                             -0.25
##   Age4wk                                              -0.25
##   Age52wk                                             -0.25

In this contrasts matrix we can see that we have taken the average effect over the other time points. Now running this using testNhoods

da_results <- testNhoods(thy.milo, design = ~ 0 + Age, design.df = thy.design, model.contrasts = ave.contrast, fdr.weighting="graph-overlap")
## Using TMM normalisation
## Performing spatial FDR correction with graph-overlap weighting
table(da_results$SpatialFDR < 0.1)
## 
## FALSE  TRUE 
##   280    80
head(da_results)
##        logFC   logCPM         F      PValue        FDR Nhood SpatialFDR
## 1 -1.3250703 13.53824 1.2519174 0.263223905 0.49119990     1 0.57823471
## 2  2.4982439 13.57337 8.4925695 0.003576980 0.02575425     2 0.02647516
## 3  2.4475734 13.50392 8.3965522 0.003770674 0.02610466     3 0.02647516
## 4  0.1180567 13.48151 0.4325074 0.510782194 0.77242085     4 0.87995233
## 5 -0.3441363 13.38044 0.1237031 0.742836770 0.99970029     5 0.96330259
## 6  1.1890675 13.54314 1.7099600 0.191032317 0.37995378     6 0.44288112

The results table In this comparison there are 80 DA nhoods - which we can visualise on a superimposed single-cell UMAP.

thy.milo <- buildNhoodGraph(thy.milo)

plotUMAP(thy.milo, colour_by="SubType") + plotNhoodGraphDA(thy.milo, da_results, alpha=0.1) +
  plot_layout(guides="auto" )

In these side-by-side UMAPs we can see that there is an enrichment of the Perinatal cTEC and Proliferating TEC populations in the 1 week old compared to the other time points.

For a more extensive description of the uses of contrasts please take a look at the edgeR documentation .

Session Info

sessionInfo()
## R version 4.4.0 beta (2024-04-15 r86425)
## Platform: x86_64-pc-linux-gnu
## Running under: Ubuntu 22.04.4 LTS
## 
## Matrix products: default
## BLAS:   /home/biocbuild/bbs-3.19-bioc/R/lib/libRblas.so 
## LAPACK: /usr/lib/x86_64-linux-gnu/lapack/liblapack.so.3.10.0
## 
## locale:
##  [1] LC_CTYPE=en_US.UTF-8       LC_NUMERIC=C              
##  [3] LC_TIME=en_US.UTF-8        LC_COLLATE=en_US.UTF-8    
##  [5] LC_MONETARY=en_US.UTF-8    LC_MESSAGES=en_US.UTF-8   
##  [7] LC_PAPER=en_US.UTF-8       LC_NAME=C                 
##  [9] LC_ADDRESS=C               LC_TELEPHONE=C            
## [11] LC_MEASUREMENT=en_US.UTF-8 LC_IDENTIFICATION=C       
## 
## time zone: America/New_York
## tzcode source: system (glibc)
## 
## attached base packages:
## [1] stats4    stats     graphics  grDevices utils     datasets  methods  
## [8] base     
## 
## other attached packages:
##  [1] MouseThymusAgeing_1.11.0    patchwork_1.2.0            
##  [3] dplyr_1.1.4                 scran_1.32.0               
##  [5] scater_1.32.0               ggplot2_3.5.1              
##  [7] scuttle_1.14.0              SingleCellExperiment_1.26.0
##  [9] SummarizedExperiment_1.34.0 Biobase_2.64.0             
## [11] GenomicRanges_1.56.0        GenomeInfoDb_1.40.0        
## [13] IRanges_2.38.0              S4Vectors_0.42.0           
## [15] BiocGenerics_0.50.0         MatrixGenerics_1.16.0      
## [17] matrixStats_1.3.0           miloR_2.0.0                
## [19] edgeR_4.2.0                 limma_3.60.0               
## [21] BiocStyle_2.32.0           
## 
## loaded via a namespace (and not attached):
##   [1] RColorBrewer_1.1-3        jsonlite_1.8.8           
##   [3] magrittr_2.0.3            magick_2.8.3             
##   [5] ggbeeswarm_0.7.2          farver_2.1.1             
##   [7] rmarkdown_2.26            zlibbioc_1.50.0          
##   [9] vctrs_0.6.5               memoise_2.0.1            
##  [11] DelayedMatrixStats_1.26.0 tinytex_0.50             
##  [13] htmltools_0.5.8.1         S4Arrays_1.4.0           
##  [15] AnnotationHub_3.12.0      curl_5.2.1               
##  [17] BiocNeighbors_1.22.0      SparseArray_1.4.0        
##  [19] sass_0.4.9                bslib_0.7.0              
##  [21] cachem_1.0.8              igraph_2.0.3             
##  [23] mime_0.12                 lifecycle_1.0.4          
##  [25] pkgconfig_2.0.3           rsvd_1.0.5               
##  [27] Matrix_1.7-0              R6_2.5.1                 
##  [29] fastmap_1.1.1             GenomeInfoDbData_1.2.12  
##  [31] digest_0.6.35             numDeriv_2016.8-1.1      
##  [33] colorspace_2.1-0          AnnotationDbi_1.66.0     
##  [35] dqrng_0.3.2               irlba_2.3.5.1            
##  [37] ExperimentHub_2.12.0      RSQLite_2.3.6            
##  [39] beachmat_2.20.0           labeling_0.4.3           
##  [41] filelock_1.0.3            fansi_1.0.6              
##  [43] httr_1.4.7                polyclip_1.10-6          
##  [45] abind_1.4-5               compiler_4.4.0           
##  [47] bit64_4.0.5               withr_3.0.0              
##  [49] BiocParallel_1.38.0       viridis_0.6.5            
##  [51] DBI_1.2.2                 highr_0.10               
##  [53] ggforce_0.4.2             MASS_7.3-60.2            
##  [55] rappdirs_0.3.3            DelayedArray_0.30.0      
##  [57] bluster_1.14.0            gtools_3.9.5             
##  [59] tools_4.4.0               vipor_0.4.7              
##  [61] beeswarm_0.4.0            glue_1.7.0               
##  [63] grid_4.4.0                cluster_2.1.6            
##  [65] generics_0.1.3            gtable_0.3.5             
##  [67] tidyr_1.3.1               BiocSingular_1.20.0      
##  [69] tidygraph_1.3.1           ScaledMatrix_1.12.0      
##  [71] metapod_1.12.0            utf8_1.2.4               
##  [73] XVector_0.44.0            ggrepel_0.9.5            
##  [75] BiocVersion_3.19.1        pillar_1.9.0             
##  [77] stringr_1.5.1             splines_4.4.0            
##  [79] tweenr_2.0.3              BiocFileCache_2.12.0     
##  [81] lattice_0.22-6            FNN_1.1.4                
##  [83] bit_4.0.5                 tidyselect_1.2.1         
##  [85] locfit_1.5-9.9            Biostrings_2.72.0        
##  [87] knitr_1.46                gridExtra_2.3            
##  [89] bookdown_0.39             xfun_0.43                
##  [91] graphlayouts_1.1.1        statmod_1.5.0            
##  [93] stringi_1.8.3             UCSC.utils_1.0.0         
##  [95] yaml_2.3.8                evaluate_0.23            
##  [97] codetools_0.2-20          ggraph_2.2.1             
##  [99] tibble_3.2.1              BiocManager_1.30.22      
## [101] cli_3.6.2                 uwot_0.2.2               
## [103] munsell_0.5.1             jquerylib_0.1.4          
## [105] Rcpp_1.0.12               dbplyr_2.5.0             
## [107] png_0.1-8                 parallel_4.4.0           
## [109] blob_1.2.4                sparseMatrixStats_1.16.0 
## [111] viridisLite_0.4.2         scales_1.3.0             
## [113] purrr_1.0.2               crayon_1.5.2             
## [115] rlang_1.1.3               cowplot_1.1.3            
## [117] KEGGREST_1.44.0