Home

Awesome

3D mountains in Colorado

Interactive Storytelling

Some stories are best told with a map. Data journalists covering changing conditions in a population's demographics, the environment, an international conflict, or telling a simple travel story frequently provide geographic context in their graphics.

This template is designed to accelerate building out a "storytelling" map story. The primary input is a story broken into sections (chapters), each hooked to a particular view of a map.

Optionally, you can input a custom Mapbox Style with layers styled in Studio and toggle the layer's opacity.

The output is an HTML and JavaScript file. These outputs can be hosted on any web-accessible location, with no extra code or infrastructure required. Note that embedding the output as an iFrame in another page will not work as expected. The scroll-driven interface requires the full page.

Live Demo

You can view a live demo of this storytelling framework at https://labs.mapbox.com/storytelling/

Prerequisites

This template is for data journalists and digital storytellers of any kind. If you are planning to include some custom map layers, you will need some familiarity with Mapbox Studio.

To configure and publish a story, you will need:

The template does not rely on any particular CSS framework, fonts, or images. There are some basic styles in the head of the HTML file that can be changed, so feel free to adapt and add to these to match your site and story brand.

example story screen capture

Run the index.html file in a development server.

Clone this repository, and open its directory in Visual Studio Code.

1: Install the Live Server Extension

To make it easier to preview changes in real-time, you can install the Live Server extension in Visual Studio Code. (This will serve the index.html from a local webserver, preventing CORS errors and other issues that can arise from opening an HTML file without using a development server):

  1. Go to the Extensions tab in Visual Studio Code (the square icon on the sidebar).
  2. Search for Live Server and click Install.
  3. Once installed, you’ll be able to start a live server to preview the HTML file.

2: Run the HTML File in Your Browser

  1. With the index.html file open, right-click in the editor window and select Open with Live Server.
  2. Your default browser will open, displaying the storytelling map using the default configuration in config.js. Because you have not yet added your Mapbox Access Token, the map will not display. Follow the steps below to update config.js.

Customize your configuration

Open config.js and make the following edits to customize the map to tell your story.

  1. Add a Mapbox access token. Replace YOUR_MAPBOX_ACCESS_TOKEN with an access token from your account at account.mapbox.com. A good practice is to create a separate token per map to be able to track traffic to your different maps.

  2. Select the map style you want to use (the default is Mapbox Standard, but you can find more styles in our Styles API documentation page, or create your own style in Mapbox Studio).

  3. Choose whether or not to display a marker at the center of each map location. If you are displaying markers, you can set the color using the markerColor property. The default color is light blue.

  4. Choose a theme for the story text. There are light and dark options.

