UNPKG

kepler.gl.geoiq

Version:

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

463 lines (418 loc) 13 kB
// Copyright (c) 2023 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 memoize from 'lodash.memoize'; import uniq from 'lodash.uniq'; import Layer, {colorMaker} from '../base-layer'; import {GeoJsonLayer as DeckGLGeoJsonLayer} from 'deck.gl'; import {hexToRgb} from 'utils/color-utils'; 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'; export const geojsonVisConfigs = { opacity: 'opacity', thickness: { type: 'number', defaultValue: 0.5, label: 'Stroke Width', isRanged: false, range: [0, 100], step: 0.1, group: 'stroke', property: 'thickness' }, strokeColor: 'strokeColor', colorRange: 'colorRange', strokeColorRange: 'strokeColorRange', radius: 'radius', sizeRange: 'strokeWidthRange', radiusRange: 'radiusRange', heightRange: 'elevationRange', elevationScale: 'elevationScale', stroked: 'stroked', filled: 'filled', enable3d: 'enable3d', wireframe: 'wireframe' }; export const geoJsonRequiredColumns = ['geojson']; export const featureAccessor = ({geojson}) => d => d[geojson.fieldIdx]; export const featureResolver = ({geojson}) => geojson.fieldIdx; export default class GeoJsonLayer extends Layer { constructor(props) { super(props); this.dataToFeature = []; this.registerVisConfig(geojsonVisConfigs); this.getFeature = memoize(featureAccessor, featureResolver); } get type() { return 'geojson'; } get name() { return 'Polygon'; } get layerIcon() { return GeojsonLayerIcon; } get requiredLayerColumns() { return geoJsonRequiredColumns; } get visualChannels() { return { ...super.visualChannels, strokeColor: { property: 'strokeColor', field: 'strokeColorField', scale: 'strokeColorScale', domain: 'strokeColorDomain', range: 'strokeColorRange', key: 'strokeColor', channelScaleType: CHANNEL_SCALES.color, condition: config => config.visConfig.stroked }, size: { ...super.visualChannels.size, property: 'stroke', condition: config => config.visConfig.stroked }, height: { property: 'height', field: 'heightField', scale: 'heightScale', domain: 'heightDomain', range: 'heightRange', key: 'height', channelScaleType: 'size', condition: config => config.visConfig.enable3d }, radius: { property: 'radius', field: 'radiusField', scale: 'radiusScale', domain: 'radiusDomain', range: 'radiusRange', key: 'radius', channelScaleType: 'radius' } }; } getPositionAccessor() { return this.getFeature(this.config.columns); } static findDefaultLayerProps({label, fields = []}) { const geojsonColumns = fields .filter(f => f.type === 'geojson') .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) { // index of allData is saved to feature.properties return object; } // TODO: fix complexity /* eslint-disable complexity */ formatLayerData(_, allData, filteredIndex, oldLayerData, opt = {}) { const { colorScale, colorField, colorDomain, strokeColorField, strokeColorScale, strokeColorDomain, color, sizeScale, sizeDomain, sizeField, heightField, heightDomain, heightScale, radiusField, radiusDomain, radiusScale, visConfig } = this.config; const { enable3d, stroked, colorRange, heightRange, sizeRange, radiusRange, strokeColorRange, strokeColor } = visConfig; const getFeature = this.getPositionAccessor(this.config.column); // geojson feature are object, if doesn't exists // create it and save to layer if (!oldLayerData || oldLayerData.getFeature !== getFeature) { this.updateLayerMeta(allData, getFeature); } let geojsonData; if ( oldLayerData && oldLayerData.data && opt.sameData && oldLayerData.getFeature === getFeature ) { // no need to create a new array of data // use updateTriggers to selectively re-calculate attributes geojsonData = oldLayerData.data; } else { // filteredIndex is a reference of index in allData which can map to feature geojsonData = filteredIndex .map(i => this.dataToFeature[i]) .filter(d => d); } // fill color const cScale = colorField && this.getVisChannelScale( colorScale, colorDomain, colorRange.colors.map(hexToRgb) ); // stroke color const scScale = strokeColorField && this.getVisChannelScale( strokeColorScale, strokeColorDomain, strokeColorRange.colors.map(hexToRgb) ); // calculate stroke scale - if stroked = true const sScale = sizeField && stroked && this.getVisChannelScale(sizeScale, sizeDomain, sizeRange); // calculate elevation scale - if extruded = true const eScale = heightField && enable3d && this.getVisChannelScale(heightScale, heightDomain, heightRange); // console.log('eScale inside formatLayerData', eScale); // point radius const rScale = radiusField && this.getVisChannelScale(radiusScale, radiusDomain, radiusRange); return { data: geojsonData, getFeature, getFillColor: d => cScale ? this.getEncodedChannelValue( cScale, allData[d.properties.index], colorField ) : d.properties.fillColor || color, getLineColor: d => scScale ? this.getEncodedChannelValue( scScale, allData[d.properties.index], strokeColorField ) : d.properties.lineColor || strokeColor || color, getLineWidth: d => sScale ? this.getEncodedChannelValue( sScale, allData[d.properties.index], sizeField, 0 ) : d.properties.lineWidth || 1, getElevation: d => eScale ? this.getEncodedChannelValue( eScale, allData[d.properties.index], heightField, 0 ) : d.properties.elevation || 500, getRadius: d => rScale ? this.getEncodedChannelValue( rScale, allData[d.properties.index], radiusField, 0 ) : d.properties.radius || 1 }; } /* eslint-enable complexity */ updateLayerMeta(allData) { const getFeature = this.getPositionAccessor(); this.dataToFeature = getGeojsonDataMaps(allData, 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(allData) { this.updateLayerMeta(allData); 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({data, idx, objectHovered, mapState, interactionConfig}) { const {fixedRadius} = this.meta; const radiusScale = this.getRadiusScaleByZoom(mapState, fixedRadius); const zoomFactor = this.getZoomFactor(mapState); const {visConfig} = this.config; const layerProps = { // multiplier applied just so it being consistent with previously saved maps lineWidthScale: visConfig.thickness * zoomFactor * 8, elevationScale: visConfig.elevationScale, pointRadiusScale: radiusScale, lineMiterLimit: 4 }; const updateTriggers = { getElevation: { heightField: this.config.heightField, heightScale: this.config.heightScale, heightRange: visConfig.heightRange }, getFillColor: { color: this.config.color, colorField: this.config.colorField, colorRange: visConfig.colorRange, colorScale: this.config.colorScale }, getLineColor: { color: visConfig.strokeColor, colorField: this.config.strokeColorField, colorRange: visConfig.strokeColorRange, colorScale: this.config.strokeColorScale }, getLineWidth: { sizeField: this.config.sizeField, sizeRange: visConfig.sizeRange }, getRadius: { radiusField: this.config.radiusField, radiusRange: visConfig.radiusRange } }; return [ new DeckGLGeoJsonLayer({ ...layerProps, id: this.id, idx, data: data.data, getFillColor: data.getFillColor, getLineColor: data.getLineColor, getLineWidth: data.getLineWidth, getRadius: data.getRadius, getElevation: data.getElevation, // highlight pickable: true, highlightColor: HIGHLIGH_COLOR_3D, autoHighlight: visConfig.enable3d, // parameters parameters: { depthTest: Boolean(visConfig.enable3d || mapState.dragRotate) }, opacity: visConfig.opacity, stroked: visConfig.stroked, filled: visConfig.filled, extruded: visConfig.enable3d, wireframe: visConfig.wireframe, lineMiterLimit: 2, rounded: true, updateTriggers }), ...(this.isLayerHovered(objectHovered) && !visConfig.enable3d ? [ new DeckGLGeoJsonLayer({ ...layerProps, id: `${this.id}-hovered`, data: [objectHovered.object], getLineWidth: data.getLineWidth, getRadius: data.getRadius, getElevation: data.getElevation, getLineColor: this.config.highlightColor, getFillColor: this.config.highlightColor, updateTriggers, stroked: true, pickable: false, filled: false }) ] : []) ]; } }