mapbox-web-integration-patterns

작성자: mapbox

Mapbox GL JS의 프로덕션 준비 통합 패턴으로, React, Vue, Svelte, Angular 및 바닐라 JavaScript를 지원합니다. React의 useEffect/useRef, Vue의 mounted/unmounted, Svelte의 onMount/onDestroy, Angular의 ngOnInit/ngOnDestroy 등 프레임워크별 생명주기 관리를 다루며, 메모리 누수를 방지하기 위한 적절한 정리 작업을 포함합니다. 프레임워크에 구애받지 않는 재사용 가능한 지도 요소와 크로스 프레임워크 호환성을 위한 Web Components 패턴을 제공합니다. 환경 변수를 통한 토큰 관리, Search JS...

npx skills add https://github.com/mapbox/mapbox-agent-skills --skill mapbox-web-integration-patterns

Mapbox Integration Patterns Skill

This skill provides official patterns for integrating Mapbox GL JS into web applications using React, Vue, Svelte, Angular, and vanilla JavaScript. These patterns are based on Mapbox's create-web-app scaffolding tool and represent production-ready best practices.

Version Requirements

Mapbox GL JS

Recommended: v3.x (latest)

  • Minimum: v3.0.0
  • Why v3.x: Modern API, improved performance, active development
  • v2.x: Legacy; no longer actively developed (see migration notes below)

Installing via npm (recommended for production):

npm install mapbox-gl@^3.0.0    # Installs latest v3.x

CDN (for prototyping only):

<!-- Replace VERSION with latest v3.x from https://docs.mapbox.com/mapbox-gl-js/ -->
<script src="https://api.mapbox.com/mapbox-gl-js/vVERSION/mapbox-gl.js"></script>
<link href="https://api.mapbox.com/mapbox-gl-js/vVERSION/mapbox-gl.css" rel="stylesheet" />

Framework Requirements

React: GL JS works with React 16.8+ (requires hooks). create-web-app scaffolds with React 19.x. Vue: GL JS works with Vue 2.x+ (Vue 3 Composition API recommended). Svelte: GL JS works with any Svelte version. create-web-app scaffolds with Svelte 5.x. Angular: GL JS works with Angular 2+. create-web-app scaffolds with Angular 19.x. Next.js: Minimum 13.x (App Router), Pages Router 12.x+.

Mapbox Search JS

npm install @mapbox/search-js-react@^1.0.0      # React
npm install @mapbox/search-js-web@^1.0.0        # Other frameworks

Version Migration Notes (v2.x to v3.x)

  • WebGL 2 now required
  • optimizeForTerrain option removed
  • Improved TypeScript types, better tree-shaking support
  • No breaking changes to core initialization patterns

Token patterns (work in v2.x and v3.x):

const token = import.meta.env.VITE_MAPBOX_ACCESS_TOKEN; // Use env vars in production

// Global token (works since v1.x)
mapboxgl.accessToken = token;
const map = new mapboxgl.Map({ container: '...' });

// Per-map token (preferred for multi-map setups)
const map = new mapboxgl.Map({
  accessToken: token,
  container: '...'
});

Core Principles

Every Mapbox GL JS integration must:

  1. Initialize the map in the correct lifecycle hook
  2. Store map instance in component state (not recreate on every render)
  3. Always call map.remove() on cleanup to prevent memory leaks
  4. Handle token management securely (environment variables)
  5. Import CSS: import 'mapbox-gl/dist/mapbox-gl.css'

React Integration (Primary Pattern)

Pattern: useRef + useEffect with cleanup

Note: These examples use Vite (the bundler used in create-web-app). If using Create React App, replace import.meta.env.VITE_MAPBOX_ACCESS_TOKEN with process.env.REACT_APP_MAPBOX_TOKEN. See Token Management Patterns for other bundlers.

import { useRef, useEffect } from 'react';
import mapboxgl from 'mapbox-gl';
import 'mapbox-gl/dist/mapbox-gl.css';

function MapComponent() {
  const mapRef = useRef(null); // Store map instance
  const mapContainerRef = useRef(null); // Store DOM reference

  useEffect(() => {
    mapboxgl.accessToken = import.meta.env.VITE_MAPBOX_ACCESS_TOKEN;

    mapRef.current = new mapboxgl.Map({
      container: mapContainerRef.current,
      center: [-71.05953, 42.3629],
      zoom: 13
    });

    // CRITICAL: Cleanup to prevent memory leaks
    return () => {
      mapRef.current.remove();
    };
  }, []); // Empty dependency array = run once on mount

  return <div ref={mapContainerRef} style={{ height: '100vh' }} />;
}

