ggplot tricks
The goal of this repository is to keep track of some neat ggplot2 tricks Iβve learned. This assumes youβve familiarised yourself with the basics of ggplot2 and can construct some nice plots of your own. If not, please peruse the book at your leasure.
Iβm not incredibly adapt in gloriously typesetting plots and expertly
finetuning themes and colour palettes, so youβd have to forgive me. The
mpg
dataset is very versatile for plotting, so youβll be seeing a lot
of that as you read on. Extension packages are great, and Iβve dabbled
myself, but Iβll try to limit myself to vanilla ggplot2 tricks here.
For now, this will be mostly a README-only bag of tricks, but I may decide later to put them into separate groups in other files.
Table of contents
Letβs begin
By loading the library and setting a plotting theme. The first trick
here is to use theme_set()
to set a theme for all your plots
throughout a document. If you find yourself setting a very verbose theme
for every plot, here is the place where you set all your common
settings. Then never write a novel of theme elements ever again1!
library(ggplot2)
library(scales)
theme_set(
# Pick a starting theme
theme_gray() +
# Add your favourite elements
theme(
axis.line = element_line(),
panel.background = element_rect(fill = "white"),
panel.grid.major = element_line("grey95", linewidth = 0.25),
legend.key = element_rect(fill = NA)
)
)
Splicing aesthetics
The ?aes
documentation doesnβt tell you this, but you can splice the
mapping
argument in ggplot2. What does that mean? Well it means that
you can compose the mapping
argument on the go with !!!
. This is
especially nifty if you need to recycle aesthetics every once in a
while.
my_mapping <- aes(x = foo, y = bar)
aes(colour = qux, !!!my_mapping)
#> Aesthetic mapping:
#> * `x` -> `foo`
#> * `y` -> `bar`
#> * `colour` -> `qux`
Relating colour and fill
My personal favourite use of this is to make the fill
colour match the
colour
colour, but slightly lighter2. Weβll use the delayed
evaluation system for this, after_scale()
in this case, which youβll
see more of in the section following this one. Iβll repeat this trick a
couple of times throughout this document.
my_fill <- aes(fill = after_scale(alpha(colour, 0.3)))
ggplot(mpg, aes(displ, hwy)) +
geom_point(aes(colour = factor(cyl), !!!my_fill), shape = 21)
Text contrast
You may find yourself in a situation wherein youβre asked to make a heatmap of a small number of variables. Typically, sequential scales run from light to dark or vice versa, which makes text in a single colour hard to read. We could devise a method to automatically write the text in white on a dark background, and black on a light background. The function below considers a lightness value for a colour, and returns either black or white depending on that lightness.
contrast <- function(colour) {
out <- rep("black", length(colour))
light <- farver::get_channel(colour, "l", space = "hcl")
out[light < 50] <- "white"
out
}
Now, we can make an aesthetic to be spliced into a layerβs mapping
argument on demand.
autocontrast <- aes(colour = after_scale(contrast(fill)))
Lastly, we can test out our automatic contrast contraption. You may notice that it adapts to the scale, so you wouldnβt need to do a bunch of conditional formatting for this.
cors <- cor(mtcars)
# Melt matrix
df <- data.frame(
col = colnames(cors)[as.vector(col(cors))],
row = rownames(cors)[as.vector(row(cors))],
value = as.vector(cors)
)
# Basic plot
p <- ggplot(df, aes(row, col, fill = value)) +
geom_raster() +
geom_text(aes(label = round(value, 2), !!!autocontrast)) +
coord_equal()
p + scale_fill_viridis_c(direction = 1)
p + scale_fill_viridis_c(direction = -1)
Half-geoms
There are some extensions that offer half-geom versions of things. Of the ones I know, gghalves and the see package offer some half-geoms.
Here is how to abuse the delayed evaluation system to make your own. This can come in handy if youβre not willing to take on an extra dependency for just this feature.
Half-boxplots
The easy case is the boxplot. You can either set xmin
or xmax
to
after_scale(x)
to keep the right and left parts of a boxplot
respectively. This still works fine with position = "dodge"
.
# A basic plot to reuse for examples
p <- ggplot(mpg, aes(class, displ, colour = class, !!!my_fill)) +
guides(colour = "none", fill = "none") +
labs(y = "Engine Displacement [L]", x = "Type of car")
p + geom_boxplot(aes(xmin = after_scale(x)))
Half-errorbars
The same thing that works for boxplots, also works for errorbars.
p + geom_errorbar(
stat = "summary",
fun.data = mean_se,
aes(xmin = after_scale(x))
)
Half-violin
We can once again do the same thing for violin plots, but the layer
complains about not knowing about the xmin
aesthetic. It does use that
aesthetic, but only after the data has been setup, so it is not
intended to be a user accessible aesthetic. We can silence the warning
by updating the xmin
default to NULL
, which means it wonβt complain,
but also doesnβt use it if absent.
update_geom_defaults("violin", list(xmin = NULL))
p + geom_violin(aes(xmin = after_scale(x)))
Combining
Not left as an exercise for the reader this time, but I just wanted to show how it would work if you were to combine two halves and want them a little bit offset from one another. Weβll abuse the errorbars to serve as staples for the boxplots.
# A small nudge offset
offset <- 0.025
# We can pre-specify the mappings if we plan on recycling some
right_nudge <- aes(
xmin = after_scale(x),
x = stage(class, after_stat = x + offset)
)
left_nudge <- aes(
xmax = after_scale(x),
x = stage(class, after_stat = x - offset)
)
# Combining
p +
geom_violin(right_nudge) +
geom_boxplot(left_nudge) +
geom_errorbar(left_nudge, stat = "boxplot", width = 0.3)
Setting midpoints in divergent scales
Letβs say you have better colour intuition than I have, and three
colours arenβt enough for your divergent colour palette needs. A
painpoint is that it is tricky to get the midpoint right if your limits
arenβt perfectly centered around it. Enter the rescaler
argument in
league with scales::rescale_mid()
.
my_palette <- c("dodgerblue", "deepskyblue", "white", "hotpink", "deeppink")
p <- ggplot(mpg, aes(displ, hwy, colour = cty - mean(cty))) +
geom_point() +
labs(
x = "Engine displacement [L]",
y = "Highway miles per gallon",
colour = "Centered\nvalue"
)
p +
scale_colour_gradientn(
colours = my_palette,
rescaler = ~ rescale_mid(.x, mid = 0)
)
An alternative is to simply center the limits on x. We can do that by providing a function to the scaleβs limits.
p +
scale_colour_gradientn(
colours = my_palette,
limits = ~ c(-1, 1) * max(abs(.x))
)
Facetted tags
Putting text annotations on facetted plots is a pain, because limits can vary on a per-panel basis, so it is very difficult to find the correct position. An extension that explores alleviating this pain is the tagger extension, but we can do a similar thing in vanilla ggplot2.
Luckily, there is a mechanic in ggplot2βs position axes that letβs
-Inf
and Inf
be interpreted as the scaleβs minimum and maximum limit
respectively3. You can exploit this by choosing x = Inf, y = Inf
to
put the labels in a corner. You can also use -Inf
instead of Inf
to
place at the bottom instead of top, or left instead of right.
We need to match the hjust
/vjust
arguments to the side of the plot.
For x/y = Inf
, they would need to be hjust/vjust = 1
, and for
x/y = -Inf
they need to be hjust/vjust = 0
.
p + facet_wrap(~ class, scales = "free") +
geom_text(
# We only need 1 row per facet, so we deduplicate the facetting variable
data = ~ subset(.x, !duplicated(class)),
aes(x = Inf, y = Inf, label = LETTERS[seq_along(class)]),
hjust = 1, vjust = 1,
colour = "black"
)
Unfortunately, this places the text straight at the border of the panel,
which may offend our sense of beauty. We can get slightly fancier by
using geom_label()
, which lets us more precisely control the spacing
between the text and the panel borders by setting the label.padding
argument.
Moreover, we can use label.size = NA, fill = NA
to hide the textbox
part of the geom. For illustration purposes, we now place the tag at the
top-left instead of top-right.
p + facet_wrap(~ class, scales = "free") +
geom_label(
data = ~ subset(.x, !duplicated(class)),
aes(x = -Inf, y = Inf, label = LETTERS[seq_along(class)]),
hjust = 0, vjust = 1, label.size = NA, fill = NA,
label.padding = unit(5, "pt"),
colour = "black"
)
Recycling plots
Letβs say weβre tasked with making a bunch of similar plots, with different datasets and columns. For example, we might want to make a series of barplots4 with some specific pre-sets: weβd like the bars to touch the x-axis and not draw vertical gridlines.
Functions
One well-known way to make a bunch of similar plots is to wrap the plot construction into a function. That way, you can use encode all the presets you want in your function.
I case you might not know, there are various methods to program with
the aes()
function,
and using {{ }}
(curly-curly) is one of the more flexible ways 5.
barplot_fun <- function(data, x) {
ggplot(data, aes(x = {{ x }})) +
geom_bar(width = 0.618) +
scale_y_continuous(expand = c(0, 0, 0.05, 0)) +
theme(panel.grid.major.x = element_blank())
}
barplot_fun(mpg, class)
One drawback of this approach is that you lock-in any aesthetics in the
function arguments. To go around this, an even simpler way is to simply
pass ...
directly to aes()
.
barplot_fun <- function(data, ...) {
ggplot(data, aes(...)) +
geom_bar(width = 0.618) +
scale_y_continuous(expand = c(0, 0, 0.1, 0)) +
theme(panel.grid.major.x = element_blank())
}
barplot_fun(mpg, class, colour = factor(cyl), !!!my_fill)
Skeletons
Another method of doing a very similar thing, is to use plot
βskeletonsβ. The idea behind a skeleton is that you can build a plot,
with or without any data
argument, and add in the specifics later.
Then, when you actually want to make a plot, you can use the %+%
to
fill in or replace the dataset, and + aes(...)
to set the relevant
aesthetics.
barplot_skelly <- ggplot() +
geom_bar(width = 0.618) +
scale_y_continuous(expand = c(0, 0, 0.1, 0)) +
theme(panel.grid.major.x = element_blank())
my_plot <- barplot_skelly %+% mpg +
aes(class, colour = factor(cyl), !!!my_fill)
my_plot
One neat thing about these skeletons is that even when youβve already
filled in the data
and mapping
arguments, you can just replace them
again and again.
my_plot %+% mtcars +
aes(factor(carb), colour = factor(cyl), !!!my_fill)
6
RibcageThe idea here is to not skeletonise the entire plot, but just a
frequently re-used set of parts. For example, we might want to label our
barplot, and pack together all the things that make up a labelled
barplot. The trick to this is to not add these components together
with +
, but simply put them in a list()
. You can then +
your list
together with the main plot call.
labelled_bars <- list(
geom_bar(my_fill, width = 0.618),
geom_text(
stat = "count",
aes(y = after_stat(count),
label = after_stat(count),
fill = NULL, colour = NULL),
vjust = -1, show.legend = FALSE
),
scale_y_continuous(expand = c(0, 0, 0.1, 0)),
theme(panel.grid.major.x = element_blank())
)
ggplot(mpg, aes(class, colour = factor(cyl))) +
labelled_bars +
ggtitle("The `mpg` dataset")
ggplot(mtcars, aes(factor(carb), colour = factor(cyl))) +
labelled_bars +
ggtitle("The `mtcars` dataset")
Session info
#> β Session info βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
#> setting value
#> version R version 4.2.2 (2022-10-31 ucrt)
#> os Windows 10 x64 (build 22621)
#> system x86_64, mingw32
#> ui RTerm
#> language (EN)
#> collate English_United Kingdom.utf8
#> ctype English_United Kingdom.utf8
#> tz Europe/Berlin
#> date 2023-02-14
#> pandoc 2.19.2
#>
#> β Packages βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
#> package * version date (UTC) lib source
#> assertthat 0.2.1 2019-03-21 [] CRAN (R 4.2.0)
#> cli 3.4.1 2022-09-23 [] CRAN (R 4.2.2)
#> colorspace 2.0-3 2022-02-21 [] CRAN (R 4.2.0)
#> DBI 1.1.3 2022-06-18 [] CRAN (R 4.2.2)
#> digest 0.6.29 2021-12-01 [] CRAN (R 4.2.0)
#> dplyr 1.0.10 2022-09-01 [] CRAN (R 4.2.1)
#> evaluate 0.19 2022-12-13 [] CRAN (R 4.2.2)
#> fansi 1.0.3 2022-03-24 [] CRAN (R 4.2.0)
#> farver 2.1.1 2022-07-06 [] CRAN (R 4.2.1)
#> fastmap 1.1.0 2021-01-25 [] CRAN (R 4.2.0)
#> generics 0.1.3 2022-07-05 [] CRAN (R 4.2.1)
#> ggplot2 * 3.4.1 2023-02-10 [] CRAN (R 4.2.2)
#> glue 1.6.2 2022-02-24 [] CRAN (R 4.2.0)
#> gtable 0.3.1 2022-09-01 [] CRAN (R 4.2.1)
#> highr 0.10 2022-12-22 [] CRAN (R 4.2.2)
#> htmltools 0.5.4 2022-12-07 [] CRAN (R 4.2.2)
#> knitr 1.41 2022-11-18 [] CRAN (R 4.2.2)
#> labeling 0.4.2 2020-10-20 [] CRAN (R 4.2.0)
#> lifecycle 1.0.3 2022-10-07 [] CRAN (R 4.2.2)
#> magrittr 2.0.3 2022-03-30 [] CRAN (R 4.2.0)
#> munsell 0.5.0 2018-06-12 [] CRAN (R 4.2.0)
#> pillar 1.8.1 2022-08-19 [] CRAN (R 4.2.1)
#> pkgconfig 2.0.3 2019-09-22 [] CRAN (R 4.2.0)
#> R6 2.5.1 2021-08-19 [] CRAN (R 4.2.0)
#> ragg 1.2.2 2022-02-21 [] CRAN (R 4.2.0)
#> rlang 1.0.6 2022-09-24 [] CRAN (R 4.2.1)
#> rmarkdown 2.19 2022-12-15 [] CRAN (R 4.2.2)
#> rstudioapi 0.14 2022-08-22 [] CRAN (R 4.2.2)
#> scales * 1.2.1 2022-08-20 [] CRAN (R 4.2.2)
#> sessioninfo 1.2.2 2021-12-06 [] CRAN (R 4.2.0)
#> stringi 1.7.6 2021-11-29 [] CRAN (R 4.2.0)
#> stringr 1.5.0 2022-12-02 [] CRAN (R 4.2.2)
#> systemfonts 1.0.4 2022-02-11 [] CRAN (R 4.2.0)
#> textshaping 0.3.6 2021-10-13 [] CRAN (R 4.2.0)
#> tibble 3.1.8 2022-07-22 [] CRAN (R 4.2.1)
#> tidyselect 1.2.0 2022-10-10 [] CRAN (R 4.2.2)
#> utf8 1.2.2 2021-07-24 [] CRAN (R 4.2.0)
#> vctrs 0.5.0 2022-10-22 [] CRAN (R 4.2.2)
#> viridisLite 0.4.1 2022-08-22 [] CRAN (R 4.2.1)
#> withr 2.5.0 2022-03-03 [] CRAN (R 4.2.0)
#> xfun 0.36 2022-12-21 [] CRAN (R 4.2.2)
#> yaml 2.3.5 2022-02-21 [] CRAN (R 4.2.0)
#>
#>
#> ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
Footnotes
-
Well, you need to do it once at the start of your document. But then never again! Except in your next document. Just write a
plot_defaults.R
script andsource()
that from your document. Copy-paste that script for every project. Then, truly, never againβ€οΈ .β© -
This is a lie. In reality, I use
aes(colour = after_scale(colorspace::darken(fill, 0.3)))
instead of lightening the fill. I didnβt want this README to have a dependency on {colorspace} though.β© -
Unless you self-sabotage your plots by setting
oob = scales::oob_censor_any
in the scale for example.β© -
In your soul of souls, do you really want to make a bunch of barplots though?
β© -
The alternative is to use the
.data
pronoun, which can be.data$var
if you want to lock in that column in advance, or.data[[var]]
whenvar
is passed as a character.β© -
This bit was originally called βpartial skeletonβ, but as a ribcage is a part of a skeleton, this title sounded more evocative.
β©