UNPKG

kepler.gl.geoiq

Version:

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

383 lines (330 loc) 40 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.tripVisConfigs = exports.geoJsonRequiredColumns = exports.featureResolver = exports.featureAccessor = exports["default"] = 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 _lodash = _interopRequireDefault(require("lodash.memoize")); var _lodash2 = _interopRequireDefault(require("lodash.uniq")); var _baseLayer = _interopRequireDefault(require("../base-layer")); var _deck = require("deck.gl"); var _defaultSettings = require("../../constants/default-settings"); var _tripLayerIcon = _interopRequireDefault(require("./trip-layer-icon")); var _geojsonUtils = require("../geojson-layer/geojson-utils"); var _tripUtils = require("./trip-utils"); var _colorUtils = require("../../utils/color-utils"); var _tripInfoModal = _interopRequireDefault(require("./trip-info-modal")); 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 _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 tripVisConfigs = exports.tripVisConfigs = { opacity: 'opacity', thickness: { type: 'number', defaultValue: 0.5, label: 'Stroke Width', isRanged: false, range: [0, 100], step: 0.1, group: 'stroke', property: 'thickness' }, colorRange: 'colorRange', trailLength: 'trailLength', sizeRange: 'strokeWidthRange' }; var geoJsonRequiredColumns = exports.geoJsonRequiredColumns = ['geojson']; var featureAccessor = exports.featureAccessor = function featureAccessor(_ref) { var geojson = _ref.geojson; return function (d) { return d[geojson.fieldIdx]; }; }; var featureResolver = exports.featureResolver = function featureResolver(_ref2) { var geojson = _ref2.geojson; return geojson.fieldIdx; }; var TripLayer = exports["default"] = /*#__PURE__*/function (_Layer) { (0, _inherits2["default"])(TripLayer, _Layer); var _super = _createSuper(TripLayer); function TripLayer(props) { var _this; (0, _classCallCheck2["default"])(this, TripLayer); _this = _super.call(this, props); _this.dataToFeature = []; _this.dataToTimeStamp = []; _this.registerVisConfig(tripVisConfigs); _this.getFeature = (0, _lodash["default"])(featureAccessor, featureResolver); _this._layerInfoModal = (0, _tripInfoModal["default"])(); return _this; } (0, _createClass2["default"])(TripLayer, [{ key: "type", get: function get() { return 'trip'; } }, { key: "name", get: function get() { return 'Trip'; } }, { key: "layerIcon", get: function get() { return _tripLayerIcon["default"]; } }, { key: "requiredLayerColumns", get: function get() { return geoJsonRequiredColumns; } }, { key: "visualChannels", get: function get() { return _objectSpread(_objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(TripLayer.prototype), "visualChannels", this)), {}, { size: _objectSpread(_objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(TripLayer.prototype), "visualChannels", this).size), {}, { property: 'stroke', condition: function condition(config) { return config.visConfig.stroked; } }) }); } }, { key: "animationDomain", get: function get() { return this.config.animation.domain; } }, { key: "layerInfoModal", get: function get() { return { id: 'iconInfo', template: this._layerInfoModal, modalProps: { title: 'How to enable trip animation' } }; } }, { key: "getPositionAccessor", value: function getPositionAccessor() { return this.getFeature(this.config.columns); } }, { key: "getDefaultLayerConfig", value: function getDefaultLayerConfig(props) { return _objectSpread(_objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(TripLayer.prototype), "getDefaultLayerConfig", this).call(this, props)), {}, { animation: { enabled: true, domain: null } }); } }, { key: "getHoverData", value: function getHoverData(object, allData) { // index of allData is saved to feature.properties return allData[object.properties.index]; } // TODO: fix complexity /* eslint-disable complexity */ }, { key: "formatLayerData", value: function formatLayerData(_, allData, filteredIndex, oldLayerData) { var _this2 = this; var opt = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : {}; // to-do: parse segment from allData var _this$config = this.config, colorScale = _this$config.colorScale, colorField = _this$config.colorField, colorDomain = _this$config.colorDomain, color = _this$config.color, sizeScale = _this$config.sizeScale, sizeDomain = _this$config.sizeDomain, sizeField = _this$config.sizeField, visConfig = _this$config.visConfig; var stroked = visConfig.stroked, colorRange = visConfig.colorRange, sizeRange = visConfig.sizeRange; var getFeature = this.getPositionAccessor(this.config.column); // geojson feature are object, if doesn't exists // create it and save to layer if (!oldLayerData || oldLayerData.getFeature !== getFeature) { this.updateLayerMeta(allData, getFeature); } var geojsonData; if (oldLayerData && oldLayerData.data && opt.sameData && oldLayerData.getFeature === getFeature) { // no need to create a new array of data // use updateTriggers to selectively re-calculate attributes geojsonData = oldLayerData.data; } else { // filteredIndex is a reference of index in allData which can map to feature geojsonData = filteredIndex.map(function (i) { return _this2.dataToFeature[i]; }).filter(function (d) { return d && d.geometry.type === 'LineString'; }); } // color var cScale = colorField && this.getVisChannelScale(colorScale, colorDomain, colorRange.colors.map(_colorUtils.hexToRgb)); // calculate stroke scale - if stroked = true var sScale = sizeField && stroked && this.getVisChannelScale(sizeScale, sizeDomain, sizeRange); return { data: geojsonData, getPath: function getPath(d) { return d.geometry.coordinates; }, getTimestamps: function getTimestamps(d) { return _this2.dataToTimeStamp[d.properties.index]; }, getColor: function getColor(d) { return cScale ? _this2.getEncodedChannelValue(cScale, allData[d.properties.index], colorField) : d.properties.fillColor || color; }, getWidth: function getWidth(d) { return sScale ? _this2.getEncodedChannelValue(sScale, allData[d.properties.index], sizeField, 0) : d.properties.lineWidth || 1; } }; } /* eslint-enable complexity */ }, { key: "updateAnimationDomain", value: function updateAnimationDomain(domain) { this.updateLayerConfig({ animation: _objectSpread(_objectSpread({}, this.config.animation), {}, { domain: domain }) }); } }, { key: "updateLayerMeta", value: function updateLayerMeta(allData) { var getFeature = this.getPositionAccessor(); if (getFeature === this.meta.getFeature) { // TODO: revisit this after gpu filtering return; } this.dataToFeature = (0, _geojsonUtils.getGeojsonDataMaps)(allData, getFeature); var _parseTripGeoJsonTime = (0, _tripUtils.parseTripGeoJsonTimestamp)(this.dataToFeature), dataToTimeStamp = _parseTripGeoJsonTime.dataToTimeStamp, animationDomain = _parseTripGeoJsonTime.animationDomain; this.dataToTimeStamp = dataToTimeStamp; this.updateAnimationDomain(animationDomain); // get bounds from features var bounds = (0, _geojsonUtils.getGeojsonBounds)(this.dataToFeature); // keep a record of what type of geometry the collection has var featureTypes = (0, _geojsonUtils.getGeojsonFeatureTypes)(this.dataToFeature); this.updateMeta({ bounds: bounds, featureTypes: featureTypes, getFeature: getFeature }); } }, { key: "setInitialLayerConfig", value: function setInitialLayerConfig(allData) { this.updateLayerMeta(allData); return this; } }, { key: "renderLayer", value: function renderLayer(_ref3) { var data = _ref3.data, idx = _ref3.idx, mapState = _ref3.mapState, animationConfig = _ref3.animationConfig; var visConfig = this.config.visConfig; var zoomFactor = this.getZoomFactor(mapState); var updateTriggers = { getColor: { color: this.config.color, colorField: this.config.colorField, colorRange: visConfig.colorRange, colorScale: this.config.colorScale }, getWidth: { sizeField: this.config.sizeField, sizeRange: visConfig.sizeRange }, getTimestamps: { columns: this.config.columns, domain0: animationConfig.domain[0] } }; return [new _deck.TripsLayer({ id: this.id, idx: idx, data: data.data, getPath: data.getPath, getColor: data.getColor, getTimestamps: function getTimestamps(d) { return data.getTimestamps(d).map(function (ts) { return ts - animationConfig.domain[0]; }); }, opacity: this.config.visConfig.opacity, widthScale: this.config.visConfig.thickness * zoomFactor * 8, highlightColor: this.config.highlightColor, getWidth: data.getWidth, rounded: true, pickable: true, autoHighlight: true, parameters: { depthTest: mapState.dragRotate, depthMask: false }, trailLength: visConfig.trailLength * 1000, currentTime: animationConfig.currentTime - animationConfig.domain[0], updateTriggers: updateTriggers })]; } }], [{ key: "findDefaultLayerProps", value: function findDefaultLayerProps(_ref4, foundLayers) { var _this3 = this; var label = _ref4.label, _ref4$fields = _ref4.fields, fields = _ref4$fields === void 0 ? [] : _ref4$fields, _ref4$allData = _ref4.allData, allData = _ref4$allData === void 0 ? [] : _ref4$allData, id = _ref4.id; var geojsonColumns = fields.filter(function (f) { return f.type === 'geojson'; }).map(function (f) { return f.name; }); var defaultColumns = { geojson: (0, _lodash2["default"])([].concat((0, _toConsumableArray2["default"])(_defaultSettings.GEOJSON_FIELDS.geojson), (0, _toConsumableArray2["default"])(geojsonColumns))) }; var geoJsonColumns = this.findDefaultColumnField(defaultColumns, fields); var tripColumns = (geoJsonColumns || []).filter(function (col) { return (0, _tripUtils.isTripGeoJsonField)(allData, fields[col.geojson.fieldIdx]); }); if (!tripColumns.length) { return { props: [] }; } return { props: tripColumns.map(function (columns) { return { label: typeof label === 'string' && label.replace(/\.[^/.]+$/, '') || _this3.type, columns: columns, isVisible: true }; }), // if a geojson layer is created from this column, delete it foundLayers: foundLayers.filter(function (prop) { return prop.type !== 'geojson' || prop.dataId !== id || !tripColumns.find(function (c) { return prop.columns.geojson.name === c.geojson.name; }); }) }; } }]); return TripLayer; }(_baseLayer["default"]); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/trip-layer/trip-layer.js"],"names":["tripVisConfigs","opacity","thickness","type","defaultValue","label","isRanged","range","step","group","property","colorRange","trailLength","sizeRange","geoJsonRequiredColumns","featureAccessor","geojson","d","fieldIdx","featureResolver","TripLayer","props","dataToFeature","dataToTimeStamp","registerVisConfig","getFeature","_layerInfoModal","TripLayerIcon","size","condition","config","visConfig","stroked","animation","domain","id","template","modalProps","title","columns","enabled","object","allData","properties","index","_","filteredIndex","oldLayerData","opt","colorScale","colorField","colorDomain","color","sizeScale","sizeDomain","sizeField","getPositionAccessor","column","updateLayerMeta","geojsonData","data","sameData","map","i","filter","geometry","cScale","getVisChannelScale","colors","hexToRgb","sScale","getPath","coordinates","getTimestamps","getColor","getEncodedChannelValue","fillColor","getWidth","lineWidth","updateLayerConfig","meta","animationDomain","updateAnimationDomain","bounds","featureTypes","updateMeta","idx","mapState","animationConfig","zoomFactor","getZoomFactor","updateTriggers","domain0","DeckGLTripsLayer","ts","widthScale","highlightColor","rounded","pickable","autoHighlight","parameters","depthTest","dragRotate","depthMask","currentTime","foundLayers","fields","geojsonColumns","f","name","defaultColumns","GEOJSON_FIELDS","geoJsonColumns","findDefaultColumnField","tripColumns","col","length","replace","isVisible","prop","dataId","find","c","Layer"],"mappings":"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","sourcesContent":["// Copyright (c) 2023 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 uniq from 'lodash.uniq';\nimport Layer from '../base-layer';\nimport {TripsLayer as DeckGLTripsLayer} from 'deck.gl';\n\nimport {GEOJSON_FIELDS} from 'constants/default-settings';\nimport TripLayerIcon from './trip-layer-icon';\n\nimport {\n  getGeojsonDataMaps,\n  getGeojsonBounds,\n  getGeojsonFeatureTypes\n} from 'layers/geojson-layer/geojson-utils';\n\nimport {\n  isTripGeoJsonField,\n  parseTripGeoJsonTimestamp\n} from './trip-utils';\n\nimport {hexToRgb} from 'utils/color-utils';\nimport TripInfoModalFactory from './trip-info-modal';\n\nexport const tripVisConfigs = {\n  opacity: 'opacity',\n  thickness: {\n    type: 'number',\n    defaultValue: 0.5,\n    label: 'Stroke Width',\n    isRanged: false,\n    range: [0, 100],\n    step: 0.1,\n    group: 'stroke',\n    property: 'thickness'\n  },\n  colorRange: 'colorRange',\n  trailLength: 'trailLength',\n  sizeRange: 'strokeWidthRange'\n};\n\nexport const geoJsonRequiredColumns = ['geojson'];\nexport const featureAccessor = ({geojson}) => d => d[geojson.fieldIdx];\nexport const featureResolver = ({geojson}) => geojson.fieldIdx;\n\nexport default class TripLayer extends Layer {\n  constructor(props) {\n    super(props);\n\n    this.dataToFeature = [];\n    this.dataToTimeStamp = [];\n    this.registerVisConfig(tripVisConfigs);\n    this.getFeature = memoize(featureAccessor, featureResolver);\n    this._layerInfoModal = TripInfoModalFactory();\n  }\n\n  get type() {\n    return 'trip';\n  }\n\n  get name() {\n    return 'Trip';\n  }\n\n  get layerIcon() {\n    return TripLayerIcon;\n  }\n\n  get requiredLayerColumns() {\n    return geoJsonRequiredColumns;\n  }\n\n  get visualChannels() {\n    return {\n      ...super.visualChannels,\n\n      size: {\n        ...super.visualChannels.size,\n        property: 'stroke',\n        condition: config => config.visConfig.stroked\n      }\n    };\n  }\n\n  get animationDomain() {\n    return this.config.animation.domain;\n  }\n\n  get layerInfoModal() {\n    return {\n      id: 'iconInfo',\n      template: this._layerInfoModal,\n      modalProps: {\n        title: 'How to enable trip animation'\n      }\n    };\n  }\n\n  getPositionAccessor() {\n    return this.getFeature(this.config.columns);\n  }\n\n  static findDefaultLayerProps({label, fields = [], allData = [], id}, foundLayers) {\n    const geojsonColumns = fields\n      .filter(f => f.type === 'geojson')\n      .map(f => f.name);\n\n    const defaultColumns = {\n      geojson: uniq([...GEOJSON_FIELDS.geojson, ...geojsonColumns])\n    };\n\n    const geoJsonColumns = this.findDefaultColumnField(defaultColumns, fields);\n\n    const tripColumns = (geoJsonColumns || []).filter(col =>\n      isTripGeoJsonField(allData, fields[col.geojson.fieldIdx])\n    );\n\n    if (!tripColumns.length) {\n      return {props: []};\n    }\n\n    return {\n      props: tripColumns.map(columns => ({\n        label:\n          (typeof label === 'string' && label.replace(/\\.[^/.]+$/, '')) ||\n          this.type,\n        columns,\n        isVisible: true\n      })),\n\n      // if a geojson layer is created from this column, delete it\n      foundLayers: foundLayers.filter(\n        prop =>\n          prop.type !== 'geojson' ||\n          prop.dataId !== id ||\n          !tripColumns.find(c => prop.columns.geojson.name === c.geojson.name)\n      )\n    };\n  }\n\n  getDefaultLayerConfig(props) {\n    return {\n      ...super.getDefaultLayerConfig(props),\n      animation: {\n        enabled: true,\n        domain: null\n      }\n    };\n  }\n\n  getHoverData(object, allData) {\n    // index of allData is saved to feature.properties\n    return allData[object.properties.index];\n  }\n\n  // TODO: fix complexity\n  /* eslint-disable complexity */\n  formatLayerData(_, allData, filteredIndex, oldLayerData, opt = {}) {\n    // to-do: parse segment from allData\n    const {\n      colorScale,\n      colorField,\n      colorDomain,\n      color,\n      sizeScale,\n      sizeDomain,\n      sizeField,\n      visConfig\n    } = this.config;\n\n    const {stroked, colorRange, sizeRange} = visConfig;\n\n    const getFeature = this.getPositionAccessor(this.config.column);\n\n    // geojson feature are object, if doesn't exists\n    // create it and save to layer\n    if (!oldLayerData || oldLayerData.getFeature !== getFeature) {\n      this.updateLayerMeta(allData, getFeature);\n    }\n\n    let geojsonData;\n\n    if (\n      oldLayerData &&\n      oldLayerData.data &&\n      opt.sameData &&\n      oldLayerData.getFeature === getFeature\n    ) {\n      // no need to create a new array of data\n      // use updateTriggers to selectively re-calculate attributes\n      geojsonData = oldLayerData.data;\n    } else {\n      // filteredIndex is a reference of index in allData which can map to feature\n      geojsonData = filteredIndex\n        .map(i => this.dataToFeature[i])\n        .filter(d => d && d.geometry.type === 'LineString');\n    }\n\n    // color\n    const cScale =\n      colorField &&\n      this.getVisChannelScale(\n        colorScale,\n        colorDomain,\n        colorRange.colors.map(hexToRgb)\n      );\n\n    // calculate stroke scale - if stroked = true\n    const sScale =\n      sizeField &&\n      stroked &&\n      this.getVisChannelScale(sizeScale, sizeDomain, sizeRange);\n\n    return {\n      data: geojsonData,\n      getPath: d => d.geometry.coordinates,\n      getTimestamps: d => this.dataToTimeStamp[d.properties.index],\n      getColor: d =>\n        cScale\n          ? this.getEncodedChannelValue(\n              cScale,\n              allData[d.properties.index],\n              colorField\n            )\n          : d.properties.fillColor || color,\n      getWidth: d =>\n        sScale\n          ? this.getEncodedChannelValue(\n              sScale,\n              allData[d.properties.index],\n              sizeField,\n              0\n            )\n          : d.properties.lineWidth || 1\n    };\n  }\n  /* eslint-enable complexity */\n\n  updateAnimationDomain(domain) {\n    this.updateLayerConfig({\n      animation: {\n        ...this.config.animation,\n        domain\n      }\n    });\n  }\n\n  updateLayerMeta(allData) {\n    const getFeature = this.getPositionAccessor();\n    if (getFeature === this.meta.getFeature) {\n      // TODO: revisit this after gpu filtering\n      return;\n    }\n\n    this.dataToFeature = getGeojsonDataMaps(allData, getFeature);\n\n    const {dataToTimeStamp, animationDomain} =\n      parseTripGeoJsonTimestamp(this.dataToFeature);\n\n    this.dataToTimeStamp = dataToTimeStamp;\n    this.updateAnimationDomain(animationDomain);\n\n    // get bounds from features\n    const bounds = getGeojsonBounds(this.dataToFeature);\n\n    // keep a record of what type of geometry the collection has\n    const featureTypes = getGeojsonFeatureTypes(this.dataToFeature);\n\n    this.updateMeta({bounds, featureTypes, getFeature});\n  }\n\n  setInitialLayerConfig(allData) {\n    this.updateLayerMeta(allData);\n    return this;\n  }\n\n  renderLayer({data, idx, mapState, animationConfig}) {\n    const {visConfig} = this.config;\n    const zoomFactor = this.getZoomFactor(mapState);\n\n    const updateTriggers = {\n      getColor: {\n        color: this.config.color,\n        colorField: this.config.colorField,\n        colorRange: visConfig.colorRange,\n        colorScale: this.config.colorScale\n      },\n      getWidth: {\n        sizeField: this.config.sizeField,\n        sizeRange: visConfig.sizeRange\n      },\n      getTimestamps: {\n        columns: this.config.columns,\n        domain0: animationConfig.domain[0]\n      }\n    };\n\n    return [\n      new DeckGLTripsLayer({\n        id: this.id,\n        idx,\n        data: data.data,\n        getPath: data.getPath,\n        getColor: data.getColor,\n        getTimestamps: d =>\n          data.getTimestamps(d).map(ts => ts - animationConfig.domain[0]),\n        opacity: this.config.visConfig.opacity,\n        widthScale: this.config.visConfig.thickness * zoomFactor * 8,\n        highlightColor: this.config.highlightColor,\n\n        getWidth: data.getWidth,\n        rounded: true,\n        pickable: true,\n        autoHighlight: true,\n        parameters: {\n          depthTest: mapState.dragRotate,\n          depthMask: false\n        },\n        trailLength: visConfig.trailLength * 1000,\n        currentTime: animationConfig.currentTime - animationConfig.domain[0],\n        updateTriggers\n      })\n    ];\n  }\n}\n"]}