Key points:

  • Use useRef for both map instance and container
  • Initialize in useEffect with empty deps []
  • Always return cleanup function that calls map.remove()
  • Never initialize map in render (causes infinite loops)

React + Search JS

import { useRef, useEffect, useState } from 'react';
import mapboxgl from 'mapbox-gl';
import { SearchBox } from '@mapbox/search-js-react';
import 'mapbox-gl/dist/mapbox-gl.css';

const accessToken = import.meta.env.VITE_MAPBOX_ACCESS_TOKEN;
const center = [-71.05953, 42.3629];

function MapWithSearch() {
  const mapRef = useRef(null);
  const mapContainerRef = useRef(null);
  const [inputValue, setInputValue] = useState('');

  useEffect(() => {
    mapboxgl.accessToken = accessToken;

    mapRef.current = new mapboxgl.Map({
      container: mapContainerRef.current,
      center: center,
      zoom: 13
    });

    return () => {
      mapRef.current.remove();
    };
  }, []);

  return (
    <>
      <div
        style={{
          margin: '10px 10px 0 0',
          width: 300,
          right: 0,
          top: 0,
          position: 'absolute',
          zIndex: 10
        }}
      >
        <SearchBox
          accessToken={accessToken}
          map={mapRef.current}
          mapboxgl={mapboxgl}
          value={inputValue}
          proximity={center}
          onChange={(d) => setInputValue(d)}
          marker
        />
      </div>
      <div ref={mapContainerRef} style={{ height: '100vh' }} />
    </>
  );
}

Search JS Integration Summary

Install:

npm install @mapbox/search-js-react      # React
npm install @mapbox/search-js-web        # Vanilla/Vue/Svelte

Both packages include @mapbox/search-js-core as a dependency. Only install -core directly if building a custom search UI.

Key configuration options:

  • accessToken: Your Mapbox public token
  • map: Map instance (must be initialized first)
  • mapboxgl: The mapboxgl library reference
  • proximity: [lng, lat] to bias results geographically
  • marker: Boolean to show/hide result marker
  • placeholder: Search box placeholder text

Positioning Search Box

Absolute positioning (overlay):

<div
  style={{
    position: 'absolute',
    top: 10,
    right: 10,
    zIndex: 10,
    width: 300
  }}
>
  <SearchBox {...props} />
</div>

Common positions:

  • Top-right: top: 10px, right: 10px
  • Top-left: top: 10px, left: 10px
  • Bottom-left: bottom: 10px, left: 10px

Common Mistakes (Critical)

Mistake 1: Forgetting to call map.remove()

// BAD - Memory leak!
useEffect(() => {
  const map = new mapboxgl.Map({ ... })
  // No cleanup function
}, [])

// GOOD - Proper cleanup
useEffect(() => {
  const map = new mapboxgl.Map({ ... })
  return () => map.remove()  // Cleanup
}, [])

Why: Every Map instance creates WebGL contexts, event listeners, and DOM nodes. Without cleanup, these accumulate and cause memory leaks.

Mistake 2: Initializing map in render

// BAD - Infinite loop in React!
function MapComponent() {
  const map = new mapboxgl.Map({ ... })  // Runs on every render
  return <div />
}

// GOOD - Initialize in effect
function MapComponent() {
  useEffect(() => {
    const map = new mapboxgl.Map({ ... })
  }, [])
  return <div />
}

Why: React components re-render frequently. Creating a new map on every render causes infinite loops and crashes.

Mistake 3: Not storing map instance properly

// BAD - map variable lost between renders
function MapComponent() {
  useEffect(() => {
    let map = new mapboxgl.Map({ ... })
    // map variable is not accessible later
  }, [])
}

// GOOD - Store in useRef
function MapComponent() {
  const mapRef = useRef()
  useEffect(() => {
    mapRef.current = new mapboxgl.Map({ ... })
    // mapRef.current accessible throughout component
  }, [])
}

Why: You need to access the map instance for operations like adding layers, markers, or calling remove().

Mistake 4: Storing map instance in Vue's data() (Vue-specific)

