A brief overview of the tidySpatialExperiment package - demonstrating the SpatialExperiment-tibble abstraction, compatibility with the tidyverse ecosystem, compatibility with the tidyomics ecosystem and a few helpful utility functions.
tidySpatialExperiment 1.5.0
tidySpatialExperiment provides a bridge between the
SpatialExperiment
package and the tidyverse ecosystem. It
creates an invisible layer that allows you to interact with a
SpatialExperiment object as if it were a tibble; enabling the use of
functions from dplyr,
tidyr,
ggplot2 and
plotly. But, underneath, your data
remains a SpatialExperiment object.
tidySpatialExperiment also provides five additional utility functions.
If you would like to learn more about tidySpatialExperiment and tidyomics, the following links are a good place to start:
The tidyomics ecosystem also includes packages for:
Working with genomic features:
Working with transcriptomic features:
SummarizedExperiment objects.SingleCellExperiment objects.Seurat objects.Working with cytometry features:
And a few associated packages:
| Package | Functions available | 
|---|---|
| SpatialExperiment | All | 
| dplyr | arrange,bind_rows,bind_cols,distinct,filter,group_by,summarise,select,mutate,rename,left_join,right_join,inner_join,slice,sample_n,sample_frac,count,add_count | 
| tidyr | nest,unnest,unite,separate,extract,pivot_longer | 
| ggplot2 | ggplot | 
| plotly | plot_ly | 
| Utility | Description | 
|---|---|
| as_tibble | Convert cell data to a tbl_df | 
| join_features | Append feature data to cell data | 
| aggregate_cells | Aggregate cell-feature abundance into a pseudobulk SummarizedExperimentobject | 
| rectangle | Select cells in a rectangular region of space | 
| ellipse | Select cells in an elliptical region of space | 
| gate_spatial | |
| gate_programmatic | 
You can install the stable version of tidySpatialExperiment from Bioconductor.
if (!requireNamespace("BiocManager", quietly = TRUE))
    install.packages("BiocManager")
BiocManager::install("tidySpatialExperiment")Or, you can install the development version of tidySpatialExperiment from GitHub.
if (!requireNamespace("pak", quietly = TRUE))
    install.packages("pak")
pak::pak("william-hutchison/tidySpatialExperiment")Here, we attach tidySpatialExperiment and an example SpatialExperiment
object.
# Load example SpatialExperiment object
library(tidySpatialExperiment)
example(read10xVisium)A SpatialExperiment object represents assay-feature values as rows and
cells as columns. Additional information about the cells is stored in the
reducedDims, colData and spatialCoords slots.
tidySpatialExperiment provides a SpatialExperiment-tibble abstraction,
representing cells as rows and cell data as columns, in accordance with the
tidy observation-variable convention. The cell data is made up of information stored in the colData and spatialCoords slots.
The default view is now of the SpatialExperiment-tibble abstraction.
spe## # A SpatialExperiment-tibble abstraction: 50 × 7
## # Features = 50 | Cells = 50 | Assays = counts
##    .cell              in_tissue array_row array_col sample_id pxl_col_in_fullres
##    <chr>              <lgl>         <int>     <int> <chr>                  <int>
##  1 AAACAACGAATAGTTC-1 FALSE             0        16 section1                2312
##  2 AAACAAGTATCTCCCA-1 TRUE             50       102 section1                8230
##  3 AAACAATCTACTAGCA-1 TRUE              3        43 section1                4170
##  4 AAACACCAATAACTGC-1 TRUE             59        19 section1                2519
##  5 AAACAGAGCGACTCCT-1 TRUE             14        94 section1                7679
##  6 AAACAGCTTTCAGAAG-1 FALSE            43         9 section1                1831
##  7 AAACAGGGTCTATATT-1 FALSE            47        13 section1                2106
##  8 AAACAGTGTTCCTGGG-1 FALSE            73        43 section1                4170
##  9 AAACATGGTGAGAGGA-1 FALSE            62         0 section1                1212
## 10 AAACATTTCCCGGATT-1 FALSE            61        97 section1                7886
## # ℹ 40 more rows
## # ℹ 1 more variable: pxl_row_in_fullres <int>But, our data maintains its status as a SpatialExperiment object.
Therefore, we have access to all SpatialExperiment functions.
spe |>
    colData() |>
    head()## DataFrame with 6 rows and 4 columns
