-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathIMD_D3_M4_Data_Viz_GIS.Rmd
451 lines (369 loc) · 27.9 KB
/
IMD_D3_M4_Data_Viz_GIS.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
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
---
author: "Kate Miller & Ellen Cheng"
date: "1/11/2022"
output:
html_document:
css: custom_styles.css
---
```{r, day3code, include = F}
#------------------------
# Day 3- GIS
#------------------------
```
#### Visualizing Spatial Data
<details open><summary class = 'drop'>GIS in R</summary>
<h4>Packages used in this section:</h4>
```{r gispkges, echo = T, results = 'hide'}
library(sf) # for working with simple features
library(tmap) # for mapping spatial data
library(leaflet) # Another package for mapping spatial data
```
If folks are having trouble installing `tmap` due to an issue with one of its dependencies, `terra`, try running the following code, and then reinstall `tmap`.
```{r installterra1, echo = T, results = 'hide', eval = FALSE}
install.packages('terra', repos='https://rspatial.r-universe.dev')
```
If you've tried to do GIS and make maps in R even a few years ago, you probably encountered the same frustrations I did. There were a ton of packages out there, each with its own file format and coding convention, and each package rarely had all the features I needed. It was not easy to navigate, and I often found myself just exporting my work out of R and doing the rest in ArcGIS... Enter the `sf` and `tmap` packages, which are the latest and greatest R packages devoted to GIS and map making! Most of the frustrations I had with earlier packages have been resolved with these two packages.
The `sf` package is the workhorse for anything you need to do with spatial vector data. File types with `sf` are called <b>simple features</b>, which follow a set of GIS standards that are becoming the universal data model for vector-based spatial data in R and that most GIS-based packages in R now employ. Simple features are also now the standard for open source GIS in general. That means if you're trying to troubleshoot something with simple features, you'll need to specify that it's for R, rather than PostGIS or some other implementation. The `sf` package is also superseding the `rgdal` package, which used to be the more common data model in R and open source GIS. The more I use this package, the more impressed I am with how intuitive it is to use, and how well documented the functions are. For vector data, I have yet to need to perform a task that `sf` couldn't do.
The main application of `tmap` package is making maps, and it was designed using a grammar of graphics philosophy similar to `ggplot2`. In practice for `tmap`, it means that maps are built as a collection of many small functions/layers that get added together with pipes (%>%), and order matters. There are also tmap-enabled functions that you can use in `ggplot2` plots too, but you can do a lot more in `tmap`. I also prefer the look of tmap's built-in compass, legends, etc. over the ones available in `ggspatial`, which is an add-on package for plotting spatial data in `ggplot2`.
The `raster` package is also an excellent package for analyzing/processing raster data. For large jobs, I find the `raster` package easier to work with than ESRI tools, and it tends to run a lot faster than ESRI built-in tools (I haven't compared with python).
Finally, the `leaflet` package in R allows you to create interactive maps as html widgets. These are often included in R shiny apps, but they can also be used in R Markdown with HTML output (for more on that, attend next Wednesday's session on R Markdown). An internet connection is required for the maps to be interactive. Without an internet connection the map will show the default extent defined by the code.
The `leaflet` package is relatively easy to use for basic mapping. For more advanced features or to customize it further, you often end up having to code in JavaScript, which is the language leaflet was originally programmed in. There's also a lot more online help available for the JavaScript version of the leaflet library than the R version. If you're really stumped about something, you may find your answer with the JavaScript help.
</details>
<br>
<details open><summary class = 'drop'> Using sf and tmap </summary>
My two favorite features of sf are 1) the attribute table of a simple feature (sf's equivalent of a shapefile) is a dataframe in R, and 2) package functions are pipeable with tidyverse functions. That means, if you want to delete points in your layer, you can use dplyr's `filter()` function to filter out those points. The sf package will update the geometry of the layer to remove the points that were filtered out.
To demonstrate the use of sf and tmap, I'm going to generate a simple GRTS sample using `spsurvey`, which now connects to `sf` instead of `sp` and `rgdal`. Then we'll filter out points that don't fall in a forest habitat to demonstrate how to work with sf data in R. Finally we'll plot the results using `tmap`.
I wasn't able to figure out how to post a shapefile that you could easily download in R with a url. I emailed them to you as data.zip the previous week. I also posted all of the files to our Scientists Team, which can be downloaded using the following link: <a href="https://doimspp.sharepoint.com/:f:/s/ScientistsTraining2022/EiXOTYV1l4RCk1sMr5yXhZUB1ZFEaAlAN4elvsYbBfYHRg?e=ktVy5n">https://doimspp.sharepoint.com/:f:/s/ScientistsTraining2022/EiXOTYV1l4RCk1sMr5yXhZUB1ZFEaAlAN4elvsYbBfYHRg?e=ktVy5n</a>. To follow along, you'll need to download (and unzip if you're using the email attachment) the shapefiles and save them to a "data" folder in your working directory.
<details open><summary class = 'drop2'>Step 1. Load and prep shapefiles using sf</summary>
The code chunk below has a lot going on, but bare with me. This is the easiest way I've found to customize the layers in tmap plots. The basic steps in the code chunk below are:
<ol>
<li>Load libraries</li>
<li>Set a seed for GRTS sample. As long as you save the seed and layers used to generate the GRTS sample, you should be able to recreate the same GRTS sample.</li>
<li>Read in park boundary and vegetation map layers.</li>
<li>View quick plot of layer in base R, just to make sure it looks right.</li>
<li>Clip the veg layer to the park boundary using intersection, so map looks better.</li>
<li>Look at the attribute tables for each layer.</li>
<li>Add 2 columns to the veg layer to simplify the habitats (simp_veg) and set their colors (fill_col) </li>
</ol>
Once those steps are completed, we're ready to generate a GRTS sample and start making a map. Note that I'm using 6-digit hex colors (i.e., "#ffffff" is white) to define the fill color for each habitat type. To find your own or see what color these look like, check out <a href="https://htmlcolorcodes.com/">htmlcolorcodes.com</a>
```{r sfimports, echo = T, results = 'hide', message = F, warning = F, fig.keep = 'none'}
#install.packages(c("sf", "spsurvey"))
library(dplyr) # for filter, select, mutate and %>%
library(sf)
library(tmap)
library(spsurvey)
# Generate a random number for the seed
set.seed(62051)
sample(1:100000, 1) #62051
# Read in shapefiles from teams data folder
sara_bound1 <- st_read('./data/SARA_boundary_4269.shp')
sara_veg1 <- st_read('./data/SARA_Veg.shp')
# Check that the projections match; fix the one that doesn't match
st_crs(sara_bound1) == st_crs(sara_veg1) # FALSE- projections don't match.
# sara_bound1 needs to be reprojected to UTM Zone 18N NAD83.
sara_bound <- st_transform(sara_bound1, crs = 26918)
st_crs(sara_bound) == st_crs(sara_veg1) # TRUE
# Quick plot of first column in attribute table
plot(sara_bound[1])
plot(sara_veg1[1]) # bigger extent than boundary
# Intersect boundary and veg to be same extend
sara_veg <- st_intersection(sara_veg1, sara_bound)
plot(sara_veg[1])
# View attribute table of layers
head(sara_bound) # 1 feature with 95 fields
str(sara_veg)
head(sara_veg)
names(sara_veg)
table(sara_veg$ANDERSONII)
# Simplify vegetation types for easier plotting
dev <- c('1. Urban or Built-up Land', '11. Residential',
'12. Commercial and Services', '13. Industrial',
'14. Transportation, Communications, and Utilities',
'17. Other Urban or Built-up Land')
crop <- c('21. Cropland and Pasture',
'22. Orchards, Groves, Vineyards, and Nurseries',
'31. Herbaceous Rangeland')
shrubland <- c('32. Shrub and Brush Rangeland')
forest_up <- c('41. Deciduous Forest Land', '42. Evergreen Forest Land',
'43. Mixed Forest Land')
forest_wet <- c('61. Forested Wetland')
open_wet <- c('62. Nonforested wetland', '62. Nonforested Wetland')
water <- c('5. Water', '51. Streams and Canals', '53. Reservoirs')
unk <- 'Unclassified'
# Create 2 fields in the veg attribute table: simp_veg, and fills
sara_veg <- sara_veg %>%
mutate(simp_veg = case_when(ANDERSONII %in% dev ~ 'Developed',
ANDERSONII %in% crop ~ 'Open field',
ANDERSONII %in% shrubland ~ 'Shrublands',
ANDERSONII %in% forest_up ~ 'Forest',
ANDERSONII %in% forest_wet ~ 'Forested wetland',
ANDERSONII %in% open_wet ~ 'Open wetland',
ANDERSONII %in% water ~ 'Open water',
ANDERSONII %in% unk ~ 'Unclassified',
TRUE ~ 'Unknown'),
fill_col = case_when(simp_veg == 'Developed' ~ '#D8D8D8',
simp_veg == 'Open field' ~ '#f5f0b0',
simp_veg == 'Shrublands' ~ '#F29839',
simp_veg == 'Powerline ROW' ~ '#F9421D',
simp_veg == 'Forest' ~ '#55785e',
simp_veg == 'Forested wetland' ~ '#9577a6',
simp_veg == 'Open wetland' ~ '#c497d4',
simp_veg == 'Open water' ~ '#AFD0F2',
simp_veg == 'Unclassified' ~ '#ffffff'))
```
</details>
<br>
<details open><summary class = 'drop2'>Sidebar on simple features and shapefiles</summary>
Before moving to the next step, I wanted to show how easy it is to create simple features from dataframes that contain X/Y coordinates. We'll read in a fake dataset in the GitHub repo for this training, and call it wetdat. The dataset contains fake species composition data for wetland monitoring sites in ACAD and includes X and Y coordinates. We'll use `dplyr` functions to calculate the number of invasive and protected species in each site by year combination. Then we'll make it a simple feature, and save it as a shapefile. Note that there are multiple formats you can save simple features as. I only show the shapefile version, in case you find yourself going between R and ArcGIS.
```{r impwetd, echo = T, eval = F}
library(dplyr)
# Import data
wetdat <- read.csv(
'https://raw.githubusercontent.com/KateMMiller/IMD_R_Training_Advanced/main/data/ACAD_wetland_data_clean.csv')
# Summarize so that each site x year combination has 1 row
wetsum <- wetdat %>% group_by(Site_Name, Year, X_Coord, Y_Coord) %>%
summarize(num_inv = sum(Invasive), num_prot = sum(Protected),
.groups = 'drop') # keeps dplyr quiet in console
# Check first 6 rows of output
head(wetsum)
# Convert dataframe to sf
wetsum_sf <- st_as_sf(wetsum, coords = c('X_Coord', 'Y_Coord'), crs = 26919)
# ACAD is UTM Zone 19N NAD83, hence the difference between SARA, which is 18N.
# Write sf to disk as shapefile
st_write(wetsum_sf, 'ACAD_wetland_data.shp')
```
</details>
<br>
<details open><summary class = 'drop2'>Step 2. Generate GRTS Sample</summary>
The `spsurvey` package has been updated to point to `sf` instead of `rgdal`. It's a code-breaking change if you have old R scripts that generated GRTS samples. However, the process is even easier now, as you can see below. Here we're just going to use the simplest of GRTS designs. The output from `grts()` has multiple slots. The one we want is `sites_base`, and you can see how we get that slot in the code below.
<b>Note:</b> While I followed the approach documented in the <a href="https://cran.r-project.org/web/packages/spsurvey/vignettes/sampling.html">`spsurvey` vignette</a> to generate reproducable GRTS samples, it does not appear to be working as I'm testing it. Despite running `set.seed(62051)` after loading `spsurvey` and then running the code chunk below, each sample appears to be different.
```{r grts, echo = T, results = 'hide'}
sara_grts <- grts(sara_bound, n_base = 100) # generate 100 points within SARA boundary
sara_grts$sites_base$priority <- as.numeric(row.names(sara_grts$sites_base)) # add priority number (same as row.name)
```
</details>
<br>
<details open><summary class = 'drop2'>Step 3. Spatial join and filter</summary>
First step here is to spatially join the sara_grts layer to the sara_veg layer. Here we only cared about the sara_veg's simp_veg field, so I used dplyr's `select()` function. After joining, you should see a simp_veg field added to the end of the grts layer (it's actually to the left of geometry, which is the last). In the next step, we then filter the points in the newly created grts_veg layer to only include points that fall in forest habitat.
```{r grtsfilt, echo = T, results = 'hide'}
# Spatial join
grts_veg <- st_join(sara_grts$sites_base, sara_veg %>% select(simp_veg))
names(grts_veg)
# Create list of forest habitats
sort(unique(sara_veg$simp_veg))
forest_veg <- c('Forest', 'Forested wetland')
# Filter out non-forest points
grts_forest <- grts_veg %>% filter(simp_veg %in% forest_veg)
nrow(grts_veg) # 100 points
nrow(grts_forest) # fewer points
```
<div class="alert alert-info">
<h4>Challenge: Filter GRTS by priority</h4>
How would you filter out GRTS points with a priority higher than 50?
<br>
<details><summary class = 'drop2'>Answer</summary>
```{r d3gis_a1, echo = T}
grts_50 <- grts_veg %>% filter(priority <= 50)
nrow(grts_50)
```
</details>
</div>
</details>
<br>
<details open><summary class='drop2'>Step 4. Create map</summary>
Now it's time to plot. The code below may seem a bit much, but we'll break it down piece by piece. The great thing about building maps this way is that you're essentially building a template in code that you can steal from and adapt for future maps. You can even turn these into a function that's even easier to use in future maps. That's beyond what we can cover here, but it's another great benefit of making maps in R instead of ArcGIS.
The code below is broken into the various pieces that make up the map. The way `tmap` works is that first, you have to add the layer via `tm_shape()`, and then you specify how you want that layer to look, `like tm_fill()`, or `tm_border()`. Each piece has its own legend as well. This is similar to how you start each `ggplot2` graph defining the data with `ggplot(data, aes(x = xvar, y = yvar))`, and then start adding `geom_point()`, or whatever else to define how the data are plotted. The difference with `tmap` is that every layer you want in the map has to be coded this way. Finally `tm_layout()` is similar to `theme()` in ggplot2, and is where you can customize the map layout.
A couple of notes about the code below:
<ul>
<li>The first line that defines `for_legend` makes a list of the habitat types in simp_veg and their associated colors. That saves time having to type them all over again, and potentially spelling them wrong.</li>
<li>The first `tm_shape()` in the map sets the projection and the bounding box. If you don't set the bounding box, the map will show the largest extent of your layers. So if you have a roads layer at the state-level, your map will be zoomed at that extent, instead of the park boundary.</li>
<li>Note the difference between `tm_fill()`, which fills in colors, and `tm_borders()`, which only plots outlines. If you want both borders and fills, use `tm_polygons()` instead.</li>
<li>The z value in the `tm_add_legend()` allows you to set the order that each legend appears in the map. Otherwise, they'll appear in the order you specify the layers.</li>
<li>The code for `tm_symbols()` allows you to change the symbol format in a similar way to `ggplot2`. We then added `tm_text()` to label each point using the numbers in the priority column of the grts_forest attribute table. The `xmod` and `ymod` allow you to offset labels from the points either horizontally and vertically. In this case, negative `xmod` moves the label to the left, and a negative `ymod` moves the label below the point. The default location for labels is directly on top of the point.</li>
<li>The `tm_layout()` code is where you can change the default settings of the figure, including font size, placement of the legend, and margins. The title name and position are also set in the `tm_layout()`.</li>
<li>The `tmap_save()` allows you to write the map to disk and to specify the height and width of the map. I prefer to write maps to disk to see what the size looks like before changing any of the layout and font size settings, because the figure on your disk will look different (and often better) than the plot view in your R Studio session.</li>
</ul>
```{r tmap, echo = T, results = 'hide', warning = F, message = F, fig.keep = 'none'}
# Creating list of simp_veg types and their fill colors for easier legend
for_legend <- unique(data.frame(simp_veg = sara_veg$simp_veg, fill_col = sara_veg$fill_col))
sara_map <-
# Vegetation map
tm_shape(sara_veg, projection = 26918, bbox = sara_bound) +
tm_fill('fill_col') +
tm_add_legend(type = 'fill', labels = for_legend$simp_veg,
col = for_legend$fill_col, z = 3) +
# Park boundary
tm_shape(sara_bound) +
tm_borders('black', lwd = 2) +
tm_add_legend(type = 'line', labels = 'Park Boundary', col = 'black',
z = 2)+
# GRTS points
tm_shape(grts_forest) +
tm_symbols(shapes = 21, col = '#EAFF16', border.lwd = 0.5, size = 0.3) +
tm_text('priority', size = 0.9, xmod = -0.4, ymod = -0.4) +
tm_add_legend(type = 'symbol', labels = 'GRTS points', shape = 21,
col = '#EAFF16', border.lwd = 1, size = 0.5,
z = 1) +
# Other map features
tm_compass(size = 2, type = 'arrow',
text.size = 1, position = c('left', 'bottom')) +
tm_scale_bar(text.size = 1.25, position = c('center', 'bottom')) +
tm_layout(inner.margins = c(0.2, 0.02, 0.02, 0.02), # make room for legend
outer.margins = 0,
legend.text.size = 1.25,
legend.just = 'right',
legend.position = c('right', 'bottom'),
title = 'Saratoga NHP GRTS points',
title.position = c('center', 'top'))
tmap_save(sara_map, 'SARA_GRTS.png', height = 10.5, width = 8,
units = 'in', dpi = 600, outer.margins = 0)
```
<div class="alert alert-info">
<h4>Challenge: Change the map</h4>
1. How would you change the vegmap layers to have an outline, instead of just fill?
2. How would you make the park boundary red instead of black?
<br>
<details><summary class = 'drop2'>Answer</summary>
```{r d3gis_a2, echo = T}
sara_map2 <-
# Vegetation map
tm_shape(sara_veg, projection = 26918, bbox = sara_bound) +
tm_polygons('fill_col') + # A1: changed tm_fill() to tm_polygon()
tm_add_legend(type = 'fill', labels = for_legend$simp_veg,
col = for_legend$fill_col, z = 3) +
# Park boundary
tm_shape(sara_bound) +
tm_borders('red', lwd = 2) + # A2A: changed 'black' to 'red'
tm_add_legend(type = 'line', labels = 'Park Boundary', col = 'red', # A2B
z = 2)+
# GRTS points
tm_shape(grts_forest) +
tm_symbols(shapes = 21, col = '#EAFF16', border.lwd = 0.5, size = 0.3) +
tm_text('priority', size = 0.9, xmod = -0.5, ymod = -0.5) +
tm_add_legend(type = 'symbol', labels = 'GRTS points', shape = 21,
col = '#EAFF16', border.lwd = 1, size = 0.5,
z = 1) +
# Other map features
tm_compass(size = 2, type = 'arrow',
text.size = 1, position = c('left', 'bottom')) +
tm_scale_bar(text.size = 1.25, position = c('center', 'bottom')) +
tm_layout(inner.margins = c(0.2, 0.02, 0.02, 0.02),
outer.margins = 0,
legend.text.size = 1.25,
legend.just = 'right',
legend.position = c('right', 'bottom'),
title = 'Saratoga NHP GRTS points',
title.position = c('center', 'top'))
```
</details>
</div>
</details>
<br>
<details open><summary class = 'drop2'>Map dimensions</summary>
The dimensions and scaling in the plot viewer of your R session tends to be a bit funky. Instead of trying to make the map perfect in the plot viewer, I usually save it to disk and tweak that version to look the way I want it to. To demonstrate, here's what sara_map looks like in your plot viewer:
```{r, plotvw, echo = T, out.width = '60%', warning = F, message = F}
sara_map
```
<br>
<br>
<br>
Here's what the map looks like after it's written to disk and the dimensions are set using `tmap_save()`:
```{r, discvw, echo = F, results = 'show', out.width = '60%', warning = F, message = F, out.extra = 'style="padding:0px 0px; margin:-4px 0;"'}
knitr::include_graphics('SARA_GRTS.png')
```
</details>
<br>
<details open><summary class = 'drop2'>Interactive mode</summary>
The last cool thing to show with tmap, is that you can include interactive HTML widgets similar to what leaflet can do (coming next). With the interactive mode, you can change baselayers, turn your layers off/on, and zoom to different extent. The legend in the interactive mode is a bit limited to only show fills, but it's still pretty cool. You can also add custom basemaps (e.g. parktiles) to the interactive mode either by adding `tm_basemap(url)`, or by changing the default basemap via `tmap_options()`, which I show below.
```{r tmview, echo = T, warning = F, message = F}
# Load park tiles
NPSbasic = 'https://atlas-stg.geoplatform.gov/styles/v1/atlas-user/ck58pyquo009v01p99xebegr9/tiles/256/{z}/{x}/{y}@2x?access_token=pk.eyJ1IjoiYXRsYXMtdXNlciIsImEiOiJjazFmdGx2bjQwMDAwMG5wZmYwbmJwbmE2In0.lWXK2UexpXuyVitesLdwUg'
NPSimagery = 'https://atlas-stg.geoplatform.gov/styles/v1/atlas-user/ck72fwp2642dv07o7tbqinvz4/tiles/256/{z}/{x}/{y}@2x?access_token=pk.eyJ1IjoiYXRsYXMtdXNlciIsImEiOiJjazFmdGx2bjQwMDAwMG5wZmYwbmJwbmE2In0.lWXK2UexpXuyVitesLdwUg'
NPSslate = 'https://atlas-stg.geoplatform.gov/styles/v1/atlas-user/ck5cpvc2e0avf01p9zaw4co8o/tiles/256/{z}/{x}/{y}@2x?access_token=pk.eyJ1IjoiYXRsYXMtdXNlciIsImEiOiJjazFmdGx2bjQwMDAwMG5wZmYwbmJwbmE2In0.lWXK2UexpXuyVitesLdwUg'
NPSlight = 'https://atlas-stg.geoplatform.gov/styles/v1/atlas-user/ck5cpia2u0auf01p9vbugvcpv/tiles/256/{z}/{x}/{y}@2x?access_token=pk.eyJ1IjoiYXRsYXMtdXNlciIsImEiOiJjazFmdGx2bjQwMDAwMG5wZmYwbmJwbmE2In0.lWXK2UexpXuyVitesLdwUg'
# Set parktiles as default basemaps for all interactive maps
tmap_options(basemaps = c(Map = NPSbasic,
Imagery = NPSimagery,
Light = NPSlight,
Slate = NPSslate))
# Switch to interactive mode, and plot the sara_map
tmap_mode('view') # turn interactive mode on
sara_map
tmap_mode('plot') # return to default plot view
```
</details>
</details>
</details>
<br>
<details open><summary class = 'drop'> Using leaflet and parktiles </summary>
Now I’ll show you how to create a simple leaflet map including NPS parktiles as a basemap. By default, layers and basemaps that feed into leaflet need to be specified in latitude and longitude using WGS84 projection (EPSG:4326). This projection may introduce some inaccuracies if projecting UTM to WGS84, such as plots that are at the edge of park boundaries appearing outside of the boundary. It has to do with the geographic transformation used to go from UTM to WGS. Recent improvements, including the Proj4Leaflet plugin, allow you to change coordinate systems, which may improve this. For more on that see the Projections section of R Studio’s leaflet page. For the purposes here, we’ll use the default WGS84, and reproject our data to match the projection.
We’re going to use some of the SARA layers we used with tmap, but we have to reproject the layers to WGS84. First we’ll define the four different park tiles that are available. These are the same tiles you see if you view the map on a given park’s website.
```{r leafprep, echo = T, results = 'hide', warning = F, message = F}
# Load library
library(leaflet)
# Load park tiles
NPSbasic = 'https://atlas-stg.geoplatform.gov/styles/v1/atlas-user/ck58pyquo009v01p99xebegr9/tiles/256/{z}/{x}/{y}@2x?access_token=pk.eyJ1IjoiYXRsYXMtdXNlciIsImEiOiJjazFmdGx2bjQwMDAwMG5wZmYwbmJwbmE2In0.lWXK2UexpXuyVitesLdwUg'
NPSimagery = 'https://atlas-stg.geoplatform.gov/styles/v1/atlas-user/ck72fwp2642dv07o7tbqinvz4/tiles/256/{z}/{x}/{y}@2x?access_token=pk.eyJ1IjoiYXRsYXMtdXNlciIsImEiOiJjazFmdGx2bjQwMDAwMG5wZmYwbmJwbmE2In0.lWXK2UexpXuyVitesLdwUg'
NPSslate = 'https://atlas-stg.geoplatform.gov/styles/v1/atlas-user/ck5cpvc2e0avf01p9zaw4co8o/tiles/256/{z}/{x}/{y}@2x?access_token=pk.eyJ1IjoiYXRsYXMtdXNlciIsImEiOiJjazFmdGx2bjQwMDAwMG5wZmYwbmJwbmE2In0.lWXK2UexpXuyVitesLdwUg'
NPSlight = 'https://atlas-stg.geoplatform.gov/styles/v1/atlas-user/ck5cpia2u0auf01p9vbugvcpv/tiles/256/{z}/{x}/{y}@2x?access_token=pk.eyJ1IjoiYXRsYXMtdXNlciIsImEiOiJjazFmdGx2bjQwMDAwMG5wZmYwbmJwbmE2In0.lWXK2UexpXuyVitesLdwUg'
# Reproject SARA layers
sara_bnd_wgs <- st_transform(sara_bound, crs = 4326)
grts_forest_wgs <- st_transform(grts_forest, crs = 4326)
# Calculate centroid for leaflet map view
sara_cent <- st_centroid(sara_bnd_wgs)$geometry
str(sara_cent) # -73.63459, 42.99543
```
<br>
Now we add the pieces together. The leaflet package operates similar to tmap and ggplot2, in that you build the map by adding consecutive layers and connecting the code with pipes.
A couple of notes on the code below:
<ul>
<li>The setView() allows you to set the center (in WGS84) and the zoom of the map. A zoom of 0 is the entire world. The higher the zoom, the closer the zoom. You can also set the min/max zoom a user can view in a map, along with a maximum bounding box that a user can pan or zoom within.</li>
<li>addTiles() is how you add individual tiles. The group = "text" names the tiles, which are then shown in the map’s layer control via addLayersControl().</li>
<li>addCircleMarkers() is how the GRTS points are added to the map. I also had to specify the lat/lng coordinates for each point, and I used sf’s st_coordinates() to extract them.</li>
<li>The radius of the point is how you specify the size of the markers.</li>
<li>The stroke refers to the outline. I turned it off in this case.</li>
<li>I also added a scalebar and set the width and units of it using addScaleBar(), and scaleBarOptions().</li>
<li>There are all kinds of bells and whistles you can add to a leaflet plot, including popups on hover or click, custom colors and symbols based on the data, etc. For more examples, check out Leaflet for R.</li>
</ul>
```{r leafmap, echo = T, results = 'hide', warning = F, message = F}
sara_map_lf <-
leaflet() %>%
setView(lng = -73.63, lat = 42.99, zoom = 13) %>%
# parktiles
addTiles(group = 'Map',
urlTemplate = NPSbasic) %>%
addTiles(group = 'Imagery',
urlTemplate = NPSimagery) %>%
addTiles(group = 'Light',
urlTemplate = NPSlight) %>%
addTiles(group = 'Slate',
urlTemplate = NPSslate) %>%
addLayersControl(map = .,
baseGroups = c('Map', 'Imagery', 'Light', 'Slate'),
options = layersControlOptions(collapsed = T)) %>%
# points
addCircleMarkers(data = grts_forest_wgs,
lng = st_coordinates(grts_forest_wgs)[,1],
lat = st_coordinates(grts_forest_wgs)[,2],
label = grts_forest_wgs$priority,
labelOptions = labelOptions(noHide=T, textOnly = TRUE,
direction = 'bottom', textsize = '12px'),
fillColor = '#EAFF16',
radius = 4,
stroke = FALSE, # turn off outline
fillOpacity = 1) %>%
# scale bar and settings
addScaleBar(position = 'bottomright') %>%
scaleBarOptions(maxWidth = 10, metric = TRUE)
```
```{r leafmap2, echo = T, message = F, warnings = F, eval = F}
sara_map_lf
```
Note that I can’t figure out why I’m not able to make this map render on the website, but I’ll demonstrate it in R Studio.
</details>
<br>
<hr>
#### Resources
```{r data_viz_res, child = "IMD_Resources_D3_Data_Viz.Rmd", eval = T}
```
<hr>