UNPKG

kepler.gl

Version:

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

390 lines (342 loc) 41.8 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.aggregateRequiredColumns = exports.getFilterDataFunc = exports.getValueAggrFunc = exports.pointPosResolver = exports.pointPosAccessor = void 0; var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray")); 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 _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn")); 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 _lodash = _interopRequireDefault(require("lodash.memoize")); var _baseLayer = _interopRequireDefault(require("./base-layer")); var _colorUtils = require("../utils/color-utils"); var _aggregateUtils = require("../utils/aggregate-utils"); var _defaultSettings = require("../constants/default-settings"); 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 pointPosAccessor = function pointPosAccessor(_ref) { var lat = _ref.lat, lng = _ref.lng; return function (d) { return [d.data[lng.fieldIdx], d.data[lat.fieldIdx]]; }; }; exports.pointPosAccessor = pointPosAccessor; var pointPosResolver = function pointPosResolver(_ref2) { var lat = _ref2.lat, lng = _ref2.lng; return "".concat(lat.fieldIdx, "-").concat(lng.fieldIdx); }; exports.pointPosResolver = pointPosResolver; var getValueAggrFunc = function getValueAggrFunc(field, aggregation) { return function (points) { return field ? (0, _aggregateUtils.aggregate)(points.map(function (p) { return p.data[field.tableFieldIndex - 1]; }), aggregation) : points.length; }; }; exports.getValueAggrFunc = getValueAggrFunc; var getFilterDataFunc = function getFilterDataFunc(filterRange, getFilterValue) { return function (pt) { return getFilterValue(pt).every(function (val, i) { return val >= filterRange[i][0] && val <= filterRange[i][1]; }); }; }; exports.getFilterDataFunc = getFilterDataFunc; var getLayerColorRange = function getLayerColorRange(colorRange) { return colorRange.colors.map(_colorUtils.hexToRgb); }; var aggregateRequiredColumns = ['lat', 'lng']; exports.aggregateRequiredColumns = aggregateRequiredColumns; var AggregationLayer = /*#__PURE__*/ function (_Layer) { (0, _inherits2["default"])(AggregationLayer, _Layer); function AggregationLayer(props) { var _this; (0, _classCallCheck2["default"])(this, AggregationLayer); _this = (0, _possibleConstructorReturn2["default"])(this, (0, _getPrototypeOf2["default"])(AggregationLayer).call(this, props)); _this.getPositionAccessor = function () { return pointPosAccessor(_this.config.columns); }; _this.getColorRange = (0, _lodash["default"])(getLayerColorRange); return _this; } (0, _createClass2["default"])(AggregationLayer, [{ key: "getVisualChannelDescription", /** * Get the description of a visualChannel config * @param key * @returns {{label: string, measure: (string|string)}} */ value: function getVisualChannelDescription(key) { // e.g. label: Color, measure: Average of ETA var _this$visualChannels$ = this.visualChannels[key], range = _this$visualChannels$.range, field = _this$visualChannels$.field, defaultMeasure = _this$visualChannels$.defaultMeasure, aggregation = _this$visualChannels$.aggregation; return { label: this.visConfigSettings[range].label, measure: this.config[field] ? "".concat(this.config.visConfig[aggregation], " of ").concat(this.config[field].name) : defaultMeasure }; } }, { key: "getHoverData", value: function getHoverData(object) { // return aggregated object return object; } /** * Aggregation layer handles visual channel aggregation inside deck.gl layer */ }, { key: "updateLayerVisualChannel", value: function updateLayerVisualChannel(_ref3, channel) { var data = _ref3.data, allData = _ref3.allData; this.validateVisualChannel(channel); } /** * Validate aggregation type on top of basic layer visual channel validation * @param channel */ }, { key: "validateVisualChannel", value: function validateVisualChannel(channel) { // field type decides aggregation type decides scale type this.validateFieldType(channel); this.validateAggregationType(channel); this.validateScale(channel); } /** * Validate aggregation type based on selected field */ }, { key: "validateAggregationType", value: function validateAggregationType(channel) { var visualChannel = this.visualChannels[channel]; var field = visualChannel.field, aggregation = visualChannel.aggregation; var aggregationOptions = this.getAggregationOptions(channel); if (!aggregation) { return; } if (!aggregationOptions.length) { // if field cannot be aggregated, set field to null this.updateLayerConfig((0, _defineProperty2["default"])({}, field, null)); } else if (!aggregationOptions.includes(this.config.visConfig[aggregation])) { // current aggregation type is not supported by this field // set aggregation to the first supported option this.updateLayerVisConfig((0, _defineProperty2["default"])({}, aggregation, aggregationOptions[0])); } } }, { key: "getAggregationOptions", value: function getAggregationOptions(channel) { var visualChannel = this.visualChannels[channel]; var field = visualChannel.field, channelScaleType = visualChannel.channelScaleType; return Object.keys(this.config[field] ? _defaultSettings.FIELD_OPTS[this.config[field].type].scale[channelScaleType] : _defaultSettings.DEFAULT_AGGREGATION[channelScaleType]); } /** * Get scale options based on current field and aggregation type * @param {string} channel * @returns {string[]} */ }, { key: "getScaleOptions", value: function getScaleOptions(channel) { var visualChannel = this.visualChannels[channel]; var field = visualChannel.field, aggregation = visualChannel.aggregation, channelScaleType = visualChannel.channelScaleType; var aggregationType = this.config.visConfig[aggregation]; return this.config[field] ? // scale options based on aggregation _defaultSettings.FIELD_OPTS[this.config[field].type].scale[channelScaleType][aggregationType] : // default scale options for point count _defaultSettings.DEFAULT_AGGREGATION[channelScaleType][aggregationType]; } /** * Aggregation layer handles visual channel aggregation inside deck.gl layer */ }, { key: "updateLayerDomain", value: function updateLayerDomain(datasets, newFilter) { return this; } }, { key: "updateLayerMeta", value: function updateLayerMeta(allData, getPosition) { // get bounds from points var bounds = this.getPointsBounds(allData, function (d) { return getPosition({ data: d }); }); this.updateMeta({ bounds: bounds }); } }, { key: "calculateDataAttribute", value: function calculateDataAttribute(_ref4, getPosition) { var allData = _ref4.allData, filteredIndex = _ref4.filteredIndex; var data = []; for (var i = 0; i < filteredIndex.length; i++) { var index = filteredIndex[i]; var pos = getPosition({ data: allData[index] }); // if doesn't have point lat or lng, do not add the point // deck.gl can't handle position = null if (pos.every(Number.isFinite)) { data.push({ index: index, data: allData[index] }); } } return data; } }, { key: "formatLayerData", value: function formatLayerData(datasets, oldLayerData) { var getPosition = this.getPositionAccessor(); // if ( var gpuFilter = datasets[this.config.dataId].gpuFilter; var getColorValue = getValueAggrFunc(this.config.colorField, this.config.visConfig.colorAggregation); var getElevationValue = getValueAggrFunc(this.config.sizeField, this.config.visConfig.sizeAggregation); var hasFilter = Object.values(gpuFilter.filterRange).some(function (arr) { return arr.some(function (v) { return v !== 0; }); }); var getFilterValue = gpuFilter.filterValueAccessor(); var filterData = hasFilter ? getFilterDataFunc(gpuFilter.filterRange, getFilterValue) : undefined; var _this$updateData = this.updateData(datasets, oldLayerData), data = _this$updateData.data; return _objectSpread({ data: data, getPosition: getPosition, _filterData: filterData }, getColorValue ? { getColorValue: getColorValue } : {}, {}, getElevationValue ? { getElevationValue: getElevationValue } : {}); } }, { key: "getDefaultDeckLayerProps", value: function getDefaultDeckLayerProps(opts) { var baseProp = (0, _get2["default"])((0, _getPrototypeOf2["default"])(AggregationLayer.prototype), "getDefaultDeckLayerProps", this).call(this, opts); return _objectSpread({}, baseProp, { highlightColor: _defaultSettings.HIGHLIGH_COLOR_3D, // gpu data filtering is not supported in aggregation layer extensions: [], autoHighlight: this.config.visConfig.enable3d }); } }, { key: "getDefaultAggregationLayerProp", value: function getDefaultAggregationLayerProp(opts) { var gpuFilter = opts.gpuFilter, mapState = opts.mapState, _opts$layerCallbacks = opts.layerCallbacks, layerCallbacks = _opts$layerCallbacks === void 0 ? {} : _opts$layerCallbacks; var visConfig = this.config.visConfig; var eleZoomFactor = this.getElevationZoomFactor(mapState); var updateTriggers = { getColorValue: { colorField: this.config.colorField, colorAggregation: this.config.visConfig.colorAggregation }, getElevationValue: { sizeField: this.config.sizeField, sizeAggregation: this.config.visConfig.sizeAggregation }, _filterData: _objectSpread({ filterRange: gpuFilter.filterRange }, gpuFilter.filterValueUpdateTriggers) }; return _objectSpread({}, this.getDefaultDeckLayerProps(opts), { coverage: visConfig.coverage, // color colorRange: this.getColorRange(visConfig.colorRange), colorScaleType: this.config.colorScale, upperPercentile: visConfig.percentile[1], lowerPercentile: visConfig.percentile[0], colorAggregation: visConfig.colorAggregation, // elevation extruded: visConfig.enable3d, elevationScale: visConfig.elevationScale * eleZoomFactor, elevationScaleType: this.config.sizeScale, elevationRange: visConfig.sizeRange, elevationLowerPercentile: visConfig.elevationPercentile[0], elevationUpperPercentile: visConfig.elevationPercentile[1], // updateTriggers updateTriggers: updateTriggers, // callbacks onSetColorDomain: layerCallbacks.onSetLayerDomain }); } }, { key: "isAggregated", get: function get() { return true; } }, { key: "requiredLayerColumns", get: function get() { return aggregateRequiredColumns; } }, { key: "columnPairs", get: function get() { return this.defaultPointColumnPairs; } }, { key: "noneLayerDataAffectingProps", get: function get() { return [].concat((0, _toConsumableArray2["default"])((0, _get2["default"])((0, _getPrototypeOf2["default"])(AggregationLayer.prototype), "noneLayerDataAffectingProps", this)), ['enable3d', 'colorRange', 'colorDomain', 'sizeRange', 'sizeScale', 'sizeDomain', 'percentile', 'coverage', 'elevationPercentile', 'elevationScale']); } }, { key: "visualChannels", get: function get() { return { color: { aggregation: 'colorAggregation', channelScaleType: _defaultSettings.CHANNEL_SCALES.colorAggr, defaultMeasure: 'property.pointCount', domain: 'colorDomain', field: 'colorField', key: 'color', property: 'color', range: 'colorRange', scale: 'colorScale' }, size: { aggregation: 'sizeAggregation', channelScaleType: _defaultSettings.CHANNEL_SCALES.sizeAggr, condition: function condition(config) { return config.visConfig.enable3d; }, defaultMeasure: 'property.pointCount', domain: 'sizeDomain', field: 'sizeField', key: 'size', property: 'height', range: 'sizeRange', scale: 'sizeScale' } }; } }]); return AggregationLayer; }(_baseLayer["default"]); exports["default"] = AggregationLayer; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/layers/aggregation-layer.js"],"names":["pointPosAccessor","lat","lng","d","data","fieldIdx","pointPosResolver","getValueAggrFunc","field","aggregation","points","map","p","tableFieldIndex","length","getFilterDataFunc","filterRange","getFilterValue","pt","every","val","i","getLayerColorRange","colorRange","colors","hexToRgb","aggregateRequiredColumns","AggregationLayer","props","getPositionAccessor","config","columns","getColorRange","key","visualChannels","range","defaultMeasure","label","visConfigSettings","measure","visConfig","name","object","channel","allData","validateVisualChannel","validateFieldType","validateAggregationType","validateScale","visualChannel","aggregationOptions","getAggregationOptions","updateLayerConfig","includes","updateLayerVisConfig","channelScaleType","Object","keys","FIELD_OPTS","type","scale","DEFAULT_AGGREGATION","aggregationType","datasets","newFilter","getPosition","bounds","getPointsBounds","updateMeta","filteredIndex","index","pos","Number","isFinite","push","oldLayerData","gpuFilter","dataId","getColorValue","colorField","colorAggregation","getElevationValue","sizeField","sizeAggregation","hasFilter","values","some","arr","v","filterValueAccessor","filterData","undefined","updateData","_filterData","opts","baseProp","highlightColor","HIGHLIGH_COLOR_3D","extensions","autoHighlight","enable3d","mapState","layerCallbacks","eleZoomFactor","getElevationZoomFactor","updateTriggers","filterValueUpdateTriggers","getDefaultDeckLayerProps","coverage","colorScaleType","colorScale","upperPercentile","percentile","lowerPercentile","extruded","elevationScale","elevationScaleType","sizeScale","elevationRange","sizeRange","elevationLowerPercentile","elevationPercentile","elevationUpperPercentile","onSetColorDomain","onSetLayerDomain","defaultPointColumnPairs","color","CHANNEL_SCALES","colorAggr","domain","property","size","sizeAggr","condition","Layer"],"mappings":"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","sourcesContent":["// Copyright (c) 2020 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 memoize from 'lodash.memoize';\nimport Layer from './base-layer';\nimport {hexToRgb} from 'utils/color-utils';\nimport {aggregate} from 'utils/aggregate-utils';\nimport {HIGHLIGH_COLOR_3D} from 'constants/default-settings';\n\nimport {CHANNEL_SCALES, FIELD_OPTS, DEFAULT_AGGREGATION} from 'constants/default-settings';\n\nexport const pointPosAccessor = ({lat, lng}) => d => [d.data[lng.fieldIdx], d.data[lat.fieldIdx]];\n\nexport const pointPosResolver = ({lat, lng}) => `${lat.fieldIdx}-${lng.fieldIdx}`;\n\nexport const getValueAggrFunc = (field, aggregation) => {\n  return points => {\n    return field\n      ? aggregate(\n          points.map(p => p.data[field.tableFieldIndex - 1]),\n          aggregation\n        )\n      : points.length;\n  };\n};\n\nexport const getFilterDataFunc = (filterRange, getFilterValue) => pt =>\n  getFilterValue(pt).every((val, i) => val >= filterRange[i][0] && val <= filterRange[i][1]);\n\nconst getLayerColorRange = colorRange => colorRange.colors.map(hexToRgb);\n\nexport const aggregateRequiredColumns = ['lat', 'lng'];\n\nexport default class AggregationLayer extends Layer {\n  constructor(props) {\n    super(props);\n\n    this.getPositionAccessor = () => pointPosAccessor(this.config.columns);\n    this.getColorRange = memoize(getLayerColorRange);\n  }\n\n  get isAggregated() {\n    return true;\n  }\n\n  get requiredLayerColumns() {\n    return aggregateRequiredColumns;\n  }\n\n  get columnPairs() {\n    return this.defaultPointColumnPairs;\n  }\n\n  get noneLayerDataAffectingProps() {\n    return [\n      ...super.noneLayerDataAffectingProps,\n      'enable3d',\n      'colorRange',\n      'colorDomain',\n      'sizeRange',\n      'sizeScale',\n      'sizeDomain',\n      'percentile',\n      'coverage',\n      'elevationPercentile',\n      'elevationScale'\n    ];\n  }\n\n  get visualChannels() {\n    return {\n      color: {\n        aggregation: 'colorAggregation',\n        channelScaleType: CHANNEL_SCALES.colorAggr,\n        defaultMeasure: 'property.pointCount',\n        domain: 'colorDomain',\n        field: 'colorField',\n        key: 'color',\n        property: 'color',\n        range: 'colorRange',\n        scale: 'colorScale'\n      },\n      size: {\n        aggregation: 'sizeAggregation',\n        channelScaleType: CHANNEL_SCALES.sizeAggr,\n        condition: config => config.visConfig.enable3d,\n        defaultMeasure: 'property.pointCount',\n        domain: 'sizeDomain',\n        field: 'sizeField',\n        key: 'size',\n        property: 'height',\n        range: 'sizeRange',\n        scale: 'sizeScale'\n      }\n    };\n  }\n\n  /**\n   * Get the description of a visualChannel config\n   * @param key\n   * @returns {{label: string, measure: (string|string)}}\n   */\n  getVisualChannelDescription(key) {\n    // e.g. label: Color, measure: Average of ETA\n    const {range, field, defaultMeasure, aggregation} = this.visualChannels[key];\n    return {\n      label: this.visConfigSettings[range].label,\n      measure: this.config[field]\n        ? `${this.config.visConfig[aggregation]} of ${this.config[field].name}`\n        : defaultMeasure\n    };\n  }\n\n  getHoverData(object) {\n    // return aggregated object\n    return object;\n  }\n\n  /**\n   * Aggregation layer handles visual channel aggregation inside deck.gl layer\n   */\n  updateLayerVisualChannel({data, allData}, channel) {\n    this.validateVisualChannel(channel);\n  }\n\n  /**\n   * Validate aggregation type on top of basic layer visual channel validation\n   * @param channel\n   */\n  validateVisualChannel(channel) {\n    // field type decides aggregation type decides scale type\n    this.validateFieldType(channel);\n    this.validateAggregationType(channel);\n    this.validateScale(channel);\n  }\n\n  /**\n   * Validate aggregation type based on selected field\n   */\n  validateAggregationType(channel) {\n    const visualChannel = this.visualChannels[channel];\n    const {field, aggregation} = visualChannel;\n    const aggregationOptions = this.getAggregationOptions(channel);\n\n    if (!aggregation) {\n      return;\n    }\n\n    if (!aggregationOptions.length) {\n      // if field cannot be aggregated, set field to null\n      this.updateLayerConfig({[field]: null});\n    } else if (!aggregationOptions.includes(this.config.visConfig[aggregation])) {\n      // current aggregation type is not supported by this field\n      // set aggregation to the first supported option\n      this.updateLayerVisConfig({[aggregation]: aggregationOptions[0]});\n    }\n  }\n\n  getAggregationOptions(channel) {\n    const visualChannel = this.visualChannels[channel];\n    const {field, channelScaleType} = visualChannel;\n\n    return Object.keys(\n      this.config[field]\n        ? FIELD_OPTS[this.config[field].type].scale[channelScaleType]\n        : DEFAULT_AGGREGATION[channelScaleType]\n    );\n  }\n\n  /**\n   * Get scale options based on current field and aggregation type\n   * @param {string} channel\n   * @returns {string[]}\n   */\n  getScaleOptions(channel) {\n    const visualChannel = this.visualChannels[channel];\n    const {field, aggregation, channelScaleType} = visualChannel;\n    const aggregationType = this.config.visConfig[aggregation];\n    return this.config[field]\n      ? // scale options based on aggregation\n        FIELD_OPTS[this.config[field].type].scale[channelScaleType][aggregationType]\n      : // default scale options for point count\n        DEFAULT_AGGREGATION[channelScaleType][aggregationType];\n  }\n\n  /**\n   * Aggregation layer handles visual channel aggregation inside deck.gl layer\n   */\n  updateLayerDomain(datasets, newFilter) {\n    return this;\n  }\n\n  updateLayerMeta(allData, getPosition) {\n    // get bounds from points\n    const bounds = this.getPointsBounds(allData, d => getPosition({data: d}));\n\n    this.updateMeta({bounds});\n  }\n\n  calculateDataAttribute({allData, filteredIndex}, getPosition) {\n    const data = [];\n\n    for (let i = 0; i < filteredIndex.length; i++) {\n      const index = filteredIndex[i];\n      const pos = getPosition({data: allData[index]});\n\n      // if doesn't have point lat or lng, do not add the point\n      // deck.gl can't handle position = null\n      if (pos.every(Number.isFinite)) {\n        data.push({\n          index,\n          data: allData[index]\n        });\n      }\n    }\n\n    return data;\n  }\n\n  formatLayerData(datasets, oldLayerData) {\n    const getPosition = this.getPositionAccessor(); // if (\n    const {gpuFilter} = datasets[this.config.dataId];\n\n    const getColorValue = getValueAggrFunc(\n      this.config.colorField,\n      this.config.visConfig.colorAggregation\n    );\n\n    const getElevationValue = getValueAggrFunc(\n      this.config.sizeField,\n      this.config.visConfig.sizeAggregation\n    );\n    const hasFilter = Object.values(gpuFilter.filterRange).some(arr => arr.some(v => v !== 0));\n    const getFilterValue = gpuFilter.filterValueAccessor();\n    const filterData = hasFilter\n      ? getFilterDataFunc(gpuFilter.filterRange, getFilterValue)\n      : undefined;\n\n    const {data} = this.updateData(datasets, oldLayerData);\n\n    return {\n      data,\n      getPosition,\n      _filterData: filterData,\n      ...(getColorValue ? {getColorValue} : {}),\n      ...(getElevationValue ? {getElevationValue} : {})\n    };\n  }\n\n  getDefaultDeckLayerProps(opts) {\n    const baseProp = super.getDefaultDeckLayerProps(opts);\n    return {\n      ...baseProp,\n      highlightColor: HIGHLIGH_COLOR_3D,\n      // gpu data filtering is not supported in aggregation layer\n      extensions: [],\n      autoHighlight: this.config.visConfig.enable3d\n    };\n  }\n\n  getDefaultAggregationLayerProp(opts) {\n    const {gpuFilter, mapState, layerCallbacks = {}} = opts;\n    const {visConfig} = this.config;\n    const eleZoomFactor = this.getElevationZoomFactor(mapState);\n\n    const updateTriggers = {\n      getColorValue: {\n        colorField: this.config.colorField,\n        colorAggregation: this.config.visConfig.colorAggregation\n      },\n      getElevationValue: {\n        sizeField: this.config.sizeField,\n        sizeAggregation: this.config.visConfig.sizeAggregation\n      },\n      _filterData: {\n        filterRange: gpuFilter.filterRange,\n        ...gpuFilter.filterValueUpdateTriggers\n      }\n    };\n\n    return {\n      ...this.getDefaultDeckLayerProps(opts),\n      coverage: visConfig.coverage,\n\n      // color\n      colorRange: this.getColorRange(visConfig.colorRange),\n      colorScaleType: this.config.colorScale,\n      upperPercentile: visConfig.percentile[1],\n      lowerPercentile: visConfig.percentile[0],\n      colorAggregation: visConfig.colorAggregation,\n\n      // elevation\n      extruded: visConfig.enable3d,\n      elevationScale: visConfig.elevationScale * eleZoomFactor,\n      elevationScaleType: this.config.sizeScale,\n      elevationRange: visConfig.sizeRange,\n      elevationLowerPercentile: visConfig.elevationPercentile[0],\n      elevationUpperPercentile: visConfig.elevationPercentile[1],\n\n      // updateTriggers\n      updateTriggers,\n\n      // callbacks\n      onSetColorDomain: layerCallbacks.onSetLayerDomain\n    };\n  }\n}\n"]}