##                    in_tissue array_row array_col   sample_id
##                    <logical> <integer> <integer> <character>
## AAACAACGAATAGTTC-1     FALSE         0        16    section1
## AAACAAGTATCTCCCA-1      TRUE        50       102    section1
## AAACAATCTACTAGCA-1      TRUE         3        43    section1
## AAACACCAATAACTGC-1      TRUE        59        19    section1
## AAACAGAGCGACTCCT-1      TRUE        14        94    section1
## AAACAGCTTTCAGAAG-1     FALSE        43         9    section1spe |> 
    spatialCoords() |>
    head()##                    pxl_col_in_fullres pxl_row_in_fullres
## AAACAACGAATAGTTC-1               2312               1252
## AAACAAGTATCTCCCA-1               8230               7237
## AAACAATCTACTAGCA-1               4170               1611
## AAACACCAATAACTGC-1               2519               8315
## AAACAGAGCGACTCCT-1               7679               2927
## AAACAGCTTTCAGAAG-1               1831               6400spe |>
    imgData()## DataFrame with 1 row and 4 columns
##     sample_id    image_id   data scaleFactor
##   <character> <character> <list>   <numeric>
## 1    section1      lowres   ####   0.0510334Most functions from dplyr are available for use with the
SpatialExperiment-tibble abstraction. For example, filter() can be used
to filter cells by a variable of interest.
spe |>
   filter(array_col < 5)## # A SpatialExperiment-tibble abstraction: 3 × 7
## # Features = 50 | Cells = 3 | Assays = counts
##   .cell              in_tissue array_row array_col sample_id pxl_col_in_fullres
##   <chr>              <lgl>         <int>     <int> <chr>                  <int>
## 1 AAACATGGTGAGAGGA-1 FALSE            62         0 section1                1212
## 2 AAACGAAGATGGAGTA-1 FALSE            58         4 section1                1487
## 3 AAAGAATGACCTTAGA-1 FALSE            64         2 section1                1349
## # ℹ 1 more variable: pxl_row_in_fullres <int>And mutate can be used to add new variables, or modify the value of an
existing variable.
spe |>
    mutate(in_region = c(in_tissue & array_row < 10))## # A SpatialExperiment-tibble abstraction: 50 × 8
## # Features = 50 | Cells = 50 | Assays = counts
##    .cell    in_tissue array_row array_col sample_id in_region pxl_col_in_fullres
##    <chr>    <lgl>         <int>     <int> <chr>     <lgl>                  <int>
##  1 AAACAAC… FALSE             0        16 section1  FALSE                   2312
##  2 AAACAAG… TRUE             50       102 section1  FALSE                   8230
##  3 AAACAAT… TRUE              3        43 section1  TRUE                    4170
##  4 AAACACC… TRUE             59        19 section1  FALSE                   2519
##  5 AAACAGA… TRUE             14        94 section1  FALSE                   7679
##  6 AAACAGC… FALSE            43         9 section1  FALSE                   1831
##  7 AAACAGG… FALSE            47        13 section1  FALSE                   2106
##  8 AAACAGT… FALSE            73        43 section1  FALSE                   4170
##  9 AAACATG… FALSE            62         0 section1  FALSE                   1212
## 10 AAACATT… FALSE            61        97 section1  FALSE                   7886
## # ℹ 40 more rows
## # ℹ 1 more variable: pxl_row_in_fullres <int>Most functions from tidyr are also available. Here, nest() is used to
group the data by sample_id, and unnest() is used to ungroup the data.
# Nest the SpatialExperiment object by sample_id
spe_nested <-
    spe |> 
    nest(data = -sample_id)
# View the nested SpatialExperiment object
spe_nested## # A tibble: 1 × 2
##   sample_id data           
##   <chr>     <list>         
## 1 section1  <SptlExpr[,50]># Unnest the nested SpatialExperiment objects
spe_nested |>
    unnest(data)## # A SpatialExperiment-tibble abstraction: 50 × 7
