kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
400 lines (340 loc) • 41.8 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = exports.pointVisConfigs = exports.iconRequiredColumns = exports.iconAccessor = exports.iconPosAccessor = exports.SVG_ICON_URL = 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 _window = _interopRequireDefault(require("global/window"));
var _extensions = require("@deck.gl/extensions");
var _colorUtils = require("../../utils/color-utils");
var _svgIconLayer = _interopRequireDefault(require("../../deckgl-layers/svg-icon-layer/svg-icon-layer"));
var _iconLayerIcon = _interopRequireDefault(require("./icon-layer-icon"));
var _defaultSettings = require("../../constants/default-settings");
var _iconInfoModal = _interopRequireDefault(require("./icon-info-modal"));
var _baseLayer = _interopRequireDefault(require("../base-layer"));
var _layerTextLabel = require("../layer-text-label");
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 brushingExtension = new _extensions.BrushingExtension();
var SVG_ICON_URL = "".concat(_defaultSettings.CLOUDFRONT, "/icons/svg-icons.json");
exports.SVG_ICON_URL = SVG_ICON_URL;
var iconPosAccessor = function iconPosAccessor(_ref) {
var lat = _ref.lat,
lng = _ref.lng;
return function (d) {
return [d.data[lng.fieldIdx], d.data[lat.fieldIdx]];
};
};
exports.iconPosAccessor = iconPosAccessor;
var iconAccessor = function iconAccessor(_ref2) {
var icon = _ref2.icon;
return function (d) {
return d.data[icon.fieldIdx];
};
};
exports.iconAccessor = iconAccessor;
var iconRequiredColumns = ['lat', 'lng', 'icon'];
exports.iconRequiredColumns = iconRequiredColumns;
var pointVisConfigs = {
radius: 'radius',
fixedRadius: 'fixedRadius',
opacity: 'opacity',
colorRange: 'colorRange',
radiusRange: 'radiusRange'
};
exports.pointVisConfigs = pointVisConfigs;
function flatterIconPositions(icon) {
// had to flip y, since @luma modal has changed
return icon.mesh.cells.reduce(function (prev, cell) {
cell.forEach(function (p) {
prev.push.apply(prev, [icon.mesh.positions[p][0], -icon.mesh.positions[p][1], icon.mesh.positions[p][2]]);
});
return prev;
}, []);
}
var IconLayer =
/*#__PURE__*/
function (_Layer) {
(0, _inherits2["default"])(IconLayer, _Layer);
function IconLayer() {
var _this;
var props = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
(0, _classCallCheck2["default"])(this, IconLayer);
_this = (0, _possibleConstructorReturn2["default"])(this, (0, _getPrototypeOf2["default"])(IconLayer).call(this, props));
_this.registerVisConfig(pointVisConfigs);
_this.getPositionAccessor = function () {
return iconPosAccessor(_this.config.columns);
};
_this.getIconAccessor = function () {
return iconAccessor(_this.config.columns);
}; // prepare layer info modal
_this._layerInfoModal = (0, _iconInfoModal["default"])();
_this.iconGeometry = props.iconGeometry || null;
_this.getSvgIcons();
return _this;
}
(0, _createClass2["default"])(IconLayer, [{
key: "getSvgIcons",
value: function getSvgIcons() {
var _this2 = this;
var fetchConfig = {
method: 'GET',
mode: 'cors',
cache: 'no-cache'
};
if (_window["default"].fetch) {
_window["default"].fetch(SVG_ICON_URL, fetchConfig).then(function (response) {
return response.json();
}).then(function () {
var parsed = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
var _parsed$svgIcons = parsed.svgIcons,
svgIcons = _parsed$svgIcons === void 0 ? [] : _parsed$svgIcons;
_this2.iconGeometry = svgIcons.reduce(function (accu, curr) {
return _objectSpread({}, accu, (0, _defineProperty2["default"])({}, curr.id, flatterIconPositions(curr)));
}, {});
_this2._layerInfoModal = (0, _iconInfoModal["default"])(svgIcons);
});
}
}
}, {
key: "calculateDataAttribute",
value: function calculateDataAttribute(_ref3, getPosition) {
var allData = _ref3.allData,
filteredIndex = _ref3.filteredIndex;
var getIcon = this.getIconAccessor();
var data = [];
for (var i = 0; i < filteredIndex.length; i++) {
var index = filteredIndex[i];
var pos = getPosition({
data: allData[index]
});
var icon = getIcon({
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) && typeof icon === 'string') {
data.push({
index: index,
icon: icon,
data: allData[index]
});
}
}
return data;
}
}, {
key: "formatLayerData",
value: function formatLayerData(datasets, oldLayerData) {
var _this3 = this;
var opt = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {};
var _this$config = this.config,
colorScale = _this$config.colorScale,
colorDomain = _this$config.colorDomain,
colorField = _this$config.colorField,
color = _this$config.color,
sizeField = _this$config.sizeField,
sizeScale = _this$config.sizeScale,
sizeDomain = _this$config.sizeDomain,
textLabel = _this$config.textLabel,
_this$config$visConfi = _this$config.visConfig,
radiusRange = _this$config$visConfi.radiusRange,
colorRange = _this$config$visConfi.colorRange;
var getPosition = this.getPositionAccessor();
var gpuFilter = datasets[this.config.dataId].gpuFilter;
var _this$updateData = this.updateData(datasets, oldLayerData),
data = _this$updateData.data,
triggerChanged = _this$updateData.triggerChanged; // point color
var cScale = colorField && this.getVisChannelScale(colorScale, colorDomain, colorRange.colors.map(_colorUtils.hexToRgb)); // point radius
var rScale = sizeField && this.getVisChannelScale(sizeScale, sizeDomain, radiusRange, 0);
var getRadius = rScale ? function (d) {
return _this3.getEncodedChannelValue(rScale, d.data, sizeField);
} : 1;
var getFillColor = cScale ? function (d) {
return _this3.getEncodedChannelValue(cScale, d.data, colorField);
} : color; // get all distinct characters in the text labels
var textLabels = (0, _layerTextLabel.formatTextLabelData)({
textLabel: textLabel,
triggerChanged: triggerChanged,
oldLayerData: oldLayerData,
data: data
});
return {
data: data,
getPosition: getPosition,
getFillColor: getFillColor,
getFilterValue: gpuFilter.filterValueAccessor(),
getRadius: getRadius,
textLabels: textLabels
};
}
}, {
key: "updateLayerMeta",
value: function updateLayerMeta(allData, getPosition) {
var bounds = this.getPointsBounds(allData, function (d) {
return getPosition({
data: d
});
});
this.updateMeta({
bounds: bounds
});
}
}, {
key: "renderLayer",
value: function renderLayer(opts) {
var _this4 = this;
var data = opts.data,
gpuFilter = opts.gpuFilter,
objectHovered = opts.objectHovered,
mapState = opts.mapState,
interactionConfig = opts.interactionConfig;
var radiusScale = this.getRadiusScaleByZoom(mapState);
var layerProps = _objectSpread({
radiusScale: radiusScale
}, this.config.visConfig.fixedRadius ? {} : {
radiusMaxPixels: 500
});
var updateTriggers = {
getFilterValue: gpuFilter.filterValueUpdateTriggers,
getRadius: {
sizeField: this.config.colorField,
radiusRange: this.config.visConfig.radiusRange,
sizeScale: this.config.sizeScale
},
getFillColor: {
color: this.config.color,
colorField: this.config.colorField,
colorRange: this.config.visConfig.colorRange,
colorScale: this.config.colorScale
}
};
var defaultLayerProps = this.getDefaultDeckLayerProps(opts);
var brushingProps = this.getBrushingExtensionProps(interactionConfig);
var getPixelOffset = (0, _layerTextLabel.getTextOffsetByRadius)(radiusScale, data.getRadius, mapState);
var extensions = [].concat((0, _toConsumableArray2["default"])(defaultLayerProps.extensions), [brushingExtension]); // shared Props between layer and label layer
var sharedProps = _objectSpread({
getFilterValue: data.getFilterValue,
extensions: extensions,
filterRange: defaultLayerProps.filterRange
}, brushingProps);
var labelLayers = (0, _toConsumableArray2["default"])(this.renderTextLabelLayer({
getPosition: data.getPosition,
sharedProps: sharedProps,
getPixelOffset: getPixelOffset,
updateTriggers: updateTriggers
}, opts));
return !this.iconGeometry ? [] : [new _svgIconLayer["default"](_objectSpread({}, defaultLayerProps, {}, brushingProps, {}, layerProps, {}, data, {
getIconGeometry: function getIconGeometry(id) {
return _this4.iconGeometry[id];
},
// update triggers
updateTriggers: updateTriggers,
extensions: extensions
}))].concat((0, _toConsumableArray2["default"])(this.isLayerHovered(objectHovered) ? [new _svgIconLayer["default"](_objectSpread({}, this.getDefaultHoverLayerProps(), {}, layerProps, {
data: [objectHovered.object],
getPosition: data.getPosition,
getRadius: data.getRadius,
getFillColor: this.config.highlightColor,
getIconGeometry: function getIconGeometry(id) {
return _this4.iconGeometry[id];
}
}))] : []), (0, _toConsumableArray2["default"])(labelLayers));
}
}, {
key: "type",
get: function get() {
return 'icon';
}
}, {
key: "requiredLayerColumns",
get: function get() {
return iconRequiredColumns;
}
}, {
key: "columnPairs",
get: function get() {
return this.defaultPointColumnPairs;
}
}, {
key: "layerIcon",
get: function get() {
return _iconLayerIcon["default"];
}
}, {
key: "visualChannels",
get: function get() {
return _objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(IconLayer.prototype), "visualChannels", this), {
size: _objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(IconLayer.prototype), "visualChannels", this).size, {
range: 'radiusRange',
property: 'radius',
channelScaleType: 'radius'
})
});
}
}, {
key: "layerInfoModal",
get: function get() {
return {
id: 'iconInfo',
template: this._layerInfoModal,
modalProps: {
title: 'modal.iconInfo.title'
}
};
}
}], [{
key: "findDefaultLayerProps",
value: function findDefaultLayerProps(_ref4) {
var _ref4$fieldPairs = _ref4.fieldPairs,
fieldPairs = _ref4$fieldPairs === void 0 ? [] : _ref4$fieldPairs,
_ref4$fields = _ref4.fields,
fields = _ref4$fields === void 0 ? [] : _ref4$fields;
var notFound = {
props: []
};
if (!fieldPairs.length || !fields.length) {
return notFound;
}
var iconFields = fields.filter(function (_ref5) {
var name = _ref5.name;
return name.replace(/[_,.]+/g, ' ').trim().split(' ').some(function (seg) {
return _defaultSettings.ICON_FIELDS.icon.some(function (t) {
return t.includes(seg);
});
});
});
if (!iconFields.length) {
return notFound;
} // create icon layers for first point pair
var ptPair = fieldPairs[0];
var props = iconFields.map(function (iconField) {
return {
label: iconField.name.replace(/[_,.]+/g, ' ').trim(),
columns: {
lat: ptPair.pair.lat,
lng: ptPair.pair.lng,
icon: {
value: iconField.name,
fieldIdx: iconField.tableFieldIndex - 1
}
},
isVisible: true
};
});
return {
props: props
};
}
}]);
return IconLayer;
}(_baseLayer["default"]);
exports["default"] = IconLayer;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/icon-layer/icon-layer.js"],"names":["brushingExtension","BrushingExtension","SVG_ICON_URL","CLOUDFRONT","iconPosAccessor","lat","lng","d","data","fieldIdx","iconAccessor","icon","iconRequiredColumns","pointVisConfigs","radius","fixedRadius","opacity","colorRange","radiusRange","flatterIconPositions","mesh","cells","reduce","prev","cell","forEach","p","push","positions","IconLayer","props","registerVisConfig","getPositionAccessor","config","columns","getIconAccessor","_layerInfoModal","iconGeometry","getSvgIcons","fetchConfig","method","mode","cache","window","fetch","then","response","json","parsed","svgIcons","accu","curr","id","getPosition","allData","filteredIndex","getIcon","i","length","index","pos","every","Number","isFinite","datasets","oldLayerData","opt","colorScale","colorDomain","colorField","color","sizeField","sizeScale","sizeDomain","textLabel","visConfig","gpuFilter","dataId","updateData","triggerChanged","cScale","getVisChannelScale","colors","map","hexToRgb","rScale","getRadius","getEncodedChannelValue","getFillColor","textLabels","getFilterValue","filterValueAccessor","bounds","getPointsBounds","updateMeta","opts","objectHovered","mapState","interactionConfig","radiusScale","getRadiusScaleByZoom","layerProps","radiusMaxPixels","updateTriggers","filterValueUpdateTriggers","defaultLayerProps","getDefaultDeckLayerProps","brushingProps","getBrushingExtensionProps","getPixelOffset","extensions","sharedProps","filterRange","labelLayers","renderTextLabelLayer","SvgIconLayer","getIconGeometry","isLayerHovered","getDefaultHoverLayerProps","object","highlightColor","defaultPointColumnPairs","IconLayerIcon","size","range","property","channelScaleType","template","modalProps","title","fieldPairs","fields","notFound","iconFields","filter","name","replace","trim","split","some","seg","ICON_FIELDS","t","includes","ptPair","iconField","label","pair","value","tableFieldIndex","isVisible","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 window from 'global/window';\nimport {BrushingExtension} from '@deck.gl/extensions';\n\nimport {hexToRgb} from 'utils/color-utils';\nimport SvgIconLayer from 'deckgl-layers/svg-icon-layer/svg-icon-layer';\nimport IconLayerIcon from './icon-layer-icon';\nimport {ICON_FIELDS, CLOUDFRONT} from 'constants/default-settings';\nimport IconInfoModalFactory from './icon-info-modal';\nimport Layer from '../base-layer';\nimport {getTextOffsetByRadius, formatTextLabelData} from '../layer-text-label';\n\nconst brushingExtension = new BrushingExtension();\n\nexport const SVG_ICON_URL = `${CLOUDFRONT}/icons/svg-icons.json`;\n\nexport const iconPosAccessor = ({lat, lng}) => d => [d.data[lng.fieldIdx], d.data[lat.fieldIdx]];\nexport const iconAccessor = ({icon}) => d => d.data[icon.fieldIdx];\n\nexport const iconRequiredColumns = ['lat', 'lng', 'icon'];\n\nexport const pointVisConfigs = {\n  radius: 'radius',\n  fixedRadius: 'fixedRadius',\n  opacity: 'opacity',\n  colorRange: 'colorRange',\n  radiusRange: 'radiusRange'\n};\n\nfunction flatterIconPositions(icon) {\n  // had to flip y, since @luma modal has changed\n  return icon.mesh.cells.reduce((prev, cell) => {\n    cell.forEach(p => {\n      prev.push(\n        ...[icon.mesh.positions[p][0], -icon.mesh.positions[p][1], icon.mesh.positions[p][2]]\n      );\n    });\n    return prev;\n  }, []);\n}\n\nexport default class IconLayer extends Layer {\n  constructor(props = {}) {\n    super(props);\n\n    this.registerVisConfig(pointVisConfigs);\n    this.getPositionAccessor = () => iconPosAccessor(this.config.columns);\n    this.getIconAccessor = () => iconAccessor(this.config.columns);\n\n    // prepare layer info modal\n    this._layerInfoModal = IconInfoModalFactory();\n    this.iconGeometry = props.iconGeometry || null;\n    this.getSvgIcons();\n  }\n\n  get type() {\n    return 'icon';\n  }\n\n  get requiredLayerColumns() {\n    return iconRequiredColumns;\n  }\n\n  get columnPairs() {\n    return this.defaultPointColumnPairs;\n  }\n\n  get layerIcon() {\n    return IconLayerIcon;\n  }\n\n  get visualChannels() {\n    return {\n      ...super.visualChannels,\n      size: {\n        ...super.visualChannels.size,\n        range: 'radiusRange',\n        property: 'radius',\n        channelScaleType: 'radius'\n      }\n    };\n  }\n\n  get layerInfoModal() {\n    return {\n      id: 'iconInfo',\n      template: this._layerInfoModal,\n      modalProps: {\n        title: 'modal.iconInfo.title'\n      }\n    };\n  }\n\n  getSvgIcons() {\n    const fetchConfig = {\n      method: 'GET',\n      mode: 'cors',\n      cache: 'no-cache'\n    };\n\n    if (window.fetch) {\n      window\n        .fetch(SVG_ICON_URL, fetchConfig)\n        .then(response => response.json())\n        .then((parsed = {}) => {\n          const {svgIcons = []} = parsed;\n          this.iconGeometry = svgIcons.reduce(\n            (accu, curr) => ({\n              ...accu,\n              [curr.id]: flatterIconPositions(curr)\n            }),\n            {}\n          );\n\n          this._layerInfoModal = IconInfoModalFactory(svgIcons);\n        });\n    }\n  }\n\n  static findDefaultLayerProps({fieldPairs = [], fields = []}) {\n    const notFound = {props: []};\n    if (!fieldPairs.length || !fields.length) {\n      return notFound;\n    }\n\n    const iconFields = fields.filter(({name}) =>\n      name\n        .replace(/[_,.]+/g, ' ')\n        .trim()\n        .split(' ')\n        .some(seg => ICON_FIELDS.icon.some(t => t.includes(seg)))\n    );\n\n    if (!iconFields.length) {\n      return notFound;\n    }\n\n    // create icon layers for first point pair\n    const ptPair = fieldPairs[0];\n\n    const props = iconFields.map(iconField => ({\n      label: iconField.name.replace(/[_,.]+/g, ' ').trim(),\n      columns: {\n        lat: ptPair.pair.lat,\n        lng: ptPair.pair.lng,\n        icon: {\n          value: iconField.name,\n          fieldIdx: iconField.tableFieldIndex - 1\n        }\n      },\n      isVisible: true\n    }));\n\n    return {props};\n  }\n\n  calculateDataAttribute({allData, filteredIndex}, getPosition) {\n    const getIcon = this.getIconAccessor();\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      const icon = getIcon({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) && typeof icon === 'string') {\n        data.push({\n          index,\n          icon,\n          data: allData[index]\n        });\n      }\n    }\n\n    return data;\n  }\n\n  formatLayerData(datasets, oldLayerData, opt = {}) {\n    const {\n      colorScale,\n      colorDomain,\n      colorField,\n      color,\n      sizeField,\n      sizeScale,\n      sizeDomain,\n      textLabel,\n      visConfig: {radiusRange, colorRange}\n    } = this.config;\n    const getPosition = this.getPositionAccessor();\n\n    const {gpuFilter} = datasets[this.config.dataId];\n    const {data, triggerChanged} = this.updateData(datasets, oldLayerData);\n\n    // point color\n    const cScale =\n      colorField &&\n      this.getVisChannelScale(colorScale, colorDomain, colorRange.colors.map(hexToRgb));\n\n    // point radius\n    const rScale = sizeField && this.getVisChannelScale(sizeScale, sizeDomain, radiusRange, 0);\n\n    const getRadius = rScale ? d => this.getEncodedChannelValue(rScale, d.data, sizeField) : 1;\n\n    const getFillColor = cScale\n      ? d => this.getEncodedChannelValue(cScale, d.data, colorField)\n      : color;\n\n    // get all distinct characters in the text labels\n    const textLabels = formatTextLabelData({\n      textLabel,\n      triggerChanged,\n      oldLayerData,\n      data\n    });\n\n    return {\n      data,\n      getPosition,\n      getFillColor,\n      getFilterValue: gpuFilter.filterValueAccessor(),\n      getRadius,\n      textLabels\n    };\n  }\n\n  updateLayerMeta(allData, getPosition) {\n    const bounds = this.getPointsBounds(allData, d => getPosition({data: d}));\n    this.updateMeta({bounds});\n  }\n\n  renderLayer(opts) {\n    const {data, gpuFilter, objectHovered, mapState, interactionConfig} = opts;\n\n    const radiusScale = this.getRadiusScaleByZoom(mapState);\n\n    const layerProps = {\n      radiusScale,\n      ...(this.config.visConfig.fixedRadius ? {} : {radiusMaxPixels: 500})\n    };\n\n    const updateTriggers = {\n      getFilterValue: gpuFilter.filterValueUpdateTriggers,\n      getRadius: {\n        sizeField: this.config.colorField,\n        radiusRange: this.config.visConfig.radiusRange,\n        sizeScale: this.config.sizeScale\n      },\n      getFillColor: {\n        color: this.config.color,\n        colorField: this.config.colorField,\n        colorRange: this.config.visConfig.colorRange,\n        colorScale: this.config.colorScale\n      }\n    };\n\n    const defaultLayerProps = this.getDefaultDeckLayerProps(opts);\n    const brushingProps = this.getBrushingExtensionProps(interactionConfig);\n    const getPixelOffset = getTextOffsetByRadius(radiusScale, data.getRadius, mapState);\n    const extensions = [...defaultLayerProps.extensions, brushingExtension];\n\n    // shared Props between layer and label layer\n    const sharedProps = {\n      getFilterValue: data.getFilterValue,\n      extensions,\n      filterRange: defaultLayerProps.filterRange,\n      ...brushingProps\n    };\n\n    const labelLayers = [\n      ...this.renderTextLabelLayer(\n        {\n          getPosition: data.getPosition,\n          sharedProps,\n          getPixelOffset,\n          updateTriggers\n        },\n        opts\n      )\n    ];\n\n    return !this.iconGeometry\n      ? []\n      : [\n          new SvgIconLayer({\n            ...defaultLayerProps,\n            ...brushingProps,\n            ...layerProps,\n            ...data,\n            getIconGeometry: id => this.iconGeometry[id],\n\n            // update triggers\n            updateTriggers,\n            extensions\n          }),\n\n          ...(this.isLayerHovered(objectHovered)\n            ? [\n                new SvgIconLayer({\n                  ...this.getDefaultHoverLayerProps(),\n                  ...layerProps,\n                  data: [objectHovered.object],\n                  getPosition: data.getPosition,\n                  getRadius: data.getRadius,\n                  getFillColor: this.config.highlightColor,\n                  getIconGeometry: id => this.iconGeometry[id]\n                })\n              ]\n            : []),\n\n          // text label layer\n          ...labelLayers\n        ];\n  }\n}\n"]}