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
JavaScript
;
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"]}