## # Features = 50 | Cells = 50 | Assays = counts
##    .cell              in_tissue array_row array_col sample_id pxl_col_in_fullres
##    <chr>              <lgl>         <int>     <int> <chr>                  <int>
##  1 AAACAACGAATAGTTC-1 FALSE             0        16 section1                2312
##  2 AAACAAGTATCTCCCA-1 TRUE             50       102 section1                8230
##  3 AAACAATCTACTAGCA-1 TRUE              3        43 section1                4170
##  4 AAACACCAATAACTGC-1 TRUE             59        19 section1                2519
##  5 AAACAGAGCGACTCCT-1 TRUE             14        94 section1                7679
##  6 AAACAGCTTTCAGAAG-1 FALSE            43         9 section1                1831
##  7 AAACAGGGTCTATATT-1 FALSE            47        13 section1                2106
##  8 AAACAGTGTTCCTGGG-1 FALSE            73        43 section1                4170
##  9 AAACATGGTGAGAGGA-1 FALSE            62         0 section1                1212
## 10 AAACATTTCCCGGATT-1 FALSE            61        97 section1                7886
## # ℹ 40 more rows
## # ℹ 1 more variable: pxl_row_in_fullres <int>The ggplot() function can be used to create a plot directly from a
SpatialExperiment object. This example also demonstrates how tidy
operations can be combined to build up more complex analysis.
spe |>
    filter(sample_id == "section1" & in_tissue) |>
    
    # Add a column with the sum of feature counts per cell
    mutate(count_sum = purrr::map_int(.cell, ~
        spe[, .x] |> 
            counts() |> 
            sum()
      )) |>
    
    # Plot with tidySpatialExperiment and ggplot2
    ggplot(aes(x = reorder(.cell, count_sum), y = count_sum)) +
    geom_point() +
    coord_flip()The plot_ly() function can also be used to create a plot from a
SpatialExperiment object.
spe |>
    filter(sample_id == "section1") |>
    plot_ly(
        x = ~ array_col, 
        y = ~ array_row, 
        color = ~ in_tissue, 
        type = "scatter"
    )The tidyomics ecosystem places an emphasis on interacting with cell
data. To interact with feature data, the join_features() function can be
used to append assay-feature values to cell data.
# Join feature data in wide format, preserving the SpatialExperiment object
spe |>
    join_features(features = c("ENSMUSG00000025915", "ENSMUSG00000042501"), shape = "wide") |> 
    head()## # A SpatialExperiment-tibble abstraction: 50 × 9
## # Features = 6 | Cells = 50 | Assays = counts
##    .cell              in_tissue array_row array_col sample_id ENSMUSG00000025915
##    <chr>              <lgl>         <int>     <int> <chr>                  <dbl>
##  1 AAACAACGAATAGTTC-1 FALSE             0        16 section1                   0
##  2 AAACAAGTATCTCCCA-1 TRUE             50       102 section1                   0
##  3 AAACAATCTACTAGCA-1 TRUE              3        43 section1                   0
##  4 AAACACCAATAACTGC-1 TRUE             59        19 section1                   0
##  5 AAACAGAGCGACTCCT-1 TRUE             14        94 section1                   0
##  6 AAACAGCTTTCAGAAG-1 FALSE            43         9 section1                   0
##  7 AAACAGGGTCTATATT-1 FALSE            47        13 section1                   0
##  8 AAACAGTGTTCCTGGG-1 FALSE            73        43 section1                   0
##  9 AAACATGGTGAGAGGA-1 FALSE            62         0 section1                   0
## 10 AAACATTTCCCGGATT-1 FALSE            61        97 section1                   0
## # ℹ 40 more rows
## # ℹ 3 more variables: ENSMUSG00000042501 <dbl>, pxl_col_in_fullres <int>,
## #   pxl_row_in_fullres <int># Join feature data in long format, discarding the SpatialExperiment object
spe |>
    join_features(features = c("ENSMUSG00000025915", "ENSMUSG00000042501"), shape = "long") |> 
    head()## tidySpatialExperiment says: A data frame is returned for independent data 
