Chapter1_Setting_Up_Interactive_Web_Maps

Chapter 1 will introduce students to the htmlwidgets package and the leaflet package. Following this introduction, students will build their first interactive web map using leaflet. Through the process of creating this first map students will be introduced to many of the core features of the leaflet package, including adding different map tiles, setting the center point and zoom level, plotting single points based on latitude and longitude coordinates, and storing leaflet maps as objects. Chapter 1 will conclude with students geocoding DataCamp’s headquarters, and creating a leaflet map that plots the headquarters and displays a popup describing the location.

1. Creating an Interactive Web Map

Similar to the packages in the tidyverse, the leaflet package makes use of the pipe operator (i.e., %>%) from the magrittr package to chain function calls together. This means we can pipe the result of one function into another without having to store the intermediate output in an object. For example, one way to find every car in the mtcars data set with a mpg >= 25 is to pipe the data through a series of functions.

{r}
mtcars  %>% 
    mutate(car = rownames(.))  %>% 
    select(car, mpg)  %>% 
    filter(mpg >= 25)

To create a web map in R, you will chain together a series of function calls using the %>% operator. Our first function leaflet() will initialize the htmlwidget then we will add a map tile using the addTiles() function.

In [1]:
# Load the leaflet library
library(leaflet)
{r}
# Create a leaflet map with default map tile using addTiles()
library(htmlwidgets)
leaflet() %>% addTiles()

2. Provider Tiles

In the previous exercise, addTiles() added the default OpenStreetMap (OSM) tile to your leaflet map. Map tiles weave multiple map images together. The map tiles presented adjust when a user zooms or pans the map enabling the interactive features you experimented with in exercise 2.

The leaflet package comes with more than 100 map tiles that you can use. These tiles are stored in a list called providers and can be added to your map using addProviderTiles() instead of addTiles().

The leaflet and tidyverse libraries have been loaded for you.

