Clustering local indicators of spatial association (LISA) functions
is a methodology for identifying consistent spatial organisation of
multiple cell-types in an unsupervised way. This can be used to enable
the characterization of interactions between multiple cell-types
simultaneously and can complement traditional pairwise analysis. In our
implementation our LISA curves are a localised summary of an L-function
from a Poisson point process model. Our framework lisaClust
can be used to provide a high-level summary of cell-type colocalization
in high-parameter spatial cytometry data, facilitating the
identification of distinct tissue compartments or identification of
complex cellular microenvironments.
To illustrate our lisaClust
framework, we consider a
very simple toy example where two cell-types are completely separated
spatially. We simulate data for two different images.
set.seed(51773)
x <- round(c(
runif(200), runif(200) + 1, runif(200) + 2, runif(200) + 3,
runif(200) + 3, runif(200) + 2, runif(200) + 1, runif(200)
), 4) * 100
y <- round(c(
runif(200), runif(200) + 1, runif(200) + 2, runif(200) + 3,
runif(200), runif(200) + 1, runif(200) + 2, runif(200) + 3
), 4) * 100
cellType <- factor(paste("c", rep(rep(c(1:2), rep(200, 2)), 4), sep = ""))
imageID <- rep(c("s1", "s2"), c(800, 800))
cells <- data.frame(x, y, cellType, imageID)
ggplot(cells, aes(x, y, colour = cellType)) +
geom_point() +
facet_wrap(~imageID) +
theme_minimal()
First we store our data in a SingleCellExperiment
object.
We can then use the convenience function lisaClust
to
simultaneously calculate local indicators of spatial association (LISA)
functions and perform k-means clustering. The number of clusters can be
specified with the k =
parameter. In the example below,
we’ve chosen k = 2
, resulting in a total of 2 clusters. The
cell type column can be specified using the cellType =
argument. By default, lisaClust
uses the column named
cellType
.
The clusters identified by lisaClust
are stored in
colData
of the SingleCellExperiment
object as
a new column called regions
.
SCE <- lisaClust(SCE, k = 2)
colData(SCE) |> head()
## DataFrame with 6 rows and 5 columns
## x y cellType imageID region
## <numeric> <numeric> <factor> <character> <character>
## 1 36.72 38.58 c1 s1 region_2
## 2 61.38 41.29 c1 s1 region_2
## 3 33.59 80.98 c1 s1 region_2
## 4 50.17 64.91 c1 s1 region_2
## 5 82.93 35.60 c1 s1 region_2
## 6 83.13 2.69 c1 s1 region_2
lisaClust
also provides the convenient
hatchingPlot
function to visualise the different regions
that have been demarcated by the clustering. hatchingPlot
outputs a ggplot
object where the regions are marked by
different hatching patterns. In a real biological dataset, this allows
us to plot both regions and cell-types on the same visualization.
In the example below, we can visualise our stimulated data where our
2 cell types have been separated neatly into 2 distinct regions based on
which cell type each region is dominated by. region_2
is
dominated by the red cell type c1
, and
region_1
is dominated by the blue cell type
c2
.
## Using other clustering methods.
While the lisaClust
function is convenient, we have not
implemented an exhaustive suite of clustering methods as it is very easy
to do this yourself. There are just two simple steps.
We can calculate local indicators of spatial association (LISA)
functions using the lisa
function. Here the LISA curves are
a localised summary of an L-function from a Poisson point process model.
The radii that will be calculated over can be set with
Rs
.
lisaCurves <- lisa(SCE, Rs = c(20, 50, 100))
head(lisaCurves)
## 20_c1 20_c2 50_c1 50_c2 100_c1 100_c2
## cell_1 5.556700 -2.764143 15.631209 -6.910357 11.733097 -9.198914
## cell_2 4.833149 -2.764143 13.940407 -6.910357 9.532662 -8.543440
## cell_3 5.918476 -2.764143 9.008588 -6.910357 9.157887 -7.813862
## cell_4 4.109597 -2.764143 11.907928 -6.910357 8.404425 -8.140036
## cell_5 3.024270 -2.764143 10.159278 -6.910357 9.006286 -8.283564
## cell_6 7.986742 -2.764143 8.675070 -6.910357 12.859615 -13.820714
The LISA curves can then be used to cluster the cells. Here we use
k-means clustering. However, other clustering methods like SOM could
also be used. We can store these cell clusters or cell “regions” in our
SingleCellExperiment
object.
# Custom clustering algorithm
kM <- kmeans(lisaCurves, 2)
# Storing clusters into colData
colData(SCE)$custom_region <- paste("region", kM$cluster, sep = "_")
colData(SCE) |> head()
## DataFrame with 6 rows and 6 columns
## x y cellType imageID region custom_region
## <numeric> <numeric> <factor> <character> <character> <character>
## 1 36.72 38.58 c1 s1 region_2 region_2
## 2 61.38 41.29 c1 s1 region_2 region_2
## 3 33.59 80.98 c1 s1 region_2 region_2
## 4 50.17 64.91 c1 s1 region_2 region_2
## 5 82.93 35.60 c1 s1 region_2 region_2
## 6 83.13 2.69 c1 s1 region_2 region_2
Next, we apply our lisaClust
framework to two images of
breast cancer obtained by Keren et al.
(2018).
We will start by reading in the data from the
SpatialDatasets
package as a
SingleCellExperiment
object. Here the data is in a format
consistent with that outputted by CellProfiler.
This data includes annotation of the cell-types of each cell. Hence,
we can move directly to performing k-means clustering on the local
indicators of spatial association (LISA) functions using the
lisaClust
function, remembering to specify the
imageID
, cellType
, and
spatialCoords
columns in colData
. For the
purpose of demonstration, we will be using only images 5 and 6 of the
kerenSPE
dataset.
These regions are stored in colData
and can be
extracted.
colData(kerenSPE)[, c("imageID", "region")] |>
head(20)
## DataFrame with 20 rows and 2 columns
## imageID region
## <character> <character>
## 21154 5 region_4
## 21155 5 region_4
## 21156 5 region_4
## 21157 5 region_3
## 21158 5 region_3
## ... ... ...
## 21169 5 region_3
## 21170 5 region_3
## 21171 5 region_1
## 21172 5 region_3
## 21173 5 region_1
lisaClust
also provides a convenient function,
regionMap
, for examining which cell types are located in
which regions. In this example, we use this to check which cell types
appear more frequently in each region than expected by chance.
Here, we clearly see that healthy epithelial and mesenchymal tissue are highly concentrated in region 1, immune cells are concentrated in regions 2 and 4, whilst tumour cells are concentrated in region 3.
We can further segregate these cells by increasing the number of
clusters, i.e., increasing the parameter k =
in the
lisaClust()
function. For the purposes of demonstration,
let’s take a look at the hatchingPlot
of these regions.
sessionInfo()
## R version 4.4.2 (2024-10-31)
## Platform: x86_64-pc-linux-gnu
## Running under: Ubuntu 24.04.1 LTS
##
## Matrix products: default
## BLAS: /usr/lib/x86_64-linux-gnu/openblas-pthread/libblas.so.3
## LAPACK: /usr/lib/x86_64-linux-gnu/openblas-pthread/libopenblasp-r0.3.26.so; LAPACK version 3.12.0
##
## locale:
## [1] LC_CTYPE=en_US.UTF-8 LC_NUMERIC=C
## [3] LC_TIME=en_US.UTF-8 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: Etc/UTC
## tzcode source: system (glibc)
##
## attached base packages:
## [1] stats4 stats graphics grDevices utils datasets methods
## [8] base
##
## other attached packages:
## [1] SpatialDatasets_1.4.0 SpatialExperiment_1.17.0
## [3] ExperimentHub_2.15.0 AnnotationHub_3.15.0
## [5] BiocFileCache_2.15.1 dbplyr_2.5.0
## [7] SingleCellExperiment_1.29.1 SummarizedExperiment_1.37.0
## [9] Biobase_2.67.0 GenomicRanges_1.59.1
## [11] GenomeInfoDb_1.43.2 IRanges_2.41.2
## [13] S4Vectors_0.45.2 BiocGenerics_0.53.3
## [15] generics_0.1.3 MatrixGenerics_1.19.1
## [17] matrixStats_1.5.0 ggplot2_3.5.1
## [19] spicyR_1.19.3 lisaClust_1.15.6
## [21] BiocStyle_2.35.0
##
## loaded via a namespace (and not attached):
## [1] splines_4.4.2 later_1.4.1
## [3] bitops_1.0-9 filelock_1.0.3
## [5] svgPanZoom_0.3.4 tibble_3.2.1
## [7] polyclip_1.10-7 lifecycle_1.0.4
## [9] Rdpack_2.6.2 rstatix_0.7.2
## [11] lattice_0.22-6 MASS_7.3-64
## [13] MultiAssayExperiment_1.33.4 backports_1.5.0
## [15] magrittr_2.0.3 sass_0.4.9
## [17] rmarkdown_2.29 jquerylib_0.1.4
## [19] yaml_2.3.10 httpuv_1.6.15
## [21] ClassifyR_3.11.4 sp_2.1-4
## [23] spatstat.sparse_3.1-0 DBI_1.2.3
## [25] buildtools_1.0.0 minqa_1.2.8
## [27] RColorBrewer_1.1-3 abind_1.4-8
## [29] purrr_1.0.2 RCurl_1.98-1.16
## [31] tweenr_2.0.3 rappdirs_0.3.3
## [33] GenomeInfoDbData_1.2.13 spatstat.utils_3.1-2
## [35] terra_1.8-10 maketools_1.3.1
## [37] pheatmap_1.0.12 goftest_1.2-3
## [39] simpleSeg_1.9.0 spatstat.random_3.3-2
## [41] svglite_2.1.3 codetools_0.2-20
## [43] DelayedArray_0.33.4 ggforce_0.4.2
## [45] tidyselect_1.2.1 raster_3.6-31
## [47] UCSC.utils_1.3.1 farver_2.1.2
## [49] viridis_0.6.5 lme4_1.1-36
## [51] spatstat.explore_3.3-4 jsonlite_1.8.9
## [53] Formula_1.2-5 survival_3.8-3
## [55] systemfonts_1.2.0 tools_4.4.2
## [57] ggnewscale_0.5.0 Rcpp_1.0.14
## [59] glue_1.8.0 gridExtra_2.3
## [61] SparseArray_1.7.4 xfun_0.50
## [63] mgcv_1.9-1 ggthemes_5.1.0
## [65] EBImage_4.49.0 HDF5Array_1.35.6
## [67] dplyr_1.1.4 shinydashboard_0.7.2
## [69] scam_1.2-18 withr_3.0.2
## [71] numDeriv_2016.8-1.1 BiocManager_1.30.25
## [73] fastmap_1.2.0 ggh4x_0.3.0
## [75] rhdf5filters_1.19.0 boot_1.3-31
## [77] digest_0.6.37 mime_0.12
## [79] R6_2.5.1 colorspace_2.1-1
## [81] tensor_1.5 jpeg_0.1-10
## [83] spatstat.data_3.1-4 RSQLite_2.3.9
## [85] tidyr_1.3.1 data.table_1.16.4
## [87] class_7.3-23 httr_1.4.7
## [89] htmlwidgets_1.6.4 S4Arrays_1.7.1
## [91] pkgconfig_2.0.3 gtable_0.3.6
## [93] blob_1.2.4 XVector_0.47.2
## [95] sys_3.4.3 htmltools_0.5.8.1
## [97] carData_3.0-5 fftwtools_0.9-11
## [99] scales_1.3.0 ggupset_0.4.0
## [101] png_0.1-8 spatstat.univar_3.1-1
## [103] reformulas_0.4.0 knitr_1.49
## [105] reshape2_1.4.4 rjson_0.2.23
## [107] nlme_3.1-166 curl_6.1.0
## [109] nloptr_2.1.1 bdsmatrix_1.3-7
## [111] rhdf5_2.51.2 cachem_1.1.0
## [113] stringr_1.5.1 BiocVersion_3.21.1
## [115] vipor_0.4.7 parallel_4.4.2
## [117] concaveman_1.1.0 AnnotationDbi_1.69.0
## [119] pillar_1.10.1 grid_4.4.2
## [121] vctrs_0.6.5 coxme_2.2-22
## [123] promises_1.3.2 ggpubr_0.6.0
## [125] car_3.1-3 xtable_1.8-4
## [127] beeswarm_0.4.0 evaluate_1.0.3
## [129] magick_2.8.5 cli_3.6.3
## [131] locfit_1.5-9.10 compiler_4.4.2
## [133] rlang_1.1.5 crayon_1.5.3
## [135] ggsignif_0.6.4 labeling_0.4.3
## [137] ggbeeswarm_0.7.2 plyr_1.8.9
## [139] stringi_1.8.4 viridisLite_0.4.2
## [141] nnls_1.6 deldir_2.0-4
## [143] BiocParallel_1.41.0 cytomapper_1.19.0
## [145] lmerTest_3.1-3 munsell_0.5.1
## [147] Biostrings_2.75.3 tiff_0.1-12
## [149] spatstat.geom_3.3-5 V8_6.0.0
## [151] Matrix_1.7-1 bit64_4.6.0-1
## [153] Rhdf5lib_1.29.0 KEGGREST_1.47.0
## [155] shiny_1.10.0 rbibutils_2.3
## [157] broom_1.0.7 memoise_2.0.1
## [159] bslib_0.8.0 bit_4.5.0.1