-
Notifications
You must be signed in to change notification settings - Fork 24
/
Copy path03-Introduction-to-Mapping.Rmd
270 lines (216 loc) · 14.7 KB
/
03-Introduction-to-Mapping.Rmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
# Part II: Statistics and Maps {-}
```{r coverlet-part-ii, echo=FALSE}
knitr::include_graphics(path = "images/part-02.jpg")
```
# Introduction to Mapping in `R` {#introduction-to-maps}
*NOTE*: The source files for this book are available with companion package [{isdas}](https://paezha.github.io/isdas/). The source files are in Rmarkdown format and packed as templates. These files allow you execute code within the notebook, so that you can work interactively with the notes.
Spatial statistics is a sub-field of spatial analysis that has grown in relevance in recent years as a result of 1) the availability of information that is geo-coded, in other words, that has geographical references; and 2) the availability of software to analyze such information.
A key technology fueling this trend is that of Geographical Information Systems (GIS). GIS are, in simplest terms, digital mapping for the 21st century. In most cases, however, GIS go beyond cartographic functions to also enable and enhance our ability to analyze data.
There are many available packages for geographical information analysis. Some are very user friendly, and widely available in many institutional contexts, including open source alternatives such as [QGIS](https://qgis.org/en/site/), or commercial software such as ESRI's Arc software. Others are fairly specialized, such as Caliper's TransCAD, which implements many operations of interest for transportation engineering and planning.
Others packages have the advantage of being more flexible and/or free.
Such is the case of the `R` statistical computing language. `R` has been adopted by many in the spatial analysis community, and a number of specialized libraries have been developed to support mapping and spatial data analysis functions.
The objective of this note is to provide an introduction to mapping in `R`. Maps are one of the fundamental tools of spatial statistics and spatial analysis, and `R` allows for many GIS-like functions.
In the previous reading/practice you created a simple proportional symbols map. In this reading/practice you will learn how to create more sophisticated maps in `R`.
## Learning Objectives
In this reading, you will:
1. Revisit how to install and load a package.
2. Learn how to invoke a data and view the data structure.
3. Learn how to easily create maps using `R`.
4. Think about how statistical maps help us understand patterns.
## Suggested Readings
- Bivand RS, Pebesma E, Gomez-Rubio V [-@Bivand2008] Applied Spatial Data Analysis with R, Chapters 2-3. Springer: New York
- Brunsdon C and Comber L [-@Brunsdon2015R] An Introduction to R for Spatial Analysis and Mapping, Chapter 3. Sage: Los Angeles
## Preliminaries
it is good practice to begin with a clean session to make sure that you do not have extraneous items there when you begin your work. The best practice is to restart the `R` session, which can be accomplished for example with `command/ctrl + shift + F10`. An alternative to _only_ purge user-created objects from memory is to use the `R` command `rm` (for "remove"), followed by a list of items to be removed:
```{r ch03-clear-workspace}
# The function `ls()` lists all objects in the Environment, that is, your current workspace;
# `rm()` removes all objects listed in the argument `list = `
rm(list = ls())
```
## Packages
According to Wickham [-@wickham2015rpackages] packages are the basic units of reproducible code in the `R` multiverse.
Now that your workspace is clear, you can proceed to load a package. In this case, the package is the one used for this book/course, called `isdas`:
```{r ch03-load-packages, message=FALSE}
#The function 'library' is used to load the data we want to work with.
# In this case, it is the isdas master package that we want to work with
library(isdas) # Companion Package for Book An Introduction to Spatial Data Analysis and Statistics
```
The package includes a few datasets that will be used throughout the book:
```{r ch03-load-data}
#The function `data` is used to check if a dataset is present within any loaded packages.
# In this case, we are looking for `snow_deaths` and `snow_pumps`
data("snow_deaths")
data("snow_pumps")
```
## Exploring Dataframes and a Simple Proportional Symbols Map
If you correctly loaded the library, you can now access the dataframes in the package `isdas`. For this section, you will need two dataframes, namely `snow_pumps` and `snow_deaths`:
```{r ch03-head-snow-deaths}
#The function `head` will display the first few rows of the dataframe, snow_deaths
head(snow_deaths)
```
These data are from the famous London cholera example by John Snow (not the one from Game of Thrones, but the British physician). John Snow is considered the father of spatial epidemiology, and his study mapping the outbreak is credited with helping find its cause.This study investigates the cholera outbreak of Soho, London, in 1854.
The dataframe `snow_deaths` includes the geocoded addresses of cholera deaths in `long` and `lat`, and the number of deaths (the `Count`) recorded at each address, as well as unique identifiers for the addresses (`Id`).
A second dataframe `snow_pumps` includes the geocoded locations of water pumps in Soho:
```{r ch03-head-snow-pumps}
head(snow_pumps)
```
As in your previous reading, it is possible to map the cases using `ggplot2`. Begin by loading the package `tidyverse`:
```{r ch03-load-tidyverse, message=FALSE}
#'Tidyverse' is a collection of R packages designed for data science used in everyday data analyses
library(tidyverse) # Easily Install and Load the 'Tidyverse'
```
Now, you can create a blank `ggplot2` object from which you can render the points for deaths and the pumps.
```{r ch03-plot-snow-deaths}
#The function `ggplot` is used for data visualization - it creates a graph.
# The function `geom_point` tells R you want to create a plot of points.
# `data = snow_deaths` tells R you want to use the `snow_deaths` dataframe.
# `aes` stands for aesthetics of your graph where `x = long` sets the x axis
# to `long`, where `y = lat` sets the y axis to `lat`, where `color = blue`
# colours the points blue and `shape = 16` assigns the shape of the points -
# in this case, `16` are circles and `17` are triangles
ggplot() +
geom_point(data = snow_deaths, aes(x = long, y = lat), color = "blue", shape = 16) +
geom_point(data = snow_pumps, aes(x = long, y = lat), color = "black", shape = 17)
```
This map is a decent example of how to represent visually some contents in the dataframe. Here, information is displayed using different colours and symbols to represent pumps and deaths from the London Cholera Example. Though this map provides useful insights, it is not of the greatest quality. We will illustrate other ways of creating maps below, including interactive maps.
## Improving on the Proportional Symbols Map
A package that extends the functionality of mapping in `R` is `leaflet`. A key feature of the `leaflet` package is the ability to make maps interactive for the user. We will see next how to enhance our proportional symbol map using this package. First you need to load the package (you need to install it first if you have not already):
```{r ch03-load-more-packages, message=FALSE}
# 'Leaflet' is a package used for visualizing data on a map in R.
if (!require("leaflet")) install.packages('leaflet')
library(leaflet) # Create Interactive Web Maps with the JavaScript 'Leaflet'
```
The first step is to create a `leaflet` object, which will be saved in `m` :
```{r ch03-leaflet-map}
# Here, we create a `leaflet` object and assign it to the variable, 'm'.
# The `setView` function sets the view of the map where `lng = -0.136`
# sets the longitutde, 'lat = 51.513' sets the latitude and the map zoom
# is set to 16. The `|>` is a pipe operator that passes the output from
# the left hand side of the operator to the first argument of the right
# hand side of the operator. In this case we are telling `R` that we want
# to center the map on the set longitude and latitude, with a zoom level
# of 16, which corresponds roughly to a neighborhood
m <- leaflet(data = snow_deaths) |>
setView(lng = -0.136, lat = 51.513, zoom = 16)
```
This map looks like this at this point:
```{r ch03-empty-leaflet-map}
m
```
The map is empty! This is because we have not yet added any geographical information to plot. We can begin by adding a basemap as follows:
```{r ch03-leaflet-add-basemap}
# We are adding a basemap or background map of the study area
# by means of the `addTiles` function to the 'm' variable
m <- m |> addTiles()
m
```
The map now shows the neighborhood in Soho where the cholera outbreak happened. Now, at long last, we can add the cases of cholera deaths to the map. For this, we indicate the coordinates (preceded by `~`), and set an option for clustering by means of the `clusterOptions` in the following fashion:
```{r ch03-leaflet-add-deaths}
# We are adding the cholera deaths to the map using 'group = Deaths'.
# The '~' symbol tells R to use the same longitude and latitude values
# used in the previous block of code and the 'clusterOptions = markerClusterOptions()'
# clusters a large number of markers on the map - in this case it is clusturing
# number of deaths into icons with numbers
m <- m |>
addMarkers(~long,
~lat,
clusterOptions = markerClusterOptions(),
group = "Deaths")
m
```
The map now displays the locations of cholera deaths on the map. If you zoom in, the clusters will rearrange accordingly. Try it! The other information that we have available is the location of the water pumps, which we can add to the map above (notice that the Broad Street Pump is already shown in the basemap!):
```{r ch03-leaflet-add-pumps}
m |>
addMarkers(data = snow_pumps,
~long,
~lat,
group = "Pumps")
```
An alternative and quicker way to run the same bit of code is by means of pipe operators (`|>`). These operators make writing code a lot faster, easier to read, and more intuitive! Recall that a pipe operator will take the output of the preceding function, and pass it on as the first argument of the next:
```{r ch03-leaflet-full-example}
m_test <- leaflet () |>
setView(lng = -0.136, lat = 51.513, zoom = 16) |>
addTiles() |>
addMarkers(data = snow_deaths,
~long,
~lat,
clusterOptions = markerClusterOptions(),
group = "Deaths") |>
addMarkers(data = snow_pumps,
~long,
~lat,
group = "Pumps")
m_test
```
The above results in a much nicer map. Is this map informative? What does it tell you about the incidence of cholera and the location of the pumps?
## Some Simple Spatial Analysis
Despite the simplicity of this map, we can begin to do some spatial analysis. For instance, we could create a _heatmap_. You have probably seen heatmaps in many different situations before, as they are a popular visualization tool. Heatmaps are created based on a spatial analytical technique called _kernel analysis_. We will cover this technique in more detail later on. For the time being, it can be illustrated by taking advantage of the `leaflet.extras` package, which contains a heatmap function. Load the package as follows:
```{r ch03-load-leaflet-extras, message=FALSE}
if (!require("leaflet.extras")) install.packages('leaflet.extras')
library(leaflet.extras) # Extra Functionality for 'leaflet' Package
```
Next, create a second leaflet object for this example, and call it `m2`. Notice that we are using the same `setView` parameters:
```{r ch03-second-leaflet-example}
m2 <- leaflet(data = snow_deaths) |>
setView(lng = -0.136,
lat = 51.513,
zoom = 16) |>
addTiles()
```
Then, add the heatmap. The function used to do this is `addHeatmap`. We specify the coordinates and the variable for the intensity (i.e., each case in the dataframe is representative of `Count` deaths at the address). Two parameters are important here, the `blur` and the `radius`. If you are working with the R notebook version of the book, experiment changing these parameters:
```{r ch03-leaflet-heatmap}
# The 'addHeatmap' function is making a heat map. We specify the coordinates,
# same as the block of code above. The 'intensity' function sets a numeric value,
# the 'blur' specifies the amount of blur to apply and the 'radius' function sets
# the radius of each point on the heatmap
m2 |>
addHeatmap(lng = ~long,
lat = ~lat,
intensity = ~Count,
blur = 40,
max = 1,
radius = 25)
```
Lastly, you can also add markers for the pumps as follows:
```{r ch03-leaflet-heatmap-with-pumps}
m2 |> addHeatmap(lng = ~long,
lat = ~lat,
intensity = ~Count,
blur = 40,
max = 1,
radius = 25) |>
addMarkers(data = snow_pumps,
~long,
~lat,
group = "Pumps")
```
And everything together:
```{r ch03-leaflet-put-everything-together}
m2_test <- leaflet(data = snow_deaths) |>
setView(lng = -0.136,
lat = 51.513,
zoom = 16) |>
addTiles() |>
addHeatmap(lng = ~long,
lat = ~lat,
intensity = ~Count,
blur = 40,
max = 1,
radius = 25) |>
addMarkers(data = snow_deaths,
~long,
~lat,
clusterOptions = markerClusterOptions(),
group = "Deaths") |>
addMarkers(data = snow_pumps,
~long,
~lat,
group = "Pumps")
m2_test
```
A heatmap (essentially a kernel density of spatial points; more on this in a later chapter) makes it very clear that most cases of cholera happend in the neighborhood of one (possibly contaminated) water pump! At the time, Snow noted with respect to this geographical pattern that:
> "It will be observed that the deaths either very much diminished, or ceased altogether, at every point where it becomes decidedly nearer to send to another pump than to the one in Broad street. It may also be noticed that the deaths are most numerous near to the pump where the water could be more readily obtained."
Snow's analysis helped to convince officials to close the pump, after which the cholera outbreak subsided. This illustrates how even some relatively simple spatial analysis can help to inform public policy and even save lives. You can read more about this case [here](https://en.wikipedia.org/wiki/1854_Broad_Street_cholera_outbreak).
In this practice you have learned how to implement some simple mapping and spatial statistical analysis using `R`. In future readings we will further explore the potential of `R` for both.
## Other Resources
For more information on the functionality of `leaflet`, please check [Leaflet for R](https://rstudio.github.io/leaflet/)