1 Installation

if (!require("BiocManager")) {
  install.packages("BiocManager")
}
BiocManager::install("spicyR")
# load required packages
library(spicyR)
library(ggplot2)
library(SpatialExperiment)
library(SpatialDatasets)
library(imcRtools)
library(dplyr)
library(survival)

2 Overview

This guide provides step-by-step instructions on how to apply a linear model to multiple segmented and labelled images to assess how the localisation of different cell types changes across different disease conditions.

3 Example data

We use the Keren et al. (2018) breast cancer dataset to compare the spatial distribution of immune cells in individuals with different levels of tumour infiltration (cold and compartmentalised).

The data is stored as a SpatialExperiment object and contains single-cell spatial data from 41 images.

kerenSPE <- SpatialDatasets::spe_Keren_2018()

The cell types in this dataset includes 11 immune cell types (double negative CD3 T cells, CD4 T cells, B cells, monocytes, macrophages, CD8 T cells, neutrophils, natural killer cells, dendritic cells, regulatory T cells), 2 structural cell types (endothelial, mesenchymal), 2 tumour cell types (keratin+ tumour, tumour) and one unidentified category.

4 Linear modelling

To investigate changes in localisation between two different cell types, we measure the level of localisation between two cell types by modelling with the L-function. The L-function is a variance-stabilised K-function given by the equation

\[ \widehat{L_{ij}} (r) = \sqrt{\frac{\widehat{K_{ij}}(r)}{\pi}} \]

with \(\widehat{K_{ij}}\) defined as

\[ \widehat{K_{ij}} (r) = \frac{|W|}{n_i n_j} \sum_{n_i} \sum_{n_j} 1 \{d_{ij} \leq r \} e_{ij} (r) \]

where \(\widehat{K_{ij}}\) summarises the degree of co-localisation of cell type \(j\) with cell type \(i\), \(n_i\) and \(n_j\) are the number of cells of type \(i\) and \(j\), \(|W|\) is the image area, \(d_{ij}\) is the distance between two cells and \(e_{ij} (r)\) is an edge correcting factor.

Specifically, the mean difference between the experimental function and the theoretical function is used as a measure for the level of localisation, defined as

