R/pspacePlots.R

Defines functions .trimRamp .trimcols .pspacePalette .getGrid .set_theme_zlim .set_theme_bks .custom_th3 .custom_th2 .custom_th1 .custom_th0 .custom_themes .get_mark_idx .add_marks .add_contour .add_image .add_grid .set_pspace .custom_annotations .scale_alpha

#' @title Plotting 2D-landscape images for the PathwaySpace package
#'
#' @description \code{plotPathwaySpace} is a wrapper function to 
#' create dedicated ggplot graphics for PathwaySpace-class objects.
#'
#' @param ps A \linkS4class{PathwaySpace} class object.
#' @param colors A vector of colors. Each color is a specific hue used to 
#' create a customized color palette, interpolated according to the provided 
#' sequence in the vector of colors. The proportion of each color hue can be 
#' adjusted by the 'trim.colors' argument. This palette is designed to 
#' fine-tune the visibility of summits and valleys within the image space.
#' To bypass this automatic palette generation and use the 'colors' 
#' input as-is, simply set 'trim.colors' to NULL.
#' @param trim.colors An vector with 5 positive integer numbers. This argument
#' can be used to adjust the proportion of each color hue in the palette.
#' @param bg.color A single color for background.
#' @param si.color A single color for silhouette.
#' (see \code{\link{silhouetteMapping}}).
#' @param si.alpha A transparency level in [0, 1], used to adjust the 
#' opacity of the silhouette. This parameter is useful for improving the 
#' perception of a background image, when one is available.
#' @param theme Name of a custom PathwaySpace theme. These themes 
#' (from 'th0' to 'th3') consist mainly of preconfigured ggplot settings, 
#' which the user can subsequently refine using \code{\link[ggplot2]{ggplot2}}.
#' @param title A string for the title.
#' @param xlab The title for the 'x' axis of a 2D-image space.
#' @param ylab The title for the 'y' axis of a 2D-image space.
#' @param zlab The title for the 'z' axis of the image signal.
#' @param font.size A single numeric value passed to ggplot themes.
#' @param font.color A single color passed to ggplot themes.
#' @param zlim The 'z' limits of the plot (a numeric vector with two numbers).
#' If NULL, limits are determined from the range of the input values.
#' @param slices A single positive integer value used to split 
#' the image signal into equally-spaced intervals.
#' @param add.grid A logical value indicating whether to add gridlines to 
#' the image space. However, gridlines will only appear when the image 
#' is decorated with graph silhouettes (see \code{\link{silhouetteMapping}}).
#' @param grid.color A color passed to \code{\link[ggplot2]{geom_point}}.
#' @param add.summits A logical value indicating whether to add contour 
#' lines to 'summits' (when summits are available; 
#' see \code{\link{summitMapping}}).
#' @param label.summits A logical value indicating whether to label summits.
#' @param summit.color A color passed to 'summits'.
#' @param add.marks A logical value indicating whether to plot vertex labels.
#' @param marks A vector of vertex names to be highlighted in the 
#' image space. This argument overrides 'add.labels'.
#' @param mark.size A size argument passed to \code{\link[ggplot2]{geom_text}}.
#' @param mark.color A color passed to \code{\link[ggplot2]{geom_text}}.
#' @param mark.padding A box padding argument passed to 
#' \code{\link[ggrepel]{geom_text_repel}}.
#' @param mark.line.width A line width argument passed to 
#' \code{\link[ggrepel]{geom_text_repel}}.
#' @param use.dotmark A logical value indicating whether "marks" should be 
#' represented as dots.
#' @param add.image A logical value indicating whether to add a background 
#' image, when one is available (see \code{\link[RGraphSpace]{GraphSpace}}).
#' @return A ggplot-class object.
#' @author Mauro Castro and TCGA Network.
#' @seealso \code{\link{circularProjection}}
#' @examples
#' # Load a demo igraph
#' data('gtoy1', package = 'RGraphSpace')
#'
#' # # Check graph validity
#' gs <- GraphSpace(gtoy1)
#' 
#' # Create a PathwaySpace object
#' ps <- buildPathwaySpace(gs, nrc = 300)
#' # note: adjust 'nrc' to increase image resolution
#'
#' # Set '1s' as vertex signal
#' vertexSignal(ps) <- 1
#'
#' # Create a 2D-landscape image
#' ps <- circularProjection(ps, k = 2, pdist = 0.4)
#'
#' # Plot a 2D-landscape image
#' plotPathwaySpace(ps)
#' 
#' @import methods
#' @docType methods
#' @importFrom ggplot2 ggplot annotate element_text theme theme_bw theme_gray
#' @importFrom ggplot2 scale_x_continuous scale_y_continuous expansion
#' @importFrom ggplot2 aes scale_fill_gradientn annotation_raster
#' @importFrom ggplot2 coord_fixed geom_raster
#' @importFrom ggplot2 margin element_blank element_rect element_line
#' @importFrom ggrepel geom_text_repel
#' @importFrom grDevices convertColor col2rgb rgb as.raster
#' @importFrom grDevices adjustcolor colorRampPalette
#' @importFrom stats runif
#' @importFrom RGraphSpace plotGraphSpace
#' @rdname plotPathwaySpace-methods
#' @aliases plotPathwaySpace
#' @export
#'
setMethod("plotPathwaySpace", "PathwaySpace", 
  function(ps, 
    colors = pspace.cols(), trim.colors = c(3, 2, 1, 2, 3), 
    bg.color = "grey95", si.color = "grey85", si.alpha = 1,
    theme = c("th0", "th1", "th2", "th3"),
    title = "PathwaySpace", 
    xlab = "Pathway coordinates 1", 
    ylab = "Pathway coordinates 2", 
    zlab = "Density", 
    font.size = 1, font.color = "white",
    zlim = NULL, slices = 25, 
    add.grid = TRUE, grid.color = "white", 
    add.summits = TRUE, label.summits = TRUE, summit.color = "white",
    add.marks = FALSE, marks = NULL, mark.size = 3, mark.color = "white", 
    mark.padding = 0.5, mark.line.width = 0.5, use.dotmark = FALSE, 
    add.image = FALSE) {
    
    #--- validate the ps object and args
    if (!.checkStatus(ps, "Projection") && !.checkStatus(ps, "Silhouette")) {
      stop("NOTE: 'ps' needs to be evaluated by a 'projection' method!",
        call. = FALSE)
    }
    .validate.args("singleNumber", "si.alpha", si.alpha)
    .validate.args("singleString", "title", title)
    .validate.args("singleString", "xlab", xlab)
    .validate.args("singleString", "ylab", ylab)
    .validate.args("singleString", "zlab", zlab)
    .validate.args("singleNumber", "font.size", font.size)
    .validate.args("singleInteger", "slices", slices)
    .validate.args("singleLogical", "add.grid", add.grid)
    .validate.args("singleLogical", "add.summits", add.summits)
    .validate.args("singleLogical", "label.summits", label.summits)
    .validate.args("singleLogical", "add.marks", add.marks)
    .validate.plot.args("marks", marks)
    .validate.args("singleNumber", "mark.size", mark.size)
    .validate.colors("singleColor", "mark.color", mark.color)
    .validate.args("singleNumber", "mark.padding", mark.padding)
    .validate.args("singleNumber","mark.line.width", mark.line.width)
    .validate.args("singleLogical", "use.dotmark", use.dotmark)
    .validate.args("singleLogical", "add.image", add.image)
    .validate.colors("allColors","colors", colors)
    .validate.colors("singleColor", "bg.color", bg.color)
    .validate.colors("singleColor", "si.color", si.color)
    .validate.colors("singleColor", "font.color", font.color)
    .validate.colors("singleColor", "grid.color", grid.color)
    .validate.colors("singleColor", "summit.color", summit.color)
    .validate.plot.args("trim.colors", trim.colors)
    theme <- match.arg(theme)
    if(!is.null(zlim)) {
      .validate.args("numeric_vec", "zlim", zlim)
      if(length(zlim)!=2) 
        stop("'zlim' should be a numeric vector of lenght 2.", call. = FALSE)
    }
    if (si.alpha < 0 || si.alpha > 1) {
      stop("'si.alpha' should be in [0,1]", call. = FALSE)
    }
    #--- get slots from ps
    summits <- getPathwaySpace(ps, "summits")
    cset <- getPathwaySpace(ps, "summit_contour")
    silstatus <- .checkStatus(ps, "Silhouette")
    gxy <- getPathwaySpace(ps, "projections")$gxy
    gxyz <- getPathwaySpace(ps, "projections")$gxyz
    pars <- getPathwaySpace(ps, "projections")$pars
    
    #--- set colors
    if(!is.null(trim.colors)){
      colors <- .pspacePalette(colors, trim.colors)
    }
    if(pars$ps$configs$scale.type=="negpos"){
      slices <- ceiling(slices/2) * 2
    }
    
    # set scales
    if(is.null(zlim)){
      zlim <- pars$ps$configs$zlim
    } else {
      gxyz[gxyz < zlim[1]] <- zlim[1]
      gxyz[gxyz > zlim[2]] <- zlim[2]
    }
    if (all(zlim == 0)) zlim[2] <- 1
    
    #--- trim colors and set theme args
    cl <- .trimcols(colors, bg.color, zlim, pars)
    cl <- .set_theme_bks(theme, cl)
    cl <- .set_theme_zlim(cl, zlim)
    # slice gxyz image
    bks <- seq(zlim[1], zlim[2], length.out = slices)
    gxyz[, ] <- bks[cut(as.numeric(gxyz), breaks = sort(unique(bks)),
      include.lowest = TRUE)]
    
    #--- get grid
    gridln <- .getGrid(gxyz, cl$axis.ticks)
    gridln <- as.numeric(gridln)
    gridln <- data.frame(
      arrayInd(seq_along(gridln), .dim = dim(gxyz)), gridln)
    colnames(gridln) <- c("Y", "X", "L")
    
    #--- set main input for ggplot
    gxyz <- data.frame(arrayInd(seq_along(gxyz), .dim = dim(gxyz)),
      as.numeric(gxyz))
    colnames(gxyz) <- c("Y", "X", "Z")
    
    #--- scale coordinates to plot space
    gxyz$X <- scales::rescale(gxyz$X)
    gxyz$Y <- scales::rescale(gxyz$Y)
    gxyz$L <- gridln$L
    
    #--- set a bg color effect, scaling alpha to z
    if(si.alpha < 1){
      si.color <- adjustcolor(si.color, si.alpha)
      gz.alpha <- .scale_alpha(si.alpha, gxyz, zlim, pars)
    } else {
      gz.alpha <- 1
    }
    
    #--- initialize a ggplot
    ggp <- .set_pspace(gxyz, xlab, ylab, zlab, cl, si.color)
    
    #--- add image
    if(pars$image.layer){
      img <- getPathwaySpace(ps, "image")
      if(add.image){
        ggp <- .add_image(ggp, img)
      } else {
        ggi <- .add_image(ggp, img)
        if(add.grid) ggi <- .add_grid(ggi, gxyz, grid.color)
        ggi <- .custom_themes(ggi, theme, font.size, bg.color)
      }
    }
    
    #--- add main projection
    ggp <- ggp + ggplot2::geom_raster(interpolate = FALSE, 
      na.rm=TRUE, alpha = gz.alpha)
    
    #--- add a grid
    if(add.grid) ggp <- .add_grid(ggp, gxyz, grid.color)
    
    #--- add contour lines if available
    bl <- add.summits || label.summits
    if (bl && !is.null(summits) && length(summits) > 0 && sum(cset) > 0) {
      ggp <- .add_contour(ggp, gxyz, summits, cset, 
        summit.color, mark.size, add.summits, label.summits)
    }
    
    #--- add marks if available
    if (!is.null(marks)) {
      ggp <- .add_marks(ggp, gxy, pars, marks, mark.size,
        mark.color, mark.padding, mark.line.width, use.dotmark)
    } else if(add.marks){
      ggp <- .add_marks(ggp, gxy, pars, marks=rownames(gxy), 
        mark.size, mark.color, mark.padding, mark.line.width, use.dotmark)
    }
    
    #--- add annotations
    if(.checkStatus(ps, "Projection")){
      ggp <- .custom_annotations(ggp, title, pars, font.size, 
        font.color, silstatus, si.color)
    }
    
    #--- apply custom theme
    ggp <- .custom_themes(ggp, theme, font.size, bg.color)
    
    if(pars$image.layer && !add.image){
      ggl <- list(graph = ggp, image = ggi)
      return(ggl)
    } else {
      return(ggp)
    }
    
  }
)

