R/graphics-renderer.R

#' Graphics Renderer
#'
#' A [Renderer] based on the [graphics][graphics::graphics-package] package. This
#' is mostly used for testing at the moment.
#'
#' @export
GraphicsRenderer <- R6Class(
  "GraphicsRenderer", inherit = Renderer,
  public = list(

    render_points = function(
      x, y, feature = NULL,
      pch = theme("points.pch", 16),
      cex = theme("points.cex", 1),
      col = theme("points.col", "black"),
      lwd = theme("points.lwd", 1),
      bg = theme("points.bg", "black"),
      ...
    ) {
      graphics::points(x, y, pch = pch, cex = cex, col = col, lwd = lwd, bg = bg)
    },

    render_path = function(
      x, y, feature = 1L, part = 1L,
      lty = theme("path.lty", 1),
      lwd = theme("path.lwd", 1),
      col = theme("path.col", "black"),
      ...
    ) {
      tbl <- tibble(x, y, feature, part, lty, lwd, col)
      private$render_grouped(tbl, graphics::lines, groups = quos(.data$feature, .data$part))
    },

    render_polygon = function(
      x, y, feature = 1L, piece = 1L, part = 1L,
      col = theme("polygon.col", "black"),
      fill = theme("polygon.fill", "grey50"),
      lty = theme("polygon.lty", 1),
      ...
    ) {
      tbl <- tibble(x, y, feature, part, col = fill, border = col, lty)
      private$render_grouped(tbl, graphics::polypath, groups = quos(.data$feature, .data$part))
    },

    render_text = function(
      x, y, label, feature = 1L,
      cex = theme("text.cex", 1),
      col = theme("text.col", "black"),
      font = theme("text.font", 1),
      ...
    ) {
      graphics::text(x, y, label, cex = cex, col = col, font = font)
    },

    render_null = function() {
      invisible()
    },

    render_stack = function(...) {
      purrr::walk(quos(...), rlang::eval_tidy)
    },

    render_panel = function(panel, ...) {
      x <- panel$scales$scale("x")
      y <- panel$scales$scale("y")
      guide_x <- panel$guides$guide("x", default = GuideNull$new())
      guide_y <- panel$guides$guide("y", default = GuideNull$new())

      # create a blank plot
      graphics::plot(
        x = 1, y = 1,
        type = "n", axes = FALSE,
        xlab = guide_x$title(), ylab = guide_y$title(),

        # sets the limits
        xlim = x$limits_continuous(),
        ylim = y$limits_continuous()
      )

      # render data
      self$render_stack(...)

      # render guides
      purrr::map(
        panel$guides$lst,
        function(guide) guide$render(panel, renderer)
      )
    },

    render_panels = function(graphic, ...) {
      dots <- quos(...)
      n_panels <- length(dots)
      withr::with_par(list(mfrow = grDevices::n2mfrow(n_panels)), {
        purrr::walk(dots, rlang::eval_tidy)
      })
    },

    default_scale = function(x, aesthetic) {
      if (aesthetic %in% c("x", "y")) {
        if (is_discrete(x))
          ScaleDiscretePosition$new(aesthetic)$
          set_guide(GuideGraphicsAxis$new())
        else
          ScaleContinuousPosition$new(aesthetic)$
          set_guide(GuideGraphicsAxis$new())

      } else if (aesthetic %in% c("col", "fill")) {
        if (is_discrete(x)) {
          ScaleDiscrete$new(aesthetic)$
            set_palette_factory(scales::hue_pal())$
            set_na_value("grey50")$
            set_guide(GraphicsGuideLegend$new())
        } else {
          ScaleContinuous$new(aesthetic)$
            set_rescaler(scales::rescale)$
            set_palette(scales::seq_gradient_pal())$
            set_na_value("grey50")$
            set_guide(GraphicsGuideLegend$new())
        }

      } else if (aesthetic == "pch") {
        if (is_discrete(x)) {
          ScaleDiscrete$new(aesthetic)$
            set_palette_factory(scales::shape_pal())$
            set_guide(GraphicsGuideLegend$new())
        } else {
          abort("Cannot map a continuous value to 'pch'")
        }

      } else {
        ScaleNull$new(aesthetic)
      }
    }
  ),

  private = list(
    render_grouped = function(tbl, fun, args = quos(), groups = dplyr::vars(.data$group)) {
      dplyr::group_walk(
        dplyr::group_by(tbl, !!!groups),
        function(tbl_group, grouping) {
          exec(fun, !!!tbl_group, !!!args)
        }
      )
      invisible()
    }
  )
)

GuideGraphicsAxis <- R6Class(
  "GuideGraphicsAxis", inherit = Guide,

  public = list(
    axis_args = NULL,

    initialize = function(...) {
      super$initialize()
      axis_args <- list(...)

      if (length(axis_args) > 0 && !rlang::is_named(axis_args)) {
        abort("All axis guide arguments must be named")
      }

      self$axis_args <- axis_args
    },

    render = function(panel, renderer) {
      if (is.null(self$position())) {
        return()
      }

      aesthetic <- if ("x" %in% self$aesthetics()) "x" else "y"
      default_position <- if ("x" %in% self$aesthetics()) 1 else 2
      position <- self$position %|W|% default_position

      arg_names <- setdiff(
        names(formals(graphics::axis)),
        c("x", "y", "labels", "at")
      )

      exec(
        graphics::axis,
        default_position,
        at = self$key[[aesthetic]],
        labels = self$key$.labels,
        !!!self$axis_args[intersect(names(self$axis_args), arg_names)]
      )
    }
  )
)

GraphicsGuideLegend <- R6Class(
  "GraphicsGuideLegend", inherit = Guide,

  public = list(
    legend_args = NULL,
    layer_defaults = NULL,

    initialize = function(...) {
      super$initialize()
      legend_args <- list(...)

      if (length(legend_args) > 0 && !rlang::is_named(legend_args)) {
        abort("All legend guide arguments must be named")
      }

      self$legend_args <- legend_args
      self$layer_defaults <- NULL
    },

    train_layers = function(layers, renderer) {
      defaults <- list()
      for (layer in layers$lst) {
        if (!any(self$aesthetics() %in% layer$mapping$aesthetics())) {
          next
        }

        layer_defaults <- layer$geom$default_aesthetic_values(renderer)
        if (inherits(layer$geom, "GeomPoint")) {
          names(layer_defaults) <- gsub("(cex|lwd|bg)", "pt.\\1", names(layer_defaults))
        }

        defaults <- c(layer_defaults, defaults)
      }

      self$layer_defaults <- defaults[unique(names(defaults))]
      invisible(self)
    },

    render = function(panel, renderer) {
      if (is.null(self$position())) {
        return()
      }

      legend_args <- self$legend_args
      key <- self$key
      key$legend <- key$.labels

      arg_names <- setdiff(
        names(formals(graphics::legend)),
        c("title", "x", "y")
      )

      # combine the key with the default arguments
      key <- c(unclass(key), self$layer_defaults)
      key <- key[unique(names(key))]

      key_arg_names <- setdiff(intersect(names(key), arg_names), names(legend_args))
      legend_arg_names <- intersect(names(legend_args), arg_names)

      exec(
        graphics::legend,
        self$position() %|W|% "bottomright",
        title = self$title() %|W|% NULL,
        !!!key[key_arg_names],
        !!!legend_args[legend_arg_names]
      )
    }
  )
)
paleolimbot/ggr6 documentation built on Feb. 5, 2020, 2:17 p.m.