\[ u = \sum_{r' = r_{\text{min}}}^{r_{\text{max}}} \widehat L_{ij, \text{Experimental}} (r') - \widehat L_{ij, \text{Poisson}} (r') \]

where \(u\) is the sum is taken over a discrete range of \(r\) between \(r_{\text{min}}\) and \(r_{\text{max}}\). Differences of the statistic \(u\) between two conditions is modelled using a weighted linear model.

4.1 Test for change in localisation for a specific pair of cells

Firstly, we can test whether one cell type tends to be more localised with another cell type in one condition compared to the other. This can be done using the spicy() function, where we specify the condition parameter.

In this example, we want to see whether or not neutrophils (to) tend to be found around CD8 T cells (from) in compartmentalised tumours compared to cold tumours. Given that there are 3 conditions, we can specify the desired conditions by setting the order of our condition factor. spicy will choose the first level of the factor as the base condition and the second level as the comparison condition. spicy will also naturally coerce the condition column into a factor if it is not already a factor. The column containing cell type annotations can be specified using the cellType argument. By default, spicy uses the column named cellType in the SpatialExperiment object.

spicyTestPair <- spicy(
  kerenSPE,
  condition = "tumour_type",
  from = "CD8_T_cell",
  to = "Neutrophils"
)

topPairs(spicyTestPair)
#>                         intercept coefficient      p.value   adj.pvalue
#> CD8_T_cell__Neutrophils  -109.081    112.0185 2.166646e-05 2.166646e-05
#>                               from          to
#> CD8_T_cell__Neutrophils CD8_T_cell Neutrophils

We obtain a spicy object which details the results of the modelling performed. The topPairs() function can be used to obtain the associated coefficients and p-value.

As the coefficient in spicyTestPair is positive, we find that neutrophils are significantly more likely to be found near CD8 T cells in the compartmentalised tumours group compared to the cold tumour group.

4.2 Test for change in localisation for all pairwise cell combinations

We can perform what we did above for all pairwise combinations of cell types by excluding the from and to parameters in spicy().

spicyTest <- spicy(
  kerenSPE,
  condition = "tumour_type"
)

topPairs(spicyTest)
#>                             intercept coefficient      p.value   adj.pvalue
#> Macrophages__dn_T_CD3       56.446064   -50.08474 1.080273e-07 3.035568e-05
#> dn_T_CD3__Macrophages       54.987151   -48.38664 2.194018e-07 3.082595e-05
#> Macrophages__DC_or_Mono     73.239404   -59.90361 5.224660e-06 4.893765e-04
#> DC_or_Mono__Macrophages     71.777087   -58.46833 7.431172e-06 5.220399e-04
#> dn_T_CD3__dn_T_CD3         -63.786032   100.61010 2.878804e-05 1.208706e-03
#> Neutrophils__dn_T_CD3      -63.141840    69.64356 2.891872e-05 1.208706e-03
#> dn_T_CD3__Neutrophils      -63.133725    70.15508 3.011012e-05 1.208706e-03
#> DC__Macrophages             96.893239   -92.55112 1.801300e-04 5.758129e-03
#> Macrophages__DC             96.896215   -93.25194 1.844241e-04 5.758129e-03
#> CD4_T_cell__Keratin_Tumour  -4.845037   -22.14995 2.834659e-04 7.409016e-03
#>                                   from             to
#> Macrophages__dn_T_CD3      Macrophages       dn_T_CD3
#> dn_T_CD3__Macrophages         dn_T_CD3    Macrophages
#> Macrophages__DC_or_Mono    Macrophages     DC_or_Mono
#> DC_or_Mono__Macrophages     DC_or_Mono    Macrophages
#> dn_T_CD3__dn_T_CD3            dn_T_CD3       dn_T_CD3
#> Neutrophils__dn_T_CD3      Neutrophils       dn_T_CD3
#> dn_T_CD3__Neutrophils         dn_T_CD3    Neutrophils
#> DC__Macrophages                     DC    Macrophages
#> Macrophages__DC            Macrophages             DC
#> CD4_T_cell__Keratin_Tumour  CD4_T_cell Keratin_Tumour

Again, we obtain a spicy object which outlines the result of the linear models performed for each pairwise combination of cell types.

We can also examine the L-function metrics of individual images by using the convenient bind() function on our spicyTest results object.

bind(spicyTest)[1:5, 1:5]
#>   imageID         condition Keratin_Tumour__Keratin_Tumour
#> 1       1             mixed                      -2.300602
#> 2       2             mixed                      -1.989699
#> 3       3 compartmentalised                      11.373530
#> 4       4 compartmentalised                      33.931133
#> 5       5 compartmentalised                      17.922818
#>   dn_T_CD3__Keratin_Tumour B_cell__Keratin_Tumour
#> 1                -5.298543             -20.827279
#> 2               -16.020022               3.025815
#> 3               -21.857447             -24.962913
#> 4               -36.438476             -40.470221
#> 5               -20.816783             -38.138076

The results can be represented as a bubble plot using the signifPlot() function.

signifPlot(
  spicyTest,
  breaks = c(-3, 3, 1),
  marksToPlot = c("Macrophages", "DC_or_Mono", "dn_T_CD3", "Neutrophils",
                  "CD8_T_cell", "Keratin_Tumour")
)

Here, we can observe that the most significant relationships occur between macrophages and double negative CD3 T cells, suggesting that the two cell types are far more dispersed in compartmentalised tumours compared to cold tumours.

To examine a specific cell type-cell type relationship in more detail, we can use spicyBoxplot() and specify either from = "Macrophages" and to = "dn_T_CD3" or rank = 1.

spicyBoxPlot(results = spicyTest, 
             # from = "Macrophages",
             # to = "dn_T_CD3"
             rank = 1)

5 Linear modelling for custom metrics

spicyR can also be applied to custom distance or abundance metrics. A kNN interactions graph can be generated with the function buildSpatialGraph from the imcRtools package. This generates a colPairs object inside of the SpatialExperiment object.

spicyR provides the function convPairs for converting a colPairs object into an abundance matrix by calculating the average number of nearby cells types for every cell type for a given k. For example, if there exists on average 5 neutrophils for every macrophage in image 1, the column Neutrophil__Macrophage would have a value of 5 for image 1.

kerenSPE <- imcRtools::buildSpatialGraph(kerenSPE, 
                                         img_id = "imageID", 
                                         type = "knn", k = 20,
                                        coords = c("x", "y"))

pairAbundances <- convPairs(kerenSPE,
                  colPair = "knn_interaction_graph")

head(pairAbundances["B_cell__B_cell"])
#>    B_cell__B_cell
#> 1      12.7349608
#> 10      0.2777778
#> 11      0.0000000
#> 12      1.3333333
#> 13      1.2200957
#> 14      0.0000000

The custom distance or abundance metrics can then be included in the analysis with the alternateResult parameter. The Statial package contains other custom distance metrics which can be used with spicy.

spicyTestColPairs <- spicy(
  kerenSPE,
  condition = "tumour_type",
  alternateResult = pairAbundances,
  weights = FALSE
)

topPairs(spicyTestColPairs)
#>                            intercept coefficient     p.value adj.pvalue
#> CD8_T_cell__Neutrophils  0.833333333  -0.7592968 0.002645466  0.3291833
#> B_cell__Tumour           0.001937984   0.2602822 0.004872664  0.3291833
#> Other_Immune__NK         0.012698413   0.2612881 0.005673068  0.3291833
#> Unidentified__CD8_T_cell 0.106626794   0.6387339 0.005906526  0.3291833
#> dn_T_CD3__NK             0.004242424   0.2148797 0.006317829  0.3291833
#> CD4_T_cell__Neutrophils  0.036213602   0.2947696 0.007902670  0.3291833
#> Tregs__CD4_T_cell        0.128876212   0.5726201 0.010207087  0.3291833
#> Endothelial__DC          0.008771930   0.3008523 0.011189533  0.3291833
#> Tumour__Neutrophils      0.021638939   0.2529045 0.011388850  0.3291833
#> Mesenchymal__Neutrophils 0.004504505   0.2494301 0.012761315  0.3291833
#>                                  from          to
#> CD8_T_cell__Neutrophils    CD8_T_cell Neutrophils
#> B_cell__Tumour                 B_cell      Tumour
#> Other_Immune__NK         Other_Immune          NK
#> Unidentified__CD8_T_cell Unidentified  CD8_T_cell
#> dn_T_CD3__NK                 dn_T_CD3          NK
#> CD4_T_cell__Neutrophils    CD4_T_cell Neutrophils
#> Tregs__CD4_T_cell               Tregs  CD4_T_cell
#> Endothelial__DC           Endothelial          DC
#> Tumour__Neutrophils            Tumour Neutrophils
#> Mesenchymal__Neutrophils  Mesenchymal Neutrophils
signifPlot(
  spicyTestColPairs,
  breaks = c(-3, 3, 1),
  marksToPlot = c("Macrophages", "dn_T_CD3", "CD4_T_cell", 
                  "B_cell", "DC_or_Mono", "Neutrophils", "CD8_T_cell")
)

6 Performing survival analysis

spicy can also be used to perform survival analysis to asses whether changes in co-localisation between cell types are associated with survival probability. spicy requires the SingleCellExperiment object being used to contain a column called survival as a Surv object.

kerenSPE$event = 1 - kerenSPE$Censored
kerenSPE$survival = Surv(kerenSPE$`Survival_days_capped*`, kerenSPE$event)

We can then perform survival analysis using the spicy function by specifying condition = "survival". We can then access the corresponding coefficients and p-values by accessing the survivalResults slot in the spicy results object.

# Running survival analysis
spicySurvival = spicy(kerenSPE,
                      condition = "survival")

# top 10 significant pairs
head(spicySurvival$survivalResults, 10)
#> # A tibble: 10 × 4
#>    test                       coef se.coef    p.value
#>    <chr>                     <dbl>   <dbl>      <dbl>
#>  1 Other_Immune__Tregs     0.0236  0.00866 0.00000893
#>  2 CD4_T_cell__Tregs       0.0177  0.00685 0.0000124 
#>  3 Tregs__Other_Immune     0.0237  0.00873 0.0000126 
#>  4 Tregs__CD4_T_cell       0.0171  0.00676 0.0000285 
#>  5 CD8_T_cell__CD8_T_cell  0.00605 0.00272 0.000332  
#>  6 Tumour__CD8_T_cell     -0.0305  0.0114  0.000617  
#>  7 CD8_T_cell__Tumour     -0.0305  0.0116  0.000721  
#>  8 CD4_T_cell__dn_T_CD3    0.00845 0.00353 0.000794  
#>  9 dn_T_CD3__CD4_T_cell    0.00840 0.00353 0.000937  
#> 10 DC__Other_Immune       -0.0289  0.0123  0.00103

7 Accounting for tissue inhomogeneity

The spicy function can also account for tissue inhomogeneity to avoid false positives or negatives. This can be done by setting the sigma = parameter within the spicy function. By default, sigma is set to NULL, and spicy assumes a homogeneous tissue structure.

For example, when we examine the L-function for Keratin_Tumour__Neutrophils when sigma = NULL and Rs = 100, the value is positive, indicating attraction between the two cell types.

# filter SPE object to obtain image 24 data
kerenSubset = kerenSPE[, colData(kerenSPE)$imageID == "24"]

pairwiseAssoc = getPairwise(kerenSubset, 
                            sigma = NULL, 
                            Rs = 100) |>
  as.data.frame()

pairwiseAssoc[["Keratin_Tumour__Neutrophils"]]
#> [1] 10.88892

When we specify sigma = 20 and re-calculate the L-function, it indicates that there is no relationship between Keratin_Tumour and Neutrophils, i.e., there is no major attraction or dispersion, as it now takes into account tissue inhomogeneity.

pairwiseAssoc = getPairwise(kerenSubset, 
                            sigma = 20, 
                            Rs = 100) |>
  as.data.frame()

pairwiseAssoc[["Keratin_Tumour__Neutrophils"]]
#> [1] 0.9024836
# obtain colData for image 24
cData = colData(kerenSPE) |> as.data.frame() |> 
          dplyr::filter(imageID == "24")

# obtain cells present in image 24
coords = spatialCoords(kerenSPE) |> as.data.frame()
coords$cellID = rownames(coords)
coords = coords |> dplyr::filter(cellID %in% cData$CellID)

cData$X = coords$x
cData$Y = coords$y

cData = cData |> 
  dplyr::mutate(cellTypeNew = ifelse(cellType %in% c("Keratin_Tumour", "Neutrophils"), 
                                     cellType, "Other"))

 pal = setNames(c("#d6b11c", "#850f07"), 
                c("Keratin_Tumour", "Neutrophils"))

ggplot() +
    stat_density_2d(data = cData, aes(x = X, y = Y, fill = after_stat(density)), 
                    geom = "raster", 
                    contour = FALSE) +
    geom_point(data = cData |> filter(cellType != "Other"),
               aes(x = X, y = Y, colour = cellTypeNew), size = 1) +
    scale_color_manual(values = pal) +
    scale_fill_distiller(palette = "Blues", direction = 1) +
    theme_classic() +
    labs(title = "image ID: 24")

Plotting image 24 shows that the supposed co-localisation occurs due to the dense cluster of cells near the top of the image.

8 Mixed effects modelling

spicyR supports mixed effects modelling when multiple images are obtained for each subject. In this case, subject is treated as a random effect and condition is treated as a fixed effect. To perform mixed effects modelling, we can specify the subject parameter in the spicy function.

spicyMixedTest <- spicy(
  diabetesData,
  condition = "stage",
  subject = "case"
)

9 References

Keren, L, M Bosse, D Marquez, R Angoshtari, S Jain, S Varma, S Yang, et al. 2018. “A Structured Tumor-Immune Microenvironment in Triple Negative Breast Cancer Revealed by Multiplexed Ion Beam Imaging.” Cell 174 (6): 1373–1387.e19.

Appendix

sessionInfo()
#> R Under development (unstable) (2024-10-21 r87258)
#> Platform: x86_64-pc-linux-gnu
#> Running under: Ubuntu 24.04.1 LTS
#> 
#> Matrix products: default
#> BLAS:   /home/biocbuild/bbs-3.21-bioc/R/lib/libRblas.so 
#> LAPACK: /usr/lib/x86_64-linux-gnu/lapack/liblapack.so.3.12.0
#> 
#> locale:
#>  [1] LC_CTYPE=en_US.UTF-8       LC_NUMERIC=C              
#>  [3] LC_TIME=en_GB              LC_COLLATE=C              
#>  [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] survival_3.7-0              dplyr_1.1.4                
#>  [3] imcRtools_1.13.0            SpatialDatasets_1.5.0      
#>  [5] ExperimentHub_2.15.0        AnnotationHub_3.15.0       
#>  [7] BiocFileCache_2.15.0        dbplyr_2.5.0               
#>  [9] SpatialExperiment_1.17.0    SingleCellExperiment_1.29.1
#> [11] SummarizedExperiment_1.37.0 Biobase_2.67.0             
#> [13] GenomicRanges_1.59.1        GenomeInfoDb_1.43.1        
#> [15] IRanges_2.41.1              S4Vectors_0.45.2           
#> [17] BiocGenerics_0.53.3         generics_0.1.3             
#> [19] MatrixGenerics_1.19.0       matrixStats_1.4.1          
#> [21] ggplot2_3.5.1               spicyR_1.19.3              
#> [23] BiocStyle_2.35.0           
#> 
#> loaded via a namespace (and not attached):
#>   [1] spatstat.sparse_3.1-0       bitops_1.0-9               
#>   [3] sf_1.0-19                   EBImage_4.49.0             
#>   [5] httr_1.4.7                  RColorBrewer_1.1-3         
#>   [7] numDeriv_2016.8-1.1         tools_4.5.0                
#>   [9] backports_1.5.0             utf8_1.2.4                 
#>  [11] R6_2.5.1                    DT_0.33                    
#>  [13] HDF5Array_1.35.1            mgcv_1.9-1                 
#>  [15] rhdf5filters_1.19.0         withr_3.0.2                
#>  [17] sp_2.1-4                    gridExtra_2.3              
#>  [19] coxme_2.2-22                ClassifyR_3.11.0           
#>  [21] cli_3.6.3                   spatstat.explore_3.3-3     
#>  [23] labeling_0.4.3              sass_0.4.9                 
#>  [25] nnls_1.6                    spatstat.data_3.1-4        
#>  [27] readr_2.1.5                 proxy_0.4-27               
#>  [29] systemfonts_1.1.0           ggupset_0.4.0              
#>  [31] svglite_2.1.3               RSQLite_2.3.8              
#>  [33] vroom_1.6.5                 spatstat.random_3.3-2      
#>  [35] car_3.1-3                   scam_1.2-17                
#>  [37] Matrix_1.7-1                ggbeeswarm_0.7.2           
#>  [39] fansi_1.0.6                 abind_1.4-8                
#>  [41] terra_1.7-83                lifecycle_1.0.4            
#>  [43] yaml_2.3.10                 carData_3.0-5              
#>  [45] rhdf5_2.51.0                SparseArray_1.7.2          
#>  [47] grid_4.5.0                  blob_1.2.4                 
#>  [49] promises_1.3.0              crayon_1.5.3               
#>  [51] bdsmatrix_1.3-7             shinydashboard_0.7.2       
#>  [53] lattice_0.22-6              beachmat_2.23.1            
#>  [55] KEGGREST_1.47.0             magick_2.8.5               
#>  [57] cytomapper_1.19.0           pillar_1.9.0               
#>  [59] knitr_1.49                  RTriangle_1.6-0.14         
#>  [61] rjson_0.2.23                boot_1.3-31                
#>  [63] codetools_0.2-20            glue_1.8.0                 
#>  [65] spatstat.univar_3.1-1       data.table_1.16.2          
#>  [67] MultiAssayExperiment_1.33.1 vctrs_0.6.5                
#>  [69] png_0.1-8                   gtable_0.3.6               
#>  [71] cachem_1.1.0                xfun_0.49                  
#>  [73] S4Arrays_1.7.1              mime_0.12                  
#>  [75] tidygraph_1.3.1             pheatmap_1.0.12            
#>  [77] tinytex_0.54                units_0.8-5                
#>  [79] nlme_3.1-166                bit64_4.5.2                
#>  [81] filelock_1.0.3              bslib_0.8.0                
#>  [83] svgPanZoom_0.3.4            KernSmooth_2.23-24         
#>  [85] vipor_0.4.7                 colorspace_2.1-1           
#>  [87] DBI_1.2.3                   raster_3.6-30              
#>  [89] tidyselect_1.2.1            bit_4.5.0                  
#>  [91] compiler_4.5.0              curl_6.0.1                 
#>  [93] BiocNeighbors_2.1.0         DelayedArray_0.33.2        
#>  [95] bookdown_0.41               scales_1.3.0               
#>  [97] classInt_0.4-10             distances_0.1.11           
#>  [99] rappdirs_0.3.3              tiff_0.1-12                
#> [101] stringr_1.5.1               digest_0.6.37              
#> [103] goftest_1.2-3               fftwtools_0.9-11           
#> [105] spatstat.utils_3.1-1        minqa_1.2.8                
#> [107] rmarkdown_2.29              XVector_0.47.0             
#> [109] htmltools_0.5.8.1           pkgconfig_2.0.3            
#> [111] jpeg_0.1-10                 lme4_1.1-35.5              
#> [113] fastmap_1.2.0               rlang_1.1.4                
#> [115] htmlwidgets_1.6.4           ggthemes_5.1.0             
#> [117] UCSC.utils_1.3.0            shiny_1.9.1                
#> [119] ggh4x_0.2.8                 farver_2.1.2               
#> [121] jquerylib_0.1.4             jsonlite_1.8.9             
#> [123] BiocParallel_1.41.0         RCurl_1.98-1.16            
#> [125] magrittr_2.0.3              Formula_1.2-5              
#> [127] scuttle_1.17.0              GenomeInfoDbData_1.2.13    
#> [129] Rhdf5lib_1.29.0             munsell_0.5.1              
#> [131] Rcpp_1.0.13-1               ggnewscale_0.5.0           
#> [133] viridis_0.6.5               stringi_1.8.4              
#> [135] ggraph_2.2.1                zlibbioc_1.53.0            
#> [137] MASS_7.3-61                 plyr_1.8.9                 
#> [139] parallel_4.5.0              ggrepel_0.9.6              
#> [141] deldir_2.0-4                Biostrings_2.75.1          
#> [143] graphlayouts_1.2.1          splines_4.5.0              
#> [145] tensor_1.5                  hms_1.1.3                  
#> [147] locfit_1.5-9.10             igraph_2.1.1               
#> [149] ggpubr_0.6.0                spatstat.geom_3.3-4        
#> [151] ggsignif_0.6.4              reshape2_1.4.4             
#> [153] BiocVersion_3.21.1          evaluate_1.0.1             
#> [155] BiocManager_1.30.25         nloptr_2.1.1               
#> [157] tzdb_0.4.0                  tweenr_2.0.3               
#> [159] httpuv_1.6.15               tidyr_1.3.1                
#> [161] purrr_1.0.2                 polyclip_1.10-7            
#> [163] ggforce_0.4.2               broom_1.0.7                
#> [165] xtable_1.8-4                e1071_1.7-16               
#> [167] rstatix_0.7.2               later_1.3.2                
#> [169] class_7.3-22                viridisLite_0.4.2          
#> [171] tibble_3.2.1                lmerTest_3.1-3             
#> [173] memoise_2.0.1               beeswarm_0.4.0             
#> [175] AnnotationDbi_1.69.0        concaveman_1.1.0