#-------------------------------------------------------------------------------
.scale_alpha <- function(si.alpha, gxyz, zlim, pars){
  az <- gxyz$Z
  mxz <- max(abs(zlim))
  if(pars$ps$configs$scale.type == "negpos") {
    slim <- 0.5 * (1 - si.alpha)
    slim <- slim * mxz
    az[az < 0 & az < -slim] <- mxz
    az[az > 0 & az > slim] <- mxz
    az <- abs(az)
  } else if(pars$ps$configs$scale.type == "neg") {
    az[az < zlim[2]] <- zlim[1]
  } else {
    az[az > zlim[1]] <- zlim[2]
  }
  pars$ps$configs$zlim
  az <- az/mxz
  alpha <- az^10 + si.alpha
  return(alpha)
}

#-------------------------------------------------------------------------------
.custom_annotations <- function(ggp, title, pars, font.size, 
  font.color, silstatus, si.color){
  if(silstatus){
    if(si.color=="grey85"){
      fcol <- "grey20"
    } else {
      fcol <- font.color
    }
    sep <- "; "
    xlab <- 0.01
    hjust <- 0
  } else {
    fcol <- font.color
    sep <- "\n"
    xlab <- 0.99
    hjust <- 1
  }
  ggp <- ggp + ggplot2::annotate("text", label = title,
    colour = fcol, size = font.size*3.5, x = 0, y = 0.99, 
    hjust = 0, vjust = 1)
  dfun <- attributes(pars$ps$decay$fun)$name
  if(!is.null(dfun)){
    if(dfun == "weibullDecay"){
      dfun <- "Weibull decay"
    } else if(dfun == "expDecay"){
      dfun <- "Exponential decay"
    } else if(dfun == "linearDecay"){
      dfun <- "Linear decay"
    } else {
      dfun <- "Custom decay"
    }
    pars$ps$dfun <- dfun
  } else {
    pars$ps$dfun <- "Custom decay"
  }
  if(pars$ps$projection=="Polar"){
    annot <- pars$ps[c("projection", "dfun", "k", "theta")]
    annot$k <- paste0("k = ", annot$k, "; ")
    annot$theta <- paste0("theta = ", pars$ps$theta)
  } else {
    annot <- pars$ps[c("projection", "dfun", "k")]
    annot$k <- paste0("k = ", annot$k)
  }
  annot$projection <- paste0(annot$projection, " projection", sep)
  annot$dfun <- paste0(annot$dfun, sep)
  annot <- paste(unlist(annot), collapse = "")
  ggp <- ggp + ggplot2::annotate("text", label = annot,
    colour = fcol, size = font.size*3, x = xlab, 
    y = 0.01, hjust = hjust, vjust = 0)
  return(ggp)
}

