#' Calculate 1 vs. All standard fold change for each gene x cell type, wrapper
#' function for scran::findMarkers
#'
#' @param sce single cell experiment object
#' @param assay_name Name of the assay to use for calculation
#' @param cellType_col Column name on colData of the sce that denotes the celltype
#' @param add_symbol Add the gene symbol column to the marker stats table
#' @param mod String specifying the model used as design in findMarkers. Can be `NULL` if there are no blocking terms with uninteresting factors as documented at [pairwiseTTests][scran::pairwiseTTests].
#' @param verbose Boolean choosing to print progress messages or not
#'
#' @return Table of 1 vs. ALL std log fold change + p-values for each gene x cell type
#' @export
#'
#' @examples
#' markers_1vAll <- findMarkers_1vAll(sce_ab)
#' head(markers_1vAll)
#'
#' @importFrom purrr map
#' @importFrom dplyr mutate
#' @importFrom scran findMarkers
#' @importFrom tibble rownames_to_column as_tibble add_column
findMarkers_1vAll <- function(sce, assay_name = "counts", cellType_col = "cellType", add_symbol = FALSE, mod = "~donor", verbose = TRUE) {
# RCMD Fix
gene <- rank_marker <- cellType.target <- std.logFC <- rowData <- Symbol <- NULL
cell_types <- unique(sce[[cellType_col]])
names(cell_types) <- cell_types
## Traditional t-test with design as in PB'd/limma approach
pd <- as.data.frame(SummarizedExperiment::colData(sce))
# message("donor" %in% colnames(pd))
if (!is.null(mod)) {
mod <- with(pd, stats::model.matrix(as.formula(mod)))
mod <- mod[, -1, drop = F] # intercept otherwise automatically dropped by `findMarkers()`
}
markers.t.1vAll <- map(cell_types, function(x) {
if (verbose) message(x, " - '", Sys.time())
sce$contrast <- ifelse(sce[[cellType_col]] == x, 1, 0)
fm <- scran::findMarkers(sce,
groups = sce$contrast,
assay.type = assay_name, design = mod, test.type = "t",
direction = "up", pval.type = "all", full.stats = TRUE
)
fm <- fm[[2]]$stats.0
fm.std <- scran::findMarkers(sce,
groups = sce$contrast,
assay.type = assay_name, design = mod, test.type = "t",
std.lfc = TRUE,
direction = "up", pval.type = "all", full.stats = TRUE
)
fm.std <- fm.std[[2]]$stats.0
colnames(fm.std)[[1]] <- "std.logFC"
return(cbind(fm, fm.std[, 1, drop = FALSE]))
})
if (verbose) message("Building Table - ", Sys.time())
markers.t.1vAll.table <- do.call("rbind", markers.t.1vAll) |>
as.data.frame() |>
tibble::rownames_to_column("gene") |>
mutate(gene = gsub("\\.\\d+", "", gene)) |>
tibble::as_tibble() |>
tibble::add_column(cellType.target = rep(cell_types, each = nrow(sce))) |>
dplyr::group_by(cellType.target) |>
mutate(
rank_marker = dplyr::row_number(),
anno_logFC = paste0(" std logFC = ", round(std.logFC, 3))
)
if (add_symbol) {
markers.t.1vAll.table$Symbol <- rowData(sce)[markers.t.1vAll.table$gene, ]$Symbol
markers.t.1vAll.table <- markers.t.1vAll.table |>
mutate(feature_marker = paste0(stringr::str_pad(rank_marker, 4, "left"), ": ", Symbol))
}
if (verbose) message("** Done! **\n")
return(markers.t.1vAll.table)
}
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.