Vector maps for the web with MapLibre GL JS¶
Free
Starter
Standard
Professional
MapLibre GL JS is a robust, fully free and open-source library with 3D rendering, dynamic styling, and many other features. Stadia Maps is a founding member of the MapLibre project, which facilitates the ongoing development of MapLibre GL JS and other rendering technologies.
In this tutorial, you'll learn how to add a map to your webpage and fill it with markers representing your store locations, delivery trucks, or whatever else you're mapping.
Drawing a map¶
Let's dive in with some example code on how to use our vector tiles in MapLibre GL JS.
For local development on a web server at localhost
or 127.0.0.1
,
you can get started without any API keys or domain setup!
For mobile, backend, and non-local web development, you'll need either domain auth or an API key. If you don't already have a Stadia Maps account, sign up for a free to get started.
Domain-based authentication
Domain-based authentication is the easiest form of authentication for production web apps. No additional application code is required, and you don't need to worry about anyone scraping your API keys. We recommend this for most browser-based applications.
- Sign in to the client dashboard.
- Click "Manage Properties."
- Under "Authentication Configuration," click the button to add your domain.
API key authentication
Authenticating requests via API key¶
You can authenticate your requests by adding an API key to the query string or HTTP header.
The simplest is to add a query string parameter api_key=YOUR-API-KEY
to your request URL.
For example, to access the /tz/lookup
API endpoint, your request URL might look like this.
https://api.stadiamaps.com/tz/lookup/v1?lat=59.43696&lng=24.75357&api_key=YOUR-API-KEY
You can also use an Authorization
HTTP header instead of a query string
as shown below.
Don't forget the Stadia-Auth
prefix!
Authorization: Stadia-Auth YOUR-API-KEY
How to get an API key¶
Don't have an API key yet? Follow these easy steps!
- Sign in to the client dashboard. (If you don't have an account yet, sign up for free; no credit card required!)
- Click "Manage Properties."
- If you have more than one property (ex: for several websites or apps), make sure you have selected the correct property from the dropdown at the top of the page.
- Under "Authentication Configuration," you can generate, view or revoke your API key.
Video: How to generate your API key¶
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.
<!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_smooth.json', // Style URL; see our documentation for more options
center: [12, 53], // Initial focus coordinate
zoom: 4
});
// 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());
</script>
</body>
</html>
Code Walkthrough¶
First, we'll add some opening HTML content that will make the document standard compliant:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<meta name="viewport" content="initial-scale=1,maximum-scale=1,user-scalable=no" />
<title>Vector Map Demo</title>
Next, we'll include the MapLibre library files:
<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" />
Now, we'll add CSS to handle the size of the body, the map ID, and then close the <head>
tag:
<style type="text/css">
body {
margin: 0;
padding: 0;
}
#map {
position: absolute;
top: 0;
bottom: 0;
width: 100%;
}
</style>
</head>
With the <head>
details in, we can now add the body for our page. This is where we'll create the map element:
<body>
<div id="map"></div>
Next, we'll add the JavaScript for:
- initializing the map, specifying the
container
,style
,center
, andzoom
:
<script type="text/javascript">
var map = new maplibregl.Map({
container: 'map',
style: 'https://tiles.stadiamaps.com/styles/alidade_smooth.json', // Style URL; see our documentation for more options
center: [12, 53], // Initial focus coordinate
zoom: 4
});
- adding RTL support:
// 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');
- and adding navigation controls:
// Add zoom and rotation controls to the map.
map.addControl(new maplibregl.NavigationControl());
</script>
Finally, we'll close up the <body>
and <html>
tags:
</body>
</html>
And that's it. Congratulations.
All the code, put together, will match the same as the code at the start of this section.
Adding marker pins to the map¶
Once you have a map, you probably want to add some markers to it! The example below can serve as a starting point,
and covers all the main components. You can easily substitute your own branded markers, style the popup (or remove it),
etc. Please refer to the Marker
documentation
for additional details.
<!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%;
}
/* We recommend using an icon that is 2x the intended display size, so that it renders nicely on retina displays. */
.marker {
/* The following icon is owned by Stadia Maps. While you maintain an account with us, we grant you royalty-free use of
* this image when displayed on our maps.
*/
background-image: url('');
background-size: cover;
width: 27px;
height: 42px;
cursor: pointer;
}
</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_smooth_dark.json', // Style URL; see our documentation for more options
center: [6.942373, 50.332852], // Initial focus coordinate (long, lat)
zoom: 14
});
// Add zoom and rotation controls to the map.
map.addControl(new maplibregl.NavigationControl());
// First, we define our marker locations. You can use whatever format you want when
// working with custom markers, but we have chosen to use GeoJSON for this example, as
// a lot of geospatial data comes in this form. If you have a lot of data, you may want to
// put it in another file that is loaded separately.
var markerCollection = {
"type": "FeatureCollection",
"features": [{
"type": "Feature",
"geometry": {
"type": "Point",
// NOTE: in GeoJSON notation, LONGITUDE comes first. GeoJSON
// uses x, y coordinate notation
"coordinates": [6.940046, 50.333947]
},
"properties": {
"title": "Nürburgring"
}
}]
};
// Next, we can add markers to the map
markerCollection.features.forEach(function(point) {
// Since these are HTML markers, we create a DOM element first, which we will later
// pass to the Marker constructor.
var elem = document.createElement('div');
elem.className = 'marker';
// Now, we construct a marker and set it's coordinates from the GeoJSON. Note the coordinate order.
var marker = new maplibregl.Marker(elem);
marker.setLngLat(point.geometry.coordinates);
// You can also create a popup that gets shown when you click on a marker. You can style this using
// CSS as well if you so desire. A minimal example is shown. The offset will depend on the height of your image.
var popup = new maplibregl.Popup({ offset: 24, closeButton: false });
popup.setHTML('<div>' + point.properties.title + '</div>');
// Set the marker's popup.
marker.setPopup(popup);
// Finally, we add the marker to the map.
marker.addTo(map);
});
</script>
</body>
</html>
Code Walkthrough¶
To set up the page, we'll need the same initial code we used to draw the map:
<!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%;
}
Next, we'll add the marker
class, along with a base64 blob of the image we want to display for the marker:
/* We recommend using an icon that is 2x the intended display size, so that it renders nicely on retina displays. */
.marker {
/* The following icon is owned by Stadia Maps. While you maintain an account with us, we grant you royalty-free use of
* this image when displayed on our maps.
*/
background-image: url('');
background-size: cover;
width: 27px;
height: 42px;
cursor: pointer;
}
Then we'll close up the <style>
and <head>
tag:
</style>
</head>
Next, we'll add the <body>
tag and map element:
<body>
<div id="map"></div>
And then we'll add the JavaScript to create the map, style
and center
it, along with setting the zoom
level and
controls:
<script type="text/javascript">
var map = new maplibregl.Map({
container: 'map',
style: 'https://tiles.stadiamaps.com/styles/alidade_smooth.json', // Style URL; see our documentation for more options
center: [6.942373, 50.332852], // Initial focus coordinate (long, lat)
zoom: 14
});
// Add zoom and rotation controls to the map.
map.addControl(new maplibregl.NavigationControl());
With the map
element created, we can now create our markerCollection
object, which will store the coordinates for
our markers as GeoJSON in this example:
// You can use whatever format you want when working with custom markers, but we have chosen
// to use GeoJSON for this example, as a lot of geospatial data comes in this form. If you
// have a lot of data, you may want to put it in another file that is loaded separately.
var markerCollection = {
"type": "FeatureCollection",
"features": [{
"type": "Feature",
"geometry": {
"type": "Point",
// NOTE: in GeoJSON notation, LONGITUDE comes first. GeoJSON
// uses x, y coordinate notation
"coordinates": [6.940046, 50.333947]
},
"properties": {
"title": "Nürburgring"
}
}]
};
Next, we'll write a function to handle the market components:
- iterating through each feature in the
markerCollection
:
// Next, we can add markers to the map
markerCollection.features.forEach(function(point) {
// Since these are HTML markers, we create a DOM element first, which we will later
// pass to the Marker constructor.
var elem = document.createElement('div');
elem.className = 'marker';
- constructing a marker and setting its coordinates from the GeoJSON object we created earlier:
var marker = new maplibregl.Marker(elem);
marker.setLngLat(point.geometry.coordinates);
- adding a popup that will display the title we defined earlier:
Note
The offset value will depend on the height of your image.
var popup = new maplibregl.Popup({ offset: 24, closeButton: false });
popup.setHTML('<div>' + point.properties.title + '</div>');
// Set the marker's popup.
marker.setPopup(popup);
- and then adding the marker to the map:
// Finally, we add the marker to the map.
marker.addTo(map);
});
Finally, we'll close up the <script>
, <body>
, and <html>
tags:
</script>
</body>
</html>
And that's it. Congratulations.
All the code, put together, will match the code at the start of this section.
Next Steps¶
We've just scratched the surface of what's possible with MapLibre GL JS. Check out our other tutorials for more ideas, like adding a 3D globe view!
You can also browse the MapLibre GL JS Examples page for inspiration. The examples have in-depth examples covering everything from GeoJSON lines to interactive time sliders. And don't forget to check out the plugins and the rest of the MapLibre GL JS documentation.
Once you're ready to move beyond localhost
testing, sign up for a free Stadia Maps account,
and we'll walk through the next steps.