In [6]:
pkgs <- c("tidyverse", "leaflet")
sapply(pkgs, require, character.only = TRUE)
Loading required package: tidyverse
── Attaching packages ─────────────────────────────────────── tidyverse 1.2.1 ──
✔ ggplot2 3.1.0     ✔ purrr   0.2.5
✔ tibble  1.4.2     ✔ dplyr   0.7.8
✔ tidyr   0.8.2     ✔ stringr 1.3.1
✔ readr   1.3.1     ✔ forcats 0.3.0
── Conflicts ────────────────────────────────────────── tidyverse_conflicts() ──
✖ dplyr::filter() masks stats::filter()
✖ dplyr::lag()    masks stats::lag()
tidyverse
TRUE
leaflet
TRUE
In [9]:
# Print the providers list included in the leaflet library
providers[1:5]
$OpenStreetMap
'OpenStreetMap'
$OpenStreetMap.Mapnik
'OpenStreetMap.Mapnik'
$OpenStreetMap.BlackAndWhite
'OpenStreetMap.BlackAndWhite'
$OpenStreetMap.DE
'OpenStreetMap.DE'
$OpenStreetMap.CH
'OpenStreetMap.CH'
In [10]:
# Print only the names of the map tiles in the providers list 
names(providers)
  1. 'OpenStreetMap'
  2. 'OpenStreetMap.Mapnik'
  3. 'OpenStreetMap.BlackAndWhite'
  4. 'OpenStreetMap.DE'
  5. 'OpenStreetMap.CH'
  6. 'OpenStreetMap.France'
  7. 'OpenStreetMap.HOT'
  8. 'OpenStreetMap.BZH'
  9. 'OpenInfraMap'
  10. 'OpenInfraMap.Power'
  11. 'OpenInfraMap.Telecom'
  12. 'OpenInfraMap.Petroleum'
  13. 'OpenInfraMap.Water'
  14. 'OpenSeaMap'
  15. 'OpenPtMap'
  16. 'OpenTopoMap'
  17. 'OpenRailwayMap'
  18. 'OpenFireMap'
  19. 'SafeCast'
  20. 'Thunderforest'
  21. 'Thunderforest.OpenCycleMap'
  22. 'Thunderforest.Transport'
  23. 'Thunderforest.TransportDark'
  24. 'Thunderforest.SpinalMap'
  25. 'Thunderforest.Landscape'
  26. 'Thunderforest.Outdoors'
  27. 'Thunderforest.Pioneer'
  28. 'OpenMapSurfer'
  29. 'OpenMapSurfer.Roads'
  30. 'OpenMapSurfer.AdminBounds'
  31. 'OpenMapSurfer.Grayscale'
  32. 'Hydda'
  33. 'Hydda.Full'
  34. 'Hydda.Base'
  35. 'Hydda.RoadsAndLabels'
  36. 'MapBox'
  37. 'Stamen'
  38. 'Stamen.Toner'
  39. 'Stamen.TonerBackground'
  40. 'Stamen.TonerHybrid'
  41. 'Stamen.TonerLines'
  42. 'Stamen.TonerLabels'
  43. 'Stamen.TonerLite'
  44. 'Stamen.Watercolor'
  45. 'Stamen.Terrain'
  46. 'Stamen.TerrainBackground'
  47. 'Stamen.TopOSMRelief'
  48. 'Stamen.TopOSMFeatures'
  49. 'Esri'
  50. 'Esri.WorldStreetMap'
  51. 'Esri.DeLorme'
  52. 'Esri.WorldTopoMap'
  53. 'Esri.WorldImagery'
  54. 'Esri.WorldTerrain'
  55. 'Esri.WorldShadedRelief'
  56. 'Esri.WorldPhysical'
  57. 'Esri.OceanBasemap'
  58. 'Esri.NatGeoWorldMap'
  59. 'Esri.WorldGrayCanvas'
  60. 'OpenWeatherMap'
  61. 'OpenWeatherMap.Clouds'
  62. 'OpenWeatherMap.CloudsClassic'
  63. 'OpenWeatherMap.Precipitation'
  64. 'OpenWeatherMap.PrecipitationClassic'
  65. 'OpenWeatherMap.Rain'
  66. 'OpenWeatherMap.RainClassic'
  67. 'OpenWeatherMap.Pressure'
  68. 'OpenWeatherMap.PressureContour'
  69. 'OpenWeatherMap.Wind'
  70. 'OpenWeatherMap.Temperature'
  71. 'OpenWeatherMap.Snow'
  72. 'HERE'
  73. 'HERE.normalDay'
  74. 'HERE.normalDayCustom'
  75. 'HERE.normalDayGrey'
  76. 'HERE.normalDayMobile'
  77. 'HERE.normalDayGreyMobile'
  78. 'HERE.normalDayTransit'
  79. 'HERE.normalDayTransitMobile'
  80. 'HERE.normalNight'
  81. 'HERE.normalNightMobile'
  82. 'HERE.normalNightGrey'
  83. 'HERE.normalNightGreyMobile'
  84. 'HERE.basicMap'
  85. 'HERE.mapLabels'
  86. 'HERE.trafficFlow'
  87. 'HERE.carnavDayGrey'
  88. 'HERE.hybridDay'
  89. 'HERE.hybridDayMobile'
  90. 'HERE.pedestrianDay'
  91. 'HERE.pedestrianNight'
  92. 'HERE.satelliteDay'
  93. 'HERE.terrainDay'
  94. 'HERE.terrainDayMobile'
  95. 'FreeMapSK'
  96. 'MtbMap'
  97. 'CartoDB'
  98. 'CartoDB.Positron'
  99. 'CartoDB.PositronNoLabels'
  100. 'CartoDB.PositronOnlyLabels'
  101. 'CartoDB.DarkMatter'
  102. 'CartoDB.DarkMatterNoLabels'
  103. 'CartoDB.DarkMatterOnlyLabels'
  104. 'HikeBike'
  105. 'HikeBike.HikeBike'
  106. 'HikeBike.HillShading'
  107. 'BasemapAT'
  108. 'BasemapAT.basemap'
  109. 'BasemapAT.grau'
  110. 'BasemapAT.overlay'
  111. 'BasemapAT.highdpi'
  112. 'BasemapAT.orthofoto'
  113. 'nlmaps'
  114. 'nlmaps.standaard'
  115. 'nlmaps.pastel'
  116. 'nlmaps.grijs'
  117. 'nlmaps.luchtfoto'
  118. 'NASAGIBS'
  119. 'NASAGIBS.ModisTerraTrueColorCR'
  120. 'NASAGIBS.ModisTerraBands367CR'
  121. 'NASAGIBS.ViirsEarthAtNight2012'
  122. 'NASAGIBS.ModisTerraLSTDay'
  123. 'NASAGIBS.ModisTerraSnowCover'
  124. 'NASAGIBS.ModisTerraAOD'
  125. 'NASAGIBS.ModisTerraChlorophyll'
  126. 'NLS'
  127. 'JusticeMap'
  128. 'JusticeMap.income'
  129. 'JusticeMap.americanIndian'
  130. 'JusticeMap.asian'
  131. 'JusticeMap.black'
  132. 'JusticeMap.hispanic'
  133. 'JusticeMap.multi'
  134. 'JusticeMap.nonWhite'
  135. 'JusticeMap.white'
  136. 'JusticeMap.plurality'
  137. 'Wikimedia'
