Add inspection map

This commit is contained in:
Lukas Martinelli 2016-12-24 14:42:57 +01:00
parent 04eab70e27
commit f332d517f3
9 changed files with 232 additions and 48 deletions

1
.gitignore vendored
View file

@ -1,6 +1,7 @@
# Logs # Logs
logs logs
*.log *.log
*.swp
# Runtime data # Runtime data
pids pids

View file

@ -2,6 +2,7 @@ import React from 'react'
import { saveAs } from 'file-saver' import { saveAs } from 'file-saver'
import Mousetrap from 'mousetrap' import Mousetrap from 'mousetrap'
import InspectionMap from './map/InspectionMap'
import MapboxGlMap from './map/MapboxGlMap' import MapboxGlMap from './map/MapboxGlMap'
import OpenLayers3Map from './map/OpenLayers3Map' import OpenLayers3Map from './map/OpenLayers3Map'
import LayerList from './layers/LayerList' import LayerList from './layers/LayerList'
@ -121,8 +122,8 @@ export default class App extends React.Component {
const mapProps = { const mapProps = {
mapStyle: this.state.mapStyle, mapStyle: this.state.mapStyle,
accessToken: this.state.accessToken, accessToken: this.state.accessToken,
onMapLoaded: (map) => { onDataChange: (e) => {
this.layerWatcher.map = map this.layerWatcher.analyzeMap(e.map)
} }
} }
@ -132,6 +133,8 @@ export default class App extends React.Component {
// Check if OL3 code has been loaded? // Check if OL3 code has been loaded?
if(renderer === 'ol3') { if(renderer === 'ol3') {
return <OpenLayers3Map {...mapProps} /> return <OpenLayers3Map {...mapProps} />
} else if(renderer === 'inspection') {
return <InspectionMap {...mapProps} sources={this.layerWatcher.sources} />
} else { } else {
return <MapboxGlMap {...mapProps} /> return <MapboxGlMap {...mapProps} />
} }

View file

@ -0,0 +1,79 @@
import React from 'react'
import MapboxGl from 'mapbox-gl'
import validateColor from 'mapbox-gl-style-spec/lib/validate/validate_color'
import colors from '../../config/colors'
import style from '../../libs/style'
import { generateColoredLayers } from '../../libs/stylegen'
function convertInspectStyle(mapStyle, sources) {
const newStyle = {
...mapStyle,
layers: [
{
"id": "background",
"type": "background",
"paint": {
"background-color": colors.black,
}
},
...generateColoredLayers(sources),
]
}
return newStyle
}
export default class InspectionMap extends React.Component {
static propTypes = {
onDataChange: React.PropTypes.func,
sources: React.PropTypes.object,
originalStyle: React.PropTypes.object,
}
static defaultProps = {
onMapLoaded: () => {},
onTileLoaded: () => {},
}
constructor(props) {
super(props)
this.state = { map: null }
}
componentWillReceiveProps(nextProps) {
if(!this.state.map) return
this.state.map.setStyle(convertInspectStyle(nextProps.mapStyle, this.props.sources), { diff: true})
}
componentDidMount() {
MapboxGl.accessToken = this.props.accessToken
const map = new MapboxGl.Map({
container: this.container,
style: convertInspectStyle(this.props.mapStyle, this.props.sources),
})
map.on("style.load", () => {
this.setState({ map });
})
map.on("data", e => {
if(e.dataType !== 'tile') return
this.props.onDataChange({
map: this.state.map
})
})
}
render() {
return <div
ref={x => this.container = x}
style={{
position: "fixed",
top: 0,
bottom: 0,
height: "100%",
width: "100%",
}}></div>
}
}

View file

@ -1,20 +0,0 @@
import React from 'react'
export default class Map extends React.Component {
static propTypes = {
mapStyle: React.PropTypes.object.isRequired,
accessToken: React.PropTypes.string,
}
render() {
return <div
ref={x => this.container = x}
style={{
position: "fixed",
top: 0,
bottom: 0,
height: "100%",
width: "100%",
}}></div>
}
}

View file

@ -2,22 +2,25 @@ import React from 'react'
import MapboxGl from 'mapbox-gl' import MapboxGl from 'mapbox-gl'
import validateColor from 'mapbox-gl-style-spec/lib/validate/validate_color' import validateColor from 'mapbox-gl-style-spec/lib/validate/validate_color'
import Map from './Map.jsx'
import style from '../../libs/style.js' import style from '../../libs/style.js'
export default class MapboxGlMap extends Map { export default class MapboxGlMap extends React.Component {
static propTypes = { static propTypes = {
onMapLoaded: React.PropTypes.func, onDataChange: React.PropTypes.func,
mapStyle: React.PropTypes.object.isRequired,
accessToken: React.PropTypes.string,
} }
static defaultProps = { static defaultProps = {
onMapLoaded: () => {} onMapLoaded: () => {},
onDataChange: () => {},
} }
constructor(props) { constructor(props) {
super(props) super(props)
this.state = { map: null } this.state = { map: null }
} }
componentWillReceiveProps(nextProps) { componentWillReceiveProps(nextProps) {
if(!this.state.map) return if(!this.state.map) return
@ -32,11 +35,29 @@ export default class MapboxGlMap extends Map {
const map = new MapboxGl.Map({ const map = new MapboxGl.Map({
container: this.container, container: this.container,
style: this.props.mapStyle, style: this.props.mapStyle,
}); })
map.on("style.load", (...args) => { map.on("style.load", () => {
this.props.onMapLoaded(map)
this.setState({ map }); this.setState({ map });
}); })
map.on("data", e => {
if(e.dataType !== 'tile') return
this.props.onDataChange({
map: this.state.map
})
})
}
render() {
return <div
ref={x => this.container = x}
style={{
position: "fixed",
top: 0,
bottom: 0,
height: "100%",
width: "100%",
}}></div>
} }
} }

View file

@ -1,10 +1,16 @@
import React from 'react' import React from 'react'
import Map from './Map'
import style from '../../libs/style.js' import style from '../../libs/style.js'
class OpenLayers3Map extends Map { class OpenLayers3Map extends React.Component {
constructor(props) { static propTypes = {
super(props) onDataChange: React.PropTypes.func,
mapStyle: React.PropTypes.object.isRequired,
accessToken: React.PropTypes.string,
}
static defaultProps = {
onMapLoaded: () => {},
onDataChange: () => {},
} }
componentWillReceiveProps(nextProps) { componentWillReceiveProps(nextProps) {
@ -63,6 +69,18 @@ class OpenLayers3Map extends Map {
this.setState({ map }); this.setState({ map });
}) })
} }
render() {
return <div
ref={x => this.container = x}
style={{
position: "fixed",
top: 0,
bottom: 0,
height: "100%",
width: "100%",
}}></div>
}
} }
export default OpenLayers3Map export default OpenLayers3Map

