kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
420 lines (411 loc) • 58.6 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.pointVisConfigs = exports.iconRequiredColumns = exports.iconPosAccessor = exports.iconOptionalColumns = exports.iconAccessor = exports["default"] = void 0;
var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray"));
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 _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _window = _interopRequireDefault(require("global/window"));
var _extensions = require("@deck.gl/extensions");
var _deckglLayers = require("@kepler.gl/deckgl-layers");
var _iconLayerIcon = _interopRequireDefault(require("./icon-layer-icon"));
var _constants = require("@kepler.gl/constants");
var _iconInfoModal = _interopRequireDefault(require("./icon-info-modal"));
var _baseLayer = _interopRequireDefault(require("../base-layer"));
var _layerUtils = require("../layer-utils");
var _utils = require("@kepler.gl/utils");
var _layerTextLabel = require("../layer-text-label");
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2["default"])(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
function _callSuper(t, o, e) { return o = (0, _getPrototypeOf2["default"])(o), (0, _possibleConstructorReturn2["default"])(t, _isNativeReflectConstruct() ? Reflect.construct(o, e || [], (0, _getPrototypeOf2["default"])(t).constructor) : o.apply(t, e)); }
function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); }
function _superPropGet(t, e, r, o) { var p = (0, _get2["default"])((0, _getPrototypeOf2["default"])(1 & o ? t.prototype : t), e, r); return 2 & o ? function (t) { return p.apply(r, t); } : p; } // SPDX-License-Identifier: MIT
// Copyright contributors to the kepler.gl project
var brushingExtension = new _extensions.BrushingExtension();
var iconPosAccessor = exports.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];
};
};
};
var iconAccessor = exports.iconAccessor = function iconAccessor(_ref2) {
var icon = _ref2.icon;
return function (dc) {
return function (d) {
return dc.valueAt(d.index, icon.fieldIdx);
};
};
};
var iconRequiredColumns = exports.iconRequiredColumns = ['lat', 'lng', 'icon'];
var iconOptionalColumns = exports.iconOptionalColumns = ['altitude'];
var pointVisConfigs = exports.pointVisConfigs = {
radius: 'radius',
fixedRadius: 'fixedRadius',
opacity: 'opacity',
colorRange: 'colorRange',
radiusRange: 'radiusRange',
billboard: 'billboard'
};
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 = exports["default"] = /*#__PURE__*/function (_Layer) {
function IconLayer(props) {
var _this;
(0, _classCallCheck2["default"])(this, IconLayer);
_this = _callSuper(this, IconLayer, [props]);
(0, _defineProperty2["default"])(_this, "getIconAccessor", void 0);
(0, _defineProperty2["default"])(_this, "_layerInfoModal", void 0);
(0, _defineProperty2["default"])(_this, "iconGeometry", void 0);
(0, _defineProperty2["default"])(_this, "iconGeometryVersion", void 0);
(0, _defineProperty2["default"])(_this, "onRedrawNeeded", void 0);
_this.registerVisConfig(pointVisConfigs);
_this.getPositionAccessor = function (dataContainer) {
return iconPosAccessor(_this.config.columns)(dataContainer);
};
_this.getIconAccessor = function (dataContainer) {
return iconAccessor(_this.config.columns)(dataContainer);
};
_this._layerInfoModal = (0, _iconInfoModal["default"])(props.svgIcons);
_this.iconGeometry = props.iconGeometry || null;
_this.iconGeometryVersion = 0;
if ((0, _utils.isTest)()) {
return (0, _possibleConstructorReturn2["default"])(_this);
}
if (props.svgIcons) {
// if svg icons are passed in then bypass fetching of remote svg icons
_this.setSvgIcons(props.svgIcons);
} else {
// prepare layer info modal and fetch remote svg icons
_this.getSvgIcons();
}
return _this;
}
(0, _inherits2["default"])(IconLayer, _Layer);
return (0, _createClass2["default"])(IconLayer, [{
key: "svgIconUrl",
get: function get() {
return "".concat((0, _utils.getApplicationConfig)().cdnUrl, "/icons/svg-icons.json");
}
}, {
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({}, _superPropGet(IconLayer, "visualChannels", this, 1).color), {}, {
accessor: 'getFillColor',
defaultValue: function defaultValue(config) {
return config.color;
}
}),
size: _objectSpread(_objectSpread({}, _superPropGet(IconLayer, "visualChannels", this, 1).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: "getZoomFactor",
value: function getZoomFactor(_ref3) {
var zoom = _ref3.zoom,
_ref3$zoomOffset = _ref3.zoomOffset,
zoomOffset = _ref3$zoomOffset === void 0 ? 0 : _ref3$zoomOffset;
return Math.pow(2, 14 - zoom + zoomOffset);
}
}, {
key: "getSvgIcons",
value: function getSvgIcons() {
var _this2 = this;
var fetchConfig = {
method: 'GET',
mode: 'cors',
cache: 'no-cache'
};
if (_window["default"].fetch && this.svgIconUrl) {
_window["default"].fetch(this.svgIconUrl, fetchConfig).then(function (response) {
if (!response.ok) {
throw new Error("Failed to load svg-icons.json: ".concat(response.status));
}
return response.json();
}).then(function () {
var parsed = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
_this2.setSvgIcons(parsed.svgIcons);
})["catch"](function (err) {
console.error('Error fetching or parsing svg-icons.json:', err);
// Fallback to empty geometry to allow default icon rendering
_this2.iconGeometry = {};
_this2.iconGeometryVersion += 1;
});
} else {
// No fetch available; set empty geometry so layer can render default icons
this.iconGeometry = {};
this.iconGeometryVersion += 1;
}
}
}, {
key: "setSvgIcons",
value: function setSvgIcons() {
var _this$onRedrawNeeded;
var svgIcons = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
this.iconGeometry = svgIcons.reduce(function (accu, curr) {
return _objectSpread(_objectSpread({}, accu), {}, (0, _defineProperty2["default"])({}, curr.id, flatterIconPositions(curr)));
}, {});
// Increment version when SVG icons are loaded to trigger layer re-render
this.iconGeometryVersion += 1;
this._layerInfoModal = (0, _iconInfoModal["default"])(svgIcons);
// Trigger a map redraw so deck.gl picks up the new geometry
(_this$onRedrawNeeded = this.onRedrawNeeded) === null || _this$onRedrawNeeded === void 0 || _this$onRedrawNeeded.call(this);
}
}, {
key: "getFilteredItemCount",
value: function getFilteredItemCount() {
// use total
if (Object.keys(this.filteredItemCount).length) {
return Object.values(this.filteredItemCount).reduce(function (total, curr) {
return Number.isFinite(curr) ? total + curr : total;
}, 0);
}
return null;
}
}, {
key: "calculateDataAttribute",
value: function calculateDataAttribute(_ref4, getPosition) {
var dataContainer = _ref4.dataContainer,
filteredIndex = _ref4.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) {
if (this.config.dataId === null) {
return {};
}
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(dataset, getPosition) {
var dataContainer = dataset.dataContainer;
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,
layerCallbacks = opts.layerCallbacks;
// Store callback to trigger map redraw when icon geometry loads asynchronously
this.onRedrawNeeded = layerCallbacks === null || layerCallbacks === void 0 ? void 0 : layerCallbacks.onRedrawNeeded;
var radiusScale = this.getRadiusScaleByZoom(mapState);
var layerProps = _objectSpread({
radiusScale: radiusScale,
billboard: this.config.visConfig.billboard
}, 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,
cullMode: _constants.CULL_MODE.FRONT
};
// Append geometry version to layer id so deck.gl treats it as new layer when geometry changes
var baseLayerId = defaultLayerProps.id || this.id;
var layerIdWithVersion = "".concat(baseLayerId, "_").concat(this.iconGeometryVersion);
return [new _deckglLayers.SvgIconLayer(_objectSpread(_objectSpread(_objectSpread(_objectSpread(_objectSpread({}, defaultLayerProps), {}, {
id: layerIdWithVersion
}, brushingProps), layerProps), data), {}, {
parameters: parameters,
getIconGeometry: function getIconGeometry(id) {
var _this3$iconGeometry;
return (_this3$iconGeometry = _this3.iconGeometry) === null || _this3$iconGeometry === void 0 ? void 0 : _this3$iconGeometry[id];
},
// update triggers
updateTriggers: updateTriggers,
extensions: extensions
}))].concat((0, _toConsumableArray2["default"])(hoveredObject ? [new _deckglLayers.SvgIconLayer(_objectSpread(_objectSpread(_objectSpread({}, this.getDefaultHoverLayerProps()), {}, {
id: "".concat(layerIdWithVersion, "-hover")
}, layerProps), {}, {
visible: defaultLayerProps.visible,
data: [hoveredObject],
parameters: parameters,
getPosition: data.getPosition,
getRadius: data.getRadius,
getFillColor: this.config.highlightColor,
getIconGeometry: function getIconGeometry(id) {
var _this3$iconGeometry2;
return (_this3$iconGeometry2 = _this3.iconGeometry) === null || _this3$iconGeometry2 === void 0 ? void 0 : _this3$iconGeometry2[id];
}
}))] : []), (0, _toConsumableArray2["default"])(labelLayers));
}
}], [{
key: "findDefaultLayerProps",
value: function findDefaultLayerProps(_ref5) {
var _ref5$fieldPairs = _ref5.fieldPairs,
fieldPairs = _ref5$fieldPairs === void 0 ? [] : _ref5$fieldPairs,
_ref5$fields = _ref5.fields,
fields = _ref5$fields === void 0 ? [] : _ref5$fields;
var notFound = {
props: []
};
if (!fieldPairs.length || !fields.length) {
return notFound;
}
var iconFields = fields.filter(function (_ref6) {
var name = _ref6.name;
return name.replace(/[_,.]+/g, ' ').trim().split(' ').some(function (seg) {
return _constants.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 columns = (0, _layerUtils.assignPointPairToLayerColumn)(ptPair, true);
var props = iconFields.map(function (iconField) {
return {
label: iconField.name.replace(/[_,.]+/g, ' ').trim(),
columns: _objectSpread(_objectSpread({}, columns), {}, {
icon: {
value: iconField.name,
fieldIdx: iconField.fieldIdx
}
}),
isVisible: true
};
});
return {
props: props
};
}
}]);
}(_baseLayer["default"]);
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_window","_interopRequireDefault","require","_extensions","_deckglLayers","_iconLayerIcon","_constants","_iconInfoModal","_baseLayer","_layerUtils","_utils","_layerTextLabel","ownKeys","e","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","defineProperty","_callSuper","_getPrototypeOf2","_possibleConstructorReturn2","_isNativeReflectConstruct","Reflect","construct","constructor","Boolean","prototype","valueOf","call","_superPropGet","p","_get2","brushingExtension","BrushingExtension","iconPosAccessor","exports","_ref","lat","lng","altitude","dc","d","valueAt","index","fieldIdx","iconAccessor","_ref2","icon","iconRequiredColumns","iconOptionalColumns","pointVisConfigs","radius","fixedRadius","opacity","colorRange","radiusRange","billboard","flatterIconPositions","mesh","cells","reduce","prev","cell","positions","IconLayer","_Layer","props","_this","_classCallCheck2","registerVisConfig","getPositionAccessor","dataContainer","config","columns","getIconAccessor","_layerInfoModal","IconInfoModalFactory","svgIcons","iconGeometry","iconGeometryVersion","isTest","setSvgIcons","getSvgIcons","_inherits2","_createClass2","key","get","concat","getApplicationConfig","cdnUrl","defaultPointColumnPairs","IconLayerIcon","color","accessor","defaultValue","size","property","range","channelScaleType","id","template","modalProps","title","value","getZoomFactor","_ref3","zoom","_ref3$zoomOffset","zoomOffset","Math","pow","_this2","fetchConfig","method","mode","cache","Window","fetch","svgIconUrl","then","response","ok","Error","status","json","parsed","undefined","err","console","error","_this$onRedrawNeeded","accu","curr","onRedrawNeeded","getFilteredItemCount","filteredItemCount","values","total","Number","isFinite","calculateDataAttribute","_ref4","getPosition","filteredIndex","getIcon","data","i","rowIndex","pos","every","formatLayerData","datasets","oldLayerData","dataId","textLabel","_datasets$this$config","gpuFilter","_this$updateData","updateData","triggerChanged","textLabels","formatTextLabelData","accessors","getAttributeAccessors","getFilterValue","filterValueAccessor","updateLayerMeta","dataset","bounds","getPointsBounds","updateMeta","renderLayer","opts","_this$config$columns$","_this3","objectHovered","mapState","interactionConfig","layerCallbacks","radiusScale","getRadiusScaleByZoom","layerProps","visConfig","radiusMaxPixels","updateTriggers","filterValueUpdateTriggers","getVisualChannelUpdateTriggers","defaultLayerProps","getDefaultDeckLayerProps","brushingProps","getBrushingExtensionProps","getPixelOffset","getTextOffsetByRadius","getRadius","extensions","_toConsumableArray2","sharedProps","filterRange","labelLayers","renderTextLabelLayer","hoveredObject","hasHoveredObject","parameters","depthTest","cullMode","CULL_MODE","FRONT","baseLayerId","layerIdWithVersion","SvgIconLayer","getIconGeometry","_this3$iconGeometry","getDefaultHoverLayerProps","visible","getFillColor","highlightColor","_this3$iconGeometry2","findDefaultLayerProps","_ref5","_ref5$fieldPairs","fieldPairs","_ref5$fields","fields","notFound","iconFields","_ref6","name","replace","trim","split","some","seg","ICON_FIELDS","includes","ptPair","assignPointPairToLayerColumn","map","iconField","label","isVisible","Layer"],"sources":["../../src/icon-layer/icon-layer.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport Window from 'global/window';\nimport {BrushingExtension} from '@deck.gl/extensions';\n\nimport {SvgIconLayer} from '@kepler.gl/deckgl-layers';\nimport IconLayerIcon from './icon-layer-icon';\nimport {ICON_FIELDS, CULL_MODE} from '@kepler.gl/constants';\nimport IconInfoModalFactory from './icon-info-modal';\nimport Layer, {LayerBaseConfig, LayerBaseConfigPartial} from '../base-layer';\nimport {assignPointPairToLayerColumn, FindDefaultLayerPropsReturnValue} from '../layer-utils';\nimport {isTest} from '@kepler.gl/utils';\nimport {getTextOffsetByRadius, formatTextLabelData} from '../layer-text-label';\nimport {default as KeplerTable} from '@kepler.gl/table';\nimport {getApplicationConfig, DataContainerInterface} from '@kepler.gl/utils';\nimport {\n  ColorRange,\n  VisConfigBoolean,\n  VisConfigColorRange,\n  VisConfigNumber,\n  VisConfigRange,\n  Merge,\n  LayerColumn,\n  BindedLayerCallbacks\n} from '@kepler.gl/types';\n\nexport type IconLayerColumnsConfig = {\n  lat: LayerColumn;\n  lng: LayerColumn;\n  altitude: LayerColumn;\n  icon: LayerColumn;\n};\n\ntype IconGeometry = object | null;\n\nexport type IconLayerVisConfigSettings = {\n  radius: VisConfigNumber;\n  fixedRadius: VisConfigBoolean;\n  opacity: VisConfigNumber;\n  colorRange: VisConfigColorRange;\n  radiusRange: VisConfigRange;\n};\n\nexport type IconLayerVisConfig = {\n  radius: number;\n  fixedRadius: boolean;\n  opacity: number;\n  colorRange: ColorRange;\n  radiusRange: [number, number];\n  billboard: boolean;\n};\n\nexport type IconLayerConfig = Merge<\n  LayerBaseConfig,\n  {columns: IconLayerColumnsConfig; visConfig: IconLayerVisConfig}\n>;\n\nexport type IconLayerData = {index: number; icon: string};\n\nconst brushingExtension = new BrushingExtension();\n\nexport const iconPosAccessor =\n  ({lat, lng, altitude}: IconLayerColumnsConfig) =>\n  (dc: DataContainerInterface) =>\n  d =>\n    [\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 =\n  ({icon}: IconLayerColumnsConfig) =>\n  (dc: DataContainerInterface) =>\n  d =>\n    dc.valueAt(d.index, icon.fieldIdx);\n\nexport const iconRequiredColumns: ['lat', 'lng', 'icon'] = ['lat', 'lng', 'icon'];\nexport const iconOptionalColumns: ['altitude'] = ['altitude'];\n\nexport const pointVisConfigs: {\n  radius: 'radius';\n  fixedRadius: 'fixedRadius';\n  opacity: 'opacity';\n  colorRange: 'colorRange';\n  radiusRange: 'radiusRange';\n  billboard: 'billboard';\n} = {\n  radius: 'radius',\n  fixedRadius: 'fixedRadius',\n  opacity: 'opacity',\n  colorRange: 'colorRange',\n  radiusRange: 'radiusRange',\n  billboard: 'billboard'\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  getIconAccessor: (dataContainer: DataContainerInterface) => (d: any) => any;\n  _layerInfoModal: () => JSX.Element;\n  iconGeometry: IconGeometry;\n  iconGeometryVersion: number;\n\n  onRedrawNeeded: BindedLayerCallbacks['onRedrawNeeded'];\n\n  declare visConfigSettings: IconLayerVisConfigSettings;\n  declare config: IconLayerConfig;\n\n  constructor(\n    props: {\n      id?: string;\n      iconGeometry?: IconGeometry;\n      svgIcons?: any[];\n    } & LayerBaseConfigPartial\n  ) {\n    super(props);\n\n    this.registerVisConfig(pointVisConfigs);\n    this.getPositionAccessor = (dataContainer: DataContainerInterface) =>\n      iconPosAccessor(this.config.columns)(dataContainer);\n    this.getIconAccessor = dataContainer => iconAccessor(this.config.columns)(dataContainer);\n\n    this._layerInfoModal = IconInfoModalFactory(props.svgIcons);\n    this.iconGeometry = props.iconGeometry || null;\n    this.iconGeometryVersion = 0;\n\n    if (isTest()) {\n      return;\n    }\n    if (props.svgIcons) {\n      // if svg icons are passed in then bypass fetching of remote svg icons\n      this.setSvgIcons(props.svgIcons);\n    } else {\n      // prepare layer info modal and fetch remote svg icons\n      this.getSvgIcons();\n    }\n  }\n\n  get svgIconUrl() {\n    return `${getApplicationConfig().cdnUrl}/icons/svg-icons.json`;\n  }\n\n  get type(): 'icon' {\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  getZoomFactor({zoom, zoomOffset = 0}: {zoom: number; zoomOffset?: number}) {\n    return Math.pow(2, 14 - zoom + zoomOffset);\n  }\n\n  getSvgIcons() {\n    const fetchConfig = {\n      method: 'GET',\n      mode: 'cors',\n      cache: 'no-cache'\n    };\n\n    if (Window.fetch && this.svgIconUrl) {\n      Window.fetch(this.svgIconUrl, fetchConfig)\n        .then(response => {\n          if (!response.ok) {\n            throw new Error(`Failed to load svg-icons.json: ${response.status}`);\n          }\n          return response.json();\n        })\n        .then((parsed: {svgIcons?: any[]} = {}) => {\n          this.setSvgIcons(parsed.svgIcons);\n        })\n        .catch(err => {\n          console.error('Error fetching or parsing svg-icons.json:', err);\n          // Fallback to empty geometry to allow default icon rendering\n          this.iconGeometry = {};\n          this.iconGeometryVersion += 1;\n        });\n    } else {\n      // No fetch available; set empty geometry so layer can render default icons\n      this.iconGeometry = {};\n      this.iconGeometryVersion += 1;\n    }\n  }\n\n  setSvgIcons(svgIcons: any[] = []) {\n    this.iconGeometry = svgIcons.reduce(\n      (accu, curr) => ({\n        ...accu,\n        [curr.id]: flatterIconPositions(curr)\n      }),\n      {}\n    );\n\n    // Increment version when SVG icons are loaded to trigger layer re-render\n    this.iconGeometryVersion += 1;\n\n    this._layerInfoModal = IconInfoModalFactory(svgIcons);\n\n    // Trigger a map redraw so deck.gl picks up the new geometry\n    this.onRedrawNeeded?.();\n  }\n\n  static findDefaultLayerProps({\n    fieldPairs = [],\n    fields = []\n  }: KeplerTable): FindDefaultLayerPropsReturnValue {\n    const notFound: FindDefaultLayerPropsReturnValue = {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    const columns = assignPointPairToLayerColumn(ptPair, true);\n\n    const props = iconFields.map(iconField => ({\n      label: iconField.name.replace(/[_,.]+/g, ' ').trim(),\n      columns: {\n        ...columns,\n        icon: {\n          value: iconField.name,\n          fieldIdx: iconField.fieldIdx\n        }\n      },\n      isVisible: true\n    }));\n\n    return {props};\n  }\n\n  getFilteredItemCount() {\n    // use total\n    if (Object.keys(this.filteredItemCount).length) {\n      return Object.values(this.filteredItemCount).reduce(\n        (total, curr) => (Number.isFinite(curr) ? total + curr : total),\n        0\n      );\n    }\n\n    return null;\n  }\n\n  calculateDataAttribute({dataContainer, filteredIndex}: KeplerTable, getPosition) {\n    const getIcon = this.getIconAccessor(dataContainer);\n    const data: IconLayerData[] = [];\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    if (this.config.dataId === null) {\n      return {};\n    }\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(dataset: KeplerTable, getPosition) {\n    const {dataContainer} = dataset;\n    const bounds = this.getPointsBounds(dataContainer, getPosition);\n    this.updateMeta({bounds});\n  }\n\n  renderLayer(opts) {\n    const {data, gpuFilter, objectHovered, mapState, interactionConfig, layerCallbacks} = opts;\n\n    // Store callback to trigger map redraw when icon geometry loads asynchronously\n    this.onRedrawNeeded = layerCallbacks?.onRedrawNeeded;\n\n    const radiusScale = this.getRadiusScaleByZoom(mapState);\n\n    const layerProps = {\n      radiusScale,\n      billboard: this.config.visConfig.billboard,\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      cullMode: CULL_MODE.FRONT\n    };\n\n    // Append geometry version to layer id so deck.gl treats it as new layer when geometry changes\n    const baseLayerId = defaultLayerProps.id || this.id;\n    const layerIdWithVersion = `${baseLayerId}_${this.iconGeometryVersion}`;\n\n    return [\n      new SvgIconLayer({\n        ...defaultLayerProps,\n        id: layerIdWithVersion,\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      // hover layer\n      ...(hoveredObject\n        ? [\n            new SvgIconLayer({\n              ...this.getDefaultHoverLayerProps(),\n              id: `${layerIdWithVersion}-hover`,\n              ...layerProps,\n              visible: defaultLayerProps.visible,\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"],"mappings":"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