Skip to contents
library(legendry)
#> Loading required package: ggplot2

Rather than information that is key, this article will discuss information about keys.

Keys in vanilla ggplot2

The way guides exchange information with scales is through so called ‘keys’. Keys are simply data frames that typically hold information about the aesthetic, what values they represent and how these should be displayed. You may have already seen keys if you’ve used the get_guide_data() function before, as it can be used to retrieved a guide’s key. In the data frame below, we can see a key for the ‘x’ aesthetic. It tells us the relative location of tick marks in the aesthetic’s x column, and the numerical values they represent in the .value column. How the values should be communicated to users is captured in the .label column. Sometimes, keys have information about additional aesthetics, like the y column in the key below.

standard <- ggplot(mpg, aes(displ, hwy)) +
  geom_point(aes(shape = drv, colour = drv)) +
  labs(
    shape  = "Drive train",
    colour = "Drive train",
    y = "Highway efficiency",
    x = "Engine Displacement"
  )

get_guide_data(standard, aesthetic = "x")
#>           x .value .label y
#> 1 0.1127946      2      2 0
#> 2 0.2811448      3      3 0
#> 3 0.4494949      4      4 0
#> 4 0.6178451      5      5 0
#> 5 0.7861953      6      6 0
#> 6 0.9545455      7      7 0

Keys in legendry

The key difference between keys in legendry and keys in ggplot2, is that legendry exposes users to keys. At first, this can be an inconvenience, but it allows for a greater degree of customisation.

Why use keys?

Before we dig into the different types of keys, it is worth noting exactly why keys have been exposed. Keys represent ‘rules’ about how to annotate a scale, whereas the guide is the display of that rule.

For example, key_log() instructs to annotate every 10x change with large ticks, and in between changes with smaller ticks. It doesn’t really matter whether this rule is applied on an axis or a colour bar. Having the rule independent of the display makes it more modular.

logkey <- key_log()

ggplot(msleep, aes(sleep_total, brainwt, colour = bodywt)) +
  geom_point(na.rm = TRUE) +
  scale_y_log10(guide = guide_axis_base(key = logkey)) +
  scale_colour_viridis_c(
    trans = "log10",
    guide = guide_colbar(key = logkey)
  )

Regular keys

The understand better how a typical key works, we can use key_manual() to manually create a key. Usually it is sufficient to just provide the aesthetic argument, as the .value and .label columns automatically derive from that.

key_manual(aesthetic = c(2, 4, 6))
#>   aesthetic .value .label
#> 1         2      2      2
#> 2         4      4      4
#> 3         6      6      6

If you want custom labels, you can set the label argument. Most guides in legendry accept a key argument, which will cause the guide to display the information in the key, rather than the information automatically derived from the scale.

my_key <- key_manual(aesthetic = c(2, 4, 6), label = c("two", "four", "six"))

standard + guides(x = guide_axis_base(key = my_key))

In addition, you can provide some automatic keys as keywords. Setting key = "minor", is the same as setting key = key_minor(). In the same fashion many other key_*() functions can be used as keyword by omitting the key_-prefix.

standard + guides(x = guide_axis_base(key = "minor"))

Some keys don’t directly return data frames, but return instructions on how these keys should interact with scales. For example key_auto(), the default key for many guides in legendry, needs to know the range in which to populate tickmarks.

key <- key_auto()
print(key)
#> function (scale, aesthetic = NULL) 
#> {
#>     aesthetic <- aesthetic %||% scale$aesthetics[1]
#>     df <- Guide$extract_key(scale, aesthetic)
#>     df <- data_frame0(df, !!!extra_args(...))
#>     class(df) <- c("key_standard", "key_guide", class(df))
#>     df
#> }
#> <bytecode: 0x558408bf64b8>
#> <environment: 0x558408bf9280>

We can preview what values they’d label by letting the key absorb a scale with known limits.

template <- scale_y_log10(limits = c(1, 1000))
key(template, "y")
#>   y .value .label
#> 1 0      0      1
#> 2 1      1     10
#> 3 2      2    100
#> 4 3      3   1000

Ranged keys