View file

@ -72,7 +72,8 @@ class SettingsModal extends React.Component {
<SelectInput {...inputProps} <SelectInput {...inputProps}
options={[ options={[
['mbgljs', 'MapboxGL JS'], ['mbgljs', 'MapboxGL JS'],
['ol3', 'Open Layers 3'] ['ol3', 'Open Layers 3'],
['inspection', 'Inspection Mode'],
]} ]}
value={(this.props.mapStyle.metadata || {})['maputnik:renderer'] || 'mbgljs'} value={(this.props.mapStyle.metadata || {})['maputnik:renderer'] || 'mbgljs'}
onChange={this.onRendererChange.bind(this)} onChange={this.onRendererChange.bind(this)}

View file

@ -6,7 +6,6 @@ export default class LayerWatcher {
constructor() { constructor() {
this._sources = {} this._sources = {}
this._vectorLayers = {} this._vectorLayers = {}
this._map= null
// Since we scan over all features we want to avoid this as much as // Since we scan over all features we want to avoid this as much as
// possible and only do it after a batch of data has loaded because // possible and only do it after a batch of data has loaded because
@ -14,27 +13,21 @@ export default class LayerWatcher {
this.throttledAnalyzeVectorLayerFields = throttle(this.analyzeVectorLayerFields, 5000) this.throttledAnalyzeVectorLayerFields = throttle(this.analyzeVectorLayerFields, 5000)
} }
/** Set the map as soon as the map is initialized */ analyzeMap(map) {
set map(m) { Object.keys(map.style.sourceCaches).forEach(sourceId => {
this._map = m
//TODO: At some point we need to unsubscribe when new map is set
this._map.on('data', (e) => {
if(e.dataType !== 'tile') return
//NOTE: This heavily depends on the internal API of Mapbox GL //NOTE: This heavily depends on the internal API of Mapbox GL
//so this breaks between Mapbox GL JS releases //so this breaks between Mapbox GL JS releases
this._sources[e.sourceId] = e.style.sourceCaches[e.sourceId]._source.vectorLayerIds this._sources[sourceId] = map.style.sourceCaches[sourceId]._source.vectorLayerIds
this.throttledAnalyzeVectorLayerFields()
}) })
this.throttledAnalyzeVectorLayerFields(map)
} }
analyzeVectorLayerFields() { analyzeVectorLayerFields(map) {
Object.keys(this._sources).forEach(sourceId => { Object.keys(this._sources).forEach(sourceId => {
this._sources[sourceId].forEach(vectorLayerId => { this._sources[sourceId].forEach(vectorLayerId => {
const knownProperties = this._vectorLayers[vectorLayerId] || {} const knownProperties = this._vectorLayers[vectorLayerId] || {}
const params = { sourceLayer: vectorLayerId } const params = { sourceLayer: vectorLayerId }
this._map.querySourceFeatures(sourceId, params).forEach(feature => { map.querySourceFeatures(sourceId, params).forEach(feature => {
Object.keys(feature.properties).forEach(propertyName => { Object.keys(feature.properties).forEach(propertyName => {
const knownPropertyValues = knownProperties[propertyName] || {} const knownPropertyValues = knownProperties[propertyName] || {}
knownPropertyValues[feature.properties[propertyName]] = {} knownPropertyValues[feature.properties[propertyName]] = {}

88
src/libs/stylegen.js Normal file
View file

@ -0,0 +1,88 @@
import randomColor from 'randomcolor'
import Color from 'color'
function assignVectorLayerColor(layerId) {
let hue = null
if(/water|ocean|lake|sea|river/.test(layerId)) {
hue = 'blue'
}
if(/road|highway|transport/.test(layerId)) {
hue = 'orange'
}
if(/building/.test(layerId)) {
hue = 'yellow'
}
if(/wood|forest|park|landcover|landuse/.test(layerId)) {
hue = 'green'
}
return randomColor({
luminosity: 'bright',
hue: hue,
seed: layerId,
})
}
function circleLayer(source, vectorLayer, color) {
return {
id: vectorLayer + Math.random(),
source: source,
'source-layer': vectorLayer,
interactive: true,
type: 'circle',
paint: {
'circle-color': color,
'circle-radius': 2,
},
filter: ["==", "$type", "Point"]
}
}
function polygonLayer(source, vectorLayer, color) {
return {
id: vectorLayer + Math.random(),
source: source,
'source-layer': vectorLayer,
interactive: true,
type: 'fill',
paint: {
'fill-color': Color(color).alpha(0.15).string(),
'fill-antialias': true,
'fill-outline-color': color,
},
filter: ["==", "$type", "Polygon"]
}
}
function lineLayer(source, vectorLayer, color) {
return {
id: vectorLayer + Math.random(),
source: source,
'source-layer': vectorLayer,
interactive: true,
layout: {
'line-join': 'round',
'line-cap': 'round'
},
type: 'line',
paint: {'line-color': color},
filter: ["==", "$type", "LineString"]
}
}
export function generateColoredLayers(sources) {
const styleLayers = []
Object.keys(sources).forEach(sourceId => {
const layers = sources[sourceId]
layers.forEach(layerId => {
const color = assignVectorLayerColor(layerId)
styleLayers.push(circleLayer(sourceId, layerId, color))
styleLayers.push(lineLayer(sourceId, layerId, color))
styleLayers.push(polygonLayer(sourceId, layerId, color))
})
})
return styleLayers
}