Chapter 13 Messmer human ESC (Smart-seq2)
13.1 Introduction
This performs an analysis of the human embryonic stem cell (hESC) dataset generated with Smart-seq2 (Messmer et al. 2019), which contains several plates of naive and primed hESCs. The chapter’s code is based on the steps in the paper’s GitHub repository, with some additional steps for cell cycle effect removal contributed by Philippe Boileau.
13.2 Data loading
Converting the batch to a factor, to make life easier later on.
13.3 Quality control
Let’s have a look at the QC statistics.
## low_lib_size low_n_features high_subsets_Mito_percent
## 107 99 22
## high_altexps_ERCC_percent discard
## 117 156
gridExtra::grid.arrange(
plotColData(original, x="experiment batch", y="sum",
colour_by=I(filtered$discard), other_field="phenotype") +
facet_wrap(~phenotype) + scale_y_log10(),
plotColData(original, x="experiment batch", y="detected",
colour_by=I(filtered$discard), other_field="phenotype") +
facet_wrap(~phenotype) + scale_y_log10(),
plotColData(original, x="experiment batch", y="subsets_Mito_percent",
colour_by=I(filtered$discard), other_field="phenotype") +
facet_wrap(~phenotype),
plotColData(original, x="experiment batch", y="altexps_ERCC_percent",
colour_by=I(filtered$discard), other_field="phenotype") +
facet_wrap(~phenotype),
ncol=1
)

Figure 13.1: Distribution of QC metrics across batches (x-axis) and phenotypes (facets) for cells in the Messmer hESC dataset. Each point is a cell and is colored by whether it was discarded.
13.4 Normalization
library(scran)
set.seed(10000)
clusters <- quickCluster(sce.mess)
sce.mess <- computeSumFactors(sce.mess, cluster=clusters)
sce.mess <- logNormCounts(sce.mess)
par(mfrow=c(1,2))
plot(sce.mess$sum, sizeFactors(sce.mess), log = "xy", pch=16,
xlab = "Library size (millions)", ylab = "Size factor",
col = ifelse(sce.mess$phenotype == "naive", "black", "grey"))
spike.sf <- librarySizeFactors(altExp(sce.mess, "ERCC"))
plot(sizeFactors(sce.mess), spike.sf, log = "xy", pch=16,
ylab = "Spike-in size factor", xlab = "Deconvolution size factor",
col = ifelse(sce.mess$phenotype == "naive", "black", "grey"))

Figure 13.2: Deconvolution size factors plotted against the library size (left) and spike-in size factors plotted against the deconvolution size factors (right). Each point is a cell and is colored by its phenotype.
13.5 Cell cycle phase assignment
Here, we use multiple cores to speed up the processing.
set.seed(10001)
hs_pairs <- readRDS(system.file("exdata", "human_cycle_markers.rds", package="scran"))
assigned <- cyclone(sce.mess, pairs=hs_pairs,
gene.names=rownames(sce.mess),
BPPARAM=BiocParallel::MulticoreParam(10))
sce.mess$phase <- assigned$phases
##
## G1 G2M S
## 460 406 322

Figure 13.3: G1 cyclone()
phase scores against the G2/M phase scores for each cell in the Messmer hESC dataset.
13.6 Feature selection
dec <- modelGeneVarWithSpikes(sce.mess, "ERCC", block = sce.mess$`experiment batch`)
top.hvgs <- getTopHVGs(dec, prop = 0.1)
par(mfrow=c(1,3))
for (i in seq_along(dec$per.block)) {
current <- dec$per.block[[i]]
plot(current$mean, current$total, xlab="Mean log-expression",
ylab="Variance", pch=16, cex=0.5, main=paste("Batch", i))
fit <- metadata(current)
points(fit$mean, fit$var, col="red", pch=16)
curve(fit$trend(x), col='dodgerblue', add=TRUE, lwd=2)
}

Figure 13.4: Per-gene variance of the log-normalized expression values in the Messmer hESC dataset, plotted against the mean for each batch. Each point represents a gene with spike-ins shown in red and the fitted trend shown in blue.
13.7 Batch correction
We eliminate the obvious batch effect between batches with linear regression, which is possible due to the replicated nature of the experimental design.
We set keep=1:2
to retain the effect of the first two coefficients in design
corresponding to our phenotype of interest.
13.8 Dimensionality Reduction
We could have set d=
and subset.row=
in correctExperiments()
to automatically perform a PCA on the the residual matrix with the subset of HVGs,
but we’ll just explicitly call runPCA()
here to keep things simple.
set.seed(1101001)
sce.mess <- runPCA(sce.mess, subset_row = top.hvgs, exprs_values = "corrected")
sce.mess <- runTSNE(sce.mess, dimred = "PCA", perplexity = 40)
From a naive PCA, the cell cycle appears to be a major source of biological variation within each phenotype.
gridExtra::grid.arrange(
plotTSNE(sce.mess, colour_by = "phenotype") + ggtitle("By phenotype"),
plotTSNE(sce.mess, colour_by = "experiment batch") + ggtitle("By batch "),
plotTSNE(sce.mess, colour_by = "CDK1", swap_rownames="SYMBOL") + ggtitle("By CDK1"),
plotTSNE(sce.mess, colour_by = "phase") + ggtitle("By phase"),
ncol = 2
)