In [11]:
# Use str_detect() to determine if the name of each provider tile contains the string "CartoDB"
str_detect(names(providers), "CartoDB")
  1. FALSE
  2. FALSE
  3. FALSE
  4. FALSE
  5. FALSE
  6. FALSE
  7. FALSE
  8. FALSE
  9. FALSE
  10. FALSE
  11. FALSE
  12. FALSE
  13. FALSE
  14. FALSE
  15. FALSE
  16. FALSE
  17. FALSE
  18. FALSE
  19. FALSE
  20. FALSE
  21. FALSE
  22. FALSE
  23. FALSE
  24. FALSE
  25. FALSE
  26. FALSE
  27. FALSE
  28. FALSE
  29. FALSE
  30. FALSE
  31. FALSE
  32. FALSE
  33. FALSE
  34. FALSE
  35. FALSE
  36. FALSE
  37. FALSE
  38. FALSE
  39. FALSE
  40. FALSE
  41. FALSE
  42. FALSE
  43. FALSE
  44. FALSE
  45. FALSE
  46. FALSE
  47. FALSE
  48. FALSE
  49. FALSE
  50. FALSE
  51. FALSE
  52. FALSE
  53. FALSE
  54. FALSE
  55. FALSE
  56. FALSE
  57. FALSE
  58. FALSE
  59. FALSE
  60. FALSE
  61. FALSE
  62. FALSE
  63. FALSE
  64. FALSE
  65. FALSE
  66. FALSE
  67. FALSE
  68. FALSE
  69. FALSE
  70. FALSE
  71. FALSE
  72. FALSE
  73. FALSE
  74. FALSE
  75. FALSE
  76. FALSE
  77. FALSE
  78. FALSE
  79. FALSE
  80. FALSE
  81. FALSE
  82. FALSE
  83. FALSE
  84. FALSE
  85. FALSE
  86. FALSE
  87. FALSE
  88. FALSE
  89. FALSE
  90. FALSE
  91. FALSE
  92. FALSE
  93. FALSE
  94. FALSE
  95. FALSE
  96. FALSE
  97. TRUE
  98. TRUE
  99. TRUE
  100. TRUE
  101. TRUE
  102. TRUE
  103. TRUE
  104. FALSE
  105. FALSE
  106. FALSE
  107. FALSE
  108. FALSE
  109. FALSE
  110. FALSE
  111. FALSE
  112. FALSE
  113. FALSE
  114. FALSE
  115. FALSE
  116. FALSE
  117. FALSE
  118. FALSE
  119. FALSE
  120. FALSE
  121. FALSE
  122. FALSE
  123. FALSE
  124. FALSE
  125. FALSE
  126. FALSE
  127. FALSE
  128. FALSE
  129. FALSE
  130. FALSE
  131. FALSE
  132. FALSE
  133. FALSE
  134. FALSE
  135. FALSE
  136. FALSE
  137. FALSE
