Due to the sparsity observed in single-cell data (e.g. RNA-seq, ATAC-seq), the
visualization of cell features (e.g. gene, peak) is frequently affected and
unclear, especially when it is overlaid with clustering to annotate cell
types. Nebulosa
is an R package to visualize data from single cells based on
kernel density estimation. It aims to recover the signal from dropped-out
features by incorporating the similarity between cells allowing a "convolution"
of the cell features.
For this vignette, let's use Nebulosa
with the scran
and scater
packages.
First, we'll do a brief/standard data processing.
library("Nebulosa") library("scater") library("scran") library("DropletUtils") library("BiocFileCache")
Let's download a dataset of 3k PBMCs (available from 10X Genomics). For the
purpose of this vignette, let's use the BiocFileChache
package to dowload
the data and store it in a temporary directory defined by the tempdir()
function.
To import the count data, we'll use the read10xCounts
from the DropletUtils
package.
bfc <- BiocFileCache(ask = FALSE) data_file <- bfcrpath(bfc, file.path( "https://s3-us-west-2.amazonaws.com/10x.files/samples/cell", "pbmc3k", "pbmc3k_filtered_gene_bc_matrices.tar.gz" )) untar(data_file, exdir = tempdir()) pbmc <- read10xCounts(file.path(tempdir(), "filtered_gene_bc_matrices", "hg19" ))
The default feature names are Ensembl ids, let's use thegene names and
set them as row names of the sce
object. The following step will use the gene
names as rownames and make them unique by appending it's corresponding
Ensemble id when a gene-name duplicate is found.
rownames(pbmc) <- uniquifyFeatureNames(rowData(pbmc)[["ID"]], rowData(pbmc)[["Symbol"]])
First, let's remove features that are not expressed in at least 3 cells.
i <- rowSums(counts(pbmc) > 0) is_expressed <- i > 3 pbmc <- pbmc[is_expressed, ]
And cells not expressing at least one UMI in at least 200 genes.
i <- colSums(counts(pbmc) > 0) is_expressed <- i > 200 pbmc <- pbmc[,is_expressed]
Finally, let's remove outlier cells based on the number of genes being
expressed in each cell, library size, and expression of mitochondrial genes
using the perCellQCMetrics
and quickPerCellQC
functions from the scater
package.
is_mito <- grepl("^MT-", rownames(pbmc)) qcstats <- perCellQCMetrics(pbmc, subsets = list(Mito = is_mito)) qcfilter <- quickPerCellQC(qcstats, percent_subsets = c("subsets_Mito_percent"))
For more information on quality control, please visit the OSCA website: https://osca.bioconductor.org/quality-control.html
Let's normalize the data by scaling the counts from each cell across all genes
by the sequencing depth of each cell and using a scaling factor of 1 x 10^4.
Then, we can stabilize the variance by calculating the pseudo-natural logarithm
using the log1p
function.
logcounts(pbmc) <- log1p(counts(pbmc) / colSums(counts(pbmc)) * 1e4)
Please refer to the OSCA website for more details on other normalization strategies: https://osca.bioconductor.org/normalization.html
A reduced set of variable genes are expected to drive the major differences
between the cell populations. To identify these genes, let's use the
modelGeneVar()
and getTopHVGsfrom()
from scran
by selecting the top 3000
most highly-variable genes.
dec <- modelGeneVar(pbmc) top_hvgs <- getTopHVGs(dec, n = 3000)
Once the data is normalized and highly-variable features have been determined, we can run a Principal Component Analysis (PCA) to reduce the dimensions of our data to 50 principal components. Then, we can run a Uniform Manifold Approximation and Projection (UMAP) using the principal components to obtain a two-dimensional representation that could be visualized in a scatter plot.
set.seed(66) pbmc <- runPCA(pbmc, scale = TRUE, subset_row = top_hvgs)
Finally, we can run the UMAP as follows:
pbmc <- runUMAP(pbmc, dimred = "PCA")
To assess cell similarity, let's cluster the data by constructing a
Shared Nearest Neighbor (SNN) Graph using the first 50 principal components
and applying cluster_louvain()
from the igraph
package.
g <- buildSNNGraph(pbmc, k = 10, use.dimred = "PCA") clust <- igraph::cluster_louvain(g)$membership colLabels(pbmc) <- factor(clust)
Nebulosa
The main function from Nebulosa
is the plot_density
.
Let's plot the kernel density estimate for CD4
as follows
plot_density(pbmc, "CD4")
For comparison, let's also create a standard scatter plot using scater
plotUMAP(pbmc, colour_by = "CD4")
By smoothing the data, Nebulosa
allows a better visualization of the global
expression of CD4 in myeloid and CD4+ T cells. Notice that the "random"
expression of CD4 in other areas of the plot is removed as the expression of
this gene is not supported by enough cells in those areas. Furthermore, CD4+
cells appear to show considerable dropout rate.
Characterizing cell populations usually relies in more than one marker. Nebulosa also allows the visualization of the joint density from multiple features in a single plot.
Users familiarized with PBMC datasets may be aware that CD8+ CCR7+ cells usually cluster next to CD4+ CCR7+ and separate from the rest of CD8+ cells. Let's aim to identify naive CD8+ T cells. To do so, we can just add another gene to the vector containing the features to visualize.
p3 <- plot_density(pbmc, c("CD8A", "CCR7")) p3 + plot_layout(ncol = 1)
Nebulosa
can return a joint density plot by multiplying the densities
from all query genes by using the joint = TRUE
parameter:
p4 <- plot_density(pbmc, c("CD8A", "CCR7"), joint = TRUE) p4 + plot_layout(ncol = 1)
When compared to the clustering results, we can identify that Naive
CD8+ T cells are contained within cluster 4
.
plotUMAP(pbmc, colour_by = "label", text_by = "label")
Nebulosa
returns the density estimates for each gene along with the joint
density across all provided genes. By setting combine = FALSE
, we can obtain
a list of ggplot objects where the last plot corresponds to the joint density
estimate.
p_list <- plot_density(pbmc, c("CD8A", "CCR7"), joint = TRUE, combine = FALSE) p_list[[length(p_list)]]
Likewise, the identification of Naive CD4+ T cells becomes clearer by
combining CD4
and CCR7
:
p4 <- plot_density(pbmc, c("CD4", "CCR7"), joint = TRUE) p4 + plot_layout(ncol = 1)
CCR7
is predominantly expressed in cluster 4
, and its expression gets
reduced in cluster 2
, which suggests that cluster 4
may consist of most
naive T cells (both CD4+ and CD8+).
In summary,Nebulosa
can be useful to recover the signal from dropped-out genes
and improve their visualization in a two-dimensional space. We recommend using
Nebulosa
particularly for dropped-out genes. For fairly well-expressed genes,
the direct visualization of the gene expression may be preferable. We encourage
users to use Nebulosa
along with the core visualization methods from the
Seurat
and Bioconductor
environments as well as other visualization methods
to draw more informed conclusions about their data.
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.