UNPKG

kepler.gl

Version:

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

269 lines (226 loc) 29 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.heatmapVisConfigs = exports.pointColResolver = exports.pointPosAccessor = exports.MAX_ZOOM_LEVEL = void 0; var _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties")); var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck")); var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass")); var _assertThisInitialized2 = _interopRequireDefault(require("@babel/runtime/helpers/assertThisInitialized")); var _get2 = _interopRequireDefault(require("@babel/runtime/helpers/get")); 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 _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray")); var _reselect = require("reselect"); var _lodash = _interopRequireDefault(require("lodash.memoize")); var _defaultSettings = require("../../constants/default-settings"); var _colorUtils = require("../../utils/color-utils"); var _mapboxglLayer = _interopRequireDefault(require("../mapboxgl-layer")); var _heatmapLayerIcon = _interopRequireDefault(require("./heatmap-layer-icon")); function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2["default"])(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } 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 MAX_ZOOM_LEVEL = 18; exports.MAX_ZOOM_LEVEL = MAX_ZOOM_LEVEL; var pointPosAccessor = function pointPosAccessor(_ref) { var lat = _ref.lat, lng = _ref.lng; return function (dc) { return function (d) { return [dc.valueAt(d.index, lng.fieldIdx), dc.valueAt(d.index, lat.fieldIdx)]; }; }; }; exports.pointPosAccessor = pointPosAccessor; var pointColResolver = function pointColResolver(_ref2) { var lat = _ref2.lat, lng = _ref2.lng; return "".concat(lat.fieldIdx, "-").concat(lng.fieldIdx); }; exports.pointColResolver = pointColResolver; var heatmapVisConfigs = { opacity: 'opacity', colorRange: 'colorRange', radius: 'heatmapRadius' }; /** * * @param {Object} colorRange * @return {Array} [ * 0, "rgba(33,102,172,0)", * 0.2, "rgb(103,169,207)", * 0.4, "rgb(209,229,240)", * 0.6, "rgb(253,219,199)", * 0.8, "rgb(239,138,98)", * 1, "rgb(178,24,43)" * ] */ exports.heatmapVisConfigs = heatmapVisConfigs; var heatmapDensity = function heatmapDensity(colorRange) { var scaleFunction = _defaultSettings.SCALE_FUNC.quantize; var colors = ['#000000'].concat((0, _toConsumableArray2["default"])(colorRange.colors)); var scale = scaleFunction().domain([0, 1]).range(colors); var colorDensity = scale.range().reduce(function (bands, level) { var invert = scale.invertExtent(level); return [].concat((0, _toConsumableArray2["default"])(bands), [invert[0], // first value in the range "rgb(".concat((0, _colorUtils.hexToRgb)(level).join(','), ")") // color ]); }, []); colorDensity[1] = 'rgba(0,0,0,0)'; return colorDensity; }; var HeatmapLayer = /*#__PURE__*/function (_MapboxGLLayer) { (0, _inherits2["default"])(HeatmapLayer, _MapboxGLLayer); var _super = _createSuper(HeatmapLayer); function HeatmapLayer(props) { var _this; (0, _classCallCheck2["default"])(this, HeatmapLayer); _this = _super.call(this, props); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "columnsSelector", function (config) { return pointColResolver(config.columns); }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "visConfigSelector", function (config) { return config.visConfig; }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "weightFieldSelector", function (config) { return config.weightField ? config.weightField.name : null; }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "weightDomainSelector", function (config) { return config.weightDomain; }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "paintSelector", (0, _reselect.createSelector)(_this.visConfigSelector, _this.weightFieldSelector, _this.weightDomainSelector, function (visConfig, weightField, weightDomain) { return { 'heatmap-weight': weightField ? ['interpolate', ['linear'], ['get', weightField], weightDomain[0], 0, weightDomain[1], 1] : 1, 'heatmap-intensity': ['interpolate', ['linear'], ['zoom'], 0, 1, MAX_ZOOM_LEVEL, 3], 'heatmap-color': ['interpolate', ['linear'], ['heatmap-density']].concat((0, _toConsumableArray2["default"])(heatmapDensity(visConfig.colorRange))), 'heatmap-radius': ['interpolate', ['linear'], ['zoom'], 0, 2, MAX_ZOOM_LEVEL, visConfig.radius // radius ], 'heatmap-opacity': visConfig.opacity }; })); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "computeHeatmapConfiguration", (0, _reselect.createSelector)(_this.sourceSelector, _this.filterSelector, _this.paintSelector, function (source, filter, paint) { return _objectSpread({ type: 'heatmap', id: _this.id, source: source, layout: { visibility: 'visible' }, maxzoom: MAX_ZOOM_LEVEL, paint: paint }, _this.isValidFilter(filter) ? { filter: filter } : {}); })); _this.registerVisConfig(heatmapVisConfigs); _this.getPosition = (0, _lodash["default"])(pointPosAccessor, pointColResolver); return _this; } (0, _createClass2["default"])(HeatmapLayer, [{ key: "type", get: function get() { return 'heatmap'; } }, { key: "visualChannels", get: function get() { return { weight: { property: 'weight', field: 'weightField', scale: 'weightScale', domain: 'weightDomain', key: 'weight', // supportedFieldTypes can be determined by channelScaleType // or specified here defaultMeasure: 'property.density', supportedFieldTypes: [_defaultSettings.ALL_FIELD_TYPES.real, _defaultSettings.ALL_FIELD_TYPES.integer], channelScaleType: _defaultSettings.CHANNEL_SCALES.size } }; } }, { key: "layerIcon", get: function get() { return _heatmapLayerIcon["default"]; } }, { key: "getVisualChannelDescription", value: function getVisualChannelDescription(channel) { return channel === 'color' ? { label: 'property.color', measure: 'property.density' } : { label: 'property.weight', measure: this.config.weightField ? this.config.weightField.name : 'property.density' }; } }, { key: "getDefaultLayerConfig", value: function getDefaultLayerConfig() { var props = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; // mapbox heatmap layer color is always based on density // no need to set colorField, colorDomain and colorScale /* eslint-disable no-unused-vars */ var _get$call$weightField = _objectSpread(_objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(HeatmapLayer.prototype), "getDefaultLayerConfig", this).call(this, props)), {}, { weightField: null, weightDomain: [0, 1], weightScale: 'linear' }), colorField = _get$call$weightField.colorField, colorDomain = _get$call$weightField.colorDomain, colorScale = _get$call$weightField.colorScale, layerConfig = (0, _objectWithoutProperties2["default"])(_get$call$weightField, ["colorField", "colorDomain", "colorScale"]); /* eslint-enable no-unused-vars */ return layerConfig; } }, { key: "getPositionAccessor", value: function getPositionAccessor(dataContainer) { return this.getPosition(this.config.columns)(dataContainer); } }, { key: "updateLayerMeta", value: function updateLayerMeta(dataContainer) { var getPosition = this.getPositionAccessor(dataContainer); var bounds = this.getPointsBounds(dataContainer, getPosition); this.updateMeta({ bounds: bounds }); } }, { key: "getGeometry", value: function getGeometry(position) { return position.every(Number.isFinite) ? { type: 'Point', coordinates: position } : null; } }, { key: "formatLayerData", value: function formatLayerData(datasets, oldLayerData) { var weightField = this.config.weightField; var dataContainer = datasets[this.config.dataId].dataContainer; var getPosition = this.getPositionAccessor(dataContainer); var _this$updateData = this.updateData(datasets, oldLayerData), data = _this$updateData.data; var newConfig = this.computeHeatmapConfiguration(this.config, datasets); newConfig.id = this.id; return { columns: this.config.columns, config: newConfig, data: data, weightField: weightField, getPosition: getPosition }; } }]); return HeatmapLayer; }(_mapboxglLayer["default"]); var _default = HeatmapLayer; exports["default"] = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/heatmap-layer/heatmap-layer.js"],"names":["MAX_ZOOM_LEVEL","pointPosAccessor","lat","lng","dc","d","valueAt","index","fieldIdx","pointColResolver","heatmapVisConfigs","opacity","colorRange","radius","heatmapDensity","scaleFunction","SCALE_FUNC","quantize","colors","scale","domain","range","colorDensity","reduce","bands","level","invert","invertExtent","join","HeatmapLayer","props","config","columns","visConfig","weightField","name","weightDomain","visConfigSelector","weightFieldSelector","weightDomainSelector","sourceSelector","filterSelector","paintSelector","source","filter","paint","type","id","layout","visibility","maxzoom","isValidFilter","registerVisConfig","getPosition","weight","property","field","key","defaultMeasure","supportedFieldTypes","ALL_FIELD_TYPES","real","integer","channelScaleType","CHANNEL_SCALES","size","HeatmapLayerIcon","channel","label","measure","weightScale","colorField","colorDomain","colorScale","layerConfig","dataContainer","getPositionAccessor","bounds","getPointsBounds","updateMeta","position","every","Number","isFinite","coordinates","datasets","oldLayerData","dataId","updateData","data","newConfig","computeHeatmapConfiguration","MapboxGLLayer"],"mappings":"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","sourcesContent":["// Copyright (c) 2021 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 {createSelector} from 'reselect';\nimport memoize from 'lodash.memoize';\nimport {CHANNEL_SCALES, SCALE_FUNC, ALL_FIELD_TYPES} from 'constants/default-settings';\nimport {hexToRgb} from 'utils/color-utils';\nimport MapboxGLLayer from '../mapboxgl-layer';\nimport HeatmapLayerIcon from './heatmap-layer-icon';\n\nexport const MAX_ZOOM_LEVEL = 18;\n\nexport const pointPosAccessor = ({lat, lng}) => dc => d => [\n  dc.valueAt(d.index, lng.fieldIdx),\n  dc.valueAt(d.index, lat.fieldIdx)\n];\n\nexport const pointColResolver = ({lat, lng}) => `${lat.fieldIdx}-${lng.fieldIdx}`;\n\nexport const heatmapVisConfigs = {\n  opacity: 'opacity',\n  colorRange: 'colorRange',\n  radius: 'heatmapRadius'\n};\n\n/**\n *\n * @param {Object} colorRange\n * @return {Array} [\n *  0, \"rgba(33,102,172,0)\",\n *  0.2, \"rgb(103,169,207)\",\n *  0.4, \"rgb(209,229,240)\",\n *  0.6, \"rgb(253,219,199)\",\n *  0.8, \"rgb(239,138,98)\",\n *  1, \"rgb(178,24,43)\"\n * ]\n */\nconst heatmapDensity = colorRange => {\n  const scaleFunction = SCALE_FUNC.quantize;\n\n  const colors = ['#000000', ...colorRange.colors];\n\n  const scale = scaleFunction()\n    .domain([0, 1])\n    .range(colors);\n\n  const colorDensity = scale.range().reduce((bands, level) => {\n    const invert = scale.invertExtent(level);\n    return [\n      ...bands,\n      invert[0], // first value in the range\n      `rgb(${hexToRgb(level).join(',')})` // color\n    ];\n  }, []);\n  colorDensity[1] = 'rgba(0,0,0,0)';\n  return colorDensity;\n};\n\nclass HeatmapLayer extends MapboxGLLayer {\n  constructor(props) {\n    super(props);\n    this.registerVisConfig(heatmapVisConfigs);\n    this.getPosition = memoize(pointPosAccessor, pointColResolver);\n  }\n\n  get type() {\n    return 'heatmap';\n  }\n\n  get visualChannels() {\n    return {\n      weight: {\n        property: 'weight',\n        field: 'weightField',\n        scale: 'weightScale',\n        domain: 'weightDomain',\n        key: 'weight',\n        // supportedFieldTypes can be determined by channelScaleType\n        // or specified here\n        defaultMeasure: 'property.density',\n        supportedFieldTypes: [ALL_FIELD_TYPES.real, ALL_FIELD_TYPES.integer],\n        channelScaleType: CHANNEL_SCALES.size\n      }\n    };\n  }\n\n  get layerIcon() {\n    return HeatmapLayerIcon;\n  }\n\n  getVisualChannelDescription(channel) {\n    return channel === 'color'\n      ? {\n          label: 'property.color',\n          measure: 'property.density'\n        }\n      : {\n          label: 'property.weight',\n          measure: this.config.weightField ? this.config.weightField.name : 'property.density'\n        };\n  }\n\n  getDefaultLayerConfig(props = {}) {\n    // mapbox heatmap layer color is always based on density\n    // no need to set colorField, colorDomain and colorScale\n    /* eslint-disable no-unused-vars */\n    const {colorField, colorDomain, colorScale, ...layerConfig} = {\n      ...super.getDefaultLayerConfig(props),\n\n      weightField: null,\n      weightDomain: [0, 1],\n      weightScale: 'linear'\n    };\n    /* eslint-enable no-unused-vars */\n\n    return layerConfig;\n  }\n\n  getPositionAccessor(dataContainer) {\n    return this.getPosition(this.config.columns)(dataContainer);\n  }\n\n  updateLayerMeta(dataContainer) {\n    const getPosition = this.getPositionAccessor(dataContainer);\n    const bounds = this.getPointsBounds(dataContainer, getPosition);\n    this.updateMeta({bounds});\n  }\n\n  columnsSelector = config => pointColResolver(config.columns);\n  visConfigSelector = config => config.visConfig;\n  weightFieldSelector = config => (config.weightField ? config.weightField.name : null);\n  weightDomainSelector = config => config.weightDomain;\n\n  paintSelector = createSelector(\n    this.visConfigSelector,\n    this.weightFieldSelector,\n    this.weightDomainSelector,\n    (visConfig, weightField, weightDomain) => ({\n      'heatmap-weight': weightField\n        ? ['interpolate', ['linear'], ['get', weightField], weightDomain[0], 0, weightDomain[1], 1]\n        : 1,\n      'heatmap-intensity': ['interpolate', ['linear'], ['zoom'], 0, 1, MAX_ZOOM_LEVEL, 3],\n      'heatmap-color': [\n        'interpolate',\n        ['linear'],\n        ['heatmap-density'],\n        ...heatmapDensity(visConfig.colorRange)\n      ],\n      'heatmap-radius': [\n        'interpolate',\n        ['linear'],\n        ['zoom'],\n        0,\n        2,\n        MAX_ZOOM_LEVEL,\n        visConfig.radius // radius\n      ],\n      'heatmap-opacity': visConfig.opacity\n    })\n  );\n\n  computeHeatmapConfiguration = createSelector(\n    this.sourceSelector,\n    this.filterSelector,\n    this.paintSelector,\n    (source, filter, paint) => {\n      return {\n        type: 'heatmap',\n        id: this.id,\n        source,\n        layout: {\n          visibility: 'visible'\n        },\n        maxzoom: MAX_ZOOM_LEVEL,\n        paint,\n        ...(this.isValidFilter(filter) ? {filter} : {})\n      };\n    }\n  );\n\n  getGeometry(position) {\n    return position.every(Number.isFinite)\n      ? {\n          type: 'Point',\n          coordinates: position\n        }\n      : null;\n  }\n\n  formatLayerData(datasets, oldLayerData) {\n    const {weightField} = this.config;\n    const {dataContainer} = datasets[this.config.dataId];\n    const getPosition = this.getPositionAccessor(dataContainer);\n    const {data} = this.updateData(datasets, oldLayerData);\n\n    const newConfig = this.computeHeatmapConfiguration(this.config, datasets);\n    newConfig.id = this.id;\n\n    return {\n      columns: this.config.columns,\n      config: newConfig,\n      data,\n      weightField,\n      getPosition\n    };\n  }\n}\n\nexport default HeatmapLayer;\n"]}