#-------------------------------------------------------------------------------
.set_pspace <- function(gxyz, xlab, ylab, zlab, cl, si.color){
  X <- Y <- Z <- NULL
  ggp <- ggplot2::ggplot(gxyz, ggplot2::aes(X, Y, fill = Z)) +
    ggplot2::scale_x_continuous(name = xlab, breaks = cl$axis.ticks,
      labels = format(cl$axis.ticks), position = cl$x.position,
      limits = cl$xylim, expand = ggplot2::expansion(mult = 0)) +
    ggplot2::scale_y_continuous(name = ylab, breaks = cl$axis.ticks,
      labels = format(cl$axis.ticks), limits = cl$xylim,
      expand = ggplot2::expansion(mult = 0)) +
    ggplot2::scale_fill_gradientn(name = zlab, limits = cl$zlim,
      breaks = cl$breaks, labels = names(cl$breaks),
      colours = cl$pal, aesthetics = "fill", na.value = si.color) +
    ggplot2::coord_fixed()
  return(ggp)
}

#-------------------------------------------------------------------------------
.add_grid <- function(ggp, gxyz, grid.color){
  dt <- gxyz[gxyz$L == 1, c("X", "Y")]
  ggp <- ggp + ggplot2::annotate(geom = "point", x = dt$X, 
    y = dt$Y, color = grid.color, size = 0.2, pch = 15)
  return(ggp)
}

