import React from 'react' import { saveAs } from 'file-saver' import Mousetrap from 'mousetrap' import InspectionMap from './map/InspectionMap' import MapboxGlMap from './map/MapboxGlMap' import OpenLayers3Map from './map/OpenLayers3Map' import LayerList from './layers/LayerList' import LayerEditor from './layers/LayerEditor' import Toolbar from './Toolbar' import AppLayout from './AppLayout' import MessagePanel from './MessagePanel' import GlSpec from 'mapbox-gl-style-spec/reference/latest.js' import validateStyleMin from 'mapbox-gl-style-spec/lib/validate_style.min' import style from '../libs/style.js' import { undoMessages, redoMessages } from '../libs/diffmessage' import { loadDefaultStyle, StyleStore } from '../libs/stylestore' import { ApiStyleStore } from '../libs/apistore' import { RevisionStore } from '../libs/revisions' import LayerWatcher from '../libs/layerwatcher' export default class App extends React.Component { constructor(props) { super(props) this.revisionStore = new RevisionStore() this.styleStore = new ApiStyleStore({ onLocalStyleChange: mapStyle => this.onStyleChanged(mapStyle, false) }) 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)) }) this.state = { errors: [], infos: [], mapStyle: style.emptyStyle, selectedLayerIndex: 0, sources: {}, vectorLayers: {}, } this.layerWatcher = new LayerWatcher({ onSourcesChange: v => this.setState({ sources: v }), onVectorLayersChange: v => this.setState({ vectorLayers: v }) }) } componentDidMount() { Mousetrap.bind(['ctrl+z'], this.onUndo.bind(this)); Mousetrap.bind(['ctrl+y'], this.onRedo.bind(this)); } componentWillUnmount() { Mousetrap.unbind(['ctrl+z'], this.onUndo.bind(this)); Mousetrap.unbind(['ctrl+y'], this.onRedo.bind(this)); } onReset() { this.styleStore.purge() loadDefaultStyle(mapStyle => this.onStyleOpen(mapStyle)) } onStyleDownload() { const mapStyle = this.state.mapStyle const blob = new Blob([JSON.stringify(mapStyle, null, 4)], {type: "application/json;charset=utf-8"}); saveAs(blob, mapStyle.id + ".json"); } saveStyle(snapshotStyle) { snapshotStyle.modified = new Date().toJSON() this.styleStore.save(snapshotStyle) } onStyleChanged(newStyle, save=true) { const errors = validateStyleMin(newStyle, GlSpec) if(errors.length === 0) { this.revisionStore.addRevision(newStyle) if(save) this.saveStyle(newStyle) this.setState({ mapStyle: newStyle, errors: [], }) } else { this.setState({ errors: errors.map(err => err.message) }) } } onUndo() { const activeStyle = this.revisionStore.undo() const messages = undoMessages(this.state.mapStyle, activeStyle) this.saveStyle(activeStyle) this.setState({ mapStyle: activeStyle, infos: messages, }) } onRedo() { const activeStyle = this.revisionStore.redo() const messages = redoMessages(this.state.mapStyle, activeStyle) this.saveStyle(activeStyle) this.setState({ mapStyle: activeStyle, infos: messages, }) } onLayersChange(changedLayers) { const changedStyle = { ...this.state.mapStyle, layers: changedLayers } this.onStyleChanged(changedStyle) } onLayerIdChange(oldId, newId) { 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) { const changedLayers = this.state.mapStyle.layers.slice(0) const idx = style.indexOfLayer(changedLayers, layer.id) changedLayers[idx] = layer this.onLayersChange(changedLayers) } mapRenderer() { const metadata = this.state.mapStyle.metadata || {} const mapProps = { mapStyle: this.state.mapStyle, accessToken: metadata['maputnik:access_token'], onDataChange: (e) => { this.layerWatcher.analyzeMap(e.map) }, //TODO: This would actually belong to the layout component style:{ top: 40, //left: 500, } } const renderer = metadata['maputnik:renderer'] || 'mbgljs' // Check if OL3 code has been loaded? if(renderer === 'ol3') { return } else if(renderer === 'inspection') { return } else { return } } onLayerSelect(layerId) { const idx = style.indexOfLayer(this.state.mapStyle.layers, layerId) this.setState({ selectedLayerIndex: idx }) } render() { const layers = this.state.mapStyle.layers || [] const selectedLayer = layers.length > 0 ? layers[this.state.selectedLayerIndex] : null const metadata = this.state.mapStyle.metadata || {} const toolbar = const layerList = const layerEditor = selectedLayer ? : null const bottomPanel = (this.state.errors.length + this.state.infos.length) > 0 ? : null return } }