In [12]:
# Use str_detect() to print only the provider tile names that include the string "CartoDB"
names(providers)[str_detect(names(providers), "CartoDB")]
  1. 'CartoDB'
  2. 'CartoDB.Positron'
  3. 'CartoDB.PositronNoLabels'
  4. 'CartoDB.PositronOnlyLabels'
  5. 'CartoDB.DarkMatter'
  6. 'CartoDB.DarkMatterNoLabels'
  7. 'CartoDB.DarkMatterOnlyLabels'

3. Adding a Custom Map Tile

Did any tile names look familiar? If you have worked with the mapping software you may recognize the name ESRI or CartoDB.

We create our first leaflet map using the default OSM map tile.

{r}
leaflet() %>% 
    addTiles()

We will primarily use CartoDB provider tiles, but feel free to try others, like Esri. To add a custom provider tile to our map we will use the addProviderTiles() function. The first argument to addProviderTiles() is your leaflet map, which allows us to pipe leaflet() output directly into addProviderTiles(). The second argument is provider, which accepts any of the map tiles included in the providers list.

Familiarize yourself with the SCRIPT.R and HTML VIEWER tabs. Click back and forth to type your code and view your maps.

{r}
leaflet() %>% 
    addProviderTiles("Esri")
{r}
leaflet() %>% 
    addProviderTiles("CartoDB.PositronNoLabels")

4. A Map with a View I

You may have noticed that, by default, maps are zoomed out to the farthest level. Rather than manually zooming and panning, we can load the map centered on a particular point using the setView() function.

{r}
leaflet()  %>% 
    addProviderTiles("CartoDB")  %>% 
    setView(lat = 40.7, lng = -74.0, zoom = 10)

Currently, DataCamp has offices at the following locations:

350 5th Ave, Floor 77, New York, NY 10118

Martelarenlaan 38, 3010 Kessel-Lo, Belgium

These addresses were converted to coordinates using the geocode() function in the ggmaps package.

NYC: (-73.98575, 40.74856) Belgium: (4.717863, 50.881363)

{r}
leaflet()  %>% 
    addProviderTiles("CartoDB")  %>% 
    setView(lng = -73.98575, lat = 40.74856, zoom = 6)
{r}
hc_dq <- data.frame(hq = c("DataCamp - NYC", "DataCamp - Belgium"), 
                   lon = c(-74.0, 4.72), 
                   lat = c(40.7, 50.9))
leaflet() %>% 
    addProviderTiles("CartoDB.PositronNoLabels") %>% 
    setView(lng = hc_dq$lon[2], lat = hc_dq$lat[2], zoom = 4)

5. A Map with a Narrower View

We can limit users' ability to pan away from the map's focus using the options argument in the leaflet() function. By setting minZoom anddragging, we can create an interactive web map that will always be focused on a specific area.

{r}
leaflet(options = 
        leafletOptions(minZoom = 14, dragging = FALSE))  %>% 
  addProviderTiles("CartoDB")  %>% 
  setView(lng = -73.98575, lat = 40.74856, zoom = 14)

Alternatively, if we want our users to be able to drag the map while ensuring that they do not stray too far, we can set the maps maximum boundaries by specifying two diagonal corners of a rectangle.

You'll use dc_hq to create a map with the "CartoDB" provider tile that is centered on DataCamp's Belgium office.

{r}
leaflet(options = leafletOptions(
                    # Set minZoom and dragging 
                    minZoom = 12, dragging = TRUE))  %>% 
  addProviderTiles("CartoDB")  %>% 

  # Set default zoom level 
  setView(lng = hc_dq$lon[2], lat = hc_dq$lat[2], zoom = 14) %>% 

  # Set max bounds of map 
  setMaxBounds(lng1 = hc_dq$lon[2] + .05, 
               lat1 = hc_dq$lat[2] + .05, 
               lng2 = hc_dq$lon[2] - .05, 
               lat2 = hc_dq$lat[2] - .05)

6. Mark it

So far we have been creating maps with a single layer: a base map. We can add layers to this base map similar to how you add layers to a plot in ggplot2. One of the most common layers to add to a leaflet map is location markers, which you can add by piping the result of addTiles() or addProviderTiles() into the add markers function.

