R/operator.R

Defines functions `%+>%` `%place%` `%<%`

##' update tree
##'
##' This operator apply the visualization directives in ggtree object (lhs)
##' to visualize another tree object (rhs), that is similar to Format Painter.
##'
##'
##' @rdname ggtree-format-painter
##' @title %<%
##' @param pg ggtree object
##' @param x tree object
##' @return updated ggplot object
##' @export
##' @examples
##' library("ggplot2")
##' nwk <- system.file("extdata", "sample.nwk", package="treeio")
##' tree <- read.tree(nwk)
##' p <- ggtree(tree) + geom_tippoint(color="#b5e521", alpha=1/4, size=10)
##' p %<% rtree(30)
##' @author Guangchuang Yu
`%<%` <- function(pg, x) {
    if (! is.tree(x)) {
        stop("input should be a tree object...")
    }
    pg %place% x
}

# ##' add annotation data to a tree
# ##'
# ##' This operator attaches annotation data to a ggtree graphic object
# ##'
# ##' @rdname attacher
# ##' @title %<+%
# ##' @param pg ggplot2 object
# ##' @param data annotation data that contains a column of “node” , or the first column of taxa labels
# ##' @return ggplot object with annotation data added
# ##' @export
# ##' @author Guangchuang Yu
# ##' @seealso geom_facet
# ##' @examples
# ##' nwk <- system.file("extdata", "sample.nwk", package="treeio")
# ##' tree <- read.tree(nwk)
# ##' p <- ggtree(tree)
# ##' dd <- data.frame(taxa=LETTERS[1:13],
# ##'    		 place=c(rep("GZ", 5), rep("HK", 3), rep("CZ", 4), NA),
# ##'              value=round(abs(rnorm(13, mean=70, sd=10)), digits=1))
# ##' row.names(dd) <- NULL
# ##' p %<+% dd + geom_text(aes(color=place, label=label), hjust=-0.5)
# ##' @references G Yu, TTY Lam, H Zhu, Y Guan (2018). Two methods for mapping and visualizing associated data
# ##' on phylogeny using ggtree. Molecular Biology and Evolution, 35(2):3041-3043.
# ##' <https://doi.org/10.1093/molbev/msy194>
# `%<+%` <- function(pg, data) {
#     if (! is.data.frame(data)) {
#         stop("input should be a data.frame...")
#     }
#     pg %add% data
# }

`%place%` <- function(pg, tree) {

    mrsd      <- get("mrsd", envir=pg$plot_env)
    layout    <- get("layout", envir = pg$plot_env)
    yscale    <- get("yscale", envir = pg$plot_env)
    ladderize <- get("ladderize", envir = pg$plot_env)
    right     <- get("right", envir = pg$plot_env)
    branch.length <- get("branch.length", envir = pg$plot_env)

    pg$data <- fortify(tree,
                       layout        = layout,
                       yscale        = yscale,
                       ladderize     = ladderize,
                       right         = right,
                       branch.length = branch.length,
                       mrsd          = mrsd)
    return(pg)
}


# `%add%` <- function(p, data) {
#     p$data <- p$data %add2% data
#     return(p)
# }
# 
# ##' @importFrom dplyr rename
# ##' @importFrom dplyr left_join
# `%add2%` <- function(d1, d2) {
#     if ("node" %in% colnames(d2)) {
#         cn <- colnames(d2)
#         ii <- which(cn %in% c("node", cn[!cn %in% colnames(d1)]))
#         d2 <- d2[, ii]
#         dd <- dplyr::left_join(d1, d2, by="node")
#     } else {
#         d2[,1] <- as.character(unlist(d2[,1])) ## `unlist` to work with tbl_df
#         d2 <- dplyr::rename(d2, label = 1) ## rename first column name to 'label'
#         dd <- dplyr::left_join(d1, d2, by="label")
#     }
#     dd <- dd[match(d1$node, dd$node),]
#     return(dd)
# }

##' update data with tree info (y coordination and panel)
##'
##' add tree information to an input data.
##' This function will setup y coordination and panel info
##' for data used in facet_plot and geom_faceet
##'
##' @rdname add_TREEINFO
##' @title %+>%
##' @param p tree view
##' @param .data data.frame
##' @return updated data.frame
##' @importFrom methods is
##' @export
##' @references G Yu, TTY Lam, H Zhu, Y Guan (2018). Two methods for mapping and visualizing associated data
##' on phylogeny using ggtree. Molecular Biology and Evolution, 35(2):3041-3043.
##' <https://doi.org/10.1093/molbev/msy194>
##' @author Guangchuang Yu
`%+>%` <- function(p, .data) {
    df <- p$data
    lv <- levels(df$.panel)
    if (inherits(.data, "GRanges") || inherits(.data, "GRangesList")) {
        names(.data) <- df$y[match(names(.data), df$label)]
        res <- .data[order(as.numeric(names(.data)))]
        mcols <- get_fun_from_pkg("GenomicRanges", "mcols")
        `mcols<-` <- get_fun_from_pkg("GenomicRanges", "`mcols<-`")
        mcols(res)$.panel <- factor(lv[length(lv)], levels=lv)
    } else if (is(.data, "data.frame") || is(.data, "tbl_df")) {
        .data <- as.data.frame(.data)
        ## res <- merge(df[, c('label', 'y')], data, by.x='label', by.y=1) ## , all.x=TRUE)
        res <- merge(df[, !names(df) %in% c('node', 'parent', 'x', 'branch', 'angle')], .data, by.x='label', by.y=1)
        res[[".panel"]] <- factor(lv[length(lv)], levels=lv)
        res <- res[order(res$y),]
    } else if (is.function(.data)){
        res <- .data(df)
        if (!is.data.frame(res)){
            rlang::abort("Data function must return a data.frame")
        }
        res[[".panel"]] <- factor(lv[length(lv)], levels=lv)
        res %<>% dplyr::filter(.data$isTip)
        res <- res[order(res$y),]
    } else {
        stop("input 'data' is not supported...")
    }

    return(res)
}
GuangchuangYu/ggtree documentation built on Oct. 31, 2024, 10:10 a.m.