maputnik/src/components/App.jsx

650 lines
18 KiB
React
Raw Normal View History

import autoBind from 'react-autobind';
import React from 'react'
import cloneDeep from 'lodash.clonedeep'
import clamp from 'lodash.clamp'
import {arrayMove} from 'react-sortable-hoc'
import url from 'url'
2016-12-20 16:37:35 +01:00
import MapboxGlMap from './map/MapboxGlMap'
import OpenLayersMap from './map/OpenLayersMap'
2016-12-20 16:37:35 +01:00
import LayerList from './layers/LayerList'
import LayerEditor from './layers/LayerEditor'
import Toolbar from './Toolbar'
2016-12-22 16:39:09 +01:00
import AppLayout from './AppLayout'
2016-12-31 14:32:04 +01:00
import MessagePanel from './MessagePanel'
import SettingsModal from './modals/SettingsModal'
import ExportModal from './modals/ExportModal'
import SourcesModal from './modals/SourcesModal'
import OpenModal from './modals/OpenModal'
import ShortcutsModal from './modals/ShortcutsModal'
import SurveyModal from './modals/SurveyModal'
import DebugModal from './modals/DebugModal'
import { downloadGlyphsMetadata, downloadSpriteMetadata } from '../libs/metadata'
2018-10-06 22:05:33 +02:00
import {latest, validate} from '@mapbox/mapbox-gl-style-spec'
import style from '../libs/style'
2018-10-25 19:37:39 +02:00
import { initialStyleUrl, loadStyleUrl, removeStyleQuerystring } from '../libs/urlopen'
2016-12-31 14:32:04 +01:00
import { undoMessages, redoMessages } from '../libs/diffmessage'
2018-08-23 04:37:05 +02:00
import { StyleStore } from '../libs/stylestore'
2016-12-20 16:37:35 +01:00
import { ApiStyleStore } from '../libs/apistore'
2016-12-23 17:17:02 +01:00
import { RevisionStore } from '../libs/revisions'
2016-12-20 16:37:35 +01:00
import LayerWatcher from '../libs/layerwatcher'
import tokens from '../config/tokens.json'
import isEqual from 'lodash.isequal'
import Debug from '../libs/debug'
import queryUtil from '../libs/query-util'
2016-12-20 11:44:22 +01:00
2018-01-25 20:12:04 +01:00
import MapboxGl from 'mapbox-gl'
// Similar functionality as <https://github.com/mapbox/mapbox-gl-js/blob/7e30aadf5177486c2cfa14fe1790c60e217b5e56/src/util/mapbox.js>
function normalizeSourceURL (url, apiToken="") {
const matches = url.match(/^mapbox:\/\/(.*)/);
if (matches) {
// mapbox://mapbox.mapbox-streets-v7
return `https://api.mapbox.com/v4/${matches[1]}.json?secure&access_token=${apiToken}`
}
else {
return url;
}
}
2018-01-25 20:12:04 +01:00
2018-11-27 13:29:47 +01:00
function setFetchAccessToken(url, mapStyle) {
const matchesTilehosting = url.match(/\.tilehosting\.com/);
2019-04-03 09:51:43 +02:00
const matchesMaptiler = url.match(/\.maptiler\.com/);
2018-11-27 13:29:47 +01:00
const matchesThunderforest = url.match(/\.thunderforest\.com/);
if (matchesTilehosting || matchesMaptiler) {
2018-11-27 13:29:47 +01:00
const accessToken = style.getAccessToken("openmaptiles", mapStyle, {allowFallback: true})
if (accessToken) {
return url.replace('{key}', accessToken)
}
}
else if (matchesThunderforest) {
const accessToken = style.getAccessToken("thunderforest", mapStyle, {allowFallback: true})
if (accessToken) {
return url.replace('{key}', accessToken)
}
}
else {
return url;
}
}
2018-01-25 20:12:04 +01:00
function updateRootSpec(spec, fieldName, newValues) {
return {
...spec,
$root: {
...spec.$root,
[fieldName]: {
...spec.$root[fieldName],
values: newValues
}
}
}
}
2015-06-15 15:21:19 +02:00
export default class App extends React.Component {
constructor(props) {
super(props)
autoBind(this);
2016-12-23 17:17:02 +01:00
this.revisionStore = new RevisionStore()
2017-01-01 14:49:32 +01:00
this.styleStore = new ApiStyleStore({
onLocalStyleChange: mapStyle => this.onStyleChanged(mapStyle, false)
})
2016-12-29 15:22:47 +01:00
const shortcuts = [
{
key: "?",
handler: () => {
this.toggleModal("shortcuts");
2018-05-28 13:03:47 +02:00
}
},
{
key: "o",
handler: () => {
this.toggleModal("open");
2018-05-28 13:03:47 +02:00
}
},
{
key: "e",
handler: () => {
this.toggleModal("export");
2018-05-28 13:03:47 +02:00
}
},
{
key: "d",
handler: () => {
this.toggleModal("sources");
2018-05-28 13:03:47 +02:00
}
},
{
key: "s",
handler: () => {
this.toggleModal("settings");
2018-05-28 13:03:47 +02:00
}
},
{
key: "i",
handler: () => {
this.setMapState(
this.state.mapState === "map" ? "inspect" : "map"
);
2018-05-28 13:03:47 +02:00
}
},
{
key: "m",
handler: () => {
document.querySelector(".mapboxgl-canvas").focus();
2018-05-28 13:03:47 +02:00
}
},
{
key: "!",
handler: () => {
this.toggleModal("debug");
}
},
]
document.body.addEventListener("keyup", (e) => {
if(e.key === "Escape") {
e.target.blur();
document.body.focus();
}
else if(this.state.isOpen.shortcuts || document.activeElement === document.body) {
const shortcut = shortcuts.find((shortcut) => {
return (shortcut.key === e.key)
})
if(shortcut) {
this.setModal("shortcuts", false);
shortcut.handler(e);
2018-05-28 13:03:47 +02:00
}
}
})
2017-01-05 19:34:32 +01:00
const styleUrl = initialStyleUrl()
2018-10-29 17:35:12 +01:00
if(styleUrl && window.confirm("Load style from URL: " + styleUrl + " and discard current changes?")) {
this.styleStore = new StyleStore()
loadStyleUrl(styleUrl, mapStyle => this.onStyleChanged(mapStyle))
2018-10-25 19:37:39 +02:00
removeStyleQuerystring()
2017-01-05 19:34:32 +01:00
} else {
2018-10-29 17:35:12 +01:00
if(styleUrl) {
removeStyleQuerystring()
}
2017-01-05 19:34:32 +01:00
this.styleStore.init(err => {
if(err) {
console.log('Falling back to local storage for storing styles')
this.styleStore = new StyleStore()
}
this.styleStore.latestStyle(mapStyle => this.onStyleChanged(mapStyle))
if(Debug.enabled()) {
Debug.set("maputnik", "styleStore", this.styleStore);
Debug.set("maputnik", "revisionStore", this.revisionStore);
}
2017-01-05 19:34:32 +01:00
})
}
if(Debug.enabled()) {
Debug.set("maputnik", "revisionStore", this.revisionStore);
Debug.set("maputnik", "styleStore", this.styleStore);
}
const queryObj = url.parse(window.location.href, true).query;
this.state = {
errors: [],
2016-12-31 14:32:04 +01:00
infos: [],
2016-12-17 15:44:42 +01:00
mapStyle: style.emptyStyle,
2016-12-20 16:08:49 +01:00
selectedLayerIndex: 0,
2016-12-29 15:22:47 +01:00
sources: {},
vectorLayers: {},
2018-09-23 15:39:02 +02:00
mapState: "map",
2018-10-06 22:05:33 +02:00
spec: latest,
isOpen: {
settings: false,
sources: false,
open: false,
shortcuts: false,
export: false,
survey: localStorage.hasOwnProperty('survey') ? false : true,
debug: false,
},
mapboxGlDebugOptions: {
showTileBoundaries: false,
showCollisionBoxes: false,
showOverdrawInspector: false,
},
}
2016-12-29 15:22:47 +01:00
this.layerWatcher = new LayerWatcher({
onVectorLayersChange: v => this.setState({ vectorLayers: v })
})
}
handleKeyPress(e) {
if(navigator.platform.toUpperCase().indexOf('MAC') >= 0) {
if(e.metaKey && e.shiftKey && e.keyCode === 90) {
this.onRedo(e);
}
else if(e.metaKey && e.keyCode === 90) {
this.onUndo(e);
}
}
else {
if(e.ctrlKey && e.keyCode === 90) {
this.onUndo(e);
}
else if(e.ctrlKey && e.keyCode === 89) {
this.onRedo(e);
}
}
}
2016-12-23 17:17:02 +01:00
componentDidMount() {
window.addEventListener("keydown", this.handleKeyPress);
2016-12-23 17:17:02 +01:00
}
componentWillUnmount() {
window.removeEventListener("keydown", this.handleKeyPress);
}
2016-12-22 21:06:32 +01:00
saveStyle(snapshotStyle) {
this.styleStore.save(snapshotStyle)
}
updateFonts(urlTemplate) {
const metadata = this.state.mapStyle.metadata || {}
const accessToken = metadata['maputnik:openmaptiles_access_token'] || tokens.openmaptiles
let glyphUrl = (typeof urlTemplate === 'string')? urlTemplate.replace('{key}', accessToken): urlTemplate;
downloadGlyphsMetadata(glyphUrl, fonts => {
this.setState({ spec: updateRootSpec(this.state.spec, 'glyphs', fonts)})
})
}
updateIcons(baseUrl) {
downloadSpriteMetadata(baseUrl, icons => {
this.setState({ spec: updateRootSpec(this.state.spec, 'sprite', icons)})
})
}
onStyleChanged = (newStyle, save=true) => {
2018-10-06 22:05:33 +02:00
const errors = validate(newStyle, latest)
2016-12-29 22:00:49 +01:00
if(errors.length === 0) {
if(newStyle.glyphs !== this.state.mapStyle.glyphs) {
this.updateFonts(newStyle.glyphs)
}
if(newStyle.sprite !== this.state.mapStyle.sprite) {
this.updateIcons(newStyle.sprite)
}
2016-12-29 22:00:49 +01:00
this.revisionStore.addRevision(newStyle)
2017-01-01 14:49:32 +01:00
if(save) this.saveStyle(newStyle)
this.setState({
mapStyle: newStyle,
errors: [],
})
2016-12-29 22:00:49 +01:00
} else {
this.setState({
errors: errors.map(err => err.message)
})
2016-12-29 22:00:49 +01:00
}
this.fetchSources();
}
onUndo = () => {
2016-12-23 17:17:02 +01:00
const activeStyle = this.revisionStore.undo()
2016-12-31 14:32:04 +01:00
const messages = undoMessages(this.state.mapStyle, activeStyle)
2016-12-23 17:17:02 +01:00
this.saveStyle(activeStyle)
2016-12-31 14:32:04 +01:00
this.setState({
mapStyle: activeStyle,
infos: messages,
})
2016-12-23 17:17:02 +01:00
}
onRedo = () => {
2016-12-23 17:17:02 +01:00
const activeStyle = this.revisionStore.redo()
2016-12-31 14:32:04 +01:00
const messages = redoMessages(this.state.mapStyle, activeStyle)
2016-12-23 17:17:02 +01:00
this.saveStyle(activeStyle)
2016-12-31 14:32:04 +01:00
this.setState({
mapStyle: activeStyle,
infos: messages,
})
2016-12-23 17:17:02 +01:00
}
onMoveLayer = (move) => {
let { oldIndex, newIndex } = move;
let layers = this.state.mapStyle.layers;
oldIndex = clamp(oldIndex, 0, layers.length-1);
newIndex = clamp(newIndex, 0, layers.length-1);
if(oldIndex === newIndex) return;
if (oldIndex === this.state.selectedLayerIndex) {
this.setState({
selectedLayerIndex: newIndex
});
}
layers = layers.slice(0);
layers = arrayMove(layers, oldIndex, newIndex);
this.onLayersChange(layers);
}
onLayersChange = (changedLayers) => {
2016-12-20 16:08:49 +01:00
const changedStyle = {
...this.state.mapStyle,
layers: changedLayers
2016-12-20 16:08:49 +01:00
}
2016-12-22 21:06:32 +01:00
this.onStyleChanged(changedStyle)
2016-12-17 21:25:00 +01:00
}
onLayerDestroy = (layerId) => {
let layers = this.state.mapStyle.layers;
const remainingLayers = layers.slice(0);
const idx = style.indexOfLayer(remainingLayers, layerId)
remainingLayers.splice(idx, 1);
this.onLayersChange(remainingLayers);
}
onLayerCopy = (layerId) => {
let layers = this.state.mapStyle.layers;
const changedLayers = layers.slice(0)
const idx = style.indexOfLayer(changedLayers, layerId)
const clonedLayer = cloneDeep(changedLayers[idx])
clonedLayer.id = clonedLayer.id + "-copy"
changedLayers.splice(idx, 0, clonedLayer)
this.onLayersChange(changedLayers)
}
onLayerVisibilityToggle = (layerId) => {
let layers = this.state.mapStyle.layers;
const changedLayers = layers.slice(0)
const idx = style.indexOfLayer(changedLayers, layerId)
const layer = { ...changedLayers[idx] }
const changedLayout = 'layout' in layer ? {...layer.layout} : {}
changedLayout.visibility = changedLayout.visibility === 'none' ? 'visible' : 'none'
layer.layout = changedLayout
changedLayers[idx] = layer
this.onLayersChange(changedLayers)
}
onLayerIdChange = (oldId, newId) => {
2016-12-22 15:49:36 +01:00
const changedLayers = this.state.mapStyle.layers.slice(0)
const idx = style.indexOfLayer(changedLayers, oldId)
changedLayers[idx] = {
...changedLayers[idx],
id: newId
}
this.onLayersChange(changedLayers)
}
onLayerChanged = (layer) => {
2016-12-20 20:50:08 +01:00
const changedLayers = this.state.mapStyle.layers.slice(0)
const idx = style.indexOfLayer(changedLayers, layer.id)
changedLayers[idx] = layer
this.onLayersChange(changedLayers)
2016-12-17 21:25:00 +01:00
}
setMapState = (newState) => {
2017-01-08 22:03:21 +01:00
this.setState({
2018-06-18 21:28:24 +02:00
mapState: newState
2017-01-08 22:03:21 +01:00
})
}
fetchSources() {
const sourceList = {...this.state.sources};
for(let [key, val] of Object.entries(this.state.mapStyle.sources)) {
if(sourceList.hasOwnProperty(key)) {
continue;
}
sourceList[key] = {
type: val.type,
layers: []
};
if(!this.state.sources.hasOwnProperty(key) && val.type === "vector" && val.hasOwnProperty("url")) {
let url = val.url;
try {
2018-07-12 13:33:40 +02:00
url = normalizeSourceURL(url, MapboxGl.accessToken);
} catch(err) {
console.warn("Failed to normalizeSourceURL: ", err);
}
2018-01-25 20:12:04 +01:00
2018-11-27 13:29:47 +01:00
try {
url = setFetchAccessToken(url, this.state.mapStyle)
} catch(err) {
console.warn("Failed to setFetchAccessToken: ", err);
}
fetch(url, {
mode: 'cors',
})
.then((response) => {
return response.json();
})
.then((json) => {
if(!json.hasOwnProperty("vector_layers")) {
return;
}
// Create new objects before setState
const sources = Object.assign({}, this.state.sources);
for(let layer of json.vector_layers) {
sources[key].layers.push(layer.id)
}
console.debug("Updating source: "+key);
this.setState({
sources: sources
});
})
.catch((err) => {
console.error("Failed to process sources for '%s'", url, err);
})
}
}
if(!isEqual(this.state.sources, sourceList)) {
console.debug("Setting sources");
this.setState({
sources: sourceList
})
}
}
_getRenderer () {
const metadata = this.state.mapStyle.metadata || {};
return metadata['maputnik:renderer'] || 'mbgljs';
}
2016-12-17 14:53:16 +01:00
mapRenderer() {
2016-12-16 15:14:20 +01:00
const mapProps = {
mapStyle: style.replaceAccessTokens(this.state.mapStyle, {allowFallback: true}),
options: this.state.mapboxGlDebugOptions,
2016-12-24 14:42:57 +01:00
onDataChange: (e) => {
this.layerWatcher.analyzeMap(e.map)
this.fetchSources();
2016-12-24 15:14:31 +01:00
},
2016-12-16 15:14:20 +01:00
}
2016-12-20 16:08:49 +01:00
const renderer = this._getRenderer();
2016-12-22 18:08:42 +01:00
let mapElement;
2018-09-24 22:01:37 +02:00
// Check if OL code has been loaded?
if(renderer === 'ol') {
mapElement = <OpenLayersMap
{...mapProps}
/>
2016-12-17 14:53:16 +01:00
} else {
mapElement = <MapboxGlMap {...mapProps}
2018-06-18 21:28:24 +02:00
inspectModeEnabled={this.state.mapState === "inspect"}
2018-01-08 22:18:30 +01:00
highlightedLayer={this.state.mapStyle.layers[this.state.selectedLayerIndex]}
onLayerSelect={this.onLayerSelect} />
2016-12-17 14:53:16 +01:00
}
let filterName;
2018-06-18 21:28:24 +02:00
if(this.state.mapState.match(/^filter-/)) {
filterName = this.state.mapState.replace(/^filter-/, "");
2018-07-17 21:40:23 +02:00
}
const elementStyle = {};
if (filterName) {
elementStyle.filter = `url('#${filterName}')`;
};
return <div style={elementStyle} className="maputnik-map__container">
{mapElement}
</div>
2016-12-17 14:53:16 +01:00
}
onLayerSelect = (layerId) => {
const idx = style.indexOfLayer(this.state.mapStyle.layers, layerId)
this.setState({ selectedLayerIndex: idx })
2016-12-17 16:09:37 +01:00
}
setModal(modalName, value) {
if(modalName === 'survey' && value === false) {
localStorage.setItem('survey', '');
}
this.setState({
isOpen: {
...this.state.isOpen,
[modalName]: value
}
})
}
toggleModal(modalName) {
this.setModal(modalName, !this.state.isOpen[modalName]);
}
onChangeMaboxGlDebug = (key, value) => {
this.setState({
mapboxGlDebugOptions: {
...this.state.mapboxGlDebugOptions,
[key]: value,
}
});
}
2016-12-17 14:53:16 +01:00
render() {
2016-12-20 16:08:49 +01:00
const layers = this.state.mapStyle.layers || []
const selectedLayer = layers.length > 0 ? layers[this.state.selectedLayerIndex] : null
2016-12-28 17:08:42 +01:00
const metadata = this.state.mapStyle.metadata || {}
2016-12-20 16:37:35 +01:00
const toolbar = <Toolbar
2018-06-18 21:28:24 +02:00
mapState={this.state.mapState}
2016-12-20 16:37:35 +01:00
mapStyle={this.state.mapStyle}
2018-06-18 21:28:24 +02:00
inspectModeEnabled={this.state.mapState === "inspect"}
2016-12-29 15:22:47 +01:00
sources={this.state.sources}
onStyleChanged={this.onStyleChanged}
onStyleOpen={this.onStyleChanged}
onSetMapState={this.setMapState}
onToggleModal={this.toggleModal.bind(this)}
2016-12-20 16:37:35 +01:00
/>
const layerList = <LayerList
onMoveLayer={this.onMoveLayer}
onLayerDestroy={this.onLayerDestroy}
onLayerCopy={this.onLayerCopy}
onLayerVisibilityToggle={this.onLayerVisibilityToggle}
onLayersChange={this.onLayersChange}
onLayerSelect={this.onLayerSelect}
2016-12-20 19:20:56 +01:00
selectedLayerIndex={this.state.selectedLayerIndex}
2016-12-20 16:37:35 +01:00
layers={layers}
2017-01-11 15:59:51 +01:00
sources={this.state.sources}
2016-12-20 16:37:35 +01:00
/>
const layerEditor = selectedLayer ? <LayerEditor
layer={selectedLayer}
layerIndex={this.state.selectedLayerIndex}
isFirstLayer={this.state.selectedLayerIndex < 1}
isLastLayer={this.state.selectedLayerIndex === this.state.mapStyle.layers.length-1}
2016-12-29 15:22:47 +01:00
sources={this.state.sources}
vectorLayers={this.state.vectorLayers}
spec={this.state.spec}
onMoveLayer={this.onMoveLayer}
onLayerChanged={this.onLayerChanged}
onLayerDestroy={this.onLayerDestroy}
onLayerCopy={this.onLayerCopy}
onLayerVisibilityToggle={this.onLayerVisibilityToggle}
onLayerIdChange={this.onLayerIdChange}
2016-12-20 16:37:35 +01:00
/> : null
2016-12-31 14:32:04 +01:00
const bottomPanel = (this.state.errors.length + this.state.infos.length) > 0 ? <MessagePanel
errors={this.state.errors}
infos={this.state.infos}
/> : null
const modals = <div>
<DebugModal
renderer={this._getRenderer()}
mapboxGlDebugOptions={this.state.mapboxGlDebugOptions}
onChangeMaboxGlDebug={this.onChangeMaboxGlDebug}
isOpen={this.state.isOpen.debug}
onOpenToggle={this.toggleModal.bind(this, 'debug')}
/>
<ShortcutsModal
ref={(el) => this.shortcutEl = el}
isOpen={this.state.isOpen.shortcuts}
onOpenToggle={this.toggleModal.bind(this, 'shortcuts')}
/>
<SettingsModal
mapStyle={this.state.mapStyle}
onStyleChanged={this.onStyleChanged}
isOpen={this.state.isOpen.settings}
onOpenToggle={this.toggleModal.bind(this, 'settings')}
/>
<ExportModal
mapStyle={this.state.mapStyle}
onStyleChanged={this.onStyleChanged}
isOpen={this.state.isOpen.export}
onOpenToggle={this.toggleModal.bind(this, 'export')}
/>
<OpenModal
isOpen={this.state.isOpen.open}
onStyleOpen={this.onStyleChanged}
onOpenToggle={this.toggleModal.bind(this, 'open')}
/>
<SourcesModal
mapStyle={this.state.mapStyle}
onStyleChanged={this.onStyleChanged}
isOpen={this.state.isOpen.sources}
onOpenToggle={this.toggleModal.bind(this, 'sources')}
/>
<SurveyModal
isOpen={this.state.isOpen.survey}
onOpenToggle={this.toggleModal.bind(this, 'survey')}
/>
</div>
2016-12-22 16:39:09 +01:00
return <AppLayout
2016-12-20 16:37:35 +01:00
toolbar={toolbar}
layerList={layerList}
layerEditor={layerEditor}
map={this.mapRenderer()}
bottom={bottomPanel}
modals={modals}
2016-12-20 16:37:35 +01:00
/>
}
2016-09-08 19:47:29 +02:00
}