# Segerstolpe human pancreas (Smart-seq2) ## Introduction This performs an analysis of the @segerstolpe2016singlecell dataset, consisting of human pancreas cells from various donors. ## Data loading ``` r library(scRNAseq) sce.seger <- SegerstolpePancreasData() ``` ``` r library(AnnotationHub) edb <- AnnotationHub()[["AH73881"]] symbols <- rowData(sce.seger)$symbol ens.id <- mapIds(edb, keys=symbols, keytype="SYMBOL", column="GENEID") ens.id <- ifelse(is.na(ens.id), symbols, ens.id) # Removing duplicated rows. keep <- !duplicated(ens.id) sce.seger <- sce.seger[keep,] rownames(sce.seger) <- ens.id[keep] ``` We simplify the names of some of the relevant column metadata fields for ease of access. Some editing of the cell type labels is necessary for consistency with other data sets. ``` r emtab.meta <- colData(sce.seger)[,c("cell type", "disease", "individual", "single cell well quality")] colnames(emtab.meta) <- c("CellType", "Disease", "Donor", "Quality") colData(sce.seger) <- emtab.meta sce.seger$CellType <- gsub(" cell", "", sce.seger$CellType) sce.seger$CellType <- paste0( toupper(substr(sce.seger$CellType, 1, 1)), substring(sce.seger$CellType, 2)) ``` ## Quality control ``` r unfiltered <- sce.seger ``` We remove low quality cells that were marked by the authors. We then perform additional quality control as some of the remaining cells still have very low counts and numbers of detected features. For some batches that seem to have a majority of low-quality cells (Figure \@ref(fig:unref-seger-qc-dist)), we use the other batches to define an appropriate threshold via `subset=`. ``` r low.qual <- sce.seger$Quality == "OK, filtered" library(scater) stats <- perCellQCMetrics(sce.seger) qc <- quickPerCellQC(stats, percent_subsets="altexps_ERCC_percent", batch=sce.seger$Donor, subset=!sce.seger$Donor %in% c("H6", "H5")) sce.seger <- sce.seger[,!(qc$discard | low.qual)] ``` ``` r colData(unfiltered) <- cbind(colData(unfiltered), stats) unfiltered$discard <- qc$discard gridExtra::grid.arrange( plotColData(unfiltered, x="Donor", y="sum", colour_by="discard") + scale_y_log10() + ggtitle("Total count") + theme(axis.text.x = element_text(angle = 90)), plotColData(unfiltered, x="Donor", y="detected", colour_by="discard") + scale_y_log10() + ggtitle("Detected features") + theme(axis.text.x = element_text(angle = 90)), plotColData(unfiltered, x="Donor", y="altexps_ERCC_percent", colour_by="discard") + ggtitle("ERCC percent") + theme(axis.text.x = element_text(angle = 90)), ncol=2 ) ```
Distribution of each QC metric across cells from each donor of the Segerstolpe pancreas dataset. Each point represents a cell and is colored according to whether that cell was discarded.

