UNPKG

kepler.gl

Version:

kepler.gl is a webgl based application to visualize large scale location data in the browser

351 lines (307 loc) 11.3 kB
// Copyright (c) 2021 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. import uniq from 'lodash.uniq'; import {DATA_TYPES} from 'type-analyzer'; import Layer, {colorMaker} from '../base-layer'; import {GeoJsonLayer as DeckGLGeoJsonLayer} from '@deck.gl/layers'; import {getGeojsonDataMaps, getGeojsonBounds, getGeojsonFeatureTypes} from './geojson-utils'; import GeojsonLayerIcon from './geojson-layer-icon'; import {GEOJSON_FIELDS, HIGHLIGH_COLOR_3D, CHANNEL_SCALES} from 'constants/default-settings'; import {LAYER_VIS_CONFIGS} from 'layers/layer-factory'; const SUPPORTED_ANALYZER_TYPES = { [DATA_TYPES.GEOMETRY]: true, [DATA_TYPES.GEOMETRY_FROM_STRING]: true, [DATA_TYPES.PAIR_GEOMETRY_FROM_STRING]: true }; export const geojsonVisConfigs = { opacity: 'opacity', strokeOpacity: { ...LAYER_VIS_CONFIGS.opacity, property: 'strokeOpacity' }, thickness: { ...LAYER_VIS_CONFIGS.thickness, defaultValue: 0.5 }, strokeColor: 'strokeColor', colorRange: 'colorRange', strokeColorRange: 'strokeColorRange', radius: 'radius', sizeRange: 'strokeWidthRange', radiusRange: 'radiusRange', heightRange: 'elevationRange', elevationScale: 'elevationScale', enableElevationZoomFactor: 'enableElevationZoomFactor', stroked: 'stroked', filled: 'filled', enable3d: 'enable3d', wireframe: 'wireframe' }; export const geoJsonRequiredColumns = ['geojson']; export const featureAccessor = ({geojson}) => dc => d => dc.valueAt(d.index, geojson.fieldIdx); // access feature properties from geojson sub layer export const defaultElevation = 500; export const defaultLineWidth = 1; export const defaultRadius = 1; export default class GeoJsonLayer extends Layer { constructor(props) { super(props); this.dataToFeature = []; this.registerVisConfig(geojsonVisConfigs); this.getPositionAccessor = dataContainer => featureAccessor(this.config.columns)(dataContainer); } get type() { return 'geojson'; } get name() { return 'Polygon'; } get layerIcon() { return GeojsonLayerIcon; } get requiredLayerColumns() { return geoJsonRequiredColumns; } get visualChannels() { const visualChannels = super.visualChannels; return { color: { ...visualChannels.color, accessor: 'getFillColor', condition: config => config.visConfig.filled, nullValue: visualChannels.color.nullValue, getAttributeValue: config => d => d.properties.fillColor || config.color, // used this to get updateTriggers defaultValue: config => config.color }, strokeColor: { property: 'strokeColor', field: 'strokeColorField', scale: 'strokeColorScale', domain: 'strokeColorDomain', range: 'strokeColorRange', key: 'strokeColor', channelScaleType: CHANNEL_SCALES.color, accessor: 'getLineColor', condition: config => config.visConfig.stroked, nullValue: visualChannels.color.nullValue, getAttributeValue: config => d => d.properties.lineColor || config.visConfig.strokeColor || config.color, // used this to get updateTriggers defaultValue: config => config.visConfig.strokeColor || config.color }, size: { ...visualChannels.size, property: 'stroke', accessor: 'getLineWidth', condition: config => config.visConfig.stroked, nullValue: 0, getAttributeValue: () => d => d.properties.lineWidth || defaultLineWidth }, height: { property: 'height', field: 'heightField', scale: 'heightScale', domain: 'heightDomain', range: 'heightRange', key: 'height', channelScaleType: CHANNEL_SCALES.size, accessor: 'getElevation', condition: config => config.visConfig.enable3d, nullValue: 0, getAttributeValue: () => d => d.properties.elevation || defaultElevation }, radius: { property: 'radius', field: 'radiusField', scale: 'radiusScale', domain: 'radiusDomain', range: 'radiusRange', key: 'radius', channelScaleType: CHANNEL_SCALES.radius, accessor: 'getRadius', nullValue: 0, getAttributeValue: () => d => d.properties.radius || defaultRadius } }; } static findDefaultLayerProps({label, fields = []}) { const geojsonColumns = fields .filter(f => f.type === 'geojson' && SUPPORTED_ANALYZER_TYPES[f.analyzerType]) .map(f => f.name); const defaultColumns = { geojson: uniq([...GEOJSON_FIELDS.geojson, ...geojsonColumns]) }; const foundColumns = this.findDefaultColumnField(defaultColumns, fields); if (!foundColumns || !foundColumns.length) { return {props: []}; } return { props: foundColumns.map(columns => ({ label: (typeof label === 'string' && label.replace(/\.[^/.]+$/, '')) || this.type, columns, isVisible: true })) }; } getDefaultLayerConfig(props = {}) { return { ...super.getDefaultLayerConfig(props), // add height visual channel heightField: null, heightDomain: [0, 1], heightScale: 'linear', // add radius visual channel radiusField: null, radiusDomain: [0, 1], radiusScale: 'linear', // add stroke color visual channel strokeColorField: null, strokeColorDomain: [0, 1], strokeColorScale: 'quantile' }; } getHoverData(object, dataContainer) { // index of dataContainer is saved to feature.properties return dataContainer.row(object.properties.index); } calculateDataAttribute({dataContainer, filteredIndex}, getPosition) { return filteredIndex.map(i => this.dataToFeature[i]).filter(d => d); } formatLayerData(datasets, oldLayerData) { const {gpuFilter, dataContainer} = datasets[this.config.dataId]; const {data} = this.updateData(datasets, oldLayerData); const customFilterValueAccessor = (dc, d, fieldIndex) => { return dc.valueAt(d.properties.index, fieldIndex); }; const indexAccessor = f => f.properties.index; const dataAccessor = dc => d => ({index: d.properties.index}); const accessors = this.getAttributeAccessors({dataAccessor, dataContainer}); return { data, getFilterValue: gpuFilter.filterValueAccessor(dataContainer)( indexAccessor, customFilterValueAccessor ), ...accessors }; } updateLayerMeta(dataContainer) { const getFeature = this.getPositionAccessor(dataContainer); this.dataToFeature = getGeojsonDataMaps(dataContainer, getFeature); // get bounds from features const bounds = getGeojsonBounds(this.dataToFeature); // if any of the feature has properties.radius set to be true const fixedRadius = Boolean( this.dataToFeature.find(d => d && d.properties && d.properties.radius) ); // keep a record of what type of geometry the collection has const featureTypes = getGeojsonFeatureTypes(this.dataToFeature); this.updateMeta({bounds, fixedRadius, featureTypes}); } setInitialLayerConfig({dataContainer}) { this.updateLayerMeta(dataContainer); const {featureTypes} = this.meta; // default settings is stroke: true, filled: false if (featureTypes && featureTypes.polygon) { // set both fill and stroke to true return this.updateLayerVisConfig({ filled: true, stroked: true, strokeColor: colorMaker.next().value }); } else if (featureTypes && featureTypes.point) { // set fill to true if detect point return this.updateLayerVisConfig({filled: true, stroked: false}); } return this; } renderLayer(opts) { const {data, gpuFilter, objectHovered, mapState, interactionConfig} = opts; const {fixedRadius, featureTypes} = this.meta; const radiusScale = this.getRadiusScaleByZoom(mapState, fixedRadius); const zoomFactor = this.getZoomFactor(mapState); const eleZoomFactor = this.getElevationZoomFactor(mapState); const {visConfig} = this.config; const layerProps = { lineWidthScale: visConfig.thickness * zoomFactor * 8, elevationScale: visConfig.elevationScale * eleZoomFactor, pointRadiusScale: radiusScale, lineMiterLimit: 4 }; const updateTriggers = { ...this.getVisualChannelUpdateTriggers(), getFilterValue: gpuFilter.filterValueUpdateTriggers }; const defaultLayerProps = this.getDefaultDeckLayerProps(opts); const opaOverwrite = { opacity: visConfig.strokeOpacity }; const pickable = interactionConfig.tooltip.enabled; const hoveredObject = this.hasHoveredObject(objectHovered); return [ new DeckGLGeoJsonLayer({ ...defaultLayerProps, ...layerProps, ...data, pickable, highlightColor: HIGHLIGH_COLOR_3D, autoHighlight: visConfig.enable3d && pickable, stroked: visConfig.stroked, filled: visConfig.filled, extruded: visConfig.enable3d, wireframe: visConfig.wireframe, wrapLongitude: false, lineMiterLimit: 2, rounded: true, updateTriggers, _subLayerProps: { ...(featureTypes.polygon ? {'polygons-stroke': opaOverwrite} : {}), ...(featureTypes.line ? {'line-strings': opaOverwrite} : {}), ...(featureTypes.point ? { points: { lineOpacity: visConfig.strokeOpacity } } : {}) } }), ...(hoveredObject && !visConfig.enable3d ? [ new DeckGLGeoJsonLayer({ ...this.getDefaultHoverLayerProps(), ...layerProps, wrapLongitude: false, data: [hoveredObject], getLineWidth: data.getLineWidth, getRadius: data.getRadius, getElevation: data.getElevation, getLineColor: this.config.highlightColor, getFillColor: this.config.highlightColor, // always draw outline stroked: true, filled: false }) ] : []) ]; } }