knitr::opts_chunk$set( collapse = TRUE, comment = "#>", out.width = "100%", dpi = 600 ) options(nflreadr.cache_warning = FALSE) options(warn = -1)
Working with publicly available NFL data often results in a desire to visualize the findings of one's analyses (often in ggplot2) and then publish these visualizations. In team-level analyses, it appears that such visualizations are more aesthetically pleasing when their logos are used instead of team abbreviations. Another way to visually distinguish teams is to use their primary or secondary team colors.
nflplotR provides ggplot2 extensions that greatly simplify these tasks and avoid typical problems.
Long story short: the only thing that is required to use NFL team logos or team colors in a ggplot is a variable in the plot data that holds official NFL team abbreviations.
The easiest way to get nflplotR is to install it from CRAN with:
install.packages("nflplotR")
To get a bug fix or to use a feature from the development version, you can install the development version of nflplotR either from GitHub with:
if (!require("pak")) install.packages("pak") pak::pak("nflverse/nflplotR")
or prebuilt from the development repo with:
install.packages("nflplotR", repos = "https://nflverse.r-universe.dev")
Let's look at some typical use cases for nflplotR using NFL play-by-play data. In a first step we load all necessary packages.
library(nflplotR) library(ggplot2) library(gt) library(nflreadr) library(dplyr, warn.conflicts = FALSE) options(nflreadr.verbose = FALSE)
We will load the data and compute each team's offensive and defensive EPA per play for the 2020 regular season. We will also compute the top 10 Quarterbacks in EPA per play.
pbp <- nflreadr::load_pbp(2020) |> dplyr::filter(season_type == "REG") |> dplyr::filter(!is.na(posteam) & (rush == 1 | pass == 1)) offense <- pbp |> dplyr::group_by(team = posteam) |> dplyr::summarise(off_epa = mean(epa, na.rm = TRUE)) defense <- pbp |> dplyr::group_by(team = defteam) |> dplyr::summarise(def_epa = mean(epa, na.rm = TRUE)) combined <- offense |> dplyr::inner_join(defense, by = "team") qbs <- pbp |> dplyr::filter(pass == 1 | rush == 1) |> dplyr::filter(down %in% 1:4) |> dplyr::group_by(id) |> dplyr::summarise( name = dplyr::first(name), team = dplyr::last(posteam), plays = dplyr::n(), qb_epa = mean(qb_epa, na.ram = TRUE) ) |> dplyr::filter(plays > 200) |> dplyr::slice_max(qb_epa, n = 10)
Offensive and Defensive EPA per Play are typically used for NFL team tiers. Let's create this scatter plot and play around with the capabilities of the logo geom.
ggplot2::ggplot(combined, aes(x = off_epa, y = def_epa)) + ggplot2::geom_abline(slope = -1.5, intercept = seq(0.4, -0.3, -0.1), alpha = .2) + nflplotR::geom_mean_lines(aes(x0 = off_epa , y0 = def_epa)) + nflplotR::geom_nfl_logos(aes(team_abbr = team), width = 0.065, alpha = 0.7) + ggplot2::labs( x = "Offense EPA/play", y = "Defense EPA/play", caption = "Data: @nflfastR", title = "2020 NFL Offensive and Defensive EPA per Play" ) + ggplot2::theme_minimal() + ggplot2::theme( plot.title = ggplot2::element_text(face = "bold"), plot.title.position = "plot", plot.background = ggplot2::element_rect(fill = "#F0F0F0") ) + ggplot2::scale_y_reverse()
There is a lot going on here:
What if we would like to highlight just a few teams? Well, there are multiple ways to do this. Either set the alpha channel for the irrelevant teams lower or give them a color (or do both). Wait...a color? Yes, it is possible to overwrite the color of logos but keep their shape! Let's do this in the next example. We would like to highlight the NFC East Teams for no specific reason...
nfc_east <- c("DAL", "NYG", "PHI", "WAS") combined |> dplyr::mutate( colour = ifelse(team %in% nfc_east, NA, "b/w"), alpha = ifelse(team %in% nfc_east, 0.9, 0.2) ) |> ggplot2::ggplot(aes(x = off_epa, y = def_epa)) + ggplot2::geom_abline(slope = -1.5, intercept = seq(0.4, -0.3, -0.1), alpha = .2) + nflplotR::geom_mean_lines(aes(x0 = off_epa , y0 = def_epa)) + nflplotR::geom_nfl_logos(aes(team_abbr = team, alpha = alpha, colour = colour), width = 0.065) + ggplot2::labs( x = "Offense EPA/play", y = "Defense EPA/play", caption = "Data: @nflfastR", title = "2020 NFL Offensive and Defensive EPA per Play" ) + ggplot2::scale_alpha_identity() + ggplot2::scale_color_identity() + ggplot2::theme_minimal() + ggplot2::theme( plot.title = ggplot2::element_text(face = "bold"), plot.title.position = "plot", plot.background = ggplot2::element_rect(fill = "#F0F0F0") ) + ggplot2::scale_y_reverse()
Please note the usage of the special color character "b/w"
which sets the colors to black and white and that we have to add ggplot2::scale_alpha_identity()
and ggplot2::scale_color_identity()
to our plot in case we are using alphas or colors as aesthetics because we want to use the actual alpha and color values in our data set and not some random defaults!
This is just a small overview and it is recommended to check the examples provided for each function.
Let's start with the offense and build a bar chart using logos as axis labels and team colors as bar colors.
ggplot2::ggplot(offense, aes(x = team, y = off_epa)) + ggplot2::geom_col(aes(color = team, fill = team), width = 0.5) + nflplotR::scale_color_nfl(type = "secondary") + nflplotR::scale_fill_nfl(alpha = 0.4) + ggplot2::labs( title = "2020 NFL Offensive EPA per Play", y = "Offense EPA/play" ) + ggplot2::theme_minimal() + ggplot2::theme( plot.title = ggplot2::element_text(face = "bold"), plot.title.position = "plot", plot.background = ggplot2::element_rect(fill = "#F0F0F0"), # it's obvious what the x-axis is so we remove the title axis.title.x = ggplot2::element_blank(), # this line triggers the replacement of team abbreviations with logos axis.text.x = nflplotR::element_nfl_logo(size = 1) )
Some notes:
axis.text.x = nflplotR::element_nfl_logo()
in the ggplot2 theme call. We can do the same thing for the y-axis (now with defensive EPA per play). Let's make them less intrusive by changing the color to black and white
ggplot2::ggplot(defense, aes(y = team, x = def_epa)) + ggplot2::geom_col(aes(color = team, fill = team), width = 0.5) + nflplotR::scale_color_nfl(type = "secondary") + nflplotR::scale_fill_nfl(alpha = 0.4) + ggplot2::labs( title = "2020 NFL Defensive EPA per Play", x = "Defense EPA/play" ) + ggplot2::theme_minimal() + ggplot2::theme( plot.title = ggplot2::element_text(face = "bold"), plot.title.position = "plot", plot.background = ggplot2::element_rect(fill = "#F0F0F0"), # it's obvious what the y-axis is so we remove the title axis.title.y = ggplot2::element_blank(), # this line triggers the replacement of team abbreviations with logos axis.text.y = nflplotR::element_nfl_logo(color = "b/w", size = 1) )
Similar to the usage of team logos as axis labels, we can also use player headshots. The only requirement is a valid NFL gsis ID as used in the nflfastR play-by-play data.
ggplot2::ggplot(qbs, aes(x = reorder(id, -qb_epa), y = qb_epa)) + ggplot2::geom_col(aes(color = team, fill = team), width = 0.5) + nflplotR::scale_color_nfl(type = "secondary") + nflplotR::scale_fill_nfl(alpha = 0.4) + ggplot2::labs( title = "2020 NFL Quarterback EPA per Play", y = "EPA/play" ) + ggplot2::theme_minimal() + ggplot2::theme( plot.title = ggplot2::element_text(face = "bold"), plot.title.position = "plot", plot.background = ggplot2::element_rect(fill = "#F0F0F0"), # it's obvious what the x-axis is so we remove the title axis.title.x = ggplot2::element_blank(), # this line triggers the replacement of gsis ids with player headshots axis.text.x = nflplotR::element_nfl_headshot(size = 1) )
We can do the above example alternatively by putting the player image on top of the columns and use team logos as axis labels instead.
ggplot2::ggplot(qbs, aes(x = reorder(team, -qb_epa), y = qb_epa)) + ggplot2::geom_col(aes(color = team, fill = team), width = 0.5) + nflplotR::geom_nfl_headshots(aes(player_gsis = id), width = 0.075, vjust = 0.45) + nflplotR::scale_color_nfl(type = "secondary") + nflplotR::scale_fill_nfl(alpha = 0.4) + ggplot2::labs( title = "2020 NFL Quarterback EPA per Play", y = "EPA/play" ) + ggplot2::ylim(0, 0.4) + ggplot2::theme_minimal() + ggplot2::theme( plot.title = ggplot2::element_text(face = "bold"), plot.title.position = "plot", plot.background = ggplot2::element_rect(fill = "#F0F0F0"), # it's obvious what the x-axis is so we remove the title axis.title.x = ggplot2::element_blank(), # this line triggers the replacement of team abbreviations with logos axis.text.x = nflplotR::element_nfl_logo(size = 1) )
The theme_elements
presented above are more powerful than it seems at first sight. We can replace text with NFL logos, NFL wordmarks, NFL player headshots at every position, where ggplot can put text!
And we can do the same with literally every image url on the internet.
To show this, we create a dataframe of example data and add some nfl team abbreviations and player gsis ids.
df <- mtcars |> dplyr::mutate( team = sample(c("LAC", "BUF", "DAL", "ARI"), nrow(mtcars), TRUE), player = sample(c("00-0033873", "00-0035228", "00-0036355", "00-0019596"), nrow(mtcars), TRUE) )
Now we plot some of the data and facet by the team abbreviations.
ggplot(df, aes(x = mpg, y = disp)) + geom_point() + facet_wrap(vars(team)) + labs( title = tools::toTitleCase("These are random teams and data"), subtitle = "I just want to show how the nflplotR theme elements work", caption = "https://github.com/nflverse/nflseedR/raw/master/man/figures/caption.png" ) + theme_minimal() + theme( plot.title.position = "plot", plot.title = ggplot2::element_text(face = "bold"), plot.background = ggplot2::element_rect(fill = "#F0F0F0"), axis.title = element_blank() )
There are valid team names in the facet titles and there is an image url in the caption. nflplotR can make images of these with two simple lines in the theme()
call.
ggplot(df, aes(x = mpg, y = disp)) + geom_point() + facet_wrap(vars(team)) + labs( title = tools::toTitleCase("These are random teams and data"), subtitle = "I just want to show how the nflplotR theme elements work", caption = "https://github.com/nflverse/nflseedR/raw/master/man/figures/caption.png" ) + theme_minimal() + theme( plot.title.position = "plot", plot.title = ggplot2::element_text(face = "bold"), axis.title = element_blank(), plot.background = ggplot2::element_rect(fill = "#F0F0F0"), # make wordmarks of team abbreviations strip.text = nflplotR::element_nfl_wordmark(size = 1), # load image from url in caption plot.caption = ggpath::element_path(hjust = 1, size = 0.4) )
If you want logos instead of wordmarks...
ggplot(df, aes(x = mpg, y = disp)) + geom_point() + facet_wrap(vars(team)) + labs( title = tools::toTitleCase("These are random teams and data"), subtitle = "I just want to show how the nflplotR theme elements work", caption = "https://github.com/nflverse/nflseedR/raw/master/man/figures/caption.png" ) + theme_minimal() + theme( plot.title.position = "plot", plot.title = ggplot2::element_text(face = "bold"), axis.title = element_blank(), plot.background = ggplot2::element_rect(fill = "#F0F0F0"), # make wordmarks of team abbreviations strip.text = nflplotR::element_nfl_logo(size = 1), # load image from url in caption plot.caption = ggpath::element_path(hjust = 1, size = 0.4) )
# the pkgdown workflow ran in cache size problems # so we clear the cache here hoping to resolve the issue nflplotR::.nflplotR_clear_cache()
Or maybe facet by player and use headshots...
ggplot(df, aes(x = mpg, y = disp)) + geom_point() + facet_wrap(vars(player)) + labs( title = tools::toTitleCase("These are random players and data"), subtitle = "I just want to show how the nflplotR theme elements work", caption = "https://github.com/nflverse/nflseedR/raw/master/man/figures/caption.png" ) + theme_minimal() + theme( plot.title.position = "plot", plot.title = ggplot2::element_text(face = "bold"), axis.title = element_blank(), plot.background = ggplot2::element_rect(fill = "#F0F0F0"), # make wordmarks of team abbreviations strip.text = nflplotR::element_nfl_headshot(size = 1), # load image from url in caption plot.caption = ggpath::element_path(hjust = 1, size = 0.4) )
One thing is particularly hard to implement in ggplot, namely the free arrangement of an image (e.g. your logo) somewhere in the plot. But with the help of the powerful theme_elements
it is possible!
All we need to do is assign the ggplot tag either an image URL or - in terms of nflplotR - a team abbreviation or player gsis_id. Then we can convert the tag with theme_elements
in plot.tag
into the corresponding image and position it anywhere with plot.tag.position
.
Let's try a logo first:
ggplot(mtcars, aes(x = mpg, y = disp)) + geom_point() + labs( title = tools::toTitleCase("These are random data"), subtitle = "I just want to show how to place an NFL logo somewhere in the plot", tag = "LAC" ) + theme_minimal() + theme( plot.tag = nflplotR::element_nfl_logo(size = 2), plot.tag.position = c(0.5, 0.85), plot.background = ggplot2::element_rect(fill = "#F0F0F0") )
Looks even better with wordmarks:
ggplot(mtcars, aes(x = mpg, y = disp)) + geom_point() + labs( title = tools::toTitleCase("These are random data"), subtitle = "I just want to show how to place an NFL wordmark top right", tag = "LAC" ) + theme_minimal() + theme( plot.tag = nflplotR::element_nfl_wordmark(size = 5, hjust = 1, vjust = 1), plot.tag.position = c(1, 1), plot.background = ggplot2::element_rect(fill = "#F0F0F0") )
And we can do player headshots:
ggplot(mtcars, aes(x = mpg, y = disp)) + geom_point() + labs( title = tools::toTitleCase("These are random data"), subtitle = "I just want to show how to place player headshot top right", tag = "00-0036355" ) + theme_minimal() + theme( plot.tag = nflplotR::element_nfl_headshot(size = 5, hjust = 1, vjust = 1), plot.tag.position = c(1, 1), plot.background = ggplot2::element_rect(fill = "#F0F0F0") )
See vignette("gt")
.
If you are used to preview your plots in RStudio you will probably get a bit impatient if a lot of logos have to be added to a plot because the rendering can take a few seconds. This is what the nflplotR function ggpreview()
is built for. It saves the plot in a temporary png file and previews it (in it's actual dimensions). Please see the function documentation for further information.
Fortunately, nflplotR was adapted for college football and released under the name cfbplotR.
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.