R/geom-doughnut.R

Defines functions geom_doughnut_temp ggplot_add.doughnut geom_doughnut

Documented in geom_doughnut

#' Doughnut Layer
#' @description layer function to draw doughnut.
#' @param percent logical. If FALSE (the default) the value will be treated as
#' source value.
#' @param rfill the fill colour of other area.
#' @inheritParams ggplot2::layer
#' @inheritParams ggplot2::geom_polygon
#' @section Aesthetics:
#' \code{geom_doughnut()} understands the following aesthetics (required aesthetics are in bold):
#'     \itemize{
#'       \item \strong{\code{x}}
#'       \item \strong{\code{y}}
#'       \item \code{r0}
#'       \item \code{r1}
#'       \item \code{value}
#'       \item \code{colour}
#'       \item \code{fill}
#'       \item \code{alpha}
#'       \item \code{size}
#'       \item \code{linetype}
#'   }
#' @importFrom ggplot2 GeomPolygon
#' @importFrom ggplot2 draw_key_polygon
#' @importFrom grid grobTree
#' @importFrom scales alpha
#' @rdname geom_doughnut
#' @author Hou Yun
#' @examples
#' library(ggplot2)
#' dd <- tibble::tibble(x = runif(20, -180, 180),
#'                      y = runif(20, -90, 90),
#'                      value = purrr::map(sample(2:10, 20, TRUE), runif)
#'                      )
#'
#' ggplot(dd, aes(x, y)) +
#'   geom_doughnut(aes(value = value))
#' @export
geom_doughnut <- function(mapping = NULL,
                          data = NULL,
                          stat = "identity",
                          position = "identity",
                          ...,
                          percent = FALSE,
                          rfill = "grey80",
                          na.rm = FALSE,
                          show.legend = NA,
                          inherit.aes = TRUE) {
  structure(list(mapping = mapping,
                 data = data,
                 stat = stat,
                 position = position,
                 percent = percent,
                 rfill = rfill,
                 na.rm = na.rm,
                 show.legend = show.legend,
                 inherit.aes = inherit.aes,
                 ...), class = "doughnut"
            )
}

#' @method ggplot_add doughnut
#' @export
ggplot_add.doughnut <- function(object, plot, object_name) {
  data <- object$data %||% plot$data
  if(!tibble::is_tibble(data)) {
    data <- tibble::as_tibble(data)
  }
  if(isTRUE(object$inherit.aes)) {
    mapping <- aes_modify(plot$mapping, object$mapping)
  } else {
    mapping <- object$mapping
  }

  nm <- names(data)
  nn <- nrow(data)

  if(empty(data)) {
    return(plot)
  }

  vv <- aes_vars(mapping, "value")
  fv <- aes_vars(mapping, "fill")

  if(is.null(vv)) {
    data$.value <- 1
    mapping <- aes_modify(mapping, aes_(value = ~.value))
    vv <- ".value"
  }
  if(!is.list(data[[vv]])) {
    data[[vv]] <- as.list(data[[vv]])
  }

  value <- data[[vv]]
  if(all(vapply(value, is.character, logical(1))) ||
     all(vapply(value, is.factor, logical(1)))) {

    if(is.null(object$fill) && is.null(fv)) {
      fill <- lapply(value, function(.value) names(table(.value)))
      data$.fill <- fill
      mapping <- aes_modify(mapping, aes_(fill = ~.fill))
      fv <- ".fill"
    }
    value <- lapply(value, function(.value) as.numeric(table(.value)))

    object$percent <- FALSE
  }

  ll <- vapply(value, length, numeric(1))

  if(is.null(fv)) {
    if(is.null(object$fill)) {
      object$fill <- colorspace::qualitative_hcl(max(ll), "Set 2")
    }
    if(is.list(object$fill)) {
      object$fill <- rep_len(unlist(object$fill), sum(ll))
    } else {
      if(length(object$fill) <= max(ll)) {
        object$fill <- unlist(lapply(ll, function(.ll) rep_len(object$fill, .ll)))
      } else {
        object$fill <- rep_len(object$fill, sum(ll))
      }
    }
  } else {
    if(!is.list(data[[fv]])) {
      data[[fv]] <- as.list(data[[fv]])
    }
    data[[fv]] <- purrr::map2(data[[fv]], ll, rep_len)
    fill <- unlist(data[[fv]])
  }

  data$.ids <- seq_len(nn)
  mapping <- aes_modify(mapping, aes_(ids = ~.ids))
  ids <- rep(seq_len(nn), ll)
  data <- data[ids, ]
  data[[vv]] <- unlist(value)
  if(!is.null(fv)) data[[fv]] <- fill
  object$data <- data
  object$mapping <- mapping
  object <- do.call(geom_doughnut_temp, object)

  ggplot_add(object, plot, object_name)
}

#' @noRd
geom_doughnut_temp <- function(mapping = NULL,
                               data = NULL,
                               stat = "identity",
                               position = "identity",
                               ...,
                               rfill = "grey80",
                               percent = FALSE,
                               na.rm = FALSE,
                               show.legend = NA,
                               inherit.aes = TRUE) {
  layer(
    data = data,
    mapping = mapping,
    stat = stat,
    geom = GeomDoughnut,
    position = position,
    show.legend = show.legend,
    inherit.aes = inherit.aes,
    params = list(
      rfill = rfill,
      percent = percent,
      na.rm = na.rm,
      ...
    )
  )
}

#' @rdname linkET-extensions
#' @format NULL
#' @usage NULL
#' @export
GeomDoughnut <- ggproto(
  "GeomDoughnut", GeomPolygon,
  default_aes = aes(r0 = 0, r1 = 5, value = 1, colour = "grey35", size = 0.5,
                    linetype = 1, fill = "grey50", alpha = NA),
  required_aes = c("x", "y", "ids"),

  draw_panel = function(self, data, panel_params, coord, percent = FALSE,
                        rfill = "grey80", na.rm = FALSE) {
    if(empty(data)) {
      return(ggplot2::zeroGrob())
    }

    data <- coord$transform(data, panel_params)
    grobs <- lapply(split(data, data$ids), function(.data) {
      first_row <- .data[1, , drop = FALSE]
      other <- grid::nullGrob()
      main <- DoughnutGrob(x = first_row$x,
                           y = first_row$y,
                           r0 = first_row$r0,
                           r1 = first_row$r1,
                           value = .data$value,
                           percent = percent,
                           default.units = "native",
                           gp = gpar(col  = scales::alpha(.data$colour,
                                                          .data$alpha),
                                     fill = scales::alpha(.data$fill,
                                                          .data$alpha),
                                     lty  = .data$linetype,
                                     size = .data$size * ggplot2::.pt))

      if(isTRUE(percent)) {
        s <- sum(.data$value, na.rm = TRUE)
        positive <- s >= 0
        value <- if(positive) s - 1 else 1 + s
        other <- DoughnutGrob(x = first_row$x,
                              y = first_row$y,
                              r0 = first_row$r0,
                              r1 = first_row$r1,
                              value = value,
                              percent = percent,
                              default.units = "native",
                              gp = gpar(col  = scales::alpha(first_row$colour,
                                                             first_row$alpha),
                                        fill = scales::alpha(rfill, first_row$alpha),
                                        lty  = first_row$linetype,
                                        size = first_row$size * ggplot2::.pt))
      }
      grid::grobTree(other, main)
    })

    ggname("geom_doughnut", do.call("grobTree", grobs))
  },
  draw_key = draw_key_polygon
)
Hy4m/linkET documentation built on June 30, 2023, 7:39 p.m.