R/lollipop.R

Defines functions plot_lollipop prepare_lollipop_data

Documented in plot_lollipop prepare_lollipop_data

#' Prepare data for `plot_lollipop()`
#'
#' `r lifecycle::badge("maturing")`
#'
#' @param .data tibble or data.frame with variables
#' @param vars variables to reshape from wide to long, uses tidyselect syntax
#' @param group grouping variable (have to be logical!), usually denoting for
#'   which school the plot should be tailored for
#'
#' @return several tibbles inside list, intended for data-mining and for
#'   plotting
#'
#' @family Making charts
#'
#' @importFrom tidyr pivot_wider
#' @importFrom dplyr ungroup slice
#' @importFrom forcats fct_reorder
#'
#' @export
#'
prepare_lollipop_data <- function(.data, vars, group) {
  # assert group - logical
  d <- .data %>%
    pivot_longer({{ vars }}) %>%
    mutate(value = sten(.data$value))

  ref_meds <- d %>%
    filter(!{{ group }}) %>%
    group_by(.data$name) %>%
    summarise(med = median(.data$value, na.rm = TRUE))

  n_vars <- nrow(ref_meds)

  diff_data <- d %>%
    filter({{ group }}) %>%
    left_join(ref_meds, by = "name") %>%
    mutate(diff = .data$value - .data$med)

  main_data <- d %>%
    group_by({{ group }}, .data$name) %>%
    summarise(value = median(.data$value, na.rm = TRUE), .groups = "keep") %>%
    pivot_wider(names_from = {{ group }}) %>%
    ungroup() %>%
    mutate(
      diff = .data$`TRUE` - .data$`FALSE`, # positive = larger foc. group value
      name = fct_reorder(.data$name, .data$diff, abs, .desc = FALSE)
    )


  # single obs data for annotation geoms, so they do not "overplot" (print nrow()-times)
  d_single <- d |> slice(1)



  list(
    d = d,
    ref_meds = ref_meds,
    n_vars = n_vars,
    diff_data = diff_data,
    main_data = main_data,
    d_single = d_single
  )
}


#' Plot lollipop
#'
#' `r lifecycle::badge("maturing")`
#'
#' @param plot_data a
#' @param direction a
#' @param var_labels a
#' @param negative_label a
#' @param positive_label a
#' @param ref_label a
#' @param xlab a
#' @param observations_alpha opacity of individual observations
#'
#' @family Making charts
#'
#' @return ggplot2 plot
#'
#' @importFrom ggtext geom_richtext
#' @importFrom forcats fct_relevel
#'
#' @export
#'
plot_lollipop <- function(plot_data, direction = "blue_larger",
                          var_labels = waiver(),
                          negative_label = NULL, positive_label = NULL, ref_label = NULL,
                          xlab = "rozd\u00edl mezi Va\u0161\u00ed \u0161kolou a ostatn\u00edmi",
                          observations_alpha = .2) {
  direction <- match.arg(direction, c("blue_larger", "red_larger"))

  diff_scale <- if (direction == "blue_larger") {
    c(`TRUE` = "#2C7BB6FF", `FALSE` = "#D7191CFF")
  } else {
    c(`FALSE` = "#2C7BB6FF", `TRUE` = "#D7191CFF")
  }

  names_order <- levels(plot_data$main_data$name)

  plot_data$main_data %>%
    ggplot(aes(.data$diff, fct_relevel(.data$name, names_order),
      col = .data$diff > 0
    )) +
    # reference group line
    geom_vline(xintercept = 0, col = "grey", linetype = "dashed", size = .75) +
    # focal group datapoints
    geom_jitter(
      height = .25, width = 0, alpha = observations_alpha, shape = 16, size = 2,
      data = plot_data$diff_data
    ) +

    # lollipop stem
    geom_linerange(aes(xmax = diff, xmin = 0), size = 2.33, alpha = .5) +
    # lollipop ball-end
    geom_point(size = 4.5) +

    # negative (left) half-plane annotation arrow
    geom_segment(
      x = -.1, xend = -.9,
      y = .333, yend = .333,
      arrow = arrow(length = unit(.06, "in"), type = "closed"),
      col = "grey",
      data = plot_data$d_single
    ) +
    # negative (left) half-plane annotation text
    geom_richtext(
      aes(x = -.1, y = .333), # without aes(), nudge does not work...
      label = positive_label,
      vjust = 1.2, hjust = 1, col = "grey", fill = NA, label.color = NA,
      data = plot_data$d_single
    ) +

    # positive (right) half-plane annotation arrow
    geom_segment(
      x = .1, xend = .9,
      y = .333, yend = .333,
      arrow = arrow(length = unit(.06, "in"), type = "closed"),
      col = "grey",
      data = plot_data$d_single
    ) +
    # positive (right) half-plane annotation text
    geom_richtext(
      aes(x = .1, y = .333),
      label = negative_label,
      vjust = 1.2, hjust = 0, col = "grey", fill = NA, label.color = NA,
      data = plot_data$d_single
    ) +

    # reference group annotation arrow
    geom_curve(
      x = .5, xend = .1,
      yend = plot_data$n_vars + .4, y = plot_data$n_vars + .5,
      arrow = arrow(length = unit(.06, "in"), type = "closed"), curvature = .2,
      col = "grey",
      data = plot_data$d_single
    ) +

    # reference group annotation text
    geom_text(
      label = ref_label,
      x = .5, y = plot_data$n_vars + .5,
      vjust = .4, hjust = -.075,
      col = "grey", lineheight = 1,
      check_overlap = TRUE,
      data = plot_data$d_single
    ) +
    scale_x_continuous(
      breaks = scales::breaks_width(2), minor_breaks = scales::breaks_width(1),
      expand = expansion(.025)
    ) +
    scale_y_discrete(
      labels = var_labels, # named vector of variable-label
      expand = expansion(add = c(1.05, .65))
    ) +
    # master color depending on diff sign
    scale_color_manual(
      values = diff_scale
    ) +
    guides(col = guide_none()) + # disable legend
    xlab(xlab) +
    ylab(NULL) +
    theme_schola("x") +
    theme(
      panel.grid.minor.x = element_line(colour = "grey92", size = .15)
    )
}
scholaempirica/reschola documentation built on June 1, 2024, 1:29 p.m.