Deck GL and 3D Tiles

Generate Colorized 3D Tiles from LiDAR and Ortho Imagery

Deck GL and 3D Tiles
Click here to view the 3D visualization.

This post walks you through the process of creating colorized 3D Tiles suitable for displaying on a web map from LiDAR and ortho imagery. For this example we will use LiDAR from the Elevation Source Data (3DEP) - Lidar, IfSAR dataset, high-resolution ortho imagery from Ohio’s open imagery program, and the command-line geospatial tools pdal, gdal, and gocesiumtiler.

To get started, download and install PDAL and gocesiumtiler:
Instructions for macOS, for other operating systems refer to the respective documentation.

  1. PDAL Installation:

    • Open Terminal and install PDAL using Homebrew:
      brew install pdal
  2. gocesiumtiler Installation:

    • Install Go using the default installer or homebrew:
      brew install go
    • Clone the gocesiumtiler repository from GitHub:
      git clone https://github.com/mfbonfigli/gocesiumtiler
    • Navigate to the repository directory:
      cd gocesiumtiler
    • Build the CLI executable:
      go build -o gocesiumtiler ./cmd/main.go

Next we need to download the data. In this example we will use data for the Ohio Buckeye’s famous stadium.

osu lidar

osu tiff

For simplicity we will rename the laz file to input.laz and the tiff file to ortho.tif.

mv USGS_LPC_OH_Columbus_2019_B19_BS822728.laz input.laz
unzip S1820725.zip
mv S1820725.tif ortho.tif && mv S1820725.tfw ortho.tfw && mv S1820725.tif.xml ortho.tif.xml

In order to ensure the imagery will color the lidar correctly, we need to verify the EPSG code for each dataset. For the LiDAR data, we can use PDAL:

pdal info --metadata input.laz

This will export a large amount of metadata about the LAS file, but in this case we are interested only in the cartesion coordinate system.

...
"id": {
"authority": "EPSG",
"code": 3754
}

Assuming we have gdal installed, we can check the EPSG code of the orthoimage (TIFF file), using gdalinfo:

gdalinfo ortho.tif
...
CS[Cartesian,2],
AXIS["easting",east,
ORDER[1],
LENGTHUNIT["US survey foot",0.304800609601219]],
AXIS["northing",north,
ORDER[2],
LENGTHUNIT["US survey foot",0.304800609601219]],
ID["EPSG",6551]

In this instance the two coordinate systems are very close - which can be verified by overlaying the two in QGIS and checking if there are any discrepancies when using the default transformation, or see if there are any transformations present at all.

Now that we know our two datasets will align in the same coordinate system, we need to create a PDAL pipeline file to colorize the point cloud and transform it into Web Mercator. We will also force the las file into version 1.3 so it is compatible with gocesiumtiler. To do this, create a PDAL pipeline JSON file with the following content:

{
"pipeline": [
{
"type": "readers.las",
"filename": "input.laz"
},
{
"type": "filters.colorization",
"raster": "ortho.tif"
},
{
"type": "filters.range",
"limits": "Red[1:],Green[1:],Blue[1:]"
},
{
"type": "filters.assign",
"value": "Alpha = 255"
},
{
"type": "filters.reprojection",
"out_srs": "EPSG:3857"
},
{
"type": "writers.las",
"minor_version": 3,
"dataformat_id": 3,
"discard_high_return_numbers": true,
"filename": "output.las"
}
]
}

Save this as pipeline.json, then run the pipeline with PDAL:

pdal pipeline pipeline.json

Now that we have the colorized LiDAR data in Web Mercator, we can use gocesiumtiler to create 3D tiles.

./gocesiumtiler file -out ./cesium-tiles -epsg 3857 -r 1 -z 0 -d 20 -m 5000 ./input.las

To clamp the tiles to the ground, retrieve the minimum elevation from the las file using PDAL:

pdal info input.laz --stats
{
"average": 749.7651415,
"count": 2954370,
"maximum": 957.48,
"minimum": 663.63,
"name": "Z",
"position": 2,
"stddev": 37.20012168,
"variance": 1383.849053
},

Look for the minimum value under the Z dimension in the output. Use this value to clamp the tiles to the ground using the -z flag:

./gocesiumtiler file -out ./cesium-tiles -epsg 3857 -r 1 -z -663 -d 20 -m 5000 --8-bit ./output.las

The –8-bit flag reads the colors in 8bit color space, which you may omit depending on your data sources. For all options see the documentation.

Finally, now that we have our 3D TIles, we can load them into a web map. The following code snippet demonstrates how to view the 3D tiles with deck.gl:

import { Deck } from "@deck.gl/core";
import { Tile3DLayer } from "@deck.gl/geo-layers";
import { Tiles3DLoader } from "@loaders.gl/3d-tiles";

new Deck({
initialViewState: {
longitude: -83.0195,
latitude: 40.002,
zoom: 17,
pitch: 50,
bearing: -30,
},
container: "map",
controller: true,
layers: [
new Tile3DLayer({
data: "./cesium-tiles/tileset.json",
onTilesetLoad: (tileset) => {
console.log(tileset);
},
loaders: [Tiles3DLoader],
}),
],
});

Here is the final result:

OSU Stadium 3D Tiles

Newer Post
Palisades Fire Before and After
Older Post
Finding Perpendicular lines with TurfJS