#' Save the cytofkit analysis results
#'
#' Save analysis results from cytofkit main function to RData, csv files and PDF files and
#' add them to a new copy of FCS files.
#'
#' @param analysis_results Result data from output of \code{\link{cytofkit}}
#' @param projectName A prefix that will be added to the names of result files.
#' @param saveToRData Boolean value determines if results object is saved into RData file, for loading back to R and to shiny APP.
#' @param saveToFCS Boolean value determines if results are saved back to FCS files, new FCS files will be generated under folder XXX_analyzedFCS.
#' @param saveToFiles Boolean value determines if results are parsed and automatically saved to csv files and pdf figures.
#' @param resultDir The directory where result files will be generated.
#' @param rawFCSdir The directory that contains fcs files to be analysed.
#' @param inverseLgclTrans If \verb{TRUE}, inverse logicle transform the cluster cor1 and cor2 channels.
#' @return Save all results in the \code{resultDir}
#' @importFrom ggplot2 ggplot ggsave aes_string facet_wrap geom_point geom_rug theme_bw theme xlab ylab ggtitle coord_fixed guides guide_legend scale_shape_manual scale_colour_manual
#' @importFrom reshape2 dcast
#' @importFrom ggplot2 ggplot ggsave aes_string geom_line geom_point xlab ylab ggtitle theme_bw
#' @importFrom flowCore write.FCS flowFrame inverseLogicleTransform
#' @importFrom grDevices dev.off pdf rainbow
#' @importFrom graphics par
#' @importFrom colourpicker colourInput
#' @importFrom utils read.table write.csv
#' @export
#' @seealso \code{\link{cytofkit}}
#' @examples
#' d <- system.file('extdata',package='cytofkit')
#' f <- list.files(d, pattern='.fcs$', full=TRUE)
#' p <- list.files(d, pattern='.txt$', full=TRUE)
#' #tr <- cytofkit(fcsFile=f,markers=p,projectName='t',saveResults=FALSE)
#' #cytof_write_results(tr,projectName = 'test',resultDir=d,rawFCSdir =d)
cytof_writeResults <- function(analysis_results,
projectName,
saveToRData = TRUE,
saveToFCS = TRUE,
saveToFiles = TRUE,
resultDir,
rawFCSdir,
inverseLgclTrans = TRUE) {
## check projectName parameter
if(missing(projectName)){
if(!is.null(analysis_results$projectName)){
projectName <- analysis_results$projectName
}else{
projectName <- "cytofkit_"
}
}
## check resultDir parameter
if(missing(resultDir)){
if(!is.null(analysis_results$resultDir)){
resultDir <- analysis_results$resultDir
}else{
resultDir <- getwd()
}
}
if(!dir.exists(resultDir)){
dir.create(resultDir)
}
## check rawFCSdir parameter
if(missing(rawFCSdir)){
if(!is.null(analysis_results$rawFCSdir)){
rawFCSdir <- analysis_results$rawFCSdir
}
}
if(!dir.exists(rawFCSdir)){
if(saveToFCS){
saveToFCS <- FALSE
warning("Can not find the path for original FCS files. Data cannnot be
saved to new copies of FCS files. Please provide the correct path
to parameter rawFCSdir.")
}
}
curwd <- getwd()
setwd(resultDir)
exprs <- as.data.frame(analysis_results$expressionData)
dimReducedData <- analysis_results$dimReducedRes
clusterData <- analysis_results$clusterRes
## save analysis results to RData files
if(saveToRData){
objFile <- paste0(projectName, ".RData")
save(analysis_results, file = objFile)
cat("R object is saved in ", objFile, "\n")
message(" **THIS R OBJECT IS THE INPUT OF SHINY APP!** ")
}
## save analysis results to csv files and pdf figures
if(saveToFiles){
## save exprs
ifMultiFCS <- length(unique(sub("_[0-9]*$", "", row.names(exprs)))) > 1
write.csv(exprs, paste0(projectName, "_markerFiltered_transformed_merged_exprssion_data.csv"))
## save dimReducedData
for(i in 1:length(dimReducedData)){
methodi <- names(dimReducedData)[i]
if(!is.null(dimReducedData[[i]])){
write.csv(dimReducedData[[i]], paste(projectName, methodi,"dimension_reduced_data.csv", sep="_"))
}
}
## save clusterData
if(!is.null(clusterData) && length(clusterData) > 0){
for(j in 1:length(clusterData)){
methodj <- names(clusterData)[j]
dataj <- clusterData[[j]]
if(!is.null(dataj)){
write.csv(dataj, paste(projectName, methodj, "clusters.csv", sep="_"))
## expression values by cluster
for(i in unique(dataj)){
ci.Table <- cytof_clusterMtrx(analysis_results, methodj, i)
write.csv(ci.Table, paste(projectName, methodj, "Cluster", i, "expression_values.csv", sep = "_"))
}
exprs_cluster_sample <- data.frame(exprs, cluster = dataj, check.names = FALSE)
## cluster mean
cluster_mean <- cytof_clusterStat(data= exprs_cluster_sample, cluster = "cluster", statMethod = "mean")
write.csv(cluster_mean, paste(projectName, methodj, "cluster_mean_data.csv", sep = "_"))
pdf(paste(projectName, methodj, "cluster_mean_heatmap.pdf", sep = "_"))
cytof_heatmap(cluster_mean, paste(projectName, methodj, "\ncluster mean", sep = " "))
dev.off()
## cluster median
cluster_median <- cytof_clusterStat(data= exprs_cluster_sample, cluster = "cluster", statMethod = "median")
write.csv(cluster_median, paste(projectName, methodj, "cluster_median_data.csv", sep = "_"))
pdf(paste(projectName, methodj, "cluster_median_heatmap.pdf", sep = "_"))
cytof_heatmap(cluster_median, paste(projectName, methodj, "\ncluster median", sep = " "))
dev.off()
## cluster percentage
if (ifMultiFCS) {
cluster_percentage <- cytof_clusterStat(data= exprs_cluster_sample, cluster = "cluster", statMethod = "percentage")
write.csv(cluster_percentage, paste(projectName, methodj, "cluster_cell_percentage.csv", sep = "_"))
pdf(paste(projectName, methodj, "cluster_percentage_heatmap.pdf", sep = "_"))
cytof_heatmap(cluster_percentage, paste(projectName, methodj, "cluster\ncell percentage", sep = " "))
dev.off()
}
}
}
}
## expression values by cluster
if(!is.null(clusterData) && length(clusterData) > 0)
## visualizationData x clusterData plot
visualizationData <- analysis_results$dimReducedRes[analysis_results$visualizationMethods]
for(i in 1:length(visualizationData)){
if(!is.null(visualizationData[[i]])){
methodi <- names(visualizationData)[i]
datai <- as.data.frame(visualizationData[[i]])
if(!is.null(clusterData) && length(clusterData) > 0){
for(j in 1:length(clusterData)){
if(!is.null(clusterData[[j]])){
methodj <- names(clusterData)[j]
dataj <- clusterData[[j]]
# combine datai and dataj
xlab <- colnames(datai)[1]
ylab <- colnames(datai)[2]
dataij <- datai
dataij$sample <- sub("_[0-9]*$", "", row.names(dataij))
dataij$cluster <- factor(dataj)
cluster <- "cluster"
sample <- "sample"
## cluster plot
figName <- paste(projectName, methodi, methodj, sep=" ")
cluster_plot <- cytof_clusterPlot(dataij, xlab, ylab, cluster, sample, figName, 1)
ggsave(filename = paste(projectName, methodi, methodj, "cluster_scatter_plot.pdf", sep = "_"),
cluster_plot, width = 12, height = 10)
## cluster grid plot if multiple files
if (ifMultiFCS) {
figName <- paste(projectName, methodi, methodj, sep=" ")
cluster_grid_plot <- cytof_clusterPlot(dataij, xlab, ylab, cluster, sample, figName, 2)
ggsave(filename = paste(projectName, methodi, methodj, "cluster_grid_scatter_plot.pdf", sep = "_"), cluster_grid_plot)
}
}
}
}
}
}
}
#create sampleinfo here instead of at shiny app
samples <- NULL
for(i in seq_along(analysis_results$sampleNames)){
samples <- c(samples, analysis_results$sampleNames[[i]][1])
}
## save analysis results to FCS files
if(saveToFCS == TRUE){
tcols <- do.call(cbind, dimReducedData)
ctols <- do.call(cbind, clusterData)
dataToAdd <- cbind(tcols, ctols)
row.names(dataToAdd) <- row.names(exprs)
trans_col_names <- colnames(tcols)
cluster_col_names <- colnames(ctols)
cytof_addToFCS(dataToAdd,
rawFCSdir = rawFCSdir,
origSampNames = samples,
analyzedFCSdir = paste(projectName, "analyzedFCS", sep = "_"),
transformed_cols = trans_col_names,
cluster_cols = cluster_col_names,
clusterIDs = ctols,
inLgclTrans = inverseLgclTrans)
}
setwd(curwd)
message(paste0("Writing results Done! Results are saved under path: ",
resultDir))
invisible(NULL)
}
#' Scatter plot of the cluster results
#'
#' Dot plot visualization of the cluster results, with color indicating different clusters,
#' and shape of different samples.
#'
#' @param data The data frame of cluster results, which should contains at least xlab, ylab and cluster.
#' @param xlab The column name of the x axis in input \code{data}.
#' @param ylab The column name of the y axis in input \code{data}.
#' @param cluster The column name of cluster in input \code{data}.
#' @param sample The column name of the sample in input \code{data}.
#' @param title The title of the plot.
#' @param type Plot type, 1 indicates combined plot, 2 indicated grid facet plot seperated by samples.
#' @param point_size Size of the dot.
#' @param addLabel If \verb{TRUE}, add cluster labels.
#' @param labelSize Size of cluster labels.
#' @param sampleLabel If \verb{TRUE}, use point shapes to represent different samples.
#' @param labelRepel If \verb{TRUE}, repel the cluste labels to avoid label overlapping.
#' @param fixCoord If \verb{TRUE}, fix the Cartesian coordinates.
#' @param clusterColor Manually specify the colour of each cluster (mainly for ShinyAPP usage).
#' @return The ggplot object of the scatter cluster plot.
#' @export
#' @importFrom ggplot2 element_text element_rect element_blank element_line element_text annotate
#' @examples
#' x <- c(rnorm(100, mean = 1), rnorm(100, mean = 3), rnorm(100, mean = 9))
#' y <- c(rnorm(100, mean = 2), rnorm(100, mean = 8), rnorm(100, mean = 5))
#' c <- c(rep(1,100), rep(2,100), rep(3,100))
#' rnames <- paste(paste('sample_', c('A','B','C'), sep = ''), rep(1:100,each = 3), sep='_')
#' data <- data.frame(dim1 = x, dim2 = y, cluster = c)
#' rownames(data) <- rnames
#' data$sample <- "data"
#' cytof_clusterPlot(data, xlab="dim1", ylab="dim2", cluster="cluster", sample = "sample")
cytof_clusterPlot <- function(data, xlab, ylab, cluster, sample, title = "cluster",
type = 1, point_size = NULL, addLabel=TRUE,
labelSize=10, sampleLabel=TRUE,
labelRepel = FALSE, fixCoord=TRUE, clusterColor) {
if(!is.data.frame(data))
data <- as.data.frame(data)
if(missing(sample)){
sample <- "sample"
data$sample <- "data"
}
paraCheck <- c(xlab, ylab, cluster, sample) %in% colnames(data)
if(any(!paraCheck)){
stop("Undefined parameters found: ",
c(xlab, ylab, cluster, sample)[!paraCheck])
}
data[[cluster]] <- as.factor(data[[cluster]])
data[[sample]] <- as.factor(data[[sample]])
cluster_num <- length(unique(data[[cluster]]))
sample_num <- length(unique(data[[sample]]))
col_legend_row <- ceiling(cluster_num/15)
size_legend_row <- ceiling(sample_num/4)
grid_row_num <- round(sqrt(sample_num))
if (sample_num >= 8) {
shape_value <- LETTERS[1:sample_num]
} else {
shape_value <- c(1:sample_num) + 15
}
if (is.null(point_size)) {
point_size <- ifelse(nrow(data) > 10000, 1, 1.5)
}
if(missing(clusterColor) || is.null(clusterColor)){
clusterColor <- rainbow(cluster_num)
}else if(length(clusterColor) == 0 || length(clusterColor) != cluster_num){
clusterColor <- rainbow(cluster_num)
}
if(type == 1){
if(sampleLabel){
cp <- ggplot(data, aes_string(x = xlab, y = ylab, colour = cluster, shape = sample)) +
geom_point(size = point_size) + scale_shape_manual(values = shape_value) +
scale_colour_manual(values = clusterColor) +
xlab(xlab) + ylab(ylab) + ggtitle(paste(title, "Scatter Plot", sep = " ")) +
theme_bw() + theme(legend.position = "bottom") +
theme(axis.text=element_text(size=14), axis.title=element_text(size=18,face="bold")) +
guides(colour = guide_legend(nrow = col_legend_row, override.aes = list(size = 4)),
shape = guide_legend(nrow = size_legend_row, override.aes = list(size = 4)))
}else{
cp <- ggplot(data, aes_string(x = xlab, y = ylab, colour = cluster)) +
geom_point(size = point_size) + scale_shape_manual(values = shape_value) +
scale_colour_manual(values = clusterColor) +
xlab(xlab) + ylab(ylab) + ggtitle(paste(title, "Scatter Plot", sep = " ")) +
theme_bw() + theme(legend.position = "bottom") +
theme(axis.text=element_text(size=14), axis.title=element_text(size=18,face="bold")) +
guides(colour = guide_legend(nrow = col_legend_row, override.aes = list(size = 4)))
}
if(addLabel){
edata <- data[ ,c(xlab, ylab, cluster)]
colnames(edata) <- c('x', "y", "z")
center <- aggregate(cbind(x,y) ~ z, data = edata, median)
if(labelRepel && !sampleLabel){
cp <- cp + geom_text_repel(data=center, aes_string(x = "x", y = "y", label = "z"),
size = labelSize, fontface = 'bold', color = "black",
box.padding = unit(0.5, 'lines'),
point.padding = unit(1.6, 'lines'),
segment.color = '#555555',
segment.size = 0.5,
arrow = arrow(length = unit(0.02, 'npc')))
}else{
cp <- cp + annotate("text", label = center[,1], x=center[,2], y = center[,3],
size = labelSize, colour = "black")
}
}
}else if (type == 2){
cp <- ggplot(data, aes_string(x = xlab, y = ylab, colour = cluster)) +
facet_wrap(~sample, nrow = grid_row_num, scales = "fixed") +
geom_point(size = point_size - 0.05 * sample_num) +
scale_colour_manual(values = clusterColor) +
xlab(xlab) + ylab(ylab) + ggtitle(paste(title, "Grid Plot", sep = " ")) +
theme_bw() + theme(legend.position = "bottom") +
theme(axis.text=element_text(size=14), axis.title=element_text(size=18,face="bold")) +
guides(colour = guide_legend(nrow = col_legend_row, override.aes = list(size = 4)),
shape = guide_legend(nrow = size_legend_row, override.aes = list(size = 4)))
if(addLabel){
edata <- data[ ,c(xlab, ylab, cluster)]
colnames(edata) <- c('x', "y", "z")
center <- aggregate(cbind(x,y) ~ z, data = edata, median)
if(labelRepel && !sampleLabel){
cp <- cp + geom_text_repel(data=center, aes_string(x = "x", y = "y", label = "z"),
size = labelSize, fontface = 'bold', color = "black",
box.padding = unit(0.5, 'lines'),
point.padding = unit(1.6, 'lines'),
segment.color = '#555555',
segment.size = 0.5,
arrow = arrow(length = unit(0.02, 'npc')))
}else{
cp <- cp + geom_text(data=center, aes_string(x = "x", y = "y", label = "z"),
size = labelSize, colour = "black")
}
}
}else{
stop("Undefined type, only 1 or 2.")
}
if(fixCoord){
cp <- cp + coord_fixed()
}
return(cp)
}
#' Heatmap plot of cluster mean value results
#'
#' @param data A matrix with rownames and colnames
#' @param baseName The name as a prefix in the title of the heatmap.
#' @param scaleMethod Method indicating if the values should be centered and scaled in either the row direction or the column direction, or none. The default is 'none'.
#' @param dendrogram Control the dengrogram on row or column, selection includes 'both', 'row', 'column', 'none'.
#' @param colPalette Use selected colour palette, includes 'bluered', 'greenred', 'spectral1' and 'spectral2'.
#' @param cex_row_label Text size for row labels.
#' @param cex_col_label Text size for column labels.
#' @param key.par Graphical parameters for the color key.
#' @param keysize Numeric value indicating the size of the key.
#' @param margins Numeric vector of length 2 containing the margins (see par(mar= *)) for column and row names, respectively.
#' @return A heatmap object from \code{gplots}
#'
#' @importFrom gplots heatmap.2 bluered
#'
#' @export
#' @examples
#' m1 <- c(rnorm(300, 10, 2), rnorm(400, 4, 2), rnorm(300, 7))
#' m2 <- c(rnorm(300, 4), rnorm(400, 16), rnorm(300, 10, 3))
#' m3 <- c(rnorm(300, 16), rnorm(400, 40, 3), rnorm(300, 10))
#' m4 <- c(rnorm(300, 7, 3), rnorm(400, 30, 2), rnorm(300, 10))
#' m5 <- c(rnorm(300, 27), rnorm(400, 40, 1),rnorm(300, 10))
#' c <- c(rep(1,300), rep(2,400), rep(3,300))
#' rnames <- paste(paste('sample_', c('A','B','C','D'), sep = ''),
#' rep(1:250,each = 4), sep='_')
#' exprs_cluster <- data.frame(cluster = c, m1 = m1, m2 = m2, m3 = m3, m4 = m4, m5 = m5)
#' row.names(exprs_cluster) <- sample(rnames, 1000)
#' cluster_mean <- aggregate(. ~ cluster, data = exprs_cluster, mean)
#' rownames(cluster_mean) <- paste("cluster_", cluster_mean$cluster, sep = "")
#' cytof_heatmap(cluster_mean[, -which(colnames(cluster_mean) == "cluster")])
cytof_heatmap <- function(data, baseName = "Cluster", scaleMethod = "none",
dendrogram = c("both","row","column","none"),
colPalette = c("bluered", "greenred", "spectral1", "spectral2"),
cex_row_label = NULL, cex_col_label = NULL,
key.par = list(mgp=c(1.5, 0.5, 0), mar=c(3, 2.5, 3.5, 1)),
keysize = 1.4,
margins = c(5, 5)){
data <- as.matrix(data)
dendrogram <- match.arg(dendrogram)
colPalette <- match.arg(colPalette)
if(is.null(cex_row_label)){
cex_row_label <- (11 - ceiling(nrow(data)/10))/10
}
if(is.null(cex_col_label)){
cex_col_label <- (11 - ceiling(ncol(data)/10))/10
}
if(dendrogram == "row"){
dendrogramRowv <- TRUE
dendrogramColv <- FALSE
}else if (dendrogram == "column"){
dendrogramRowv <- FALSE
dendrogramColv <- TRUE
}else if(dendrogram == "none"){
dendrogramRowv <- FALSE
dendrogramColv <- FALSE
}else{
dendrogramRowv <- TRUE
dendrogramColv <- TRUE
}
heatmap.2(x = data,
Rowv = dendrogramRowv,
Colv= dendrogramColv,
dendrogram = dendrogram,
col = colPalette,
trace = "none",
symbreaks = FALSE,
scale = scaleMethod,
cexRow = cex_row_label,
cexCol = cex_col_label,
srtCol = 30, symkey = FALSE,
key.par=key.par,
margins = margins,
keysize = keysize,
main = paste(baseName, "Heat Map"))
}
#' First Function for spectral color palette
#'
#' @param n Number of colors.
#'
#' @return Hex colour values
#' @export
#'
#' @examples
#' spectral1(2)
spectral1 <- function(n){
spectralPalette <- colorRampPalette(c("#5E4FA2", "#3288BD", "#66C2A5", "#ABDDA4",
"#E6F598", "#FFFFBF", "#FEE08B", "#FDAE61",
"#F46D43", "#D53E4F", "#9E0142"))
return(spectralPalette(n))
}
#' Second Function for spectral color palette
#'
#' @param n Number of colors.
#'
#' @return Hex colour values
#' @export
#'
#' @examples
#' spectral2(2)
spectral2 <- function(n){
spectralPalette <- colorRampPalette(rev(c("#7F0000","red","#FF7F00","yellow","white",
"cyan", "#007FFF", "blue","#00007F")))
return(spectralPalette(n))
}
#' Plot the data with color-coded marker values
#'
#' @param data A dataframe containing the xlab, ylab and zlab.
#' @param xlab The column name of data for x lab.
#' @param ylab The column name of data for y lab.
#' @param zlab The column name of data for z lab.
#' @param colorPalette Color Palette.
#' @param limits Range for z lab, defaults to existing min, and max set at 98th percentile value.
#' @param pointSize Size of the point.
#' @param alpha Transparency of point. 1 is opaque, 0 is completely transparent.
#' @param removeOutlier If \verb{TRUE}, remove the outliers.
#' @return A ggplot object.
#'
#' @importFrom ggplot2 scale_colour_gradientn
#' @importFrom grDevices colorRampPalette topo.colors heat.colors terrain.colors cm.colors
#'
#' @export
#' @examples
#' x <- c(rnorm(100, mean = 1), rnorm(100, mean = 3), rnorm(100, mean = 9))
#' y <- c(rnorm(100, mean = 2), rnorm(100, mean = 8), rnorm(100, mean = 5))
#' c <- rnorm(300, 10, 5)
#' data <- data.frame(dim1 = x, dim2 = y, marker = c)
#' cytof_colorPlot(data = data, xlab = "dim1", ylab = "dim2", zlab = "marker")
cytof_colorPlot <- function(data, xlab, ylab, zlab,
colorPalette = c("bluered", "spectral1", "spectral2", "heat"),
limits = c(quantile(data[,zlab], .02), quantile(data[,zlab], .98)),
pointSize=0.5,
alpha = 1,
removeOutlier = TRUE){
remove_outliers <- function(x, na.rm = TRUE, ...) {
qnt <- quantile(x, probs=c(.02, .98), na.rm = na.rm, ...)
y <- x
y[x < qnt[1]] <- qnt[1]
y[x > qnt[2]] <- qnt[2]
y
}
data <- as.data.frame(data)
title <- paste(zlab, "Expression Level Plot")
data <- data[,c(xlab, ylab, zlab)]
limits <- limits
if(removeOutlier)
data[,zlab] <- remove_outliers(data[,zlab])
colorPalette <- match.arg(colorPalette)
switch(colorPalette,
bluered = {
myPalette <- colorRampPalette(c("blue", "white", "red"))
},
spectral1 = {
myPalette <- colorRampPalette(c("#5E4FA2", "#3288BD", "#66C2A5", "#ABDDA4",
"#E6F598", "#FFFFBF", "#FEE08B", "#FDAE61",
"#F46D43", "#D53E4F", "#9E0142"))
},
spectral2 = {
myPalette <- colorRampPalette(rev(c("#7F0000","red","#FF7F00","yellow","white",
"cyan", "#007FFF", "blue","#00007F")))
},
heat = {
myPalette <- colorRampPalette(heat.colors(50))
}
)
zlength <- nrow(data)
exp <- "Expression"
colnames(data) <- c(xlab, ylab, exp)
gp <- ggplot(data, aes_string(x = xlab, y = ylab, colour = exp)) +
scale_colour_gradientn(limits = limits, name = zlab, colours = myPalette(zlength * 2)) +
geom_point(size = pointSize, alpha = alpha) + theme_bw() + coord_fixed() +
theme(legend.position = "right") + xlab(xlab) + ylab(ylab) + ggtitle(title) +
theme(panel.grid.major = element_blank(), panel.grid.minor = element_blank()) +
theme(axis.text=element_text(size=8), axis.title=element_text(size=12,face="bold"))
return(gp)
}
#' Statistics of the cluster results
#'
#' Calculate the mean or median expression level of each marker for each cluster, or percentage
#' of cell numbers of each cluster for each sample.
#'
#' @param data Input data frame.
#' @param markers The names of markers used for calcualtion.
#' @param cluster The column name contatining cluster labels.
#' @param sample The samples used for calculation.
#' @param statMethod Statistics containing mean, median or percentage.
#'
#' @return A matrix of the statistics results
#'
#' @importFrom stats aggregate
#' @importFrom reshape2 dcast
#' @export
#' @examples
#' m1 <- c(rnorm(300, 10, 2), rnorm(400, 4, 2), rnorm(300, 7))
#' m2 <- c(rnorm(300, 4), rnorm(400, 16), rnorm(300, 10, 3))
#' m3 <- c(rnorm(300, 16), rnorm(400, 40, 3), rnorm(300, 10))
#' m4 <- c(rnorm(300, 7, 3), rnorm(400, 30, 2), rnorm(300, 10))
#' m5 <- c(rnorm(300, 27), rnorm(400, 40, 1),rnorm(300, 10))
#' c <- c(rep(1,300), rep(2,400), rep(3,300))
#' rnames <- paste(paste('sample_', c('A','B','C','D'), sep = ''),
#' rep(1:250,each = 4), sep='_')
#' exprs_cluster <- data.frame(cluster = c, m1 = m1, m2 = m2, m3 = m3, m4 = m4, m5 = m5)
#' row.names(exprs_cluster) <- rnames
#' cytof_clusterStat(data = exprs_cluster, cluster = "cluster", statMethod = "mean")
cytof_clusterStat <- function(data, markers, cluster = "cluster", sample,
statMethod = c("mean", "median", "percentage", "NULL")){
data <- as.data.frame(data)
statMethod <- match.arg(statMethod)
if(missing(sample)){
sample <- "sample"
data$sample <- sub("_[0-9]*$", "", row.names(data))
}
if(missing(markers)){
markers <- setdiff(colnames(data), c(cluster, sample))
}
exprs_cluster <- data[ ,markers, drop=FALSE]
exprs_cluster$cluster <- data[[cluster]]
switch(statMethod,
mean = {
statData <- aggregate(. ~ cluster, data = exprs_cluster, mean)
},
median = {
statData <- aggregate(. ~ cluster, data = exprs_cluster, median)
},
percentage = {
cluster_sample <- data.frame(cluster = data[[cluster]], sample = data[[sample]])
cluster_sample$value <- 1
clust_sample_count <- dcast(cluster_sample, cluster ~ sample, fun.aggregate = length)
statData <- apply(clust_sample_count[,-1], 2, function(x){round(x/sum(x)*100, 2)})
statData <- data.frame(statData, cluster = clust_sample_count$cluster, check.names = FALSE)
})
if(is.numeric(statData$cluster)){
rownames(statData) <- paste0("cluster_", statData$cluster)
}else{
rownames(statData) <- statData$cluster
}
statData$cluster <- NULL ## remove cluster column
return(as.matrix(statData))
}
#' Expression values by cluster
#'
#' Generate a matrix of expression values for cells in a cluster
#'
#' @param analysis_results analysis_results output of cytofkit to extract expression values from.
#' @param clusterMethod The cluster method of interest. If none is selected, the first one will be used.
#' @param cluster The cluster of interest.
#'
#' @return A data table with marker expression values of cells in a given cluster
#'
#' @export
#'
#' @examples
#' dir <- system.file('extdata',package='cytofkit')
#' file <- list.files(dir ,pattern='.RData$', full=TRUE)
#' load(file)
#' cluster2_table <- cytof_clusterMtrx(analysis_results, "Rphenograph", 2)
cytof_clusterMtrx <- function(analysis_results, clusterMethod = NULL, cluster){
exprs <- analysis_results$expressionData
clusterRes <- analysis_results$clusterRes
clustMeths <- names(analysis_results$clusterRes)
if(is.null(clustMeths)){
stop("No clustering data, please check your analysis results!")
}
if(is.null(clusterMethod)){
clusterMethod <- clustMeths[1]
}
if(!clusterMethod %in% clustMeths){
stop("Selected clusterMethod not used in chosen analysis results, please check again")
}
if(!cluster %in% unique(analysis_results$clusterRes[[clusterMethod]])){
stop("Selected cluster not found!")
}
cluster_cells <- names(which(clusterRes[[clusterMethod]] == cluster))
cluster_table <- exprs[cluster_cells,]
return(cluster_table)
}
#' Progression plot
#'
#' Plot the expression trend along the estimated cell progressing order
#'
#' @param data The data frame for progression plot.
#' @param markers The column names of the selected markers for visualization.
#' @param clusters Select clusters for plotting, default selects all.
#' @param orderCol The column name of the estimated cell progression order.
#' @param clusterCol The column name of the cluster results.
#' @param reverseOrder If \verb{TRUE}, reverse the value of orderCol.
#' @param addClusterLabel If \verb{TRUE}, add the cluster label on the plot.
#' @param clusterLabelSize Size of the cluster label.
#' @param segmentSize Size of the cluster label arrow.
#' @param min_expr The threshold of the minimal expression value for markers.
#' @param trend_formula A symbolic description of the model to be fit.
#'
#' @return A ggplot2 object
#' @importFrom VGAM vgam sm.ns
#' @importFrom ggplot2 arrow unit
#' @importFrom reshape2 melt
#' @importFrom plyr ddply .
#' @importFrom ggrepel geom_text_repel
#'
#' @export
#'
#' @examples
#' m1 <- c(rnorm(300, 10, 2), rnorm(400, 4, 2), rnorm(300, 7))
#' m2 <- c(rnorm(300, 4), rnorm(400, 16), rnorm(300, 10, 3))
#' m3 <- c(rnorm(300, 16), rnorm(400, 40, 3), rnorm(300, 10))
#' m4 <- c(rnorm(300, 7, 3), rnorm(400, 30, 2), rnorm(300, 10))
#' m5 <- c(rnorm(300, 27), rnorm(400, 40, 1),rnorm(300, 10))
#' c <- c(rep(1,300), rep(2,400), rep(3,300))
#' rnames <- paste(paste('sample_', c('A','B','C','D'), sep = ''),
#' rep(1:250,each = 4), sep='_')
#' exprs_cluster <- data.frame(cluster = c, m1 = m1, m2 = m2, m3 = m3, m4 = m4, isomap_1 = m5)
#' row.names(exprs_cluster) <- sample(rnames, 1000)
#' cytof_progressionPlot(exprs_cluster, markers = c("m1","m2","m3","m4"))
cytof_progressionPlot <- function(data, markers, clusters,
orderCol="isomap_1",
clusterCol = "cluster",
reverseOrder = FALSE,
addClusterLabel = TRUE,
clusterLabelSize = 5,
segmentSize = 0.5,
min_expr = NULL,
trend_formula="expression ~ sm.ns(Pseudotime, df=3)"){
if(!is.data.frame(data)) data <- data.frame(data, check.names = FALSE)
if(!all(markers %in% colnames(data))) stop("Unmatching markers found!")
if(!(length(orderCol)==1 && orderCol %in% colnames(data)))
stop("Can not find orderCol in data!")
if(!(length(clusterCol)==1 && clusterCol %in% colnames(data)))
stop("Can not find clusterCol in data!")
if(!missing(clusters)){
if(!all(clusters %in% data[[clusterCol]]))
stop("Wrong clusters selected!")
data <- data[data[[clusterCol]] %in% clusters, , drop=FALSE]
}
if(reverseOrder){
newOrderCol <- paste0(orderCol, "(reverse)")
data[[newOrderCol]] <- -data[[orderCol]]
orderCol <- newOrderCol
}
orderValue <- data[[orderCol]]
data <- data[order(orderValue), c(markers, clusterCol)]
data$Pseudotime <- sort(orderValue)
mdata <- melt(data, id.vars = c("Pseudotime", clusterCol),
variable.name = "markers", value.name= "expression")
colnames(mdata) <- c("Pseudotime", clusterCol, "markers", "expression")
mdata$markers <- factor(mdata$markers)
mdata[[clusterCol]] <- factor(mdata[[clusterCol]])
min_expr <- min(mdata$expression)
## tobit regression
vgamPredict <- ddply(mdata, .(markers), function(x) {
fit_res <- tryCatch({
vg <- suppressWarnings(vgam(formula = as.formula(trend_formula),
family = VGAM::tobit(Lower = min_expr, lmu = "identitylink"),
data = x, maxit=30, checkwz=FALSE))
res <- VGAM::predict(vg, type="response")
res[res < min_expr] <- min_expr
res
}
,error = function(e) {
print("Error!")
print(e)
res <- rep(NA, nrow(x))
res
}
)
expectation = fit_res
data.frame(Pseudotime=x[["Pseudotime"]], expectation=expectation)
})
color_by <- clusterCol
plot_cols <- round(sqrt(length(markers)))
cell_size <- 1
x_lab <- orderCol
y_lab <- "Expression"
legend_title <- "Cluster"
## copied from monocle package
monocle_theme_opts <- function(){
theme(strip.background = element_rect(colour = 'white', fill = 'white')) +
#theme(panel.border = element_blank(), axis.line = element_line()) +
theme(panel.grid.minor.x = element_blank(), panel.grid.minor.y = element_blank()) +
theme(panel.grid.major.x = element_blank(), panel.grid.major.y = element_blank()) +
theme(panel.background = element_rect(fill='white')) +
theme(legend.position = "right") +
theme(axis.title = element_text(size = 15)) +
theme(axis.text=element_text(size=8), axis.title=element_text(size=12,face="bold"))}
q <- ggplot(aes_string(x="Pseudotime", y="expression"), data=mdata)
q <- q + geom_point(aes_string(color=color_by), size=I(cell_size))
q <- q + geom_line(aes_string(x="Pseudotime", y="expectation"), data=vgamPredict, size = 0.9)
q <- q + facet_wrap(~markers, ncol=plot_cols, scales="free_y")
q <- q + ylab(y_lab) + xlab(x_lab) + theme_bw()
q <- q + guides(colour = guide_legend(title = legend_title, override.aes = list(size = cell_size*3)))
q <- q + monocle_theme_opts()
if(addClusterLabel){
mdata$cluster <- mdata[[clusterCol]]
center <- aggregate(cbind(Pseudotime, expression) ~ cluster + markers, data = mdata, median)
q <- q + geom_text_repel(data=center, aes(x=Pseudotime, y=expression, label=cluster),
size = clusterLabelSize, fontface = 'bold',
box.padding = unit(0.5, 'lines'),
point.padding = unit(1.6, 'lines'),
segment.color = '#555555',
segment.size = segmentSize,
arrow = arrow(length = unit(0.02, 'npc')))
}
q
}
#' Add data to the original FCS files
#'
#' Store the new dimension transformed data and cluster data into the exprs
#' matrix in new fcs files under \code{analyzedFCSdir}
#'
#' @param data The new data matrix to be added in.
#' @param rawFCSdir The directory containing the original fcs files.
#' @param origSampNames Vector of original names of samples, if samples were renamed.
#' @param analyzedFCSdir The directory to store the new fcs files.
#' @param transformed_cols The column name of the dimension transformed data in \code{data}.
#' @param cluster_cols The column name of the cluster data in \code{data}.
#' @param clusterIDs Table of cell cluster IDs for each clustering method
#' @param specifySampleNames Used only if sample names differ from those in raw fcs.
#' @param inLgclTrans If \verb{TRUE}, apply the inverse lgcl transformation to the the cluster data before saving
#'
#' @export
#'
#' @return New fcs files stored under \code{analyzedFCSdir}
#' @importMethodsFrom flowCore keyword
#' @importFrom Biobase exprs exprs<- description description<- pData pData<-
cytof_addToFCS <- function(data,
rawFCSdir,
origSampNames = NULL,
analyzedFCSdir,
transformed_cols = c("tsne_1", "tsne_2"),
cluster_cols = c("cluster"),
clusterIDs = NULL,
specifySampleNames = NULL,
inLgclTrans = TRUE) {
lgcl <- logicleTransform(w = 0.1, t = 4000, m = 4.5, a = 0)
ilgcl <- inverseLogicleTransform(trans = lgcl)
if (!dir.exists(analyzedFCSdir)) {
dir.create(analyzedFCSdir)
}
## transform transformed_cols
if (!is.null(transformed_cols)) {
transformed <- data[, transformed_cols, drop=FALSE]
## ilgcl transformation of t-SNE
N_transformed <- apply(transformed, 2, function(x) ((x-min(x))/(max(x)-min(x)))*4.4)
R_N_transformed <- apply(N_transformed,2,ilgcl)
## linear transformation of tSNE
R_N_transformed_l <- apply(transformed, 2, function(x) (x - min(x)) + 0.1)
colnames(R_N_transformed_l) <- paste0(colnames(R_N_transformed_l), "_linear", sep = "")
## output both ilgcl and linear transformaton of tSNE for visualization on flowJo
R_N_transformed <- cbind(R_N_transformed, R_N_transformed_l)
row.names(R_N_transformed) <- row.names(data)
}
## use original sample names
if (!is.null(specifySampleNames)) {
originalSample <- specifySampleNames
}
## transform cluster_cols
if (!is.null(cluster_cols)) {
if(inLgclTrans){
for(i in 1:length(cluster_cols)){
cCol <- data[, cluster_cols[i]]
clust_cor_1 <- as.numeric(cCol)%%10
clust_cor_2 <- floor(as.numeric(cCol)/10)
clust_cor_1 <- clust_cor_1 + runif(length(clust_cor_1), 0, 0.2)
clust_cor_2 <- clust_cor_2 + runif(length(clust_cor_2), 0, 0.2)
cluster_cor12 <- cbind(clust_cor_1, clust_cor_2)
N_clust_cor <- apply(cluster_cor12, 2, function(x) ((x - min(x))/(max(x) - min(x))) * 4.4)
clust_cor <- apply(N_clust_cor, 2, ilgcl)
colnames(clust_cor) <- paste(cluster_cols[i], c("cor_1", "cor_2"), sep="_")
if(i == 1){
R_N_clust_cor <- clust_cor
}else{
R_N_clust_cor <- cbind(R_N_clust_cor, clust_cor)
}
}
}else{
R_N_clust_cor <- data[, cluster_cols, drop = FALSE]
}
row.names(R_N_clust_cor) <- row.names(data)
}
## write data to FCS
if ((!is.null(transformed_cols)) && (!is.null(cluster_cols))) {
to_add <- cbind(R_N_transformed, R_N_clust_cor)
} else if (!is.null(transformed_cols)) {
to_add <- R_N_transformed
} else if (!is.null(cluster_cols)) {
to_add <- R_N_clust_cor
}
if(!is.null(clusterIDs)){
colnames(clusterIDs) <- paste0(colnames(clusterIDs), "_clusterIDs")
to_add <- cbind(to_add, clusterIDs)
}
addColNames <- colnames(to_add)
sample <- unique(sub("_[0-9]*$", "", row.names(to_add)))
# add argument for old sample names use match()
for (i in 1:length(sample)) {
# refer to old sample name
if(!is.null(origSampNames)){
fn <- paste0(rawFCSdir, .Platform$file.sep, origSampNames[i], ".fcs")
}else{
fn <- paste0(rawFCSdir, .Platform$file.sep, sample[i], ".fcs")
}
if(!file.exists(fn)){
## stop the writing if cannot find the file
message(paste("Cannot find raw FCS file:", fn))
return(NULL)
}
cat("Save to file:", fn, "\n")
fcs <- read.FCS(fn, transformation = FALSE)
pattern <- paste(sample[i], "_", sep = "")
to_add_i <- as.data.frame(to_add[grep(pattern, row.names(to_add), fixed = TRUE), ])
m <- regexpr("_[0-9]*$", row.names(to_add_i))
cellNo_i <- as.integer(substring(regmatches(row.names(to_add_i), m), 2))
# subscript out of bounds
sub_exprs <- fcs@exprs[cellNo_i, ]
params <- parameters(fcs)
pd <- pData(params)
keyval <- keyword(fcs)
for (j in 1:length(addColNames)) {
## update the parameters
if (addColNames[j] %in% colnames(fcs@exprs)) {
addColNames[j] <- paste(addColNames[j], "_new", sep = "")
}
addColName <- addColNames[j]
channel_number <- nrow(pd) + 1
channel_id <- paste("$P", channel_number, sep = "")
channel_name <- addColName
minRange <- ceiling(min(to_add_i[[j]]))
maxRange <- ceiling(max(to_add_i[[j]]))
channel_range <- maxRange - minRange
plist <- matrix(c(channel_name, "<NA>", channel_range,
minRange, maxRange))
rownames(plist) <- c("name", "desc", "range", "minRange",
"maxRange")
colnames(plist) <- c(channel_id)
pd <- rbind(pd, t(plist))
## update the expression value
out_col_names <- colnames(sub_exprs)
sub_exprs <- cbind(sub_exprs, to_add_i[[j]])
colnames(sub_exprs) <- c(out_col_names, addColName)
## update the description remove '\' in the keywords
keyval <- lapply(keyval, function(x) {
if (class(x) == "character") {
gsub("\\", "", x, fixed = TRUE)
} else {
x
}
})
keyval[[paste("$P", channel_number, "B", sep = "")]] <- "32" # Number of bits
keyval[[paste("$P", channel_number, "R", sep = "")]] <- toString(channel_range) # Range
keyval[[paste("$P", channel_number, "E", sep = "")]] <- "0,0" # Exponent
keyval[[paste("$P", channel_number, "N", sep = "")]] <- channel_name # Name
keyval[[paste("P", channel_number, "BS", sep = "")]] <- 0
keyval[[paste("P", channel_number, "MS", sep = "")]] <- 0
keyval[[paste("P", channel_number, "DISPLAY", sep = "")]] <- "LIN" # data display
#add keyval for clusterID and annotation
}
pData(params) <- pd
out_frame <- flowFrame(exprs = sub_exprs, parameters = params, description = keyval)
suppressWarnings(write.FCS(out_frame, paste0(analyzedFCSdir, "/cytofkit_", sample[i], ".fcs")))
}
}
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.