mirror of
https://github.com/a-nyx/maputnik-with-pmtiles.git
synced 2024-12-27 09:35:25 +01:00
Add inspection map
This commit is contained in:
parent
04eab70e27
commit
f332d517f3
9 changed files with 232 additions and 48 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -1,6 +1,7 @@
|
||||||
# Logs
|
# Logs
|
||||||
logs
|
logs
|
||||||
*.log
|
*.log
|
||||||
|
*.swp
|
||||||
|
|
||||||
# Runtime data
|
# Runtime data
|
||||||
pids
|
pids
|
||||||
|
|
|
@ -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} />
|
||||||
}
|
}
|
||||||
|
|
79
src/components/map/InspectionMap.jsx
Normal file
79
src/components/map/InspectionMap.jsx
Normal 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>
|
||||||
|
}
|
||||||
|
}
|
|
@ -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>
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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)}
|
||||||
|
|
|
@ -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
88
src/libs/stylegen.js
Normal 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
|
||||||
|
}
|
Loading…
Reference in a new issue