##               analysis.## # A tibble: 6 × 7
##   .cell       in_tissue array_row array_col sample_id .feature .abundance_counts
##   <chr>       <lgl>         <int>     <int> <chr>     <chr>                <dbl>
## 1 AAACAACGAA… FALSE             0        16 section1  ENSMUSG…                 0
## 2 AAACAACGAA… FALSE             0        16 section1  ENSMUSG…                 0
## 3 AAACAAGTAT… TRUE             50       102 section1  ENSMUSG…                 0
## 4 AAACAAGTAT… TRUE             50       102 section1  ENSMUSG…                 1
## 5 AAACAATCTA… TRUE              3        43 section1  ENSMUSG…                 0
## 6 AAACAATCTA… TRUE              3        43 section1  ENSMUSG…                 0Sometimes, it is necessary to aggregate the gene-transcript abundance from a group of cells into a single value. For example, when comparing groups of cells across different samples with fixed-effect models.
The aggregate_cells() function can be used to aggregate cells by a
specified variable and assay, returning a SummarizedExperiment object.
spe |>
    aggregate_cells(in_tissue, assays = "counts")## class: SummarizedExperiment 
## dim: 50 2 
## metadata(0):
## assays(1): counts
## rownames(50): ENSMUSG00000002459 ENSMUSG00000005886 ...
##   ENSMUSG00000104217 ENSMUSG00000104328
## rowData names(1): feature
## colnames(2): FALSE TRUE
## colData names(3): in_tissue .aggregated_cells sample_idThe ellipse() and rectangle() functions can be used to select cells by
their position in space.
spe |>
    filter(sample_id == "section1") |>
    mutate(in_ellipse = ellipse(array_col, array_row, c(20, 40), c(20, 20))) |>
    ggplot(aes(x = array_col, y = array_row, colour = in_ellipse)) +
    geom_point()For the interactive selection of cells in space, tidySpatialExperiment experiment provides gate(). This function uses tidygate, shiny and plotly to launch an interactive plot overlaying cells in position with image data. Additional parameters can be used to specify point colour, shape, size and alpha, either with a column in the SpatialExperiment object or a constant value.
spe_gated <- 
  spe |>
  gate(colour = "in_tissue", alpha = 0.8)A record of which points appear in which gates is appended to the SpatialExperiment object in the .gated column. To select cells which appear within any gates, filter for non-NA values. To select cells which appear within a specific gate, string pattern matching can be used.
# Select cells within any gate
spe_gated |> 
  filter(!is.na(.gated))## # A SpatialExperiment-tibble abstraction: 4 × 8
## # Features = 50 | Cells = 4 | Assays = counts
##   .cell        in_tissue array_row array_col sample_id .gated pxl_col_in_fullres
##   <chr>        <lgl>         <int>     <int> <chr>     <chr>               <int>
## 1 AAACGAGACGG… TRUE             35        79 section1  2                    6647
## 2 AAACTGCTGGC… TRUE             45        67 section1  2                    5821
## 3 AAAGGGATGTA… TRUE             24        62 section1  1,2                  5477
## 4 AAAGGGCAGCT… TRUE             24        26 section1  1                    3000
## # ℹ 1 more variable: pxl_row_in_fullres <int># Select cells within gate 2
spe_gated |>
  filter(stringr::str_detect(.gated, "2"))## # A SpatialExperiment-tibble abstraction: 3 × 8
## # Features = 50 | Cells = 3 | Assays = counts
##   .cell        in_tissue array_row array_col sample_id .gated pxl_col_in_fullres
##   <chr>        <lgl>         <int>     <int> <chr>     <chr>               <int>
## 1 AAACGAGACGG… TRUE             35        79 section1  2                    6647
## 2 AAACTGCTGGC… TRUE             45        67 section1  2                    5821
## 3 AAAGGGATGTA… TRUE             24        62 section1  1,2                  5477
## # ℹ 1 more variable: pxl_row_in_fullres <int>Details of the interactively drawn gates are saved to tidygate_env$gates. This variable is overwritten each time interactive gates are drawn, so save it right away if you would like to access it later.
# Inspect previously drawn gates
tidygate_env$gates |>
  head()## # A tibble: 6 × 3
