Skip to content

Add 3D Buildings to your Maps

Free

Starter

Standard

Professional

Most maps are flat, looking straight down at the earth from above. But sometimes you need to look at things from another angle. Navigation apps, for example, often have a camera pitch between 45 and 60 degrees to show the area around and in front of the vehicle better.

At this angle, your maps can really come to life with 3D buildings. This tutorial will show you how to add 3D buildings to any map style using MapLibre and vector basemap tiles from Stadia Maps.

A map with 3D extruded buildings in central London

Build the Layer

The look and feel of your map is controlled by a stylesheet using a JSON MapLibre style. It defines the data sources and layers, which order rendering should happen in, and various attributes of the layers that make up your map.

A fill-extrusion layer lets us create a 3D effect by extruding portions of buildings from height information already present in the vector tile. The result isn't as detailed as a 3D model or point cloud, but it's fast, and saves precious data for mobile users.

A 3D building layer definition
{
  "id": "3d-buildings",
  "source": "openmaptiles",
  "source-layer": "building",
  "filter": [
    "!",
    [
      "to-boolean",
      ["get", "hide_3d"]
    ]
  ],
  "type": "fill-extrusion",
  "minzoom": 13,
  "paint": {
    "fill-extrusion-color": "lightgray",
    "fill-extrusion-height": [
      "interpolate",
      ["linear"],
      ["zoom"],
      13,
      0,
      16,
      ["get", "render_height"]
    ],
    "fill-extrusion-base": [
      "case",
      [">=",
        ["get", "zoom"],
        16
      ],
      ["get", "render_min_height"],
      0
    ]
  }
}

If you're familiar with MapLibre style documents already, this might look familiar. The first few property names describe themselves: an id gives a layer a unique identifier, minzoom specifies the minimum zoom at which the layer will render, and so on.

If you're starting from a style developed internally at Stadia Maps, it has an OpenMapTiles schema-compatible source named openmaptiles. If you're adding buildings to one of our Stamen-designed styles, change the source identifier to stamen-omt.

The filter is an example of an expression, which lets you make a decision according to some rules. If you've written LISP before, this might look like a predicate function with weird syntax. In our example, we are excluding features in the building layer with hide_3d set to a truth-y value. (The hide_3d property is set on building parts which are underground.)

The paint property is where we specify how the building will look. The fill-extrusion-color determines the exterior color. We've selected a neutral light gray. The fill-extrusion-height and fill-extrusion-base properties determine how the building is extruded. The effect of our choices here is to make the building "scale in" as you zoom.

Add the Layer to a Style

Integrating into a Custom Style

If you're building a custom style already, just add the new layer where you'd like it in your JSON! This currently requires a bit of work, but we're working to make that easier. Let us know if you'd like to be kept up to date on custom style hosting.

Modifying an Existing Style Dynamically

You can also modify an existing style in-place after it's loaded into the map! Use this approach with care, as things like layer identifiers or ordering may change over time.

Here's an example of how you could do this in MapLibre GL JS. MapLibre Native offers similar APIs.

map.addLayer({
  'id': '3d-buildings',
  'source': 'openmaptiles',
  'source-layer': 'building',
  'filter': [
    "!",
    ["to-boolean",
      ["get", "hide_3d"]
    ]
  ],
  'type': 'fill-extrusion',
  'minzoom': 13,
  'paint': {
    'fill-extrusion-color': 'lightgray',
    'fill-extrusion-height': [
      'interpolate',
      ['linear'],
      ['zoom'],
      13,
      0,
      16,
      ['get', 'render_height']
    ],
    'fill-extrusion-base': ['case',
      ['>=', ['get', 'zoom'], 16],
      ['get', 'render_min_height'], 0
    ]
  }
},
labelLayerId  // Replace this with the ID of a layer in the style; the new layer will be added below this
);

Full Example in MapLibre GL JS

Bringing everything together, here is some sample code which adds a 3D buildings layer to one of our styles. To keep it relatively simple and future-proof, we add the buildings directly under the first symbol layer.

To make experimenting easy, we've packaged the example code as a JSFiddle playground. Click the "Try it in JSFiddle" button to try it right from your web browser.

Try it in JSFiddle
<!DOCTYPE html>
<html>
    <head>
        <meta charset="utf-8" />
        <title>Vector Map Demo</title>
        <meta name="viewport" content="initial-scale=1,maximum-scale=1,user-scalable=no" />
        <script type="text/javascript" src="//unpkg.com/maplibre-gl@5.0.1/dist/maplibre-gl.js"></script>
        <link href="//unpkg.com/maplibre-gl@5.0.1/dist/maplibre-gl.css" rel="stylesheet" />
        <style type="text/css">
            body {
              margin: 0;
              padding: 0;
            }

            #map {
              position: absolute;
              top: 0;
              bottom: 0;
              width: 100%;
            }
        </style>
    </head>
    <body>
        <div id="map"></div>
        <script type="text/javascript">
         var map = new maplibregl.Map({
           container: 'map',
           style: 'https://tiles.stadiamaps.com/styles/alidade_bright.json',  // Style URL; see our documentation for more options
           center: [-0.13710, 51.5], // Initial focus coordinate
           zoom: 16,
           pitch: 60,
           bearing: 82
         });

         // MapLibre GL JS does not handle RTL text by default,
         // so we recommend adding this dependency to fully support RTL rendering if your style includes RTL text
         maplibregl.setRTLTextPlugin('https://unpkg.com/@mapbox/mapbox-gl-rtl-text@0.2.3/mapbox-gl-rtl-text.min.js');

         // Add zoom and rotation controls to the map.
         map.addControl(new maplibregl.NavigationControl());

         // Wait for the map to load, then manipulate the style
         map.on('load', () => {
          const layers = map.getStyle().layers;

          // Find the first symbol layer.
          // We'll add buildings just under this for the example.
          let labelLayerId;
          for (let i = 0; i < layers.length; i++) {
            if (layers[i].type === 'symbol' && layers[i].layout['text-field']) {
              labelLayerId = layers[i].id;
              break;
            }
          }
          map.addLayer({
              'id': '3d-buildings',
              'source': 'openmaptiles',
              'source-layer': 'building',
              'filter': [
                "!",
                ["to-boolean",
                  ["get", "hide_3d"]
                ]
              ],
              'type': 'fill-extrusion',
              'minzoom': 13,
              'paint': {
                'fill-extrusion-color': 'lightgray',
                'fill-extrusion-height': [
                  'interpolate',
                  ['linear'],
                  ['zoom'],
                  13,
                  0,
                  16,
                  ['get', 'render_height']
                ],
                'fill-extrusion-base': ['case',
                  ['>=', ['get', 'zoom'], 16],
                  ['get', 'render_min_height'], 0
                ]
              }
            },
            labelLayerId
          );
        });
        </script>
    </body>
</html>

Next Steps

Ready to add maps with 3D buildings to your website or app? Sign up for a free Stadia Maps account to get started!

Get Started With a Free Account

If you want to craft even more detailed building styles, the MapLibre style documentation covers other styling options including patterns, gradients, and more.

Finally, if you're building a navigation experience, check out Ferrostar, our highly extensible navigation SDK.