{
    style: 'mapbox://styles/mapbox/streets-v11',
    accessToken: 'YOUR_MAPBOX_ACCESS_TOKEN',
    showMarkers: true,
    markerColor: '#3FB1CE',
    theme: 'light',
    use3dTerrain: false,
    title: 'The Title Text of this Story',
    subtitle: 'A descriptive and interesting subtitle to draw in the reader',
    byline: 'By a Digital Storyteller',
    footer: 'Source: source citations, etc.',
    chapters: [
    ...
    ]
{
  1. Add as many chapters in your template as needed. You'll need a , between each section, but no comma at the end. Here is what a chapter looks like:
{
    id: 'slug-style-id',
    alignment: 'left',
    hidden: false,
    title: 'Display Title',
    image: './path/to/image/source.png',
    description: 'Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.',
    location: {
        center: [-122.418398, 37.759483],
        zoom: 8.5,
        pitch: 60,
        bearing: 0
    },
    mapAnimation: 'flyTo',
    rotateAnimation: false,
    callback: '',
    onChapterEnter: [],
    onChapterExit: []
},
  1. Fill out your sections as needed. Give each section a unique name in the section id property. This will become the HTML div id, so avoid spaces in the name. The title, description properties are optional. The description supports HTML tags. If you have an image that goes with that section of the story, add the path to the image in the image property.

  2. For location, you can use the helper.html file to help you determine the map's position. This tool prints the location settings of the map on the screen in a format ready for copy/paste into the template. Optionally, you can change the style in this file to your custom style.

  3. Repeat until you have the location entered for each of your sections.

  4. Open index.html in a browser, and scroll. Voila!

  5. There are more options available in config.js. Refer to the documentation in this README and experiment with the different values as you craft your story. If you are familiar with HTML, CSS, and JavaScript, you can customize every aspect of the storytelling map, well beyond the options we've provided in this template. Have fun, and please share your work!

Generate Map Position

Use the Mapbox Location Helper to search for locations and get the center, zoom, pitch, and bearing for use in your storytelling map.

location helper screen capture

Configuration File and Layer Settings

Here is a sample configuration:

var config = {
    style: 'mapbox://styles/branigan/cjz37rcb003ib1cr3s8rnkt2d',
    accessToken: 'pk.eyJ1IjoibWJ4c29sdXRpb25zIiwiYSI6ImNrMm01aG9hdTBlZGwzbXQ1ZXVrNHNmejAifQ.QHQA0N6XPWddCXtvoODHZg',
    showMarkers: false,
    theme: 'dark',
    use3dTerrain: true,
    title: 'Glaciers of Glacier National Park',
    subtitle: 'Change in coverage from 1998 to 2015',
    byline: '',
    footer: 'Source: Story text from Wikipedia, August 2019. Data from <a href="https://www.usgs.gov/centers/norock/science/retreat-glaciers-glacier-national-park">USGS</a>',
    chapters: [
        {
            id: 'glacier-np',
            alignment: 'full',
            title: 'Glacier National Park Glaciers',
            image: 'https://upload.wikimedia.org/wikipedia/commons/thumb/e/ea/2015-06-19_Glacier_National_Park_%28U.S.%29_8633.jpg/800px-2015-06-19_Glacier_National_Park_%28U.S.%29_8633.jpg',
            description: 'Glacier National Park is dominated by mountains which were carved into their present shapes by the huge glaciers of the last ice age...',
            location: {
                center: [-113.91666, 48.66451],
                zoom: 8,
                pitch: 0.00,
                bearing: 0.00
            },
            onChapterEnter: [
                {
                    layer: 'gnpglaciers-1998',
                    opacity: 0.25
                },
                {
                    layer: 'glaciernp-boundary',
                    opacity: 0.25
                }
            ],
            onChapterExit: [
                {
                    layer: 'glaciernp-boundary',
                    opacity: 0
                }
            ]
        },
        {
            id: 'harrison1998',
            alignment: 'left',
            title: 'Harrison Glacier, 1998',
            image: '',
            description: 'Harrison Glacier is located in the US state of Montana in Glacier National Park. Situated on a southeast facing ridge immediately south of Mount Jackson, Harrison Glacier is the largest glacier in Glacier National Park...',
            location: {
                center: [-113.72917, 48.58938],
                zoom: 12.92,
                pitch: 39.50,
                bearing: 36.00
            },
            onChapterEnter: [],
            onChapterExit: [
                // {
                //     layer: 'gnpglaciers-2015',
                //     opacity: 0
                // }
            ]
        }
    ]
}

Configuration Options

OptionTypeDescription
style (Required)StringThis is the Mapbox style url to use for the app. It can be a standard style, or a custom style from your Mapbox account. Use a custom style if you want to include custom data or layers.
accessToken (Required)StringYour Mapbox access token.
showMarkersStringControls whether markers are shown at the centerpoint of each chapter. If true, the map will display a default blue, inverted-teardrop icon.
markerColorStringAccepts hexadecimal, RGB, and color names compatible with CSS standards. Overrides the default light blue marker color if showMarkers is true.
themeStringTwo basic themes (light and dark) are available.
use3dTerrainStringEnables 3D terrain. (Optional)
insetStringEnables a globe minimap. (Optional)
insetOptionsObject GlobeMiniMap options
insetPositionSrtingA string representing the position of the inset map on the map. Valid options are top-left, top-right, bottom-left, bottom-right.
projectionStringSets the Map object's projection parameter to create a map with a non-Mercator projection. (Optional)
autoStringEnables automatic advancement through the chapters. (Optional)
titleStringThe title of the overall story. (Optional)
subtitleStringA subtitle for the story. (Optional)
bylineStringCredit the author of the story. (Optional)
footerStringCitations, credits, etc., displayed at the bottom of the story.
chapters (Required)StringContains all of the story content and map controls for each section of the story. Array of objects

Chapters Options

OptionTypeDescription
id (Required)StringA slug-style ID for the chapter. Used by the app's JavaScript and assigned as an HTML id for the div containing the story. Best practice: use kebab-case, e.g., my-story-chapter-1.
alignment (Required)StringDefines where the story text should appear over the map. Options: center, left, right, full. Defaults to center for browser windows less than 750 pixels wide.
hiddenStringSets the visibility of the chapter to hidden when true. The chapter will still trigger a map and layer transition.
titleStringTitle of the section, displayed in an h3 element.
imageStringPath to an image to display in this section.
descriptionStringMain story content for the section. Aligned with what the reader sees on the map. Supports HTML for images, links, etc.
location (Required)StringDetails about the map display and camera view (e.g., center, zoom, pitch, bearing).
mapAnimationStringDefines the animation type for transitions. Options: flyTo, easeTo, jumpTo. Defaults to flyTo.
rotateAnimationStringStarts a slow rotation animation at the end of the map transition when true. Rotates 90 degrees over 24 seconds.
callbackStringName of a JavaScript function to execute custom code for the chapter, e.g., turning a legend on/off, adding API data, or displaying an interactive graph.
onChapterEnterStringLayers to be displayed/hidden/muted when the section becomes active. Array of objects (e.g., layer name, opacity, duration).
onChapterExitStringSame as onChapterEnter, triggered when the section becomes inactive. Array of objects.

Location Details

OptionTypeDescription
center (Required)StringCenter coordinates of the map, as longitude, latitude.
zoom (Required)StringZoom level of the map.
pitchStringAngle of the map view. 0 is straight down, and 60 is highly tilted.
bearingStringDegrees of rotation clockwise from North (0). Negative values represent counter-clockwise rotation.
speedStringSpeed of the flyTo animation.
curveStringCurve factor for the flyTo animation.

Layer Configuration in your Mapbox Studio Style

Add and style each custom layer in your Studio style. Before the final publish, set any layers's style to be hidden with 0 opacity. Do not hide the layer. For example, if you have a circle layer, makes sure the color-opacity and/or the stroke-opacity is set to 0.

This will ensure that the map appears correctly when the story page loads. To adjust the opacity of the layers as the reader scrolls through the story, use the onChapterEnter or onChapterExit configuration options to set your desired opacity for the layer.

Deployment

Host the index.html and config.js files in the same directory in a web-accessible location. If you don't know where to start, look into GitHub Pages or Netlify.

Built With

Authors

John Branigan on the Mapbox Solutions Architecture Team

License

BSD 3-Clause License

Acknowledgments

Notable Examples

mapbox.com/resources#solutions