R/BetaDiv.R

Defines functions BetaDiv

Documented in BetaDiv

# Beta diversity calculate
#
# The function named 'BetaDiv'
# which do beta-diversity analysis including PCoA, NMDS, LDA, DCA, CCA, RDA, MDS, PCA and t-sne
#
# You can learn more about package at:
#
#   https://github.com/microbiota/amplicon

#' @title Beta diversity plotting and statistics
#' @description Input otutab, metadata and tree or phyloseq object; support 47 distance types (bray, unifrac, wunifrac ...),  8 ordination method (PCoA, NMDS, ...); output ggplot2 figure, data and statistical test result.
#' @param otu OTU/ASV table;
#' @param map Sample metadata;
#' @param tree tree/nwk file;
#' @param dist distance type, including "unifrac" "wunifrac" "dpcoa" "jsd" "manhattan" "euclidean"   "canberra" "bray" "kulczynski"  "jaccard" "gower" "altGower" "morisita" "horn" "mountford"  "raup" "binomial"  "chao"  "cao" "w"  "-1"  "c" "wb"  "r"   "I"  "e" "t" "me"   "j"  "sor"  "m"   "-2"  "co";
#' @param group group ID;
#' @param method DCA, CCA, RDA, NMDS, MDS, PCoA, PCA, LDA;
#' @param pvalue.cutoff Pvalue threshold;
#' @param Micromet statistics by adonis/anosim/MRPP;
#' @details
#' By default, input phyloseq object include metadata, otutab and tree
#' The available diversity indices include the following:
#' \itemize{
#' \item{most used indices: bray, unifrac, wunifrac}
#' \item{other used indices: manhattan, euclidean, jaccard ...}
#' }
#' @return list object including plot, stat table
#' @author Contact: Tao Wen \email{2018203048@@njau.edu.cn}, Yong-Xin Liu \email{yxliu@@genetics.ac.cn}
#' @references
#'
#' Yong-Xin Liu, Yuan Qin, Tong Chen, Meiping Lu, Xubo Qian, Xiaoxuan Guo & Yang Bai.
#' A practical guide to amplicon and metagenomic analysis of microbiome data.
#' Protein Cell, 2020(41), 1-16, DOI: \url{https://doi.org/10.1007/s13238-020-00724-8}
#'
#' Jingying Zhang, Yong-Xin Liu, Na Zhang, Bin Hu, Tao Jin, Haoran Xu, Yuan Qin, Pengxu Yan, Xiaoning Zhang, Xiaoxuan Guo, Jing Hui, Shouyun Cao, Xin Wang, Chao Wang, Hui Wang, Baoyuan Qu, Guangyi Fan, Lixing Yuan, Ruben Garrido-Oter, Chengcai Chu & Yang Bai.
#' NRT1.1B is associated with root microbiota composition and nitrogen use in field-grown rice.
#' Nature Biotechnology, 2019(37), 6:676-684, DOI: \url{https://doi.org/10.1038/s41587-019-0104-4}
#'
#' @seealso beta_pcoa beta_cpcoa
#' @examples
#'
#' BetaDiv(otu=otutab_rare, map=metadata, tree=tree, group="Group", dist="bray", method="NMDS", Micromet="adonis", pvalue.cutoff=0.05)
#'
#' @export