For example, if we plot DataCamp's NYC HQ by passing the coordinates to addMarkers() as numeric vectors with one element, our web map will place a blue drop pin at the coordinate. In chapters 2 and 3, we will review some options for customizing these markers.

{r}
leaflet()  %>% 
    addProviderTiles("CartoDB")  %>% 
    addMarkers(lng = -73.98575, lat = 40.74856)

The dc_hq tibble is available in your work space.

{r}
# Plot DataCamp's NYC HQ
hc_dq <- data.frame(hq = c("DataCamp - NYC", "DataCamp - Belgium"), 
                   lon = c(-74.0, 4.72), 
                   lat = c(40.7, 50.9))

leaflet() %>% 
    addProviderTiles("CartoDB") %>% 
    addMarkers(lng = hc_dq$lon[1], lat = hc_dq$lat[1])
{r}
# Plot DataCamp's NYC HQ with zoom of 12    
leaflet() %>% 
    addProviderTiles("CartoDB") %>% 
    addMarkers(lng = -73.98575, lat = 40.74856)  %>% 
    setView(lng = -73.98575, lat = 40.74856, zoom = 12)
{r}
# Plot both DataCamp's NYC and Belgium locations
leaflet() %>% 
    addProviderTiles("CartoDB") %>% 
    addMarkers(lng = dc_hq$lon, lat = dc_hq$lat)

7. Adding Popups and Storing your Map

To make our map more informative we can add popups. To add popups that appear when a marker is clicked we need to specify the popup argument in the addMarkers() function. Once we have a map we would like to preserve, we can store it in an object. Then we can pipe this object into functions to add or edit the map's layers.

{r}
dc_nyc <- 
    leaflet() %>% 
        addTiles() %>% 
        addMarkers(lng = -73.98575, lat = 40.74856, 
                   popup = "DataCamp - NYC") 

dc_nyc %>% 
    setView(lng = -73.98575, lat = 40.74856, 
            zoom = 2)

Let's try adding popups to both DataCamp location markers and storing our map in an object.

In [1]:
# Store leaflet hq map in an object called map
# Plot DataCamp's NYC HQ
pkgs <- c("tidyverse", "leaflet", "htmlwidgets", "webshot")
sapply(pkgs, require, character.only = TRUE)

dc_hq <- data.frame(hq = c("DataCamp - NYC", "DataCamp - Belgium"), 
                   lon = c(-74.0, 4.72), 
                   lat = c(40.7, 50.9))

map <- leaflet() %>%
          addProviderTiles("CartoDB") %>%
          # Use dc_hq to add the hq column as popups
          addMarkers(lng = dc_hq$lon, lat = dc_hq$lat,
                     popup = dc_hq$hq)

# Center the view of map on the Belgium HQ with a zoom of 5 
map_zoom <- map %>%
      setView(lat = 50.881363, lng = 4.717863,
              zoom = 5)

# Print map_zoom
# map_zoom

# saving leaflet
## create .html and .png
## save html to png
saveWidget(map_zoom, "chapter1_mapZoom.html", selfcontained = FALSE)
webshot("chapter1_mapZoom.html", file = "chapter1_mapZoom.png",
        cliprect = "viewport")
Loading required package: tidyverse
── Attaching packages ─────────────────────────────────────── tidyverse 1.2.1 ──
✔ ggplot2 3.1.0       ✔ purrr   0.2.5  
✔ tibble  2.0.1       ✔ dplyr   0.8.0.1
✔ tidyr   0.8.2       ✔ stringr 1.3.1  
✔ readr   1.3.1       ✔ forcats 0.3.0  
── Conflicts ────────────────────────────────────────── tidyverse_conflicts() ──
✖ dplyr::filter() masks stats::filter()
✖ dplyr::lag()    masks stats::lag()
Loading required package: leaflet
Loading required package: htmlwidgets
Loading required package: webshot
tidyverse
TRUE
leaflet
TRUE
htmlwidgets
TRUE
webshot
TRUE



+ Recent posts