##       x     y .gate
##   <dbl> <dbl> <dbl>
## 1 4310. 3125.     1
## 2 3734. 3161.     1
## 3 2942. 3521.     1
## 4 2834. 3665.     1
## 5 2834. 4385.     1
## 6 2978. 4493.     1# Save if needed
tidygate_env$gates |>
  write_rds("important_gates.rds")If previously drawn gates are supplied to the programmatic_gates argument, cells will be gated programmatically. This feature allows the reproduction of previously drawn interactive gates.
important_gates <-
  read_rds("important_gates.rds")
spe |>
  gate(programmatic_gates = important_gates)) |>
  filter(!is.na(.gated))## # A SpatialExperiment-tibble abstraction: 4 × 8
## # Features = 50 | Cells = 4 | Assays = counts
##   .cell        in_tissue array_row array_col sample_id .gated pxl_col_in_fullres
##   <chr>        <lgl>         <int>     <int> <chr>     <chr>               <int>
## 1 AAACGAGACGG… TRUE             35        79 section1  2                    6647
## 2 AAACTGCTGGC… TRUE             45        67 section1  2                    5821
## 3 AAAGGGATGTA… TRUE             24        62 section1  1,2                  5477
## 4 AAAGGGCAGCT… TRUE             24        26 section1  1                    3000
## # ℹ 1 more variable: pxl_row_in_fullres <int>Removing the .cell column will return a tibble. This is consistent
with the behaviour in other tidyomics packages.
spe |>
    select(-.cell) |>
    head()## tidySpatialExperiment says: Key columns are missing. A data frame is 
##                         returned for independent data analysis.## # A tibble: 6 × 4
##   in_tissue array_row array_col sample_id
##   <lgl>         <int>     <int> <chr>    
## 1 FALSE             0        16 section1 
## 2 TRUE             50       102 section1 
## 3 TRUE              3        43 section1 
## 4 TRUE             59        19 section1 
## 5 TRUE             14        94 section1 
## 6 FALSE            43         9 section1The sample_id column cannot be removed with tidyverse functions, and
can only be modified if the changes are accepted by SpatialExperiment’s
colData() function.
# sample_id is not removed, despite the user's request
spe |>
    select(-sample_id)## # A SpatialExperiment-tibble abstraction: 50 × 7
## # Features = 50 | Cells = 50 | Assays = counts
##    .cell              in_tissue array_row array_col sample_id pxl_col_in_fullres
##    <chr>              <lgl>         <int>     <int> <chr>                  <int>
##  1 AAACAACGAATAGTTC-1 FALSE             0        16 section1                2312
##  2 AAACAAGTATCTCCCA-1 TRUE             50       102 section1                8230
##  3 AAACAATCTACTAGCA-1 TRUE              3        43 section1                4170
##  4 AAACACCAATAACTGC-1 TRUE             59        19 section1                2519
##  5 AAACAGAGCGACTCCT-1 TRUE             14        94 section1                7679
##  6 AAACAGCTTTCAGAAG-1 FALSE            43         9 section1                1831
##  7 AAACAGGGTCTATATT-1 FALSE            47        13 section1                2106
##  8 AAACAGTGTTCCTGGG-1 FALSE            73        43 section1                4170
##  9 AAACATGGTGAGAGGA-1 FALSE            62         0 section1                1212
## 10 AAACATTTCCCGGATT-1 FALSE            61        97 section1                7886
## # ℹ 40 more rows
## # ℹ 1 more variable: pxl_row_in_fullres <int># This change maintains separation of sample_ids and is permitted
spe |> 
    mutate(sample_id = stringr::str_c(sample_id, "_modified")) |>
    head()## # A SpatialExperiment-tibble abstraction: 50 × 7