#-------------------------------------------------------------------------------
.add_image <- function(ggp, image){
  ggp <- ggp + annotation_raster(image, xmin = 0, xmax = 1, ymin = 0, ymax = 1)
  return(ggp)
}

#-------------------------------------------------------------------------------
.add_contour <- function(ggp, gxyz, summits, cset, summit.color, 
  mark.size, add.summits, label.summits) {
  setnames <- names(summits)
  if (is.null(setnames)) {
    setnames <- seq_along(setnames)
  }
  cset <- data.frame(which(!is.na(cset) | is.na(cset), arr.ind = TRUE),
    as.numeric(cset))
  colnames(cset) <- c("Y", "X", "C")
  gxyz <- cbind(gxyz, C = cset$C)
  xy.tx <- NULL
  concav <- sort(unique(gxyz$C))[-1]
  for (i in seq_along(concav)) {
    xy.cv <- gxyz[gxyz$C == i, c("X", "Y")]
    xy.tx <- rbind(xy.tx, colMeans(xy.cv))
    if(add.summits){
      ggp <- ggp + ggplot2::annotate(geom = "tile", x = xy.cv[, 1], 
        y = xy.cv[, 2], color = summit.color,
        fill = summit.color, linewidth = 0.2)
    }
  }
  rownames(xy.tx) <- setnames
  xy.tx <- as.data.frame(xy.tx)
  if(label.summits){
    if(add.summits){
      ggp <- ggp + ggplot2::annotate(geom = "text", x = xy.tx[, 1],
        y = xy.tx[, 2], label = rownames(xy.tx), vjust = 0.5, hjust = 0.5,
        color = summit.color, size = mark.size,
        fontface = "bold")
    } else {
      nudgex <- sign(xy.tx$X - 0.5) * xy.tx$X
      nudgey <- sign(xy.tx$Y - 0.5) * xy.tx$Y
      ggp <- ggp + ggrepel::geom_text_repel(mapping = aes(label = rownames(xy.tx),
        segment.size = 0.4), data = xy.tx, min.segment.length = 0.1,
        fontface = "bold", force = 3, segment.linetype = 1,
        max.overlaps = nrow(xy.tx) + 5, point.padding = 0, seed = 123,
        max.iter = 20000, max.time = 30, nudge_x = nudgex * 0.075,
        nudge_y = nudgey * 0.075, size = mark.size, colour = summit.color,
        segment.colour = summit.color)
    }
  }
  return(ggp)
}

