ggplot: The remix


Before you start…

Make sure you are still in your project that you created for doing exercises and then make a new R script. Save it as soon as you make it and give it a good name like exercise_2_day_3.R or herriot.R and you’ll be ready to go!

Always more you can do

Because of our excellent work so far both as an analytic bot and as the head baker, we’ve been gifted the wonderful R Graphics Cookbook: Practical Recipes for Visualing Data.

Wow, there’s a bunch of cool stuff in here, in fact, there’s a bunch of cool stuff that could make the work that we’ve already done even better.

Instead of using penguins, we’re going to be using penguins_raw for this exercise and we’re going to be making some graphs on summarized data. But first, let’s take a look at that data!

library(tidyverse)
library(palmerpenguins)

penguins_raw <- penguins_raw # this loads data from the palmerpenguins package

glimpse(penguins_raw)
## Rows: 344
## Columns: 17
## $ studyName             <chr> "PAL0708", "PAL0708", "PAL0708", "PAL0708", "PAL…
## $ `Sample Number`       <dbl> 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 1…
## $ Species               <chr> "Adelie Penguin (Pygoscelis adeliae)", "Adelie P…
## $ Region                <chr> "Anvers", "Anvers", "Anvers", "Anvers", "Anvers"…
## $ Island                <chr> "Torgersen", "Torgersen", "Torgersen", "Torgerse…
## $ Stage                 <chr> "Adult, 1 Egg Stage", "Adult, 1 Egg Stage", "Adu…
## $ `Individual ID`       <chr> "N1A1", "N1A2", "N2A1", "N2A2", "N3A1", "N3A2", …
## $ `Clutch Completion`   <chr> "Yes", "Yes", "Yes", "Yes", "Yes", "Yes", "No", …
## $ `Date Egg`            <date> 2007-11-11, 2007-11-11, 2007-11-16, 2007-11-16,…
## $ `Culmen Length (mm)`  <dbl> 39.1, 39.5, 40.3, NA, 36.7, 39.3, 38.9, 39.2, 34…
## $ `Culmen Depth (mm)`   <dbl> 18.7, 17.4, 18.0, NA, 19.3, 20.6, 17.8, 19.6, 18…
## $ `Flipper Length (mm)` <dbl> 181, 186, 195, NA, 193, 190, 181, 195, 193, 190,…
## $ `Body Mass (g)`       <dbl> 3750, 3800, 3250, NA, 3450, 3650, 3625, 4675, 34…
## $ Sex                   <chr> "MALE", "FEMALE", "FEMALE", NA, "FEMALE", "MALE"…
## $ `Delta 15 N (o/oo)`   <dbl> NA, 8.94956, 8.36821, NA, 8.76651, 8.66496, 9.18…
## $ `Delta 13 C (o/oo)`   <dbl> NA, -24.69454, -25.33302, NA, -25.32426, -25.298…
## $ Comments              <chr> "Not enough blood for isotopes.", NA, NA, "Adult…

If we want to do anything else with this data, it might be a good idea to clean the names because some of these look like they would be annoying to type…Delta 15 N (o/oo)? No thank you.

Good news! There’s a package that’s made just for cleaning up things and it’s called janitor and it’s incredibly useful, just like real-life janitors!

It can help clean your column names, find duplicate rows, and even make spiffy tables.

install.packages("janitor") # Comment this line out if you already installed this package 

library(janitor) # It's good practice to load libraries at the top of your script, so go ahead and move this up with the rest of your library calls.

penguins_raw_clean <- clean_names(penguins_raw)

names(penguins_raw_clean)
##  [1] "study_name"        "sample_number"     "species"          
##  [4] "region"            "island"            "stage"            
##  [7] "individual_id"     "clutch_completion" "date_egg"         
## [10] "culmen_length_mm"  "culmen_depth_mm"   "flipper_length_mm"
## [13] "body_mass_g"       "sex"               "delta_15_n_o_oo"  
## [16] "delta_13_c_o_oo"   "comments"

Still long, but the names are so much better!

Distribution of species by island revisted

Well, let’s see what we can do to make this more exciting. Let’s create a summary table in additional to making a new and exciting visual.

We’ll be using the tidyverse pipe %>% that we showed last session - it makes our code easier to read and reduces the number of nested parenthesis we have to deal with.

penguin_island_dist <- penguins_raw_clean %>% 
  group_by(region, island, species) %>% 
  summarise(count = n())

penguin_island_dist
## # A tibble: 5 × 4
## # Groups:   region, island [3]
##   region island    species                                   count
##   <chr>  <chr>     <chr>                                     <int>
## 1 Anvers Biscoe    Adelie Penguin (Pygoscelis adeliae)          44
## 2 Anvers Biscoe    Gentoo penguin (Pygoscelis papua)           124
## 3 Anvers Dream     Adelie Penguin (Pygoscelis adeliae)          56
## 4 Anvers Dream     Chinstrap penguin (Pygoscelis antarctica)    68
## 5 Anvers Torgersen Adelie Penguin (Pygoscelis adeliae)          52