BetaDiv=function(otu=otutab, map=metadata, tree=tree, ps=NULL,
                   group="Group", dist="bray", method="NMDS",
                   Micromet="adonis", pvalue.cutoff=0.05){

  # 需要的R包
  library(phyloseq)
  library(vegan)
  library(ggplot2)

  # # 读取默认参数
  # otu=otutab
  # map=metadata
  # tree=tree
  # ps=NULL
  # group="Group"
  # dist="bray"
  # method ="NMDS"
  # Micromet="adonis"
  # pvalue.cutoff=0.05

  # 数据导入PhyloSeq#----
  if (is.null(otu)&is.null(map)) {
    ps=ps
  }else{
    otu=as.matrix(otu)
    map$Group=as.factor(map[, group])
    ps=phyloseq(otu_table(otu, taxa_are_rows=TRUE), sample_data(map))
  }

  # 读取树#----
  if (dist %in% c("unifrac", "wunifrac", "dpcoa")) {
    phy_tree(ps)=tree
  }

  #转换为相对丰度#----
  ps1_rela=transform_sample_counts(ps, function(x) x / sum(x) )

  #提取标准化的特征表#----
  vegan_otu= function(physeq){
    OTU=otu_table(physeq)
    if(taxa_are_rows(OTU)){
      OTU=t(OTU)
    }
    return(as(OTU,"matrix"))
  }
  otu_table=as.data.frame(t(vegan_otu(ps1_rela )))

#排序方法选择#----

#---------DCA排序#----
if (method == "DCA") {
  ordi=phyloseq::ordinate(ps1_rela, method=method, distance=dist)
  #提取样本坐标
  points=ordi$rproj[,1:2]
  #命名行名
  colnames(points)=c("x", "y")
  #提取特征值
  eig=ordi$evals^2
}

#---------CCA排序#----
if (method == "CCA") {
  ordi=ordinate(ps1_rela, method=method, distance=dist)
  #样本坐标,这里可选u或者v矩阵
  points=ordi$CA$v[,1:2]
  colnames(points)=c("x", "y")
  eig=ordi$CA$eig^2
}

#---------RDA排序#----
if (method == "RDA") {
  ordi=ordinate(ps1_rela, method=method, distance=dist)
  #样本坐标,这里可选u或者v矩阵
  points=ordi$CA$v[,1:2]
  colnames(points)=c("x", "y") #命名行名
  #提取特征值
  eig=ordi$CA$eig
}

#---------MDS排序#----
if (method == "MDS") {
  ordi=ordinate(ps1_rela, method=method, distance=dist)
  points=ordi$vectors[,1:2]
  colnames(points)=c("x", "y")
  eig=ordi$values[,1]
}

#---------PCoA排序#----
if (method == "PCoA") {
  unif=phyloseq::distance(ps1_rela , method=dist, type="samples")
  pcoa=cmdscale(unif, k=2, eig=T)
  # 获得坐标点get coordinate string, format to dataframme
  points=as.data.frame(pcoa$points)
  colnames(points)=c("x", "y")
  eig=pcoa$eig
}

#---------PCA排序#----
if (method == "PCA") {
  otu.pca=prcomp(t(otu_table), scale.=TRUE)
  points=otu.pca$x[,1:2]
  colnames(points)=c("x", "y")
  # #提取荷载坐标
  # otu.pca$rotation
  # 提取解释度,这里提供的并不是特征值而是标准差,需要求其平方才是特征值
  eig=otu.pca$sdev
  eig=eig*eig
}

#---------------LDA排序#----
if (method == "LDA") {
  #拟合模型
  library(MASS)
  data=t(otu_table)
  data=as.data.frame(data)
  data=scale(data, center=TRUE, scale=TRUE)
  dim(data)
  data1=data[,1:10]
  map=as.data.frame(sample_data(ps1_rela))
  model=lda(data, map$Group)
  # 提取坐标
  ord_in=model
  axes=c(1:2)
  points=data.frame(predict(ord_in)$x[, axes])
  colnames(points)=c("x", "y") #命名行名
  # 提取解释度
  eig= ord_in$svd^2
}

#---------------NMDS排序#----
if (method == "NMDS") {
  ordi=ordinate(ps1_rela, method=method, distance=dist)
  #样本坐标,
  points=ordi$points[,1:2]
  colnames(points)=c("x", "y")
  #提取stress
  stress=ordi$stress
  stress=paste("stress", ": ", round(stress,2), sep="")
}


#---------------t-sne排序#----
# method="t-sne"
if (method == "t-sne") {
  data=as.data.frame(t(otu_table))
  data=scale(data, center=TRUE, scale=TRUE)
  map=as.data.frame(sample_data(ps1_rela))
  if (!requireNamespace("Rtsne", quietly=TRUE))
    install.packages("Rtsne")
  library(Rtsne)
  tsne=Rtsne(data,perplexity=3)

  # 提取坐标
  points=as.data.frame(tsne$Y)
  row.names(points)= row.names(map)
  colnames(points)=c("x", "y") #命名行名
  stress= NULL
}


#----差异分析#----

#----整体差异分析#----
title1=MicroTest(ps=ps1_rela, Micromet=Micromet, dist=dist)

#----两两比较#----
pairResult=pairMicroTest(ps=ps1_rela, Micromet=Micromet, dist=dist)


#----绘图#----
map=as.data.frame(sample_data(ps1_rela))
map$Group=as.factor(map$Group)
colbar=length(levels(map$Group))

points=cbind(points, map[match(rownames(points), rownames(map)), ])
# head(points)
points$ID=row.names(points)

#---定义配色#----
# 改为使用Rcolorbrewer的颜色组合
mi=colorRampPalette(c( "#CBD588", "#599861", "orange","#DA5724", "#508578", "#CD9BCD","#AD6F3B", "#673770","#D14285", "#652926", "#C84248","#8569D5", "#5E738F","#D1A33D", "#8A7C64","black"))(colbar)

#----定义图形通用样式#----
main_theme=theme(panel.grid.major=element_blank(),
  panel.grid.minor=element_blank(),
  plot.title=element_text(vjust=-8.5,hjust=0.1),
  text=element_text(family="sans", size=7)
  # axis.title.y =element_text(size=7,face="bold",colour="black"),
  # axis.title.x =element_text(size=7,face="bold",colour="black"),
  # axis.text=element_text(size=7,face="bold"),
  # axis.text.x=element_text(colour="black",size=7),
  # axis.text.y=element_text(colour="black",size=7),
  # legend.text=element_text(size=7,face="bold")
)

if (method %in% c("DCA", "CCA", "RDA",  "MDS", "PCoA","PCA","LDA")) {
  p2 =ggplot(points, aes(x=x, y=y, fill=Group)) +
    geom_point(alpha=.7, size=2, pch=21) +
    labs(x=paste0(method," 1 (",format(100*eig[1]/sum(eig),digits=4),"%)"),
         y=paste0(method," 2 (",format(100*eig[2]/sum(eig),digits=4),"%)"),
         title=title1) +
    stat_ellipse(linetype=2,level=0.68,aes(group=Group, colour=Group))+
    scale_colour_manual(values=mi,guide=guide_legend(title=NULL))+
    scale_fill_manual(values=mi,guide=guide_legend(title=NULL))+
    guides(color=guide_legend(title=NULL),shape=guide_legend(title=NULL))
  p2=p2+theme_bw()+
    geom_hline(aes(yintercept=0), colour="black", linetype=2) +
    geom_vline(aes(xintercept=0), colour="black", linetype="dashed") +
    main_theme
  library(ggrepel)
  p3=p2+geom_text_repel(aes(label=points$ID),size=2)
}


if (method %in% c("NMDS","t-sne")) {
  p2 =ggplot(points, aes(x=x, y=y, fill=Group)) +
    geom_point(alpha=.7, size=2, pch=21) +
    labs(x=paste(method,"1", sep=""),
         y=paste(method,"2",sep=""),
         title=stress)+
    stat_ellipse( linetype=2,level=0.65,aes(group  =Group, colour= Group))+
    scale_colour_manual(values=mi,guide=guide_legend(title=NULL))+
    scale_fill_manual(values=mi,guide=guide_legend(title=NULL))+
    guides(color=guide_legend(title=NULL),shape=guide_legend(title=NULL))
  p2=p2+theme_bw()+
    geom_hline(aes(yintercept=0), colour="black", linetype=2) +
    geom_vline(aes(xintercept=0), colour="black", linetype="dashed") +
    main_theme
  library(ggrepel)
  p3=p2+geom_text_repel(aes(label=points$ID),size=2)
  if (method %in% c("t-sne")) {
    supp_lab=labs(x=paste(method,"1", sep=""),y=paste(method,"2",sep=""),title=title)
   p2=p2 + supp_lab
   p3=p3 + supp_lab
  }
  p2
}

# 返回结果:标准图,数据,标签图,成对比较结果,整体结果
return(list(p2,points,p3,pairResult,title1))

}
microbiota/amplicon documentation built on April 30, 2023, 1:48 p.m.