(\#fig:unref-seger-qc-dist)Distribution of each QC metric across cells from each donor of the Segerstolpe pancreas dataset. Each point represents a cell and is colored according to whether that cell was discarded.

``` r colSums(as.matrix(qc)) ``` ``` ## low_lib_size low_n_features high_altexps_ERCC_percent ## 788 1056 1031 ## discard ## 1246 ``` ## Normalization We don't normalize the spike-ins at this point as there are some cells with no spike-in counts. ``` r library(scran) clusters <- quickCluster(sce.seger) sce.seger <- computeSumFactors(sce.seger, clusters=clusters) sce.seger <- logNormCounts(sce.seger) ``` ``` r summary(sizeFactors(sce.seger)) ``` ``` ## Min. 1st Qu. Median Mean 3rd Qu. Max. ## 0.014 0.390 0.708 1.000 1.332 11.182 ``` ``` r plot(librarySizeFactors(sce.seger), sizeFactors(sce.seger), pch=16, xlab="Library size factors", ylab="Deconvolution factors", log="xy") ```
Relationship between the library size factors and the deconvolution size factors in the Segerstolpe pancreas dataset.

(\#fig:unref-seger-norm)Relationship between the library size factors and the deconvolution size factors in the Segerstolpe pancreas dataset.

## Variance modelling We do not use cells with no spike-ins for variance modelling. Donor H1 also has very low spike-in counts and is subsequently ignored. ``` r for.hvg <- sce.seger[,librarySizeFactors(altExp(sce.seger)) > 0 & sce.seger$Donor!="H1"] dec.seger <- modelGeneVarWithSpikes(for.hvg, "ERCC", block=for.hvg$Donor) chosen.hvgs <- getTopHVGs(dec.seger, n=2000) ``` ``` r par(mfrow=c(3,3)) blocked.stats <- dec.seger$per.block for (i in colnames(blocked.stats)) { current <- blocked.stats[[i]] plot(current$mean, current$total, main=i, pch=16, cex=0.5, xlab="Mean of log-expression", ylab="Variance of log-expression") curfit <- metadata(current) points(curfit$mean, curfit$var, col="red", pch=16) curve(curfit$trend(x), col='dodgerblue', add=TRUE, lwd=2) } ```
Per-gene variance as a function of the mean for the log-expression values in the Grun pancreas dataset. Each point represents a gene (black) with the mean-variance trend (blue) fitted to the spike-in transcripts (red) separately for each donor.

(\#fig:unref-seger-variance)Per-gene variance as a function of the mean for the log-expression values in the Grun pancreas dataset. Each point represents a gene (black) with the mean-variance trend (blue) fitted to the spike-in transcripts (red) separately for each donor.

## Dimensionality reduction We pick the first 25 PCs for downstream analyses, as it's a nice square number. ``` r library(BiocSingular) set.seed(101011001) sce.seger <- runPCA(sce.seger, subset_row=chosen.hvgs, ncomponents=25) sce.seger <- runTSNE(sce.seger, dimred="PCA") ``` ## Clustering ``` r library(bluster) clust.out <- clusterRows(reducedDim(sce.seger, "PCA"), NNGraphParam(), full=TRUE) snn.gr <- clust.out$objects$graph colLabels(sce.seger) <- clust.out$clusters ``` We see a strong donor effect in Figures \@ref(fig:unref-seger-heat-1) and \@ref(fig:unref-grun-tsne). This might be due to differences in cell type composition between donors, but the more likely explanation is that of a technical difference in plate processing or uninteresting genotypic differences. The implication is that we should have called `fastMNN()` at some point. ``` r tab <- table(Cluster=colLabels(sce.seger), Donor=sce.seger$Donor) library(pheatmap) pheatmap(log10(tab+10), color=viridis::viridis(100)) ```
Heatmap of the frequency of cells from each donor in each cluster.

(\#fig:unref-seger-heat-1)Heatmap of the frequency of cells from each donor in each cluster.

``` r gridExtra::grid.arrange( plotTSNE(sce.seger, colour_by="label"), plotTSNE(sce.seger, colour_by="Donor"), ncol=2 ) ```
Obligatory $t$-SNE plots of the Segerstolpe pancreas dataset. Each point represents a cell that is colored by cluster (left) or batch (right).

(\#fig:unref-seger-tsne)Obligatory $t$-SNE plots of the Segerstolpe pancreas dataset. Each point represents a cell that is colored by cluster (left) or batch (right).

## Data integration We repeat the clustering after running `fastMNN()` on the donors. This yields a more coherent set of clusters in Figure \@ref(fig:unref-seger-tsne-correct) where each cluster contains contributions from all donors. ``` r library(batchelor) set.seed(10001010) corrected <- fastMNN(sce.seger, batch=sce.seger$Donor, subset.row=chosen.hvgs) set.seed(10000001) corrected <- runTSNE(corrected, dimred="corrected") colLabels(corrected) <- clusterRows(reducedDim(corrected, "corrected"), NNGraphParam()) tab <- table(Cluster=colLabels(corrected), Donor=corrected$batch) tab ``` ``` ## Donor ## Cluster H1 H2 H3 H4 H5 H6 T2D1 T2D2 T2D3 T2D4 ## 1 4 20 80 3 2 4 8 29 24 13 ## 2 14 53 37 41 14 19 13 20 11 70 ## 3 3 19 67 8 27 11 3 78 124 46 ## 4 8 21 2 6 11 6 9 6 5 34 ## 5 2 1 0 1 2 9 1 2 2 1 ## 6 29 114 26 136 49 72 140 121 85 96 ## 7 1 1 2 6 3 10 3 12 13 4 ## 8 4 20 16 2 1 8 70 8 10 34 ``` ``` r gridExtra::grid.arrange( plotTSNE(corrected, colour_by="label"), plotTSNE(corrected, colour_by="batch"), ncol=2 ) ```
Yet another $t$-SNE plot of the Segerstolpe dataset, this time after batch correction across donors. Each point represents a cell and is colored by the assigned cluster identity.

(\#fig:unref-seger-tsne-correct)Yet another $t$-SNE plot of the Segerstolpe dataset, this time after batch correction across donors. Each point represents a cell and is colored by the assigned cluster identity.

## Multi-sample comparisons {#segerstolpe-comparison} This particular dataset contains both healthy donors and those with type II diabetes. It is thus of some interest to identify genes that are differentially expressed upon disease in each cell type. To keep things simple, we use the author-provided annotation rather than determining the cell type for each of our clusters. ``` r summed <- aggregateAcrossCells(sce.seger, ids=colData(sce.seger)[,c("Donor", "CellType")]) summed ``` ``` ## class: SingleCellExperiment ## dim: 25454 105 ## metadata(0): ## assays(1): counts ## rownames(25454): ENSG00000118473 ENSG00000142920 ... ENSG00000278306 ## eGFP ## rowData names(2): refseq symbol ## colnames: NULL ## colData names(9): CellType Disease ... CellType ncells ## reducedDimNames(2): PCA TSNE ## mainExpName: endogenous ## altExpNames(0): ``` Here, we will use the `voom` pipeline from the *[limma](https://bioconductor.org/packages/3.21/limma)* package instead of the QL approach with *[edgeR](https://bioconductor.org/packages/3.21/edgeR)*. This allows us to use sample weights to better account for the variation in the precision of each pseudo-bulk profile. We see that insulin is downregulated in beta cells in the disease state, which is sensible enough. ``` r summed.beta <- summed[,summed$CellType=="Beta"] library(edgeR) y.beta <- DGEList(counts(summed.beta), samples=colData(summed.beta), genes=rowData(summed.beta)[,"symbol",drop=FALSE]) y.beta <- y.beta[filterByExpr(y.beta, group=y.beta$samples$Disease),] y.beta <- calcNormFactors(y.beta) design <- model.matrix(~Disease, y.beta$samples) v.beta <- voomWithQualityWeights(y.beta, design) fit.beta <- lmFit(v.beta) fit.beta <- eBayes(fit.beta, robust=TRUE) res.beta <- topTable(fit.beta, sort.by="p", n=Inf, coef="Diseasetype II diabetes mellitus") head(res.beta) ``` ``` ## symbol logFC AveExpr t P.Value adj.P.Val B ## ENSG00000254647 INS -2.728 16.680 -7.671 3.191e-06 0.03902 4.842 ## ENSG00000137731 FXYD2 -2.595 7.265 -6.705 1.344e-05 0.08219 3.353 ## ENSG00000169297 NR0B1 -2.092 6.790 -5.789 5.810e-05 0.09916 1.984 ## ENSG00000181029 TRAPPC5 -2.127 7.046 -5.678 7.007e-05 0.09916 1.877 ## ENSG00000105707 HPN -1.803 6.118 -5.654 7.298e-05 0.09916 1.740 ## LOC284889 LOC284889 -2.113 6.652 -5.515 9.259e-05 0.09916 1.571 ``` We also create some diagnostic plots to check for potential problems in the analysis. The MA plots exhibit the expected shape (Figure \@ref(fig:unref-ma-plots)) while the differences in the sample weights in Figure \@ref(fig:unref-voom-plots) justify the use of `voom()` in this context. ``` r par(mfrow=c(5, 2)) for (i in colnames(y.beta)) { plotMD(y.beta, column=i) } ```
MA plots for the beta cell pseudo-bulk profiles. Each MA plot is generated by comparing the corresponding pseudo-bulk profile against the average of all other profiles

(\#fig:unref-ma-plots)MA plots for the beta cell pseudo-bulk profiles. Each MA plot is generated by comparing the corresponding pseudo-bulk profile against the average of all other profiles

``` r # Easier to just re-run it with plot=TRUE than # to try to make the plot from 'v.beta'. voomWithQualityWeights(y.beta, design, plot=TRUE) ```
Diagnostic plots for `voom` after estimating observation and quality weights from the beta cell pseudo-bulk profiles. The left plot shows the mean-variance trend used to estimate the observation weights, while the right plot shows the per-sample quality weights.

(\#fig:unref-voom-plots)Diagnostic plots for `voom` after estimating observation and quality weights from the beta cell pseudo-bulk profiles. The left plot shows the mean-variance trend used to estimate the observation weights, while the right plot shows the per-sample quality weights.

## Session Info {-}
``` 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] edgeR_4.5.0 limma_3.63.2 [3] batchelor_1.23.0 pheatmap_1.0.12 [5] bluster_1.17.0 BiocSingular_1.23.0 [7] scran_1.35.0 scater_1.35.0 [9] ggplot2_3.5.1 scuttle_1.17.0 [11] ensembldb_2.31.0 AnnotationFilter_1.31.0 [13] GenomicFeatures_1.59.1 AnnotationDbi_1.69.0 [15] AnnotationHub_3.15.0 BiocFileCache_2.15.0 [17] dbplyr_2.5.0 scRNAseq_2.21.0 [19] SingleCellExperiment_1.29.1 SummarizedExperiment_1.37.0 [21] Biobase_2.67.0 GenomicRanges_1.59.0 [23] GenomeInfoDb_1.43.0 IRanges_2.41.0 [25] S4Vectors_0.45.1 BiocGenerics_0.53.1 [27] generics_0.1.3 MatrixGenerics_1.19.0 [29] matrixStats_1.4.1 BiocStyle_2.35.0 [31] rebook_1.17.0 loaded via a namespace (and not attached): [1] RColorBrewer_1.1-3 jsonlite_1.8.9 [3] CodeDepends_0.6.6 magrittr_2.0.3 [5] ggbeeswarm_0.7.2 gypsum_1.3.0 [7] farver_2.1.2 rmarkdown_2.29 [9] BiocIO_1.17.0 zlibbioc_1.53.0 [11] vctrs_0.6.5 DelayedMatrixStats_1.29.0 [13] memoise_2.0.1 Rsamtools_2.23.0 [15] RCurl_1.98-1.16 htmltools_0.5.8.1 [17] S4Arrays_1.7.1 curl_6.0.0 [19] BiocNeighbors_2.1.0 Rhdf5lib_1.29.0 [21] SparseArray_1.7.1 rhdf5_2.51.0 [23] sass_0.4.9 alabaster.base_1.7.1 [25] bslib_0.8.0 alabaster.sce_1.7.0 [27] httr2_1.0.6 cachem_1.1.0 [29] ResidualMatrix_1.17.0 GenomicAlignments_1.43.0 [31] igraph_2.1.1 mime_0.12 [33] lifecycle_1.0.4 pkgconfig_2.0.3 [35] rsvd_1.0.5 Matrix_1.7-1 [37] R6_2.5.1 fastmap_1.2.0 [39] GenomeInfoDbData_1.2.13 digest_0.6.37 [41] colorspace_2.1-1 dqrng_0.4.1 [43] irlba_2.3.5.1 ExperimentHub_2.15.0 [45] RSQLite_2.3.7 beachmat_2.23.0 [47] labeling_0.4.3 filelock_1.0.3 [49] fansi_1.0.6 httr_1.4.7 [51] abind_1.4-8 compiler_4.5.0 [53] bit64_4.5.2 withr_3.0.2 [55] BiocParallel_1.41.0 viridis_0.6.5 [57] DBI_1.2.3 HDF5Array_1.35.1 [59] alabaster.ranges_1.7.0 alabaster.schemas_1.7.0 [61] rappdirs_0.3.3 DelayedArray_0.33.1 [63] rjson_0.2.23 tools_4.5.0 [65] vipor_0.4.7 beeswarm_0.4.0 [67] glue_1.8.0 restfulr_0.0.15 [69] rhdf5filters_1.19.0 grid_4.5.0 [71] Rtsne_0.17 cluster_2.1.6 [73] gtable_0.3.6 metapod_1.15.0 [75] ScaledMatrix_1.15.0 utf8_1.2.4 [77] XVector_0.47.0 ggrepel_0.9.6 [79] BiocVersion_3.21.1 pillar_1.9.0 [81] dplyr_1.1.4 lattice_0.22-6 [83] rtracklayer_1.67.0 bit_4.5.0 [85] tidyselect_1.2.1 locfit_1.5-9.10 [87] Biostrings_2.75.1 knitr_1.49 [89] gridExtra_2.3 bookdown_0.41 [91] ProtGenerics_1.39.0 xfun_0.49 [93] statmod_1.5.0 UCSC.utils_1.3.0 [95] lazyeval_0.2.2 yaml_2.3.10 [97] evaluate_1.0.1 codetools_0.2-20 [99] tibble_3.2.1 alabaster.matrix_1.7.0 [101] BiocManager_1.30.25 graph_1.85.0 [103] cli_3.6.3 munsell_0.5.1 [105] jquerylib_0.1.4 Rcpp_1.0.13-1 [107] dir.expiry_1.15.0 png_0.1-8 [109] XML_3.99-0.17 parallel_4.5.0 [111] blob_1.2.4 sparseMatrixStats_1.19.0 [113] bitops_1.0-9 viridisLite_0.4.2 [115] alabaster.se_1.7.0 scales_1.3.0 [117] purrr_1.0.2 crayon_1.5.3 [119] rlang_1.1.4 cowplot_1.1.3 [121] KEGGREST_1.47.0 ```