That’s okay looking, but let’s try the tabyl function from the janitor package as well. The benefits of the tabyl function are that you can use the adorn_ functions from janitor to do things like add row and column totals really easily. This makes summary tables like the one below a breeze!

tabyl(penguins_raw_clean, species, island) %>% 
  adorn_totals(where = 'row') %>% 
  adorn_totals(where = 'col')
##                                    species Biscoe Dream Torgersen Total
##        Adelie Penguin (Pygoscelis adeliae)     44    56        52   152
##  Chinstrap penguin (Pygoscelis antarctica)      0    68         0    68
##          Gentoo penguin (Pygoscelis papua)    124     0         0   124
##                                      Total    168   124        52   344

Nice! A quick way to generate a two-by-two table that we could include in a report. Now we just need a pretty chart to go with it.

Graphing summary data

So below is a recreation of our previous chart using the raw data. What would we need to change to make this chart using the summarized data we stored in penguin_island_dist?

Remember that helpful tip from the help of geom_bar?

“There are two types of bar charts: geom_bar() and geom_col(). geom_bar() makes the height of the bar proportional to the number of cases in each group (or if the weight aesthetic is supplied, the sum of the weights). If you want the heights of the bars to represent values in the data, use geom_col() instead.”

help(geom_bar)

Column charts

Recreate your previous bar-chart using the summarized data that you stored in penguin_island_dist.



ggplot(data = __________, 
       aes(x = ______, y = ______, fill = ________)) +
  geom_xxx(position = position_dodge(preserve = "single")) + 
  scale_fill_brewer(type = "qual", palette = 3) +
  labs(title = "Distribution of penguin species by island",
       x = "Island",
       y = "Number of penguins observed") +
  theme_bw()

ggplot(data = penguin_island_dist, 
       aes(x = island, y = count, fill = species)) +
  geom_col(position = position_dodge(preserve = "single")) + 
  scale_fill_brewer(type = "qual", palette = 3) +
  labs(title = "Distribution of penguin species by island",
       x = "Island",
       y = "Number of penguins observed") +
  theme_bw()
  


Adding labels

Something special about summarized data is that now we can add labels to our columns as well. We can do this using geom_text but there’s a couple of things that we need to know before we get started.

  1. You can’t nudge and dodge text, so instead adjust the y position
  2. ggplot2 doesn’t know you want to give the labels the same virtual width as the bars so you have to make your position width smaller.
  3. geom_text() adds only text to the plot. geom_label() draws a rectangle behind the text, making it easier to read.
  4. To add labels at specified points use annotate() with annotate(geom = "text", ...) or annotate(geom = "label", ...)
  5. geom_text() requires the following aesthetics (i.e. pass these to the aes function if you haven’t yet!): x, y, and label (this is probably the one you’ll need to add)

Bar labels

Add text labels to your columns of the total count of each species per island. You might need to position them higher than the top of the bar and you shrink the width to make them look nice, do not be afraid to experiment.



ggplot(data = penguin_island_dist, 
       aes(x = island, y = count, fill = species)) +
  geom_col(position = position_dodge2(preserve = "single")) + 
  geom_xxxx(aes(label = ______, y = _____ + 5), 
            position = position_dodge2(width = ______)) +
  scale_fill_brewer(type = "qual", palette = 3) +
  labs(title = "Distribution of penguin species by island",
       x = "Island",
       y = "Number of penguins observed") +
  theme_bw()

ggplot(data = penguin_island_dist, 
       aes(x = island, y = count, fill = species)) +
  geom_col(position = position_dodge2(preserve = "single")) + 
  geom_text(aes(label = count, y = count + 5), 
            position = position_dodge2(width = 0.9)) +
  scale_fill_brewer(type = "qual", palette = 3) +
  labs(title = "Distribution of penguin species by island",
       x = "Island",
       y = "Number of penguins observed") +
  theme_bw()
  


We had to do three things when using geom_text or geom_label - try both and see which one you like the best!

  1. Make sure the positions matched, so if your geom_ is using dodge2, your geom_text should be using dodge2.
  2. Increase the y value by some amount so that the label will sit higher than the top of the labeled bar.
  3. For bar/column charts, decrease the width so they sit nicer on top.


Pick your challenge!

Now that you have some experience under your belt, it’s important to shake things up and have some fun. Depending on how much time you have left, pick 1 or several of the challenges below and see what you can make!

The links below will take you to the specific section of the R Graphics Cookbook that will answer the question of how you do the thing.