#-------------------------------------------------------------------------------
.add_marks <- function(ggp, gxy, pars, marks, mark.size,
  mark.color, mark.padding, mark.line.width, use.dotmark) {
  
  # scale coordinates to plot space
  gxy_df <- as.data.frame(gxy)
  gxy_df <- gxy_df[, c("X", "Y")]
  gxy_df$X <- scales::rescale(gxy_df$X, from = c(1, pars$ps$nrc))
  gxy_df$Y <- scales::rescale(gxy_df$Y, from = c(1, pars$ps$nrc))
  
  # match  marks
  marks <- marks[!duplicated(marks)]
  if (is.null(names(marks))) names(marks) <- marks
  names(marks) <- ifelse(is.na(names(marks)), marks, names(marks))
  names(marks) <- ifelse(names(marks) == "", marks, names(marks))
  idx_df <- .get_mark_idx(marks, gxy_df)
  if(any(is.na(idx_df$idx))){
    stop("All 'marks' should be annotated in the 'PathwaySpace' object.",
      call. = FALSE)
  }
  
  # set df
  gxy_df <- gxy_df[idx_df$idx, , drop = FALSE]
  gxy_df$ID <- idx_df$label
  ID <- NULL
  
  # call to ggplot
  if (use.dotmark) {
    ggp <- ggp + ggplot2::annotate("point", shape = 5, 
      colour = mark.color, size = mark.size * 0.4, stroke = 0.4, 
      x = gxy_df$X, y = gxy_df$Y)
  }
  nudgex <- sign(gxy_df$X - 0.5) * gxy_df$X
  nudgey <- sign(gxy_df$Y - 0.5) * gxy_df$Y
  ggp <- ggp + ggrepel::geom_text_repel(mapping = aes(label = ID,
    segment.size = mark.line.width), data = gxy_df, min.segment.length = 0.1,
    fontface = "bold", force = 3, segment.linetype = "2121", 
    max.overlaps = nrow(gxy_df) + 5, point.padding = 0, seed = 123, 
    max.iter = 20000, max.time = 30, nudge_x = nudgex * 0.15, 
    nudge_y = nudgey * 0.15, size = mark.size, colour = mark.color, 
    segment.colour = mark.color, box.padding = mark.padding)
  
  return(ggp)
}
.get_mark_idx <- function(marks, gxy_df){
  idx_df <- data.frame(name=marks, label=names(marks))
  idx_df$idx1 <- match(marks, rownames(gxy_df))
  idx_df$idx2 <- match(names(marks), rownames(gxy_df))
  idx_df$idx <- sapply(seq_len(nrow(idx_df)), function(i){
    idx <- idx_df[i, c(3,4)]
    idx[!is.na(idx)][1]
  })
  idx_df <- idx_df[,c(1,2,5)]
  return(idx_df)
}

