class: center, middle, inverse, title-slide .title[ # Choropleth Maps ] .author[ ### Nicholas Sim ] .date[ ### 10 April 2024 ] --- class: center, middle, inverse # Introduction --- ### Topics * Using in-built maps from `library(maps)` * Map construction using `geom_polygon()` * Combine map data with spatial variables * Map data from `rnatrualearth` * The `sf` (simple features) package and `geom_sf()` * `grid.arrange()` for laying multiple plots in a single graph --- ### Required Libraries ```r library(tidyverse) library(socviz) library(maps) # To extract map data library(mapproj) library(ggthemes) # For standard themes library(ggpubr) # To use ggarrange library(sf) # To read and plot simple features data frame library(gridExtra) # For grid.arrange library(ggrepel) # To include labels theme_set(theme_bw()) ``` --- ### Choropleth Maps Ref: Chapter 7 KH A **choropleth map** is a thematic map where regions or areas on the map are colored in proportion to the values of a variable of interest. .pull-left[For instance, the map here shows the 2016 US elections results. The states are represented by two colors, red if won by the Republicans and blue if won by the Democrats. Additionally, the states with a larger number of electoral seats are represented by a deeper shade of color. ] .pull-right[ data:image/s3,"s3://crabby-images/6bfd7/6bfd72d381fcb327e05fe0a4f58dbb49080a5c28" alt="US Elections" ] --- ### Constructing a Choropleth Map The choropleth map is constructed in two steps. 1. Plot the boundaries, a.k.a. "polygons", of the regions. We may do so with `geom_polygon()` (later, we will explore `geom_sf()`). 2. Use a variable as a `fill` aesthetic to fill the area of the polygons. --- ### What is a Boundary? A boundary is a closed line plot constructed by tracing out points on an x-y axis. For example, the subplot on the left shown below plots out the boundary of a region using `geom_path()`. A map is essentially a boundary plot, except that the x-aesthetic represents longitude (i.e., east-west) and the y-aesthetic represents latitude (i.e., north-south). This is shown in the right subplot illustrating the boundaries of New Caledonia. <img src="ChoroplethMaps_files/figure-html/unnamed-chunk-1-1.png" style="display: block; margin: auto;" /> --- ### The `maps` Package Map data consists of points defined by longitudes and latitudes, which delineate the boundaries of various regions like countries, states, or counties. The maps package conveniently organizes latitude and longitude coordinates into a data frame, facilitating the plotting of sub-regions within countries such as the United States, France, and New Zealand. Additionally, it includes a low-resolution world map where points outlining regions are further apart. However, not all map data is readily available in a standard data frame format with separate longitude and latitude columns. Frequently, map data sourced from providers are formatted differently, often as shape (`shp`) files, simple features (`sf`) files, or geojson (`geojson`) files. In this exercise, we'll utilize the maps package to generate choropleth maps for the United States. Let's start by saving the map data for US states. ```r library(maps) us_states <- map_data("state") # you may select county too. ``` --- class: center, middle, inverse # Map Data Frame --- ### Necessary Components for Plotting Polygons .pull-left[ Let's look at the `us_states` map data. Notice that `us_states` is a standard data frame containing a location's **longitude** (east-west), **latitude** (north-south), and a **group** identifier (id). ] .pull-right[ ```r head(us_states[,1:4],3) ``` ``` ## long lat group order ## 1 -87.46201 30.38968 1 1 ## 2 -87.48493 30.37249 1 2 ## 3 -87.52503 30.37249 1 3 ``` ] These three variables are required for `geom_polygon()` to trace out the boundaries of an area by connecting the coordinates. The coordinates representing the boundaries of an area will be joined by the line to form a **polygon**. To ensure that the coordinates are connected by the line in the correct sequence, there is a variable called `order`. The variable contains the order by which the coordinates are to be connected to form a polygon. Since `geom_polygon()` works by plotting a line that connects the longitude and latitude of one row with those of the next row in the data frame, we need to ensure that the rows are correctly ordered first before plotting. --- ### Drawing a Map `geom_polygon()` in `ggplot2` is utilized to render polygons on a plot. It requires the longitude to be assigned to the x-aesthetic (representing east-west orientation), latitude to the y-aesthetic (representing north-south orientation), and a group aesthetic, which typically serves as a group identifier. The group ID is crucial for delineating the boundaries of regions. It helps in identifying where a boundary starts and ends, ensuring the correct formation of polygons. .pull-left[ ```r p <- ggplot(data = us_states, mapping = aes(x = long, y = lat, group = group)) p + geom_polygon(fill = "white", color = "black") ``` ] .pull-right[ <img src="ChoroplethMaps_files/figure-html/map.1-out-1.png" style="display: block; margin: auto;" /> ] --- ### Importance of Group ID As discussed, `geom_polygon()` will simply connect the longitude and latitude contained in one row with those of the next row. Without declaring the states as groups using a group aesthetic, the polygon will not terminate at the end of the state boundary. For instance, Row 202 is the last observation for Alabama and Row 204 is the first observation for Arizona. Without declaring the group id, `geom_polygon()` will mistakenly connect the longitude and latitude of Row 202 with those of Row 204. This will result in a line connecting Alabama and Arizona, instead of the boundary line terminating at Row 202, the last record for Alabama. ```r us_states[201:204,] ``` ``` ## long lat group order region subregion ## 201 -87.45055 30.40114 1 201 alabama <NA> ## 202 -87.46201 30.38968 1 202 alabama <NA> ## 204 -114.63739 35.01918 2 204 arizona <NA> ## 205 -114.64313 35.10512 2 205 arizona <NA> ``` --- ### Importance of Group ID Thus, if the group id is not declared, we will have the following plot: .pull-left[ ```r p <- ggplot(data = us_states, mapping = aes(x = long, y = lat)) p + geom_polygon(fill = "white", color="black") ``` ] .pull-right[ <img src="ChoroplethMaps_files/figure-html/map.2-out-1.png" style="display: block; margin: auto;" /> ] --- ### Importance of Ordering Since `geom_polygon()` connects the longitudes and latitudes values in the rows of the data frame in sequence, the boundary will not be correctly traced out if the rows are not properly ordered. For instance, let's shuffle the rows and see what happens. ```r index <- sample(nrow(us_states)) #random index us_states.1 <- us_states[index,] %>% arrange(group) head(us_states.1) # Order within a group is shuffled ``` ``` ## long lat group order region subregion ## 1 -87.46774 30.52719 1 192 alabama <NA> ## 2 -88.16675 34.95043 1 87 alabama <NA> ## 3 -86.31036 34.99053 1 96 alabama <NA> ## 4 -85.23893 33.10550 1 110 alabama <NA> ## 5 -87.63390 30.85951 1 181 alabama <NA> ## 6 -85.49677 31.00847 1 168 alabama <NA> ``` --- ### Importance of Ordering Now, let's plot the map with the rows ordered out-of-sequence: .pull-left[ ```r p <- ggplot(data = us_states.1, mapping = aes(x = long, y = lat, group = group)) p + geom_polygon(aes(color = group), fill = "white") + scale_color_viridis_c() + theme(legend.position = "none") ``` ] .pull-right[ <img src="ChoroplethMaps_files/figure-html/map.3-out-1.png" style="display: block; margin: auto;" /> ] --- class: center, middle, inverse # Coordinates and Labels --- ### Adjusting the Coordinates Maps projected onto a flat surface will be distorted as the earth is elliptical. The distortion will be more severe for locations further away from the Equator. To reduce such distortions, we should project a map on a non-flat surface such as an open cone. For a list of possible projections, see the help file for `mapproject()`. Below, we employ the "Albers" projection. .pull-left[ ```r p <- ggplot(data = us_states, mapping = aes(x = long, y = lat, group = group, fill = region)) # use region as a fill aesthetic p + geom_polygon(color = "gray90", size = 0.1) + coord_map(projection = "albers", lat0 = 39, lat1 = 45) + # Albers projection requires lat0 and lat1 guides(fill = FALSE) ``` ] .pull-right[ <img src="ChoroplethMaps_files/figure-html/map.4-out-1.png" style="display: block; margin: auto;" /> ] --- ### Adding Labels (The Wrong Way) To label the states on the map, we might be tempted to pass in `region` as a label aesthetic. This is incorrect as each state has multiple rows of longitudes and latitudes for drawing the state boundary, and each row has a state label. If we pass `region` in as a label, the state label will appear on the map for each row that the state label exists in the data frame. .pull-left[ ```r p <- ggplot(data = us_states, mapping = aes(x = long, y = lat, group = group, fill = region)) # use region as a fill aesthetic p + geom_polygon(color = "gray90", size = 0.1) + coord_map(projection = "albers", lat0 = 39, lat1 = 45) + # Albers projection requires lat0 and lat1 guides(fill = FALSE) + geom_text(mapping = aes(label = region), size = 3, hjust = 0.5) ``` ] .pull-right[ <img src="ChoroplethMaps_files/figure-html/map.5-out-1.png" style="display: block; margin: auto;" /> ] --- ### Adding Labels (The Right Way) To avoid this, we should create a new data frame containing the longitude and latitude of the **centroid** of each state, which is the center of the state. The centroid can be approximated by the mean of the state's longitudes and latitudes. Below, we create a new data frame `us_states.lab` containing the centroid of each state. .pull-left[ ```r us_states.lab <- us_states %>% group_by(region) %>% # Can group by the group id summarise(long = mean(long), lat = mean(lat)) head(us_states.lab, 3) ``` ] .pull-right[ ``` ## # A tibble: 3 × 3 ## region long lat ## <chr> <dbl> <dbl> ## 1 alabama -86.9 31.7 ## 2 arizona -113. 34.5 ## 3 arkansas -91.3 34.6 ``` ] Here, there is only one set of longitude and latitude for each state, and therefore, only one label for each state. By declaring the new data frame locally in `geom_text_repel()`, we may plot the state label next to the centroid for each state (and avoid having repeated labels). --- ### Adding Labels (The Right Way) Using `us_states.lab`, we pass the `region` variable into `geom_text_repel()` to label the centroid's longitude and latitude, specified as the `x` and `y` aesthetic in `geom_text_repel()`: .pull-left[ ```r p <- ggplot(data = us_states, mapping = aes(x = long, y = lat, group = group, fill = region)) # use region as a fill aesthetic p + geom_polygon(color = "gray90", size = 0.1) + coord_map(projection = "albers", lat0 = 39, lat1 = 45) + # Albers projection requires lat0 and lat1 guides(fill = FALSE) + geom_text_repel(data = us_states.lab, mapping = aes(x = long, y=lat, group = region, label = region), size = 3, hjust = 0.5) # Notice we pass in a different dataset into geom_text_repel. ``` ] .pull-right[ <img src="ChoroplethMaps_files/figure-html/map.6-out-1.png" style="display: block; margin: auto;" /> ] --- class: center, middle, inverse # Visualizing A Variable Spatially --- ### Merging Map Data with Data on Regional Attributes A choropleth map uses color gradients to differentiate the values of a variable (categorical or numerical) when visualized on a map. To visualize a variable on a map, we declare it as a `fill` aesthetic. Before a choropleth map is constructed, we need to first merge the map data with our dataset. To illustrate, we will construct two choropleth maps. The first will indicate the winning party for each state during the 2016 US election and second will display Donald Trump's winning percentage instead. The party and winning percentage data can be found in the `election` dataset from the `socviz` package.
--- ### Merging Map Data with Data on Regional Attributes To merge the `election` dataset with the map data, `us_states`, we need a variable to serve as the identifier for joining the rows of both datasets. The ID variable must be found in both datasets, although they can be named differently. From the `election` and `us_states` datasets, the common variable is `state` in `election` and `region` in `us_states`. However, the names of the states are in title case in the `election` dataset but lower case in the `us_states` dataset. For example, R does not know that `alabama` and `Alabama` are the same thing. Thus, we need to convert the names of the states in the `election` dataset to lower case using the `tolower()` function, so that the names are consistent between `election` and `us_states`. The adjusted names of the states are saved as the `region` variable in the `election` dataset. ```r election$region <- tolower(election$state) ``` --- ### Merging Map Data with Data on Regional Attributes To join `election` with `us_states`, we may now use `region` as the identifier. We will employ `left_join()`, where `left_join()` returns all rows from the first dataset, and all columns from the first and second datasets. In other words, the `left_join()` function uses the left dataset as the master (i.e. no records will be dropped), and joins it with data from the right dataset. .pull-left[ Here, the left dataset is `us_states`. The combined dataset is saved as `us_states_elec`. Notice that we do not need to declare what the identifier is. Since `region` is the common variable across the two datasets, it will be recognized as the ID variable (otherwise, you will need to provide the "by" argument). ] .pull-right[ ```r # region is the only common variable us_states_elec <- left_join(us_states, election) # returns all rows from us_states, and all columns in both str(us_states_elec) # Notice the recycling property of R ``` ``` ## 'data.frame': 15537 obs. of 28 variables: ## $ long : num -87.5 -87.5 -87.5 -87.5 -87.6 ... ## $ lat : num 30.4 30.4 30.4 30.3 30.3 ... ## $ group : num 1 1 1 1 1 1 1 1 1 1 ... ## $ order : int 1 2 3 4 5 6 7 8 9 10 ... ## $ region : chr "alabama" "alabama" "alabama" "alabama" ... ## $ subregion : chr NA NA NA NA ... ## $ state : chr "Alabama" "Alabama" "Alabama" "Alabama" ... ## $ st : chr "AL" "AL" "AL" "AL" ... ## $ fips : num 1 1 1 1 1 1 1 1 1 1 ... ## $ total_vote : num 2123372 2123372 2123372 2123372 2123372 ... ## $ vote_margin : num 588708 588708 588708 588708 588708 ... ## $ winner : chr "Trump" "Trump" "Trump" "Trump" ... ## $ party : chr "Republican" "Republican" "Republican" "Republican" ... ## $ pct_margin : num 0.277 0.277 0.277 0.277 0.277 ... ## $ r_points : num 27.7 27.7 27.7 27.7 27.7 ... ## $ d_points : num -27.7 -27.7 -27.7 -27.7 -27.7 ... ## $ pct_clinton : num 34.4 34.4 34.4 34.4 34.4 ... ## $ pct_trump : num 62.1 62.1 62.1 62.1 62.1 ... ## $ pct_johnson : num 2.09 2.09 2.09 2.09 2.09 2.09 2.09 2.09 2.09 2.09 ... ## $ pct_other : num 1.46 1.46 1.46 1.46 1.46 1.46 1.46 1.46 1.46 1.46 ... ## $ clinton_vote: num 729547 729547 729547 729547 729547 ... ## $ trump_vote : num 1318255 1318255 1318255 1318255 1318255 ... ## $ johnson_vote: num 44467 44467 44467 44467 44467 ... ## $ other_vote : num 31103 31103 31103 31103 31103 ... ## $ ev_dem : num 0 0 0 0 0 0 0 0 0 0 ... ## $ ev_rep : num 9 9 9 9 9 9 9 9 9 9 ... ## $ ev_oth : num 0 0 0 0 0 0 0 0 0 0 ... ## $ census : chr "South" "South" "South" "South" ... ``` ] --- ### Categorical `fill` Aesthetic Using a choropleth map, let's visualize the states that were won by the Democrats and Republicans. The win/loss information is contained in `party`, which is a categorical variable with two classes: "Republican" or "Democrat" (whoever won). To visualize `party` on the US map, we employ `party` as a `fill` aesthetic. .pull-left[ ```r # The fill aesthetic makes a plain map a choropleth map. p <- ggplot(data = us_states_elec, aes(x = long, y = lat, group = group, fill = party)) p + geom_polygon(color = "gray90", size = 0.1) + coord_map(projection = "albers", lat0 = 39, lat1 = 45) ``` ] .pull-right[ <img src="ChoroplethMaps_files/figure-html/election.1-out-1.png" style="display: block; margin: auto;" /> ] --- ### Tidying Up Let's tidy up the map by using the party colors and adding titles and axis labels. .panelset[ .panel[.panel-name[R Code] ```r party_colors <- c("#2E74C0", "#CB454A") # Democrats and Republicatns colors p <- ggplot(data = us_states_elec, mapping = aes(x = long, y = lat, group = group, fill = party)) p + geom_polygon(color = "gray90", size = 0.1) + coord_map(projection = "albers", lat0 = 39, lat1 = 45) + scale_fill_manual(values = party_colors) + # Change the colors of the fill aesthetic labs(title = "Election Results 2016", fill = NULL) ``` ] .panel[.panel-name[Plot] <img src="ChoroplethMaps_files/figure-html/election.2-out-1.png" style="display: block; margin: auto;" /> ] ] --- ### Numerical `fill` Aesthetic Instead of a categorical variable like `party`, we may use a numerical variable as a `fill` aesthetic. Let's show Donald Trump's winning percentage, contained in the variable `pct_trump`, on the US map. .panelset[ .panel[.panel-name[R Code] ```r p <- ggplot(data = us_states_elec, mapping = aes(x = long, y = lat, group = group, fill = pct_trump)) p + geom_polygon(color = "gray90", size = 0.1) + # color is the color of the boundaries and size is the size of the lines coord_map(projection = "albers", lat0 = 39, lat1 = 45) + labs(title = "Trump vote", fill= "Percent") ``` ] .panel[.panel-name[Plot] <img src="ChoroplethMaps_files/figure-html/election.3-out-1.png" style="display: block; margin: auto;" /> ] ] --- ### Example: Visualizing Population Density by County In this example, let's visualize population data at the county level using a choropleth map. For this example, the map data for counties are already available as `county_map` when you load the `socviz` library. There is a variable named `id`, which is the county identification number. ```r glimpse(county_map) ``` ``` ## Rows: 191,382 ## Columns: 7 ## $ long <dbl> 1225889, 1235324, 1244873, 1244129, 1272010, 1276797, 1273832, 1… ## $ lat <dbl> -1275020, -1274008, -1272331, -1267515, -1262889, -1295514, -129… ## $ order <int> 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 1… ## $ hole <lgl> FALSE, FALSE, FALSE, FALSE, FALSE, FALSE, FALSE, FALSE, FALSE, F… ## $ piece <fct> 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1… ## $ group <fct> 0500000US01001.1, 0500000US01001.1, 0500000US01001.1, 0500000US0… ## $ id <chr> "01001", "01001", "01001", "01001", "01001", "01001", "01001", "… ``` --- ### Data Let's use `county_data` from `socviz`, which contains a county identification variable (i.e. identifier) named `id`. ```r glimpse(county_data) ``` ``` ## Rows: 3,195 ## Columns: 32 ## $ id <chr> "0", "01000", "01001", "01003", "01005", "01007", "01… ## $ name <chr> NA, "1", "Autauga County", "Baldwin County", "Barbour… ## $ state <fct> NA, AL, AL, AL, AL, AL, AL, AL, AL, AL, AL, AL, AL, A… ## $ census_region <fct> NA, South, South, South, South, South, South, South, … ## $ pop_dens <fct> "[ 50, 100)", "[ 50, 100)", "[ 50, 100)", "[… ## $ pop_dens4 <fct> "[ 45, 118)", "[ 45, 118)", "[ 45, 118)", "[118,71… ## $ pop_dens6 <fct> "[ 82, 215)", "[ 82, 215)", "[ 82, 215)", "[ 82, … ## $ pct_black <fct> "[10.0,15.0)", "[25.0,50.0)", "[15.0,25.0)", "[ 5.0,1… ## $ pop <int> 318857056, 4849377, 55395, 200111, 26887, 22506, 5771… ## $ female <dbl> 50.8, 51.5, 51.5, 51.2, 46.5, 46.0, 50.6, 45.2, 53.4,… ## $ white <dbl> 77.7, 69.8, 78.1, 87.3, 50.2, 76.3, 96.0, 27.2, 54.3,… ## $ black <dbl> 13.2, 26.6, 18.4, 9.5, 47.6, 22.1, 1.8, 69.9, 43.6, 2… ## $ travel_time <dbl> 25.5, 24.2, 26.2, 25.9, 24.6, 27.6, 33.9, 26.9, 24.0,… ## $ land_area <dbl> 3531905.43, 50645.33, 594.44, 1589.78, 884.88, 622.58… ## $ hh_income <int> 53046, 43253, 53682, 50221, 32911, 36447, 44145, 3203… ## $ su_gun4 <fct> NA, NA, "[11,54]", "[11,54]", "[ 5, 8)", "[11,54]", "… ## $ su_gun6 <fct> NA, NA, "[10,12)", "[10,12)", "[ 7, 8)", "[10,12)", "… ## $ fips <dbl> 0, 1000, 1001, 1003, 1005, 1007, 1009, 1011, 1013, 10… ## $ votes_dem_2016 <int> NA, NA, 5908, 18409, 4848, 1874, 2150, 3530, 3716, 13… ## $ votes_gop_2016 <int> NA, NA, 18110, 72780, 5431, 6733, 22808, 1139, 4891, … ## $ total_votes_2016 <int> NA, NA, 24661, 94090, 10390, 8748, 25384, 4701, 8685,… ## $ per_dem_2016 <dbl> NA, NA, 0.23956855, 0.19565310, 0.46660250, 0.2142203… ## $ per_gop_2016 <dbl> NA, NA, 0.7343579, 0.7735147, 0.5227141, 0.7696616, 0… ## $ diff_2016 <int> NA, NA, 12202, 54371, 583, 4859, 20658, 2391, 1175, 1… ## $ per_dem_2012 <dbl> NA, NA, 0.2657577, 0.2156657, 0.5125229, 0.2621857, 0… ## $ per_gop_2012 <dbl> NA, NA, 0.7263374, 0.7738975, 0.4833755, 0.7306638, 0… ## $ diff_2012 <int> NA, NA, 11012, 47443, 334, 3931, 17780, 2808, 714, 14… ## $ winner <chr> NA, NA, "Trump", "Trump", "Trump", "Trump", "Trump", … ## $ partywinner16 <chr> NA, NA, "Republican", "Republican", "Republican", "Re… ## $ winner12 <chr> NA, NA, "Romney", "Romney", "Obama", "Romney", "Romne… ## $ partywinner12 <chr> NA, NA, "Republican", "Republican", "Democrat", "Repu… ## $ flipped <chr> NA, NA, "No", "No", "Yes", "No", "No", "No", "No", "N… ``` --- ### Merging Map and County Data Given that `county_map` and `county_data` have the common county identifier called `id`, we will merge these datasets by `id`. Note that the `by = "id"` option is not necessary, as `id` is contained in both datasets and there are no other variables with a common name. ```r # by = "id" not necessary since id is the only common variable county_full <- left_join(county_map, county_data, by = "id") glimpse(county_full) ``` ``` ## Rows: 191,382 ## Columns: 38 ## $ long <dbl> 1225889, 1235324, 1244873, 1244129, 1272010, 1276797,… ## $ lat <dbl> -1275020, -1274008, -1272331, -1267515, -1262889, -12… ## $ order <int> 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16… ## $ hole <lgl> FALSE, FALSE, FALSE, FALSE, FALSE, FALSE, FALSE, FALS… ## $ piece <fct> 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,… ## $ group <fct> 0500000US01001.1, 0500000US01001.1, 0500000US01001.1,… ## $ id <chr> "01001", "01001", "01001", "01001", "01001", "01001",… ## $ name <chr> "Autauga County", "Autauga County", "Autauga County",… ## $ state <fct> AL, AL, AL, AL, AL, AL, AL, AL, AL, AL, AL, AL, AL, A… ## $ census_region <fct> South, South, South, South, South, South, South, Sout… ## $ pop_dens <fct> "[ 50, 100)", "[ 50, 100)", "[ 50, 100)", "[… ## $ pop_dens4 <fct> "[ 45, 118)", "[ 45, 118)", "[ 45, 118)", "[ 45, … ## $ pop_dens6 <fct> "[ 82, 215)", "[ 82, 215)", "[ 82, 215)", "[ 82, … ## $ pct_black <fct> "[15.0,25.0)", "[15.0,25.0)", "[15.0,25.0)", "[15.0,2… ## $ pop <int> 55395, 55395, 55395, 55395, 55395, 55395, 55395, 5539… ## $ female <dbl> 51.5, 51.5, 51.5, 51.5, 51.5, 51.5, 51.5, 51.5, 51.5,… ## $ white <dbl> 78.1, 78.1, 78.1, 78.1, 78.1, 78.1, 78.1, 78.1, 78.1,… ## $ black <dbl> 18.4, 18.4, 18.4, 18.4, 18.4, 18.4, 18.4, 18.4, 18.4,… ## $ travel_time <dbl> 26.2, 26.2, 26.2, 26.2, 26.2, 26.2, 26.2, 26.2, 26.2,… ## $ land_area <dbl> 594.44, 594.44, 594.44, 594.44, 594.44, 594.44, 594.4… ## $ hh_income <int> 53682, 53682, 53682, 53682, 53682, 53682, 53682, 5368… ## $ su_gun4 <fct> "[11,54]", "[11,54]", "[11,54]", "[11,54]", "[11,54]"… ## $ su_gun6 <fct> "[10,12)", "[10,12)", "[10,12)", "[10,12)", "[10,12)"… ## $ fips <dbl> 1001, 1001, 1001, 1001, 1001, 1001, 1001, 1001, 1001,… ## $ votes_dem_2016 <int> 5908, 5908, 5908, 5908, 5908, 5908, 5908, 5908, 5908,… ## $ votes_gop_2016 <int> 18110, 18110, 18110, 18110, 18110, 18110, 18110, 1811… ## $ total_votes_2016 <int> 24661, 24661, 24661, 24661, 24661, 24661, 24661, 2466… ## $ per_dem_2016 <dbl> 0.2395685, 0.2395685, 0.2395685, 0.2395685, 0.2395685… ## $ per_gop_2016 <dbl> 0.7343579, 0.7343579, 0.7343579, 0.7343579, 0.7343579… ## $ diff_2016 <int> 12202, 12202, 12202, 12202, 12202, 12202, 12202, 1220… ## $ per_dem_2012 <dbl> 0.2657577, 0.2657577, 0.2657577, 0.2657577, 0.2657577… ## $ per_gop_2012 <dbl> 0.7263374, 0.7263374, 0.7263374, 0.7263374, 0.7263374… ## $ diff_2012 <int> 11012, 11012, 11012, 11012, 11012, 11012, 11012, 1101… ## $ winner <chr> "Trump", "Trump", "Trump", "Trump", "Trump", "Trump",… ## $ partywinner16 <chr> "Republican", "Republican", "Republican", "Republican… ## $ winner12 <chr> "Romney", "Romney", "Romney", "Romney", "Romney", "Ro… ## $ partywinner12 <chr> "Republican", "Republican", "Republican", "Republican… ## $ flipped <chr> "No", "No", "No", "No", "No", "No", "No", "No", "No",… ``` --- ### Constructing a Choropleth Map Note that `county_full$pop_dens` is a factor variable ```r class(county_full$pop_dens) ``` ``` ## [1] "factor" ``` ```r summary(county_full$pop_dens) ``` ``` ## [ 0, 10) [ 10, 50) [ 50, 100) [ 100, 500) [ 500, 1000) ## 40507 60778 35235 37857 6787 ## [ 1000, 5000) [ 5000,71672] ## 9232 986 ``` To visualize population density, we employ `pop_dens` as a `fill` aesthetic. --- ### Constructing a Choropleth Map .panelset[ .panel[.panel-name[R Code] ```r p <- ggplot(data = county_full, mapping = aes(x = long, y = lat, group = group, fill = pop_dens)) p + geom_polygon(color = "gray90", size = 0.05) + coord_equal() + # `1:1 aspect for the figure labs(fill = "Population per\nsquare mile") + theme_map() + guides(fill = guide_legend(nrow = 1)) ``` ] .panel[.panel-name[Plot] <img src="ChoroplethMaps_files/figure-html/county.1-out-1.png" style="display: block; margin: auto;" /> ] ] --- ### Constructing a Choropleth Map To tidy up, we change the scale of the `fill` aesthetic by using the `brewer` palette and changing the "fill" labels. .panelset[ .panel[.panel-name[R Code] ```r p <- ggplot(data = county_full, mapping = aes(x = long, y = lat, group = group, fill = pop_dens)) p + geom_polygon(color = "gray90", size = 0.05) + coord_equal() + # `1:1 aspect for the figure scale_fill_brewer(palette = "Blues", labels = c("0-10", "10-50", "50-100", "100-500", "500-1,000", "1,000-5,000", ">5,000")) + # Specify levels labs(fill = "Population per\nsquare mile") + theme_map() + guides(fill = guide_legend(nrow = 1)) ggsave("usdensity.png", dpi = 300) ``` ] .panel[.panel-name[Plot] <img src="ChoroplethMaps_files/figure-html/county.2-out-1.png" style="display: block; margin: auto;" /> ] ] --- class: center, middle, inverse # Simple Features --- ### Multipolygon We demonstrate how a choropleth map can be constructed from a `sf` (simple features) data frame. The `sf` data frame looks like a standard data frame. However, unlike the standard data frame, it contains a column called `geometry`, which contains an entry for each region (i.e. row) called `MULTIPOLYGON`. `MULTIPOLYGON` is a collection of the longitudes and latitudes that trace the boundary of a region. The boundary of a region may not contiguous, e.g. if there are exclaves. The `sf` data frame is easy to handle as we may just add a variable to a data frame like how we add a variable to a regular data frame. As an example, let's visualize ASEAN's GDP using a choropleth map. To obtain the national boundary lines for the ASEAN countries, we use the `rnaturalearth` package. ```r library(rnaturalearth) ``` --- ### Merging Our Data The `rnaturalearth` package provides map data as an `sf` data frame (the default is a spatial points "sp" data frame). Below, we extract an `sf` data frame from `rnaturalearth` and save it as `asean.data`. ```r # Choose a vector countries asean <- c("Brunei", "Cambodia", "Indonesia", "Malaysia", "Myanmar", "Philippines","Singapore", "Thailand", "Vietnam") # Get data from rnaturalearth asean.data <- rnaturalearth::ne_countries(scale = "medium", type = "countries", country = asean, returnclass = "sf") # simple features # "ne_countries" stands for natural earth countries ``` --- ### Merging Data with a SF Data Frame Notice that `asean.data` is a simple features data frame, as shown below: ```r class(asean.data) ``` ``` ## [1] "sf" "data.frame" ``` ```r head(asean.data[,60:63]) ``` ``` ## Simple feature collection with 6 features and 4 fields ## Geometry type: MULTIPOLYGON ## Dimension: XY ## Bounding box: xmin: 92.17959 ymin: -10.90967 xmax: 140.9762 ymax: 28.51704 ## Geodetic CRS: +proj=longlat +datum=WGS84 +no_defs +ellps=WGS84 +towgs84=0,0,0 ## long_len abbrev_len tiny homepart geometry ## 34 17 6 2 1 MULTIPOLYGON (((115.0268 4.... ## 96 9 5 NA 1 MULTIPOLYGON (((122.9489 -1... ## 115 8 5 NA 1 MULTIPOLYGON (((103.3178 10... ## 144 7 5 NA 1 MULTIPOLYGON (((98.18262 9.... ## 153 8 6 NA 1 MULTIPOLYGON (((111.3893 2.... ## 171 11 5 NA 1 MULTIPOLYGON (((120.2504 5.... ``` --- ### Merging Data with a SF Data Frame Let's add GDP per capita of the ASEAN countries, contained in `GDPPerCap.csv`, to the `asean.data`. .pull-left[ To do so, we simply assign the values of GDP per capita as a new column in `asean.data`. To be safe, it would be better to use `left_join()` as it will match both datasets using an identifier. Country names could be used as the identifier for joining the two datasets (try this as an exercise). ] .pull-right[ ```r # Import GDP per Capita Data GDP.data <- read.csv("GDPPerCap.csv",check.names=FALSE) # Arrange the countries in ascending order asean.data <- arrange(asean.data, subunit) # # Write the GDP data for 1993 and 2018 into asean.data (left_join is better) asean.data$GDPperCap1993 <- GDP.data$'1993' asean.data$GDPperCap2018 <- GDP.data$'2018' ``` ] --- ### ASEAN GDP, 1993 To visualize the GDP per capita of ASEAN countries spatially using an `sf` data frame, we employ `geom_sf()` and declare the 1993 GDP per capita as a `fill` aesthetic. .panelset[ .panel[.panel-name[R Code] ```r map1993 <- ggplot() + geom_sf(data = asean.data, aes(fill = GDPperCap1993)) + scale_fill_gradient(low = "white", high = "black", limits = c(0, 55000)) + ggtitle("GDP Per Capita, 1993") + guides(fill = guide_legend(nrow = 1)) + theme_map() + theme(legend.title = element_blank()) map1993 ``` ] .panel[.panel-name[Plot] <img src="ChoroplethMaps_files/figure-html/asean.1-out-1.png" style="display: block; margin: auto;" /> ] ] --- ### ASEAN GDP, 2018 Let's repeat the visualization for 2018 GDP per capita. .panelset[ .panel[.panel-name[R Code] ```r map2018 <- ggplot()+ geom_sf(data = asean.data, aes(fill = GDPperCap2018)) + scale_fill_gradient(low = "white", high = "black", limits = c(0, 55000)) + ggtitle("GDP Per Capita, 2018") + guides(fill = guide_legend(nrow = 1)) + theme_map() + theme(legend.title = element_blank()) map2018 ``` ] .panel[.panel-name[Plot] <img src="ChoroplethMaps_files/figure-html/asean.2-out-1.png" style="display: block; margin: auto;" /> ] ] --- ### Putting 1993 and 2008 Maps Together Let's stitch the plots together using `grid.arrange()`. ```r grid.arrange(map1993, map2018, nrow = 1) ``` <img src="ChoroplethMaps_files/figure-html/unnamed-chunk-16-1.png" style="display: block; margin: auto;" /> --- ### Exercise Let's construct a choropleth map to visualize life expectancy in Africa. First, install and load the `spData` package. We will be using the dataset `world`. Attempt the questions below. **1.** Using `filter()`, construct a (data) subset from `world` containing African countries only (i.e. filter based on the condition `continent == "Africa"`). Save this into the data frame named `africa`. What do you observe about the dataset? ``` ## [1] "sf" "tbl_df" "tbl" "data.frame" ``` --- ### Exercise **2.** Using `geom_sf()`, visualize `lifeExp` in Africa. To label the countries, pass `name_long` into `geom_sf_text()` as a label aesthetic. Use "red" as a color setting. Use `theme_map()` from the `ggthemes` package. Tidy up the plots as much as you can and describe what you observe. <img src="ChoroplethMaps_files/figure-html/unnamed-chunk-18-1.png" style="display: block; margin: auto;" /> --- ### Remark: Using `geom_text_repel()` to Label Note: `geom_sf_label()` or `geom_sf_text()` are the most straightforward way of adding text labels for `sf` objects. However, without the repel option, the map may look messy. To label the countries the "traditional" way by applying `geom_text_repel()` on the centroids, we need to first obtain the centroids from the `sf` data frame. This is achieved by passing the data frame into `st_centroid()`, and then `st_coordinates(geom)` to obtain the centroid coordinates. --- ### Remark: Using `geom_text_repel()` to Label The process of obtaining the longitude and latitude of the centroids for each country is described below. ```r africa.coord <- africa %>% st_centroid() %>% # generate the centroid, mutate(long = st_coordinates(geom)[,1], lat = st_coordinates(geom)[,2]) # get the coordinates from the centroid. This needs some inspection ``` --- ### Remark: Using `geom_text_repel()` to Label Once we have the centroids, we may use `geom_text_repel()` with the dataset `africa.coord` <img src="ChoroplethMaps_files/figure-html/unnamed-chunk-20-1.png" style="display: block; margin: auto;" />