## # Features = 6 | Cells = 50 | Assays = counts
##    .cell              in_tissue array_row array_col sample_id pxl_col_in_fullres
##    <chr>              <lgl>         <int>     <int> <chr>                  <int>
##  1 AAACAACGAATAGTTC-1 FALSE             0        16 section1…               2312
##  2 AAACAAGTATCTCCCA-1 TRUE             50       102 section1…               8230
##  3 AAACAATCTACTAGCA-1 TRUE              3        43 section1…               4170
##  4 AAACACCAATAACTGC-1 TRUE             59        19 section1…               2519
##  5 AAACAGAGCGACTCCT-1 TRUE             14        94 section1…               7679
##  6 AAACAGCTTTCAGAAG-1 FALSE            43         9 section1…               1831
##  7 AAACAGGGTCTATATT-1 FALSE            47        13 section1…               2106
##  8 AAACAGTGTTCCTGGG-1 FALSE            73        43 section1…               4170
##  9 AAACATGGTGAGAGGA-1 FALSE            62         0 section1…               1212
## 10 AAACATTTCCCGGATT-1 FALSE            61        97 section1…               7886
## # ℹ 40 more rows
## # ℹ 1 more variable: pxl_row_in_fullres <int># This change does not maintain separation of sample_ids and produces an error
spe |>
    mutate(sample_id = "new_sample")## # A SpatialExperiment-tibble abstraction: 50 × 7
## # Features = 50 | Cells = 50 | Assays = counts
##    .cell              in_tissue array_row array_col sample_id pxl_col_in_fullres
##    <chr>              <lgl>         <int>     <int> <chr>                  <int>
##  1 AAACAACGAATAGTTC-1 FALSE             0        16 new_samp…               2312
##  2 AAACAAGTATCTCCCA-1 TRUE             50       102 new_samp…               8230
##  3 AAACAATCTACTAGCA-1 TRUE              3        43 new_samp…               4170
##  4 AAACACCAATAACTGC-1 TRUE             59        19 new_samp…               2519
##  5 AAACAGAGCGACTCCT-1 TRUE             14        94 new_samp…               7679
##  6 AAACAGCTTTCAGAAG-1 FALSE            43         9 new_samp…               1831
##  7 AAACAGGGTCTATATT-1 FALSE            47        13 new_samp…               2106
##  8 AAACAGTGTTCCTGGG-1 FALSE            73        43 new_samp…               4170
##  9 AAACATGGTGAGAGGA-1 FALSE            62         0 new_samp…               1212
## 10 AAACATTTCCCGGATT-1 FALSE            61        97 new_samp…               7886
## # ℹ 40 more rows
## # ℹ 1 more variable: pxl_row_in_fullres <int>The pxl_col_in_fullres and px_row_in_fullres columns cannot be
removed or modified with tidyverse functions. This is consistent with
the behaviour of dimension reduction data in other tidyomics packages.
# Attempting to remove pxl_col_in_fullres produces an error
spe |>
    select(-pxl_col_in_fullres)## Error in `select_helper()`:
## ! Can't select columns that don't exist.
## ✖ Column `pxl_col_in_fullres` doesn't exist.# Attempting to modify pxl_col_in_fullres produces an error
spe |> 
    mutate(pxl_col_in_fullres)## Error in `dplyr::mutate()`:
## ℹ In argument: `pxl_col_in_fullres`.
## Caused by error:
## ! object 'pxl_col_in_fullres' not foundIf you use tidySpatialExperiment in published research, please cite The tidyomics ecosystem: enhancing omic data analyses.
sessionInfo()## R version 4.5.0 Patched (2025-04-21 r88169)
## Platform: x86_64-apple-darwin20
## Running under: macOS Monterey 12.7.6
## 
## Matrix products: default
## BLAS:   /Library/Frameworks/R.framework/Versions/4.5-x86_64/Resources/lib/libRblas.0.dylib 
## LAPACK: /Library/Frameworks/R.framework/Versions/4.5-x86_64/Resources/lib/libRlapack.dylib;  LAPACK version 3.12.1
## 
## locale:
## [1] C/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8
## 
## time zone: America/New_York
## tzcode source: internal
## 
## attached base packages:
## [1] stats4    stats     graphics  grDevices utils     datasets  methods  
## [8] base     
## 
## other attached packages:
##  [1] tidySpatialExperiment_1.5.0     ggplot2_3.5.2                  
##  [3] ttservice_0.4.1                 tidyr_1.3.1                    
##  [5] dplyr_1.1.4                     tidySingleCellExperiment_1.19.0
##  [7] SpatialExperiment_1.19.0        SingleCellExperiment_1.31.0    
##  [9] SummarizedExperiment_1.39.0     Biobase_2.69.0                 
## [11] GenomicRanges_1.61.0            GenomeInfoDb_1.45.0            
## [13] IRanges_2.43.0                  S4Vectors_0.47.0               
## [15] BiocGenerics_0.55.0             generics_0.1.3                 
## [17] MatrixGenerics_1.21.0           matrixStats_1.5.0              
## [19] BiocStyle_2.37.0               
## 
## loaded via a namespace (and not attached):
##  [1] gridExtra_2.3             rlang_1.1.6              
##  [3] magrittr_2.0.3            compiler_4.5.0           
##  [5] DelayedMatrixStats_1.31.0 vctrs_0.6.5              
##  [7] stringr_1.5.1             pkgconfig_2.0.3          
##  [9] crayon_1.5.3              fastmap_1.2.0            
## [11] magick_2.8.6              XVector_0.49.0           
## [13] ellipsis_0.3.2            labeling_0.4.3           
## [15] scuttle_1.19.0            utf8_1.2.4               
## [17] promises_1.3.2            rmarkdown_2.29           
## [19] UCSC.utils_1.5.0          tinytex_0.57             
## [21] purrr_1.0.4               xfun_0.52                
## [23] beachmat_2.25.0           cachem_1.1.0             
## [25] jsonlite_2.0.0            later_1.4.2              
## [27] rhdf5filters_1.21.0       DelayedArray_0.35.1      
## [29] Rhdf5lib_1.31.0           BiocParallel_1.43.0      
## [31] parallel_4.5.0            R6_2.6.1                 
## [33] bslib_0.9.0               stringi_1.8.7            
## [35] RColorBrewer_1.1-3        limma_3.65.0             
## [37] jquerylib_0.1.4           Rcpp_1.0.14              
## [39] bookdown_0.43             knitr_1.50               
## [41] R.utils_2.13.0            httpuv_1.6.16            
## [43] Matrix_1.7-3              tidyselect_1.2.1         
## [45] abind_1.4-8               yaml_2.3.10              
## [47] viridis_0.6.5             codetools_0.2-20         
## [49] lattice_0.22-7            tibble_3.2.1             
## [51] shiny_1.10.0              withr_3.0.2              
## [53] evaluate_1.0.3            pillar_1.10.2            
## [55] BiocManager_1.30.25       plotly_4.10.4            
## [57] sparseMatrixStats_1.21.0  munsell_0.5.1            
## [59] scales_1.3.0              xtable_1.8-4             
## [61] glue_1.8.0                lazyeval_0.2.2           
## [63] tools_4.5.0               data.table_1.17.0        
## [65] locfit_1.5-9.12           rhdf5_2.53.0             
## [67] grid_4.5.0                edgeR_4.7.0              
## [69] DropletUtils_1.29.0       colorspace_2.1-1         
## [71] GenomeInfoDbData_1.2.14   HDF5Array_1.37.0         
## [73] cli_3.6.4                 tidygate_1.0.14          
## [75] fansi_1.0.6               S4Arrays_1.9.0           
## [77] viridisLite_0.4.2         gtable_0.3.6             
## [79] R.methodsS3_1.8.2         sass_0.4.10              
## [81] digest_0.6.37             dqrng_0.4.1              
## [83] SparseArray_1.9.0         farver_2.1.2             
## [85] rjson_0.2.23              htmlwidgets_1.6.4        
## [87] R.oo_1.27.0               htmltools_0.5.8.1        
## [89] lifecycle_1.0.4           h5mread_1.1.0            
## [91] httr_1.4.7                statmod_1.5.0            
## [93] mime_0.13