UNPKG

kepler.gl.geoiq

Version:

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

307 lines (266 loc) 34.2 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck")); var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass")); var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits")); var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn")); var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf")); var _deck = require("deck.gl"); var _geoViewport = _interopRequireDefault(require("@mapbox/geo-viewport")); var _d3Array = require("d3-array"); var _dataScaleUtils = require("../../utils/data-scale-utils"); var _utils = require("../layer-utils/utils"); var _colorRanges = require("../../constants/color-ranges"); var _layerFactory = require("../../layers/layer-factory"); var _defaultSettings = require("../../constants/default-settings"); var _clusterUtils = require("../layer-utils/cluster-utils"); function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2["default"])(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = (0, _getPrototypeOf2["default"])(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = (0, _getPrototypeOf2["default"])(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return (0, _possibleConstructorReturn2["default"])(this, result); }; } function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } } var defaultRadius = _layerFactory.LAYER_VIS_CONFIGS.clusterRadius.defaultValue; var defaultRadiusRange = _layerFactory.LAYER_VIS_CONFIGS.clusterRadiusRange.defaultValue; var defaultProps = { clusterRadius: defaultRadius, colorDomain: null, colorRange: _colorRanges.DefaultColorRange, colorScale: _defaultSettings.SCALE_TYPES.quantize, radiusRange: defaultRadiusRange, // maybe later... lowerPercentile: 0, upperPercentile: 100, getPosition: function getPosition(x) { return x.position; }, // if want to have color based on customized aggregator, instead of count getColorValue: function getColorValue(points) { return points.length; }, // if want to have radius based on customized aggregator, instead of count getRadiusValue: function getRadiusValue(cell) { return cell.properties.point_count; }, fp64: false }; var ClusterLayer = exports["default"] = /*#__PURE__*/function (_CompositeLayer) { (0, _inherits2["default"])(ClusterLayer, _CompositeLayer); var _super = _createSuper(ClusterLayer); function ClusterLayer() { (0, _classCallCheck2["default"])(this, ClusterLayer); return _super.apply(this, arguments); } (0, _createClass2["default"])(ClusterLayer, [{ key: "initializeState", value: function initializeState() { this.state = { clusters: null, geoJSON: null }; } }, { key: "shouldUpdateState", value: function shouldUpdateState(_ref) { var changeFlags = _ref.changeFlags; return changeFlags.somethingChanged; } }, { key: "updateState", value: function updateState(_ref2) { var context = _ref2.context, oldProps = _ref2.oldProps, props = _ref2.props, changeFlags = _ref2.changeFlags; if (changeFlags.dataChanged || // changeFlags.viewportChanged || this.needsReProjectPoints(oldProps, props)) { // project data into clusters, and get clustered data this.processGeoJSON(); this.getClusters(); // this needs clustered data to be set this.getColorValueDomain(); } else if (this.needsReclusterPoints(oldProps, props)) { this.getClusters(); this.getColorValueDomain(); } else if (this.needsRecalculateScaleFunction(oldProps, props)) { this.getColorValueDomain(); } } }, { key: "needsReProjectPoints", value: function needsReProjectPoints(oldProps, props) { return oldProps.clusterRadius !== props.clusterRadius || oldProps.getPosition !== props.getPosition; } }, { key: "needsReclusterPoints", value: function needsReclusterPoints(oldProps, props) { return Math.round(oldProps.zoom) !== Math.round(props.zoom); } }, { key: "needsRecalculateScaleFunction", value: function needsRecalculateScaleFunction(oldProps, props) { return (0, _utils.needsRecalculateColorDomain)(oldProps, props) || (0, _utils.needReCalculateScaleFunction)(oldProps, props) || (0, _utils.needsRecalculateRadiusRange)(oldProps, props) || oldProps.getColorValue !== props.getColorValue; } }, { key: "processGeoJSON", value: function processGeoJSON() { var _this$props = this.props, data = _this$props.data, getPosition = _this$props.getPosition; this.setState({ geoJSON: (0, _clusterUtils.getGeoJSON)(data, getPosition) }); (0, _clusterUtils.clearClustererCache)(); } }, { key: "getClusters", value: function getClusters() { var geoJSON = this.state.geoJSON; var clusterRadius = this.props.clusterRadius; var _this$context = this.context, viewport = _this$context.viewport, _this$context$viewpor = _this$context.viewport, longitude = _this$context$viewpor.longitude, latitude = _this$context$viewpor.latitude, height = _this$context$viewpor.height, width = _this$context$viewpor.width; // zoom needs to be an integer for the different map utils. Also helps with cache key. var zoom = Math.round(viewport.zoom); var bbox = _geoViewport["default"].bounds([longitude, latitude], zoom, [width, height]); var clusters = (0, _clusterUtils.clustersAtZoom)({ bbox: bbox, clusterRadius: clusterRadius, geoJSON: geoJSON, zoom: zoom }); this.setState({ clusters: clusters }); } }, { key: "getColorValueDomain", value: function getColorValueDomain() { var _this$props2 = this.props, colorScale = _this$props2.colorScale, getColorValue = _this$props2.getColorValue, getRadiusValue = _this$props2.getRadiusValue, onSetColorDomain = _this$props2.onSetColorDomain; var clusters = this.state.clusters; var radiusDomain = [0, (0, _d3Array.max)(clusters, getRadiusValue)]; var colorValues = clusters.map(function (d) { return getColorValue(d.properties.points); }); var identity = function identity(d) { return d; }; var colorDomain = colorScale === _defaultSettings.SCALE_TYPES.ordinal ? (0, _dataScaleUtils.getOrdinalDomain)(colorValues, identity) : colorScale === _defaultSettings.SCALE_TYPES.quantile ? (0, _dataScaleUtils.getQuantileDomain)(colorValues, identity, _d3Array.ascending) : (0, _dataScaleUtils.getLinearDomain)(colorValues, identity); this.setState({ colorDomain: colorDomain, radiusDomain: radiusDomain }); (0, _utils.getColorScaleFunction)(this); (0, _utils.getRadiusScaleFunction)(this); onSetColorDomain(colorDomain); } }, { key: "getUpdateTriggers", value: function getUpdateTriggers() { return { getColor: { colorRange: this.props.colorRange, colorDomain: this.props.colorDomain, getColorValue: this.props.getColorValue, colorScale: this.props.colorScale, lowerPercentile: this.props.lowerPercentile, upperPercentile: this.props.upperPercentile }, getRadius: { radiusRange: this.props.radiusRange, radiusDomain: this.props.radiusDomain, getRadiusValue: this.props.getRadiusValue } }; } /* * override default layer method to calculate cell color based on color scale function */ }, { key: "_onGetSublayerColor", value: function _onGetSublayerColor(cell) { var getColorValue = this.props.getColorValue; var _this$state = this.state, colorScaleFunc = _this$state.colorScaleFunc, colorDomain = _this$state.colorDomain; var cv = getColorValue(cell.properties.points); // if cell value is outside domain, set alpha to 0 var color = cv >= colorDomain[0] && cv <= colorDomain[colorDomain.length - 1] ? colorScaleFunc(cv) : [0, 0, 0, 0]; // add final alpha to color color[3] = Number.isFinite(color[3]) ? color[3] : 255; return color; } }, { key: "_onGetSublayerRadius", value: function _onGetSublayerRadius(cell) { var getRadiusValue = this.props.getRadiusValue; var radiusScaleFunc = this.state.radiusScaleFunc; return radiusScaleFunc(getRadiusValue(cell)); } }, { key: "getPickingInfo", value: function getPickingInfo(_ref3) { var info = _ref3.info; var clusters = this.state.clusters; var isPicked = info.picked && info.index > -1; var object = null; if (isPicked) { // add cluster colorValue to object var cluster = clusters[info.index]; var colorValue = this.props.getColorValue(cluster.properties.points); object = _objectSpread(_objectSpread({}, cluster.properties), {}, { colorValue: colorValue, radius: this._onGetSublayerRadius(cluster), position: cluster.geometry.coordinates }); } return _objectSpread(_objectSpread({}, info), {}, { picked: Boolean(object), // override object with picked cluster property object: object }); } }, { key: "renderLayers", value: function renderLayers() { // for subclassing, override this method to return // customized sub layer props var _this$props3 = this.props, id = _this$props3.id, radiusScale = _this$props3.radiusScale, fp64 = _this$props3.fp64; // base layer props var _this$props4 = this.props, opacity = _this$props4.opacity, pickable = _this$props4.pickable, autoHighlight = _this$props4.autoHighlight, highlightColor = _this$props4.highlightColor; // return props to the sublayer constructor return new _deck.ScatterplotLayer({ id: "".concat(id, "-cluster"), data: this.state.clusters, radiusScale: radiusScale, fp64: fp64, opacity: opacity, pickable: pickable, autoHighlight: autoHighlight, highlightColor: highlightColor, getPosition: function getPosition(d) { return d.geometry.coordinates; }, getRadius: this._onGetSublayerRadius.bind(this), getColor: this._onGetSublayerColor.bind(this), updateTriggers: this.getUpdateTriggers() }); } }]); return ClusterLayer; }(_deck.CompositeLayer); ClusterLayer.layerName = 'ClusterLayer'; ClusterLayer.defaultProps = defaultProps; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/deckgl-layers/cluster-layer/cluster-layer.js"],"names":["defaultRadius","LAYER_VIS_CONFIGS","clusterRadius","defaultValue","defaultRadiusRange","clusterRadiusRange","defaultProps","colorDomain","colorRange","DefaultColorRange","colorScale","SCALE_TYPES","quantize","radiusRange","lowerPercentile","upperPercentile","getPosition","x","position","getColorValue","points","length","getRadiusValue","cell","properties","point_count","fp64","ClusterLayer","state","clusters","geoJSON","changeFlags","somethingChanged","context","oldProps","props","dataChanged","needsReProjectPoints","processGeoJSON","getClusters","getColorValueDomain","needsReclusterPoints","needsRecalculateScaleFunction","Math","round","zoom","data","setState","viewport","longitude","latitude","height","width","bbox","geoViewport","bounds","onSetColorDomain","radiusDomain","colorValues","map","d","identity","ordinal","quantile","ascending","getColor","getRadius","colorScaleFunc","cv","color","Number","isFinite","radiusScaleFunc","info","isPicked","picked","index","object","cluster","colorValue","radius","_onGetSublayerRadius","geometry","coordinates","Boolean","id","radiusScale","opacity","pickable","autoHighlight","highlightColor","ScatterplotLayer","bind","_onGetSublayerColor","updateTriggers","getUpdateTriggers","CompositeLayer","layerName"],"mappings":"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","sourcesContent":["// Copyright (c) 2023 Uber Technologies, Inc.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n// of this software and associated documentation files (the \"Software\"), to deal\n// in the Software without restriction, including without limitation the rights\n// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n// copies of the Software, and to permit persons to whom the Software is\n// furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n// all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n// THE SOFTWARE.\n\nimport {CompositeLayer, ScatterplotLayer} from 'deck.gl';\nimport geoViewport from '@mapbox/geo-viewport';\nimport {ascending, max} from 'd3-array';\nimport {\n  getQuantileDomain,\n  getOrdinalDomain,\n  getLinearDomain\n} from 'utils/data-scale-utils';\nimport {\n  getColorScaleFunction,\n  getRadiusScaleFunction,\n  needsRecalculateRadiusRange,\n  needsRecalculateColorDomain,\n  needReCalculateScaleFunction\n} from '../layer-utils/utils';\nimport {DefaultColorRange} from 'constants/color-ranges';\nimport {LAYER_VIS_CONFIGS} from 'layers/layer-factory';\nimport {SCALE_TYPES} from 'constants/default-settings';\n\nimport {\n  clearClustererCache,\n  clustersAtZoom,\n  getGeoJSON\n} from '../layer-utils/cluster-utils';\n\nconst defaultRadius = LAYER_VIS_CONFIGS.clusterRadius.defaultValue;\nconst defaultRadiusRange = LAYER_VIS_CONFIGS.clusterRadiusRange.defaultValue;\n\nconst defaultProps = {\n  clusterRadius: defaultRadius,\n  colorDomain: null,\n  colorRange: DefaultColorRange,\n  colorScale: SCALE_TYPES.quantize,\n  radiusRange: defaultRadiusRange,\n\n  // maybe later...\n  lowerPercentile: 0,\n  upperPercentile: 100,\n\n  getPosition: x => x.position,\n\n  // if want to have color based on customized aggregator, instead of count\n  getColorValue: points => points.length,\n\n  //  if want to have radius based on customized aggregator, instead of count\n  getRadiusValue: cell => cell.properties.point_count,\n  fp64: false\n};\n\nexport default class ClusterLayer extends CompositeLayer {\n  initializeState() {\n    this.state = {\n      clusters: null,\n      geoJSON: null\n    };\n  }\n\n  shouldUpdateState({changeFlags}) {\n    return changeFlags.somethingChanged;\n  }\n\n  updateState({context, oldProps, props, changeFlags}) {\n    if (\n      changeFlags.dataChanged ||\n      // changeFlags.viewportChanged ||\n      this.needsReProjectPoints(oldProps, props)\n    ) {\n      // project data into clusters, and get clustered data\n      this.processGeoJSON();\n      this.getClusters();\n\n      // this needs clustered data to be set\n      this.getColorValueDomain();\n    } else if (this.needsReclusterPoints(oldProps, props)) {\n      this.getClusters();\n      this.getColorValueDomain();\n    } else if (this.needsRecalculateScaleFunction(oldProps, props)) {\n      this.getColorValueDomain();\n    }\n  }\n\n  needsReProjectPoints(oldProps, props) {\n    return (\n      oldProps.clusterRadius !== props.clusterRadius ||\n      oldProps.getPosition !== props.getPosition\n    );\n  }\n\n  needsReclusterPoints(oldProps, props) {\n    return Math.round(oldProps.zoom) !== Math.round(props.zoom);\n  }\n\n  needsRecalculateScaleFunction(oldProps, props) {\n    return (\n      needsRecalculateColorDomain(oldProps, props) ||\n      needReCalculateScaleFunction(oldProps, props) ||\n      needsRecalculateRadiusRange(oldProps, props) ||\n      oldProps.getColorValue !== props.getColorValue\n    );\n  }\n\n  processGeoJSON() {\n    const {data, getPosition} = this.props;\n    this.setState({geoJSON: getGeoJSON(data, getPosition)});\n    clearClustererCache();\n  }\n\n  getClusters() {\n    const {geoJSON} = this.state;\n    const {clusterRadius} = this.props;\n    const {\n      viewport,\n      viewport: {longitude, latitude, height, width}\n    } = this.context;\n\n    // zoom needs to be an integer for the different map utils. Also helps with cache key.\n    const zoom = Math.round(viewport.zoom);\n    const bbox = geoViewport.bounds([longitude, latitude], zoom, [\n      width,\n      height\n    ]);\n\n    const clusters = clustersAtZoom({bbox, clusterRadius, geoJSON, zoom});\n\n    this.setState({clusters});\n  }\n\n  getColorValueDomain() {\n    const {\n      colorScale,\n      getColorValue,\n      getRadiusValue,\n      onSetColorDomain\n    } = this.props;\n    const {clusters} = this.state;\n\n    const radiusDomain = [0, max(clusters, getRadiusValue)];\n\n    const colorValues = clusters.map(d => getColorValue(d.properties.points));\n\n    const identity = d => d;\n\n    const colorDomain =\n      colorScale === SCALE_TYPES.ordinal\n        ? getOrdinalDomain(colorValues, identity)\n        : colorScale === SCALE_TYPES.quantile\n        ? getQuantileDomain(colorValues, identity, ascending)\n        : getLinearDomain(colorValues, identity);\n\n    this.setState({\n      colorDomain,\n      radiusDomain\n    });\n\n    getColorScaleFunction(this);\n    getRadiusScaleFunction(this);\n\n    onSetColorDomain(colorDomain);\n  }\n\n  getUpdateTriggers() {\n    return {\n      getColor: {\n        colorRange: this.props.colorRange,\n        colorDomain: this.props.colorDomain,\n        getColorValue: this.props.getColorValue,\n        colorScale: this.props.colorScale,\n        lowerPercentile: this.props.lowerPercentile,\n        upperPercentile: this.props.upperPercentile\n      },\n      getRadius: {\n        radiusRange: this.props.radiusRange,\n        radiusDomain: this.props.radiusDomain,\n        getRadiusValue: this.props.getRadiusValue\n      }\n    };\n  }\n\n  /*\n   * override default layer method to calculate cell color based on color scale function\n   */\n  _onGetSublayerColor(cell) {\n    const {getColorValue} = this.props;\n    const {colorScaleFunc, colorDomain} = this.state;\n\n    const cv = getColorValue(cell.properties.points);\n\n    // if cell value is outside domain, set alpha to 0\n    const color =\n      cv >= colorDomain[0] && cv <= colorDomain[colorDomain.length - 1]\n        ? colorScaleFunc(cv)\n        : [0, 0, 0, 0];\n\n    // add final alpha to color\n    color[3] = Number.isFinite(color[3]) ? color[3] : 255;\n\n    return color;\n  }\n\n  _onGetSublayerRadius(cell) {\n    const {getRadiusValue} = this.props;\n    const {radiusScaleFunc} = this.state;\n    return radiusScaleFunc(getRadiusValue(cell));\n  }\n\n  getPickingInfo({info}) {\n    const {clusters} = this.state;\n    const isPicked = info.picked && info.index > -1;\n\n    let object = null;\n    if (isPicked) {\n      // add cluster colorValue to object\n      const cluster = clusters[info.index];\n      const colorValue = this.props.getColorValue(cluster.properties.points);\n\n      object = {\n        ...cluster.properties,\n        colorValue,\n        radius: this._onGetSublayerRadius(cluster),\n        position: cluster.geometry.coordinates\n      };\n    }\n\n    return {\n      ...info,\n      picked: Boolean(object),\n      // override object with picked cluster property\n      object\n    };\n  }\n\n  renderLayers() {\n    // for subclassing, override this method to return\n    // customized sub layer props\n    const {id, radiusScale, fp64} = this.props;\n\n    // base layer props\n    const {opacity, pickable, autoHighlight, highlightColor} = this.props;\n\n    // return props to the sublayer constructor\n    return new ScatterplotLayer({\n      id: `${id}-cluster`,\n      data: this.state.clusters,\n      radiusScale,\n      fp64,\n      opacity,\n      pickable,\n      autoHighlight,\n      highlightColor,\n      getPosition: d => d.geometry.coordinates,\n      getRadius: this._onGetSublayerRadius.bind(this),\n      getColor: this._onGetSublayerColor.bind(this),\n      updateTriggers: this.getUpdateTriggers()\n    });\n  }\n}\n\nClusterLayer.layerName = 'ClusterLayer';\nClusterLayer.defaultProps = defaultProps;\n"]}