import React from 'react' import ReactDOM from 'react-dom' import MapboxGl from 'mapbox-gl/dist/mapbox-gl.js' import MapboxInspect from 'mapbox-gl-inspect' import FeatureLayerPopup from './FeatureLayerPopup' import FeaturePropertyPopup from './FeaturePropertyPopup' import validateColor from 'mapbox-gl-style-spec/lib/validate/validate_color' import colors from '../../config/colors' import style from '../../libs/style.js' import { colorHighlightedLayer } from '../../libs/highlight' import 'mapbox-gl/dist/mapbox-gl.css' import '../../mapboxgl.css' function renderLayerPopup(features) { var mountNode = document.createElement('div'); ReactDOM.render(, mountNode) return mountNode.innerHTML; } function renderPropertyPopup(features) { var mountNode = document.createElement('div'); ReactDOM.render(, mountNode) return mountNode.innerHTML; } function buildInspectStyle(originalMapStyle, coloredLayers, highlightedLayer) { const backgroundLayer = { "id": "background", "type": "background", "paint": { "background-color": colors.black, } } const layer = colorHighlightedLayer(highlightedLayer) if(layer) { coloredLayers.push(layer) } const sources = {} Object.keys(originalMapStyle.sources).forEach(sourceId => { const source = originalMapStyle.sources[sourceId] if(source.type !== 'raster') { sources[sourceId] = source } }) const inspectStyle = { ...originalMapStyle, sources: sources, layers: [backgroundLayer].concat(coloredLayers) } return inspectStyle } export default class MapboxGlMap extends React.Component { static propTypes = { onDataChange: React.PropTypes.func, mapStyle: React.PropTypes.object.isRequired, accessToken: React.PropTypes.string, style: React.PropTypes.object, inspectModeEnabled: React.PropTypes.bool.isRequired, highlightedLayer: React.PropTypes.object, } static defaultProps = { onMapLoaded: () => {}, onDataChange: () => {}, } constructor(props) { super(props) MapboxGl.accessToken = props.accessToken this.state = { map: null, inspect: null, isPopupOpen: false, popupX: 0, popupY: 0, } } componentWillReceiveProps(nextProps) { MapboxGl.accessToken = nextProps.accessToken if(!this.state.map) return if(!nextProps.inspectModeEnabled) { //Mapbox GL now does diffing natively so we don't need to calculate //the necessary operations ourselves! this.state.map.setStyle(nextProps.mapStyle, { diff: true}) } } componentDidUpdate(prevProps) { if(this.props.inspectModeEnabled !== prevProps.inspectModeEnabled) { this.state.inspect.toggleInspector() } if(this.props.inspectModeEnabled) { this.state.inspect.render() } } componentDidMount() { const map = new MapboxGl.Map({ container: this.container, style: this.props.mapStyle, hash: true, }) const nav = new MapboxGl.NavigationControl(); map.addControl(nav, 'top-right'); const inspect = new MapboxInspect({ popup: new MapboxGl.Popup({ closeButton: false, closeOnClick: false }), showMapPopup: true, showInspectButton: false, buildInspectStyle: (originalMapStyle, coloredLayers) => buildInspectStyle(originalMapStyle, coloredLayers, this.props.highlightedLayer), renderPopup: features => { if(this.props.inspectModeEnabled) { return renderPropertyPopup(features) } else { return renderLayerPopup(features) } } }) map.addControl(inspect) map.on("style.load", () => { this.setState({ map, inspect }); }) map.on("data", e => { if(e.dataType !== 'tile') return this.props.onDataChange({ map: this.state.map }) }) } render() { return
this.container = x} >
} }