Helper functions to draw ‘rayshader’ scenes. - Using elevation data from ‘Mapzen’ and ‘Mapbox’ - With map and satellite overlays - Blending between different overlays at different altitudes - with added GPS tracks - From EU Copernicus, NASA ASTER or any other DEM (Digital Elevation Model) data - With elevation shading (green valleys and snow capped peaks, or anything else you want)
‘Rayshader’ is an awesome bit of kit! I’m just doing some colouring in.
geoviz is on CRAN:
install.packages("geoviz")
Or for the latest development version:
::install_github("neilcharles/geoviz") devtools
Read news to understand the latest updates and bug fixes.
Geoviz helps you to draw images like these.
library(geoviz)
library(rayshader)
#Load an example IGC (GPS track log) file
<- example_igc()
igc
#Load a small example elevation raster showing a piece of the English Lake district
<- example_raster()
DEM
= 270
sunangle
= 25
zscale
#Get a Stamen map that will cover our DEM
<- slippy_overlay(DEM, image_source = "stamen", image_type = "watercolor", png_opacity = 0.3)
stamen_overlay
#Make an elevation shading layer with dark valleys and light peaks (not essential but I like it!)
<- elevation_shade(DEM, elevation_palette = c("#000000", "#FFFFFF"), png_opacity = 0.6)
elevation_overlay
#Calculate the 'rayshader' scene (see 'rayshader' documentation)
= matrix(
elmat ::extract(DEM, raster::extent(DEM), method = 'bilinear'),
rasternrow = ncol(DEM),
ncol = nrow(DEM)
)
<- elmat %>%
scene sphere_shade(sunangle = sunangle, texture = "bw") %>%
add_overlay(elevation_overlay) %>%
add_overlay(stamen_overlay)
#Render the 'rayshader' scene
::plot_3d(
rayshader
scene,
elmat,zscale = zscale,
solid = FALSE,
shadow = TRUE,
shadowdepth = -100
)
#Add the gps track
add_gps_to_rayshader(
DEM,$lat,
igc$long,
igc$altitude,
igcline_width = 1.5,
lightsaber = TRUE,
colour = "red",
zscale = zscale,
ground_shadow = TRUE
)
To draw scenes using sub 2m resolution DEM’s, you’ll need to download your own data (see below), but geoviz also has helpful functions to obtain DEM data from Mapbox and Mapzen. Mapzen doesn’t require an API key and gives access to higher resolution data, depending on where in the world you request.
library(rayshader)
library(geoviz)
<- "YOUR MAPBOX KEY"
mapbox_key
<- 54.4502651
lat <- -3.1767946
long <- 20
square_km
#Get elevation data from Mapbox
<- mapbox_dem(lat, long, square_km, api_key = mapbox_key)
dem
#Note: You can get elevation data from Mapzen instead, which doesn't require an API key.
#You'll still need an API key for any mapbox image overlays.
#Get a DEM from mapzen with:
#dem <- mapzen_dem(lat, long, square_km)
#Get an overlay image (Stamen for this example because it doesn't need an API key)
<-
overlay_image slippy_overlay(dem, image_source = "stamen", image_type = "watercolor", png_opacity = 0.5)
#Optionally, turn mountainous parts of the overlay transparent
<-
overlay_image elevation_transparency(overlay_image,
dem,pct_alt_high = 0.5,
alpha_max = 0.9)
#Draw the 'rayshader' scene
= matrix(
elmat ::extract(dem, raster::extent(dem), method = 'bilinear'),
rasternrow = ncol(dem),
ncol = nrow(dem)
)
<- elmat %>%
scene sphere_shade(sunangle = 270, texture = "desert") %>%
add_overlay(overlay_image)
::plot_3d(
rayshader
scene,
elmat,zscale = raster_zscale(dem),
solid = FALSE,
shadow = TRUE,
shadowdepth = -150
)
# You can also visualise your data in ggplot2 rather than 'rayshader'.
<-
gg_overlay_image slippy_overlay(
dem,image_source = "stamen",
image_type = "watercolor",
return_png = FALSE
)
::ggplot() +
ggplot2ggslippy(gg_overlay_image)
DEM files can be downloaded from various sources, usually in .asc or .tif format. Often, they will be small files that need to be stitched together to render the scene that you want.
If you have downloaded a set of DEM files, use mosaic_files() to create a single raster for use with ‘rayshader’. The mosaic_files() function is flexible and will accept a directory of files or zipped files, using any naming convention and file extension.
mosaic_files(
"path/to/zip/files",
extract_zip = TRUE,
file_match = ".*.TIF",
raster_output_file = "mosaic_out.raster"
)
<- raster::raster("mosaic_out.gri") raster_mosaic
The following is by no means an exhaustive list of data sources, but it will get you started.
EU Copernicus
EU coverage.
Copernicus map tiles are large, typically 3-5GB each and covering a country sized area. Download Copernicus
<- 25 zscale
OS Terrain 50
UK coverage. Copernicus also covers the UK and comes as a single file covering the whole UK if you want to use that instead.
Download OS Terrain 50
mosaic_files(
"path/to/zip/files",
extract_zip = TRUE,
zip_file_match = ".*GRID.*.zip"
file_match = ".*.asc",
raster_output_file = "mosaic_out.raster"
)
<- raster::raster("mosaic_out.gri")
raster_mosaic
<- 50 zscale
NASA ASTER
Whole world coverage but quite noisy. Copernicus is better if you’re mapping in the EU.
Download NASA Aster. Search for “ASTER” in the top left box and select “ASTER Global Digital Elevation Model V002” underneath the map. You won’t realistically be able to stitch together a single file of the whole world - it would be enormous - so just download the areas you need.
Stitching together the separate files is the same process as for OS Terrain 50.
<- 30 zscale
You probably don’t want to render everything in your DEM data, you’ll want to cut out a piece. Geoviz has two functions to help you do this.
Crop out a square around a point…
library(ggmap)
register_google(key = your_google_key)
#Note that the below will only work if you point it at DEM data that contains Keswick!
<- geocode("Keswick, UK")
coords
<- crop_raster_square(big_DEM, coords$lat, coords$lon, square_km) DEM
Or crop a section from your DEM to fit a GPS track…
<- example_igc()
igc
<- crop_raster_track(example_raster(), igc$lat, igc$long, width_buffer = 2) DEM
You can load GPS track data any way that you like and pass decimal lat-longs as vectors to geoviz functions (see code examples above).
If your GPS data is in IGC format - commonly used for glider flight data - then geoviz has a function read_igc(), which will do all the formatting work for you.
If your GPS data is in .gpx format, the plotKML package has a handy function readGPX().
<- read_igc("path/to/your/file.igc") igc
Geoviz converts decimal lat-long GPS traces into the ‘rayshader’ coordinate system and then plots the GPS track using the function add_gps_to_rayshader(). Rather than adding a trace to a scene, if you just want to convert lat-long points into ‘rayshader’ coordinates and see the converted data (e.g. so you can add your own rgl shapes to the scene or for use with ‘rayshder’ render_label() function), use latlong_to_rayshader_coords().