#-------------------------------------------------------------------------------
.custom_themes <- function(gg, theme, font.size, bg.color) {
  et1 <- ggplot2::element_text(size = 14 * font.size)
  et2 <- ggplot2::element_text(size = 12 * font.size)
  if (theme == "th0") {
    gg <- .custom_th0(gg, font.size, bg.color)
  } else if (theme == "th1") {
    gg <- .custom_th1(gg, font.size, bg.color)
  } else if (theme == "th2") {
    gg <- .custom_th2(gg, font.size, bg.color)
  } else {
    gg <- .custom_th3(gg, font.size, bg.color)
  }
  return(gg)
}
.custom_th0 <- function(gg, font.size, bg.color) {
  et1 <- ggplot2::element_text(size = 14 * font.size)
  et2 <- ggplot2::element_text(size = 12 * font.size)
  gg <- gg + ggplot2::theme(axis.title = et1, axis.text = et2,
    legend.title = et2, legend.text = et2,
    panel.background = element_rect(fill = bg.color))
  return(gg)
}
.custom_th1 <- function(gg, font.size,
  bg.color) {
  et1 <- ggplot2::element_text(size = 14 * font.size)
  et2 <- ggplot2::element_text(size = 12 * font.size)
  gg <- gg + ggplot2::theme_bw() +
    ggplot2::theme(axis.title = et1,
      axis.text = et2, legend.title = et2,
      legend.text = et2, legend.margin = margin(0, 0, 0, 0), 
      plot.margin = margin(1, 1, 1, 1), 
      legend.background = element_blank(),
      legend.box.background = element_blank(),
      panel.grid.minor = element_line(linewidth = 0.7, 
        colour = bg.color),
      panel.grid.major = element_line(linewidth = 0.7,
        colour = bg.color),
      axis.ticks = element_line(linewidth = 0.7),
      axis.line = element_blank(),
      panel.border = element_rect(linewidth = 1.2))
  return(gg)
}
.custom_th2 <- function(gg, font.size, bg.color) {
  et1 <- ggplot2::element_text(size = 14 * font.size)
  et2 <- ggplot2::element_text(size = 12 * font.size)
  gg <- gg + ggplot2::theme_gray() + ggplot2::theme(axis.title = et1,
    axis.text = et2, legend.title = et2,
    legend.text = et2, legend.margin = margin(0, 0, 0, 0), 
    plot.margin = margin(5, 10, 0, 10), 
    legend.background = element_blank(),
    legend.box.background = element_blank(),
    panel.grid.minor = element_blank(),
    panel.grid.major = element_blank(),
    axis.ticks = element_line(linewidth = 0.7),
    axis.line = element_blank(), panel.border = element_blank(),
    panel.background = element_rect(fill = bg.color))
  return(gg)
}
.custom_th3 <- function(gg, font.size, bg.color) {
  et1 <- ggplot2::element_text(size = 14 * font.size)
  et2 <- ggplot2::element_text(size = 12 * font.size, hjust=0.5)
  gg <- gg + ggplot2::theme_gray() + 
    ggplot2::theme(axis.title = et1, axis.text = et2, 
      legend.title = element_text(size = 12 * font.size, vjust = 1), 
      legend.text = et2,
      legend.margin = margin(0, 0, 0, 0),
      legend.position = "bottom", 
      plot.margin = margin(5, 5, 5, 5), 
      legend.box.margin = margin(0, 0, 0, 0),
      legend.background = element_blank(),
      legend.box.background = element_blank(),
      panel.grid.minor = element_blank(),
      panel.grid.major = element_blank(),
      axis.ticks = element_line(linewidth = 0.5),
      axis.line = element_blank(), panel.border = element_blank(),
      panel.background = element_rect(fill = bg.color))
  return(gg)
}
.set_theme_bks <- function(theme, cl=list()){
  if (theme %in% c("th3")) {
    cl$axis.ticks <- c(0.25, 0.5, 0.75)
    cl$xylim <- c(-0.01, 1.01)
    cl$x.position <- "top"
    cl$justify <- "centre"
  } else if (theme %in% c("th2")) {
    cl$axis.ticks <- seq(0.1, 0.9, 0.2)
    cl$xylim <- c(-0.01, 1.01)
    cl$x.position <- "bottom"
    cl$justify <- "right"
  } else {
    cl$axis.ticks <- seq(0, 1, 0.2)
    cl$xylim <- c(-0.05, 1.05)
    cl$x.position <- "bottom"
    cl$justify <- "right"
  }
  return(cl)
}
.set_theme_zlim <- function(cl, zlim){
  # adjust labels for z-axis midle and tips
  bks_names <- cl$breaks
  bks_names <- format(bks_names,  trim = TRUE)
  n <- length(bks_names)
  bks_names[!seq_len(n) %in% c(1, ceiling(n/2), n)] <- ""
  # bks_names <- format(bks_names, justify=cl$justify)
  names(cl$breaks) <- bks_names
  # expand 'zlim' and palette tips
  expand <- TRUE
  if(expand){
    tips <- (zlim[2] - zlim[1]) * 0.1
    cl$zlim <- c(zlim[1] - tips, zlim[2] + tips)
    cl$pal <- c(cl$pal[1], cl$pal, cl$pal[length(cl$pal)])
  } else {
    cl$zlim <- zlim
  }
  return(cl)
}
#-------------------------------------------------------------------------------
#--- get grid lines
.getGrid <- function(gxyz, ticks = c(0.2, 0.4, 0.6, 0.8), ndots = 100) {
  ticks <- ticks[ticks>0 & ticks <1]
  nc <- ncol(gxyz)
  nr <- nrow(gxyz)
  by <- ceiling(nc/ndots)
  ic <- ceiling(ticks * nc)
  ir <- ceiling(ticks * nr)
  grid1 <- grid2 <- array(0, dim = dim(gxyz))
  grid1[ir, seq(1, ncol(grid1), by = by)] <- 1L
  grid2[seq(1, nrow(grid2), by = by), ic] <- 1L
  grid1 <- (grid1 + grid2) > 0
  grid1[grid1] <- 1
  grid1[!is.na(gxyz)] <- 0
  return(grid1)
}