Figure 13.5: Obligatory \(t\)-SNE plots of the Messmer hESC dataset, where each point is a cell and is colored by various attributes.
We perform contrastive PCA (cPCA) and sparse cPCA (scPCA) on the corrected log-expression data to obtain the same number of PCs. Given that the naive hESCs are actually reprogrammed primed hESCs, we will use the single batch of primed-only hESCs as the “background” dataset to remove the cell cycle effect.
library(scPCA)
is.bg <- sce.mess$`experiment batch`=="3"
target <- sce.mess[,!is.bg]
background <- sce.mess[,is.bg]
mat.target <- t(assay(target, "corrected")[top.hvgs,])
mat.background <- t(assay(background, "corrected")[top.hvgs,])
set.seed(1010101001)
con_out <- scPCA(
target = mat.target,
background = mat.background,
penalties = 0, # no penalties = non-sparse cPCA.
n_eigen = 50,
contrasts = 100
)
reducedDim(target, "cPCA") <- con_out$x
set.seed(101010101)
sparse_con_out <- scPCA(
target = mat.target,
background = mat.background,
penalties = 1e-4,
n_eigen = 50,
contrasts = 100,
alg = "rand_var_proj" # for speed.
)
reducedDim(target, "scPCA") <- sparse_con_out$x
We see greater intermingling between phases within both the naive and primed cells after cPCA and scPCA.
set.seed(1101001)
target <- runTSNE(target, dimred = "cPCA", perplexity = 40, name="cPCA+TSNE")
target <- runTSNE(target, dimred = "scPCA", perplexity = 40, name="scPCA+TSNE")
gridExtra::grid.arrange(
plotReducedDim(target, "cPCA+TSNE", colour_by = "phase") + ggtitle("After cPCA"),
plotReducedDim(target, "scPCA+TSNE", colour_by = "phase") + ggtitle("After scPCA"),
ncol=2
)