// BAD - Vue's reactivity wraps data() objects in a Proxy, breaking mapbox-gl internals!
export default {
  data() {
    return {
      map: null  // Will be wrapped in a Proxy
    }
  },
  mounted() {
    this.map = new mapboxgl.Map({ ... })  // Proxy breaks GL internals
  }
}

// GOOD - Assign map as a plain instance property, not in data()
export default {
  mounted() {
    this.map = new mapboxgl.Map({
      container: this.$refs.mapContainer,
      center: [-71.05953, 42.3629],
      zoom: 13
    })
  },
  unmounted() {
    this.map?.remove()
  }
}

Why: In Vue (especially Vue 3), data() properties are wrapped in a Proxy for reactivity. Mapbox GL JS internally checks object identity and uses properties that don't survive proxy wrapping. Storing the map in data() causes subtle, hard-to-debug failures. Instead, assign the map instance directly as this.map in mounted() — properties assigned outside data() are not made reactive.

Reference Files

Load these for framework-specific patterns and additional details:

  • references/vue.md — Vue Integration (mounted/unmounted lifecycle)
  • references/svelte.md — Svelte Integration (onMount/onDestroy)
  • references/angular.md — Angular Integration with SSR handling
  • references/vanilla.md — Vanilla JS (Vite) + Vanilla JS (CDN)
  • references/web-components.md — Web Components (basic + reactive + usage in React/Vue/Svelte)
  • references/nextjs.md — Next.js App Router + Pages Router
  • references/common-mistakes.md — Common Mistakes 4-7 + Testing Patterns
  • references/token-management.md — Token Management per bundler + Style Configuration

When to Use This Skill

Invoke this skill when:

  • Setting up Mapbox GL JS in a new project
  • Integrating Mapbox into a specific framework (React, Vue, Svelte, Angular, Next.js)
  • Building framework-agnostic Web Components
  • Creating reusable map components for component libraries
  • Debugging map initialization issues
  • Adding Mapbox Search functionality
  • Implementing proper cleanup and lifecycle management
  • Converting between frameworks (e.g., React to Vue)
  • Reviewing code for Mapbox integration best practices

Related Skills

  • mapbox-cartography: Map design principles and styling
  • mapbox-token-security: Token management and security
  • mapbox-style-patterns: Common map style patterns

Resources

mapbox의 다른 스킬

mapbox-android-patterns
mapbox
Android용 Mapbox Maps SDK의 공식 통합 패턴입니다. 설치, 마커 추가, 사용자 위치, 사용자 정의 데이터, 스타일, 카메라 제어 등을 다룹니다.
official
mapbox-cartography
mapbox
지도 디자인 원칙, 색채 이론, 시각적 위계, 타이포그래피, 효과적이고 아름다운 지도를 제작하기 위한 지도학적 모범 사례에 대한 전문적인 안내…
official
mapbox-data-visualization-patterns
mapbox
지도 위에 데이터를 시각화하는 패턴으로, 코로플레스 맵, 히트 맵, 3D 시각화, 데이터 기반 스타일링, 애니메이션 데이터를 포함합니다. 레이어 유형 등을 다룹니다.
official
mapbox-geospatial-operations
mapbox
문제 유형, 정확도 요구 사항 및 성능 요구 사항에 따라 적합한 지리공간 도구를 선택하는 전문가 안내
official
mapbox-google-maps-migration
mapbox
Google Maps Platform에서 Mapbox GL JS로 전환하는 개발자를 위한 마이그레이션 가이드로, API 대응, 패턴 변환 및 주요 차이점을 다룹니다.
official
mapbox-ios-patterns
mapbox
iOS용 Mapbox Maps SDK의 공식 통합 패턴입니다. 설치, 마커 추가, 사용자 위치, 사용자 정의 데이터, 스타일, 카메라 제어 등을 다룹니다.
official
mapbox-location-grounding
mapbox
Mapbox MCP 도구를 구성하여 훈련 데이터 대신 실시간 데이터로부터 근거가 있고 인용된 위치 인식 응답을 생성합니다.
official
mapbox-maplibre-migration
mapbox
MapLibre GL JS에서 Mapbox GL JS로 마이그레이션하기 위한 가이드로, API 호환성, 토큰 설정, 스타일 구성 및 Mapbox 공식 기능의 이점을 다룹니다.
official