kepler.gl.geoiq
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
252 lines (215 loc) • 28.2 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = exports.heatmapVisConfigs = 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 _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
var _assertThisInitialized2 = _interopRequireDefault(require("@babel/runtime/helpers/assertThisInitialized"));
var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
var _get2 = _interopRequireDefault(require("@babel/runtime/helpers/get"));
var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray"));
var _reselect = require("reselect");
var _defaultSettings = require("../../constants/default-settings");
var _colorUtils = require("../../utils/color-utils");
var _mapboxUtils = require("../mapbox-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; }
var MAX_ZOOM_LEVEL = 18;
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 shouldRebuild = function shouldRebuild(sameData, sameConfig) {
return !(sameData && sameConfig);
};
var HeatmapLayer =
/*#__PURE__*/
function (_MapboxGLLayer) {
(0, _inherits2["default"])(HeatmapLayer, _MapboxGLLayer);
function HeatmapLayer(props) {
var _this;
(0, _classCallCheck2["default"])(this, HeatmapLayer);
_this = (0, _possibleConstructorReturn2["default"])(this, (0, _getPrototypeOf2["default"])(HeatmapLayer).call(this, props));
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "isSameData", function (_ref, config) {
var allData = _ref.allData,
filteredIndex = _ref.filteredIndex,
oldLayerData = _ref.oldLayerData,
_ref$opt = _ref.opt,
opt = _ref$opt === void 0 ? {} : _ref$opt;
return Boolean(oldLayerData && oldLayerData.columns === config.columns && opt.sameData);
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "isSameConfig", function (_ref2) {
var oldLayerData = _ref2.oldLayerData,
config = _ref2.config;
// columns must use the same filedIdx
// this is a fast way to compare columns object
var columns = config.columns,
weightField = config.weightField;
if (!oldLayerData) {
return false;
}
var sameColumns = columns === oldLayerData.columns;
var sameWeightField = weightField === oldLayerData.weightField;
return sameColumns && sameWeightField;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "datasetSelector", function (config) {
return config.dataId;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "isVisibleSelector", function (config) {
return config.isVisible;
});
(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), "computeHeatmapConfiguration", (0, _reselect.createSelector)(_this.datasetSelector, _this.isVisibleSelector, _this.visConfigSelector, _this.weightFieldSelector, _this.weightDomainSelector, function (datasetId, isVisible, visConfig, weightField, weightDomain) {
var layer = {
type: 'heatmap',
id: _this.id,
source: datasetId,
layout: {
visibility: isVisible ? 'visible' : 'none'
},
maxzoom: MAX_ZOOM_LEVEL,
paint: {
'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
}
};
return layer;
}));
_this.registerVisConfig(heatmapVisConfigs);
return _this;
}
(0, _createClass2["default"])(HeatmapLayer, [{
key: "getVisualChannelDescription",
value: function getVisualChannelDescription(channel) {
return channel === 'color' ? {
label: 'color',
measure: 'Density'
} : {
label: 'weight',
measure: this.config.weightField ? this.config.weightField.name : '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({}, (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: "formatLayerData",
value: function formatLayerData(_, allData, filteredIndex, oldLayerData) {
var opt = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : {};
var options = {
allData: allData,
filteredIndex: filteredIndex,
oldLayerData: oldLayerData,
opt: opt,
config: this.config
};
var weightField = this.config.weightField;
var isSameData = this.isSameData(options, this.config);
var isSameConfig = this.isSameConfig(options);
var data = !shouldRebuild(isSameData, isSameConfig) ? null : (0, _mapboxUtils.geojsonFromPoints)(allData, filteredIndex, this.config.columns, weightField ? [weightField] : []);
var newConfig = this.computeHeatmapConfiguration(this.config);
newConfig.id = this.id;
return {
columns: this.config.columns,
config: newConfig,
data: data,
weightField: weightField
};
}
}, {
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: '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"];
}
}]);
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","heatmapVisConfigs","opacity","colorRange","radius","heatmapDensity","scaleFunction","SCALE_FUNC","quantize","colors","scale","domain","range","colorDensity","reduce","bands","level","invert","invertExtent","join","shouldRebuild","sameData","sameConfig","HeatmapLayer","props","config","allData","filteredIndex","oldLayerData","opt","Boolean","columns","weightField","sameColumns","sameWeightField","dataId","isVisible","visConfig","name","weightDomain","datasetSelector","isVisibleSelector","visConfigSelector","weightFieldSelector","weightDomainSelector","datasetId","layer","type","id","source","layout","visibility","maxzoom","paint","registerVisConfig","channel","label","measure","weightScale","colorField","colorDomain","colorScale","layerConfig","_","options","isSameData","isSameConfig","data","newConfig","computeHeatmapConfiguration","weight","property","field","key","defaultMeasure","supportedFieldTypes","ALL_FIELD_TYPES","real","integer","channelScaleType","CHANNEL_SCALES","size","HeatmapLayerIcon","MapboxGLLayer"],"mappings":"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","sourcesContent":["// Copyright (c) 2019 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 {CHANNEL_SCALES, SCALE_FUNC, ALL_FIELD_TYPES} from 'constants/default-settings';\nimport {hexToRgb} from 'utils/color-utils';\nimport {geojsonFromPoints} from '../mapbox-utils';\nimport MapboxGLLayer from '../mapboxgl-layer';\nimport HeatmapLayerIcon from './heatmap-layer-icon';\n\nconst MAX_ZOOM_LEVEL = 18;\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\nconst shouldRebuild = (sameData, sameConfig) => !(sameData && sameConfig);\n\nclass HeatmapLayer extends MapboxGLLayer {\n  constructor(props) {\n    super(props);\n    this.registerVisConfig(heatmapVisConfigs);\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: '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      label: 'color',\n      measure: 'Density'\n    } : {\n      label: 'weight',\n      measure: this.config.weightField ? this.config.weightField.name : 'Density'\n    }\n  }\n\n  getDefaultLayerConfig(props = {}) {\n\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  isSameData = ({allData, filteredIndex, oldLayerData, opt = {}}, config) => {\n    return Boolean(oldLayerData && oldLayerData.columns === config.columns &&\n      opt.sameData\n    );\n  };\n\n  isSameConfig = ({oldLayerData, config}) => {\n    // columns must use the same filedIdx\n    // this is a fast way to compare columns object\n    const {\n      columns,\n      weightField\n    } = config;\n\n    if (!oldLayerData) {\n      return false;\n    }\n\n    const sameColumns = columns === oldLayerData.columns;\n    const sameWeightField = weightField === oldLayerData.weightField;\n    return sameColumns && sameWeightField;\n  };\n\n  datasetSelector = config => config.dataId;\n  isVisibleSelector = config => config.isVisible;\n  visConfigSelector = config => config.visConfig;\n  weightFieldSelector = config => config.weightField ? config.weightField.name : null;\n  weightDomainSelector = config => config.weightDomain;\n\n  computeHeatmapConfiguration = createSelector(\n    this.datasetSelector,\n    this.isVisibleSelector,\n    this.visConfigSelector,\n    this.weightFieldSelector,\n    this.weightDomainSelector,\n\n    (datasetId, isVisible, visConfig, weightField, weightDomain) => {\n\n      const layer = {\n        type: 'heatmap',\n        id: this.id,\n        source: datasetId,\n        layout: {\n          visibility: isVisible ? 'visible' : 'none'\n        },\n        maxzoom: MAX_ZOOM_LEVEL,\n        paint: {\n          'heatmap-weight': weightField ? [\n            'interpolate',\n            ['linear'],\n            ['get', weightField],\n            weightDomain[0], 0,\n            weightDomain[1], 1\n          ] : 1,\n          'heatmap-intensity': [\n            'interpolate',\n            ['linear'],\n            ['zoom'],\n            0, 1,\n            MAX_ZOOM_LEVEL, 3\n          ],\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, 2,\n            MAX_ZOOM_LEVEL, visConfig.radius // radius\n          ],\n          'heatmap-opacity': visConfig.opacity\n        }\n      };\n\n      return layer;\n    }\n  );\n\n  formatLayerData(_, allData, filteredIndex, oldLayerData, opt = {}) {\n    const options = {\n      allData,\n      filteredIndex,\n      oldLayerData,\n      opt,\n      config: this.config\n    };\n\n    const {weightField} = this.config;\n    const isSameData = this.isSameData(options, this.config);\n    const isSameConfig = this.isSameConfig(options);\n\n    const data = !shouldRebuild(isSameData, isSameConfig) ?\n      null :\n      geojsonFromPoints(\n        allData,\n        filteredIndex,\n        this.config.columns,\n        weightField ? [weightField] : []\n      );\n\n    const newConfig = this.computeHeatmapConfiguration(this.config);\n    newConfig.id = this.id;\n\n    return {\n      columns: this.config.columns,\n      config: newConfig,\n      data,\n      weightField\n    };\n  }\n}\n\nexport default HeatmapLayer;\n"]}