A special type of guide you may find in legendry are so called ‘ranged’ guides. The only difference with regular guides is that they do not mark a single point for an aesthetic, but rather use a start- and end-point to mark a range of the aesthetic. This can be convenient to annotate co-occurrances between the data you are plotting and other events. For example, we can annotate the airtimes of TV shows in timeseries data.

ranges <- key_range_manual(
  start = as.Date(c("1985-09-14", "1993-09-16")),
  end   = as.Date(c("1992-05-09", "2004-05-13")), 
  name  = c("Golden Girls", "Frasier"), 
  level = 1:2
)
ranges
#>        start        end       .label .level
#> 1 1985-09-14 1992-05-09 Golden Girls      1
#> 2 1993-09-16 2004-05-13      Frasier      2

Compared to a regular key, we don’t have an aesthetic column, which is replaced by the start and end columns. In these cases, we cannot indicate a single .value, but we can still use the .label column. The .level column indicates how far we have to offset a range, so we’ll display “Frasier” farther away than “Golden Girls”.

ggplot(economics, aes(date, unemploy)) +
  geom_line() +
  guides(x.sec = primitive_bracket(ranges))

There is also an ‘automatic’ ranged key, which attempts to find patterns in the key labels.

plot <- ggplot(mpg, aes(interaction(drv, year), displ, fill = drv)) +
  geom_boxplot() +
  labs(
    x = "Drive train by year",
    y = "Engine displacement",
    fill = "Drive train"
  )
plot

For example an obvious pattern in the x-axis labels of the plot above is that you first have 3 entries for the 3 drive trains in 1999, followed by 3 drive trains in 2008. By default, key_range_auto() tries to split the label on any non-alphanumeric character, but you give explicit split instructions by using the sep argument.

# Split on literal periods
key <- key_range_auto(sep = "\\.")

plot + guides(x = primitive_bracket(key = key))

Futher gimmicks

Piping keys

The key_manual() and key_range_manual() functions have equivalents that are easy to pipe. They are called key_map() and key_range_map() respectively, and they can replace doing the following:

key <- key_range_manual(
  start = presidential$start,
  end   = presidential$end,
  name  = presidential$name
)

By the following, more pipe-friendly version:

key <- presidential |>
  key_range_map(
    start = start,
    end   = end,
    name  = name
  )

Both of these keys would display as something like this:

ggplot(economics, aes(date, unemploy)) +
  geom_line() +
  guides(x.sec = primitive_bracket(key))

Formatting keys

In addition to having a lot of control over what the keys display, you also have control over common text formatting operations in keys. Most key options have an ... argument that allows many arguments to element_text() to be passed on to the labels.

ggplot(mpg, aes(displ, hwy)) +
  geom_point() +
  guides(x = guide_axis_base(key = key_auto(colour = "red", face = "bold")))

In some cases where you know the label in advance, which is almost every time one uses key_manual(), key_map() or their ranged equivalents, you can even vectorise these formatting options.

guide <- presidential |>
  key_range_map(
    start = start,
    end   = end,
    name  = name,
    colour = ifelse(party == "Republican", "tomato", "dodgerblue"),
    face  = "bold"
  ) |>
  primitive_bracket()

ggplot(economics, aes(date, unemploy)) +
  geom_line() +
  guides(x.sec = guide)

Forbidden keys

There are, at the time of writing, two keys that you probably shouldn’t use in your code. These are key_sequence() and key_bins(). The hope is that mentioning their use here will prevent experimenting and subsequent frustration with these keys. You can see that key_sequence() does not produce an informative axis.

my_sequence_key <- key_sequence(n = 20)
standard +
  guides(x = guide_axis_base(key = my_sequence_key))

The reason for this is that this key was designed for colour gradients

ggplot(mpg, aes(displ, hwy, colour = cty)) +
  geom_point() +
  scale_colour_viridis_c(
    guide = gizmo_barcap(key = my_sequence_key)
  )

Likewise, key_bins() was not designed for regular guides, but is specific to colour steps.

my_bins_key <- key_bins()

ggplot(mpg, aes(displ, hwy, colour = cty)) +
  geom_point() +
  scale_colour_viridis_c(
    guide = gizmo_stepcap(key = my_bins_key)
  )