#-------------------------------------------------------------------------------
.pspacePalette <- function(colors, trim.colors) {
  if(length(colors) != 5){
    colors <- colorRampPalette(colors)(5)
  } 
  tms <- trim.colors * 3
  offset <- list()
  offset[[1]] <- c(0.03, 0.09, 0.07, 0)
  offset[[2]] <- c(0.12, 0.26, 0.08, 0)
  offset[[3]] <- c(0.25, 0.11, 0.09, 0)
  offset[[4]] <- c(0.00, 0.31, 0.10, 0)
  offset[[5]] <- c(0.31, 0.30, 0.09, 0)
  cols <- lapply(seq_along(colors), function(i){
    cl <- adjustcolor(colors[i], offset = offset[[i]])
    colorRampPalette(c(colors[i], cl))(tms[i])
  })
  cols[[4]] <- rev(cols[[4]])
  cols[[5]] <- rev(cols[[5]])
  cols <- unlist(cols)
  cols <- colorRampPalette(cols)(25)
  return(cols)
}

#-------------------------------------------------------------------------------
.trimcols <- function(colors, bg.color, zlim, pars) {
  if(pars$ps$configs$scale.type == "negpos") {
    
    if (is.null(bg.color)) {
      if (length(colors) %% 2 == 1){
        bg.color <- colors[(length(colors)+1)/2]
      } else {
        bg.color <- colors[(length(colors)/2)+1]
      } 
    } else {
      if (length(colors) %% 2 == 1){
        colors[(length(colors)+1)/2] <- bg.color
      } else {
        n <- length(colors)/2
        colors <- c(colors[seq_len(n)], bg.color, 
          colors[(n+1):(n*2)])
      } 
    }
    cols <- colorRampPalette(colors)(17)
    bg <- cols[9]
    
  } else if(pars$ps$configs$scale.type=="neg") {
    
    if (is.null(bg.color)) {
      bg.color <- colors[length(colors)]
      colors <- colors[-length(colors)]
    }
    cols <- colorRampPalette(c(colors,bg.color))(16)
    bg <- cols[length(cols)]
    cols <- cols[-length(cols)]
    
  } else {
    
    if (is.null(bg.color)) {
      bg.color <- colors[1]
      colors <- colors[-1]
    }
    cols <- colorRampPalette(c(bg.color, colors))(16)
    bg <- cols[1]
    cols <- cols[-1]
    
  }
  
  bkIn <- seq(zlim[1], zlim[2], length.out = length(cols))
  bkOut <- pretty(zlim, n = 11)
  pal <- .trimRamp(bkIn, bkOut, cols)
  
  return(list(breaks = bkOut, pal = pal, bg = bg))
}
.trimRamp <- function(bkIn, bkOut, cols) {
  cols <- t(col2rgb(cols)/255)
  bkOut <- ifelse(bkOut < bkIn[1], bkIn[1], bkOut)
  bkOut <- ifelse(bkOut > bkIn[length(bkIn)], bkIn[length(bkIn)], bkOut)
  bnout <- .bincode(bkOut, bkIn, right = TRUE, include.lowest = TRUE)
  rcol <- lapply(unique(bnout), function(i){
    j <- bnout == i
    .getcolor(bkOut[j], bkIn[i], bkIn[i+1], cols[i, ], cols[i+1, ])
  })
  rcol <- unlist(rcol)
  return(rcol)
}
.getcolor <- function (x, bk1, bk2, c1, c2) {
  c1 <- grDevices::convertColor(c1, "sRGB", "Lab")
  c2 <- grDevices::convertColor(c2, "sRGB", "Lab")
  rcol <- matrix(ncol = 3, nrow = length(x))
  for (i in seq_len(3)) {
    xx <- (x - bk2) * (c2[i] - c1[i]) / (bk2 - bk1) + c2[i]
    rcol[, i] <- xx
  }
  rcol <- grDevices::convertColor(rcol, "Lab", "sRGB")
  .xlim <- function(x) {
    x[x < 0] <- 0; x[x > 1] <- 1
    return(x)
  }
  rcol[, ] <- .xlim(as.numeric(rcol))
  .rgb2hex <- function(r, g, b){
    grDevices::rgb(r, g, b, maxColorValue = 1)
  }
  rcol <- .rgb2hex(rcol[, 1], rcol[, 2], rcol[, 3])
  return(rcol)
}

Try the PathwaySpace package in your browser

Any scripts or data that you put into this service are public.

PathwaySpace documentation built on Aug. 8, 2025, 6:47 p.m.