UNPKG

kepler.gl

Version:

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

395 lines (335 loc) 41.9 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.pointVisConfigs = exports.iconOptionalColumns = 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 _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 _window = _interopRequireDefault(require("global/window")); var _extensions = require("@deck.gl/extensions"); 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; } 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 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, altitude = _ref.altitude; return function (dc) { return function (d) { return [dc.valueAt(d.index, lng.fieldIdx), dc.valueAt(d.index, lat.fieldIdx), (altitude === null || altitude === void 0 ? void 0 : altitude.fieldIdx) > -1 ? dc.valueAt(d.index, altitude.fieldIdx) : 0]; }; }; }; exports.iconPosAccessor = iconPosAccessor; var iconAccessor = function iconAccessor(_ref2) { var icon = _ref2.icon; return function (dc) { return function (d) { return dc.valueAt(d.index, icon.fieldIdx); }; }; }; exports.iconAccessor = iconAccessor; var iconRequiredColumns = ['lat', 'lng', 'icon']; exports.iconRequiredColumns = iconRequiredColumns; var iconOptionalColumns = ['altitude']; exports.iconOptionalColumns = iconOptionalColumns; 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); var _super = _createSuper(IconLayer); function IconLayer() { var _this; var props = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; (0, _classCallCheck2["default"])(this, IconLayer); _this = _super.call(this, props); _this.registerVisConfig(pointVisConfigs); _this.getPositionAccessor = function (dataContainer) { return iconPosAccessor(_this.config.columns)(dataContainer); }; _this.getIconAccessor = function (dataContainer) { return iconAccessor(_this.config.columns)(dataContainer); }; // prepare layer info modal _this._layerInfoModal = (0, _iconInfoModal["default"])(); _this.iconGeometry = props.iconGeometry || null; _this.getSvgIcons(); return _this; } (0, _createClass2["default"])(IconLayer, [{ key: "type", get: function get() { return 'icon'; } }, { key: "requiredLayerColumns", get: function get() { return iconRequiredColumns; } }, { key: "optionalColumns", get: function get() { return iconOptionalColumns; } }, { key: "columnPairs", get: function get() { return this.defaultPointColumnPairs; } }, { key: "layerIcon", get: function get() { return _iconLayerIcon["default"]; } }, { key: "visualChannels", get: function get() { return { color: _objectSpread(_objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(IconLayer.prototype), "visualChannels", this).color), {}, { accessor: 'getFillColor', defaultValue: function defaultValue(config) { return config.color; } }), size: _objectSpread(_objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(IconLayer.prototype), "visualChannels", this).size), {}, { property: 'radius', range: 'radiusRange', channelScaleType: 'radius', accessor: 'getRadius', defaultValue: 1 }) }; } }, { key: "layerInfoModal", get: function get() { return { id: 'iconInfo', template: this._layerInfoModal, modalProps: { title: 'modal.iconInfo.title' } }; } }, { 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(_objectSpread({}, accu), {}, (0, _defineProperty2["default"])({}, curr.id, flatterIconPositions(curr))); }, {}); _this2._layerInfoModal = (0, _iconInfoModal["default"])(svgIcons); }); } } }, { key: "calculateDataAttribute", value: function calculateDataAttribute(_ref3, getPosition) { var dataContainer = _ref3.dataContainer, filteredIndex = _ref3.filteredIndex; var getIcon = this.getIconAccessor(dataContainer); var data = []; for (var i = 0; i < filteredIndex.length; i++) { var index = filteredIndex[i]; var rowIndex = { index: index }; var pos = getPosition(rowIndex); var icon = getIcon(rowIndex); // 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 }); } } return data; } }, { key: "formatLayerData", value: function formatLayerData(datasets, oldLayerData) { var textLabel = this.config.textLabel; var _datasets$this$config = datasets[this.config.dataId], gpuFilter = _datasets$this$config.gpuFilter, dataContainer = _datasets$this$config.dataContainer; var getPosition = this.getPositionAccessor(dataContainer); var _this$updateData = this.updateData(datasets, oldLayerData), data = _this$updateData.data, triggerChanged = _this$updateData.triggerChanged; // get all distinct characters in the text labels var textLabels = (0, _layerTextLabel.formatTextLabelData)({ textLabel: textLabel, triggerChanged: triggerChanged, oldLayerData: oldLayerData, data: data, dataContainer: dataContainer }); var accessors = this.getAttributeAccessors({ dataContainer: dataContainer }); return _objectSpread({ data: data, getPosition: getPosition, getFilterValue: gpuFilter.filterValueAccessor(dataContainer)(), textLabels: textLabels }, accessors); } }, { key: "updateLayerMeta", value: function updateLayerMeta(dataContainer, getPosition) { var bounds = this.getPointsBounds(dataContainer, getPosition); this.updateMeta({ bounds: bounds }); } }, { key: "renderLayer", value: function renderLayer(opts) { var _this$config$columns$, _this3 = 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 = _objectSpread({ getPosition: this.config.columns, getFilterValue: gpuFilter.filterValueUpdateTriggers }, this.getVisualChannelUpdateTriggers()); 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)); var hoveredObject = this.hasHoveredObject(objectHovered); var parameters = { // icons will be flat on the map when the altitude column is not used depthTest: ((_this$config$columns$ = this.config.columns.altitude) === null || _this$config$columns$ === void 0 ? void 0 : _this$config$columns$.fieldIdx) > -1 }; return !this.iconGeometry ? [] : [new _svgIconLayer["default"](_objectSpread(_objectSpread(_objectSpread(_objectSpread(_objectSpread({}, defaultLayerProps), brushingProps), layerProps), data), {}, { parameters: parameters, getIconGeometry: function getIconGeometry(id) { return _this3.iconGeometry[id]; }, // update triggers updateTriggers: updateTriggers, extensions: extensions }))].concat((0, _toConsumableArray2["default"])(hoveredObject ? [new _svgIconLayer["default"](_objectSpread(_objectSpread(_objectSpread({}, this.getDefaultHoverLayerProps()), layerProps), {}, { data: [hoveredObject], parameters: parameters, getPosition: data.getPosition, getRadius: data.getRadius, getFillColor: this.config.highlightColor, getIconGeometry: function getIconGeometry(id) { return _this3.iconGeometry[id]; } }))] : []), (0, _toConsumableArray2["default"])(labelLayers)); } }], [{ 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.fieldIdx } }, 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","altitude","dc","d","valueAt","index","fieldIdx","iconAccessor","icon","iconRequiredColumns","iconOptionalColumns","pointVisConfigs","radius","fixedRadius","opacity","colorRange","radiusRange","flatterIconPositions","mesh","cells","reduce","prev","cell","forEach","p","push","positions","IconLayer","props","registerVisConfig","getPositionAccessor","dataContainer","config","columns","getIconAccessor","_layerInfoModal","iconGeometry","getSvgIcons","defaultPointColumnPairs","IconLayerIcon","color","accessor","defaultValue","size","property","range","channelScaleType","id","template","modalProps","title","fetchConfig","method","mode","cache","window","fetch","then","response","json","parsed","svgIcons","accu","curr","getPosition","filteredIndex","getIcon","data","i","length","rowIndex","pos","every","Number","isFinite","datasets","oldLayerData","textLabel","dataId","gpuFilter","updateData","triggerChanged","textLabels","accessors","getAttributeAccessors","getFilterValue","filterValueAccessor","bounds","getPointsBounds","updateMeta","opts","objectHovered","mapState","interactionConfig","radiusScale","getRadiusScaleByZoom","layerProps","visConfig","radiusMaxPixels","updateTriggers","filterValueUpdateTriggers","getVisualChannelUpdateTriggers","defaultLayerProps","getDefaultDeckLayerProps","brushingProps","getBrushingExtensionProps","getPixelOffset","getRadius","extensions","sharedProps","filterRange","labelLayers","renderTextLabelLayer","hoveredObject","hasHoveredObject","parameters","depthTest","SvgIconLayer","getIconGeometry","getDefaultHoverLayerProps","getFillColor","highlightColor","fieldPairs","fields","notFound","iconFields","filter","name","replace","trim","split","some","seg","ICON_FIELDS","t","includes","ptPair","map","iconField","label","pair","value","isVisible","Layer"],"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 window from 'global/window';\nimport {BrushingExtension} from '@deck.gl/extensions';\n\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, altitude}) => dc => d => [\n  dc.valueAt(d.index, lng.fieldIdx),\n  dc.valueAt(d.index, lat.fieldIdx),\n  altitude?.fieldIdx > -1 ? dc.valueAt(d.index, altitude.fieldIdx) : 0\n];\n\nexport const iconAccessor = ({icon}) => dc => d => dc.valueAt(d.index, icon.fieldIdx);\n\nexport const iconRequiredColumns = ['lat', 'lng', 'icon'];\nexport const iconOptionalColumns = ['altitude'];\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 = dataContainer => iconPosAccessor(this.config.columns)(dataContainer);\n    this.getIconAccessor = dataContainer => iconAccessor(this.config.columns)(dataContainer);\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 optionalColumns() {\n    return iconOptionalColumns;\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      color: {\n        ...super.visualChannels.color,\n        accessor: 'getFillColor',\n        defaultValue: config => config.color\n      },\n      size: {\n        ...super.visualChannels.size,\n        property: 'radius',\n        range: 'radiusRange',\n        channelScaleType: 'radius',\n        accessor: 'getRadius',\n        defaultValue: 1\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.fieldIdx\n        }\n      },\n      isVisible: true\n    }));\n\n    return {props};\n  }\n\n  calculateDataAttribute({dataContainer, filteredIndex}, getPosition) {\n    const getIcon = this.getIconAccessor(dataContainer);\n    const data = [];\n\n    for (let i = 0; i < filteredIndex.length; i++) {\n      const index = filteredIndex[i];\n      const rowIndex = {index};\n      const pos = getPosition(rowIndex);\n      const icon = getIcon(rowIndex);\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        });\n      }\n    }\n\n    return data;\n  }\n\n  formatLayerData(datasets, oldLayerData) {\n    const {textLabel} = this.config;\n    const {gpuFilter, dataContainer} = datasets[this.config.dataId];\n\n    const getPosition = this.getPositionAccessor(dataContainer);\n\n    const {data, triggerChanged} = this.updateData(datasets, oldLayerData);\n\n    // get all distinct characters in the text labels\n    const textLabels = formatTextLabelData({\n      textLabel,\n      triggerChanged,\n      oldLayerData,\n      data,\n      dataContainer\n    });\n\n    const accessors = this.getAttributeAccessors({dataContainer});\n\n    return {\n      data,\n      getPosition,\n      getFilterValue: gpuFilter.filterValueAccessor(dataContainer)(),\n      textLabels,\n      ...accessors\n    };\n  }\n\n  updateLayerMeta(dataContainer, getPosition) {\n    const bounds = this.getPointsBounds(dataContainer, getPosition);\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      getPosition: this.config.columns,\n      getFilterValue: gpuFilter.filterValueUpdateTriggers,\n      ...this.getVisualChannelUpdateTriggers()\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    const hoveredObject = this.hasHoveredObject(objectHovered);\n\n    const parameters = {\n      // icons will be flat on the map when the altitude column is not used\n      depthTest: this.config.columns.altitude?.fieldIdx > -1\n    };\n\n    return !this.iconGeometry\n      ? []\n      : [\n          new SvgIconLayer({\n            ...defaultLayerProps,\n            ...brushingProps,\n            ...layerProps,\n            ...data,\n            parameters,\n            getIconGeometry: id => this.iconGeometry[id],\n\n            // update triggers\n            updateTriggers,\n            extensions\n          }),\n\n          ...(hoveredObject\n            ? [\n                new SvgIconLayer({\n                  ...this.getDefaultHoverLayerProps(),\n                  ...layerProps,\n                  data: [hoveredObject],\n                  parameters,\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"]}