Figure 13.6: More \(t\)-SNE plots of the Messmer hESC dataset after cPCA and scPCA, where each point is a cell and is colored by its assigned cell cycle phase.
We can quantify the change in the separation between phases within each phenotype using the silhouette coefficient.
library(bluster)
naive <- target[,target$phenotype=="naive"]
primed <- target[,target$phenotype=="primed"]
N <- approxSilhouette(reducedDim(naive, "PCA"), naive$phase)
P <- approxSilhouette(reducedDim(primed, "PCA"), primed$phase)
c(naive=mean(N$width), primed=mean(P$width))
## naive primed
## 0.02032 0.03025
cN <- approxSilhouette(reducedDim(naive, "cPCA"), naive$phase)
cP <- approxSilhouette(reducedDim(primed, "cPCA"), primed$phase)
c(naive=mean(cN$width), primed=mean(cP$width))
## naive primed
## 0.007696 0.011941
scN <- approxSilhouette(reducedDim(naive, "scPCA"), naive$phase)
scP <- approxSilhouette(reducedDim(primed, "scPCA"), primed$phase)
c(naive=mean(scN$width), primed=mean(scP$width))
## naive primed
## 0.006614 0.014601
Session Info
R version 4.5.1 Patched (2025-08-23 r88802)
Platform: x86_64-pc-linux-gnu
Running under: Ubuntu 24.04.3 LTS
Matrix products: default
BLAS: /home/biocbuild/bbs-3.22-bioc/R/lib/libRblas.so
LAPACK: /usr/lib/x86_64-linux-gnu/lapack/liblapack.so.3.12.0 LAPACK version 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] bluster_1.19.0 scPCA_1.23.0
[3] batchelor_1.25.0 scran_1.37.0
[5] scater_1.37.0 ggplot2_4.0.0
[7] scuttle_1.19.0 AnnotationHub_3.99.6
[9] BiocFileCache_2.99.6 dbplyr_2.5.1
[11] ensembldb_2.33.2 AnnotationFilter_1.33.0
[13] GenomicFeatures_1.61.6 AnnotationDbi_1.71.2
[15] scRNAseq_2.23.1 SingleCellExperiment_1.31.1
[17] SummarizedExperiment_1.39.2 Biobase_2.69.1
[19] GenomicRanges_1.61.5 Seqinfo_0.99.2
[21] IRanges_2.43.5 S4Vectors_0.47.4
[23] BiocGenerics_0.55.3 generics_0.1.4
[25] MatrixGenerics_1.21.0 matrixStats_1.5.0
[27] BiocStyle_2.37.1 rebook_1.19.0
loaded via a namespace (and not attached):
[1] BiocIO_1.19.0 bitops_1.0-9
[3] filelock_1.0.3 tibble_3.3.0
[5] CodeDepends_0.6.6 graph_1.87.0
[7] XML_3.99-0.19 lifecycle_1.0.4
[9] httr2_1.2.1 Rdpack_2.6.4
[11] edgeR_4.7.6 globals_0.18.0
[13] lattice_0.22-7 alabaster.base_1.9.5
[15] magrittr_2.0.4 limma_3.65.5
[17] sass_0.4.10 rmarkdown_2.30
[19] jquerylib_0.1.4 yaml_2.3.10
[21] metapod_1.17.0 cowplot_1.2.0
[23] DBI_1.2.3 RColorBrewer_1.1-3
[25] ResidualMatrix_1.19.0 abind_1.4-8
[27] Rtsne_0.17 purrr_1.1.0
[29] RCurl_1.98-1.17 rappdirs_0.3.3
[31] ggrepel_0.9.6 irlba_2.3.5.1
[33] listenv_0.9.1 alabaster.sce_1.9.0
[35] RSpectra_0.16-2 parallelly_1.45.1
[37] dqrng_0.4.1 DelayedMatrixStats_1.31.0
[39] codetools_0.2-20 DelayedArray_0.35.3
[41] tidyselect_1.2.1 UCSC.utils_1.5.0
[43] farver_2.1.2 ScaledMatrix_1.17.0
[45] viridis_0.6.5 GenomicAlignments_1.45.4
[47] jsonlite_2.0.0 BiocNeighbors_2.3.1
[49] tools_4.5.1 Rcpp_1.1.0
[51] glue_1.8.0 gridExtra_2.3
[53] SparseArray_1.9.1 xfun_0.53
[55] GenomeInfoDb_1.45.12 dplyr_1.1.4
[57] HDF5Array_1.37.0 gypsum_1.5.0
[59] withr_3.0.2 BiocManager_1.30.26
[61] fastmap_1.2.0 sparsepca_0.1.2
[63] rhdf5filters_1.21.4 digest_0.6.37
[65] rsvd_1.0.5 R6_2.6.1
[67] dichromat_2.0-0.1 RSQLite_2.4.3
[69] h5mread_1.1.1 data.table_1.17.8
[71] rtracklayer_1.69.1 httr_1.4.7
[73] S4Arrays_1.9.1 pkgconfig_2.0.3
[75] gtable_0.3.6 blob_1.2.4
[77] S7_0.2.0 XVector_0.49.1
[79] htmltools_0.5.8.1 bookdown_0.45
[81] ProtGenerics_1.41.0 scales_1.4.0
[83] alabaster.matrix_1.9.0 png_0.1-8
[85] knitr_1.50 rjson_0.2.23
[87] curl_7.0.0 cachem_1.1.0
[89] rhdf5_2.53.6 stringr_1.5.2
[91] BiocVersion_3.22.0 KernSmooth_2.23-26
[93] parallel_4.5.1 vipor_0.4.7
[95] restfulr_0.0.16 pillar_1.11.1
[97] grid_4.5.1 alabaster.schemas_1.9.0
[99] vctrs_0.6.5 origami_1.0.7
[101] BiocSingular_1.25.0 beachmat_2.25.5
[103] cluster_2.1.8.1 beeswarm_0.4.0
[105] evaluate_1.0.5 cli_3.6.5
[107] locfit_1.5-9.12 compiler_4.5.1
[109] Rsamtools_2.25.3 rlang_1.1.6
[111] crayon_1.5.3 future.apply_1.20.0
[113] labeling_0.4.3 ggbeeswarm_0.7.2
[115] stringi_1.8.7 alabaster.se_1.9.0
[117] viridisLite_0.4.2 BiocParallel_1.43.4
[119] assertthat_0.2.1 Biostrings_2.77.2
[121] lazyeval_0.2.2 coop_0.6-3
[123] Matrix_1.7-4 dir.expiry_1.17.0
[125] ExperimentHub_2.99.5 future_1.67.0
[127] sparseMatrixStats_1.21.0 bit64_4.6.0-1
[129] Rhdf5lib_1.31.1 KEGGREST_1.49.2
[131] statmod_1.5.1 alabaster.ranges_1.9.1
[133] kernlab_0.9-33 rbibutils_2.3
[135] igraph_2.2.0 memoise_2.0.1
[137] bslib_0.9.0 bit_4.6.0
References
Messmer, T., F. von Meyenn, A. Savino, F. Santos, H. Mohammed, A. T. L. Lun, J. C. Marioni, and W. Reik. 2019. “Transcriptional heterogeneity in naive and primed human pluripotent stem cells at single-cell resolution.” Cell Rep 26 (4): 815–24.