UNPKG

kepler.gl

Version:

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

402 lines (345 loc) 39.7 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.featureResolver = exports.featureAccessor = exports.geoJsonRequiredColumns = exports.tripVisConfigs = exports.defaultLineWidth = exports.defaultThickness = 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 _geoLayers = require("@deck.gl/geo-layers"); 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 _tripInfoModal = _interopRequireDefault(require("./trip-info-modal")); 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 zoomFactorValue = 8; var defaultThickness = 0.5; exports.defaultThickness = defaultThickness; var defaultLineWidth = 1; exports.defaultLineWidth = defaultLineWidth; var tripVisConfigs = { opacity: 'opacity', thickness: { type: 'number', defaultValue: defaultThickness, label: 'Stroke Width', isRanged: false, range: [0, 100], step: 0.1, group: 'stroke', property: 'thickness' }, colorRange: 'colorRange', trailLength: 'trailLength', sizeRange: 'strokeWidthRange' }; exports.tripVisConfigs = tripVisConfigs; var geoJsonRequiredColumns = ['geojson']; exports.geoJsonRequiredColumns = geoJsonRequiredColumns; var featureAccessor = function featureAccessor(_ref) { var geojson = _ref.geojson; return function (dc) { return function (d) { return dc.valueAt(d.index, geojson.fieldIdx); }; }; }; exports.featureAccessor = featureAccessor; var featureResolver = function featureResolver(_ref2) { var geojson = _ref2.geojson; return geojson.fieldIdx; }; exports.featureResolver = featureResolver; var TripLayer = /*#__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() { var visualChannels = (0, _get2["default"])((0, _getPrototypeOf2["default"])(TripLayer.prototype), "visualChannels", this); return _objectSpread(_objectSpread({}, visualChannels), {}, { color: _objectSpread(_objectSpread({}, visualChannels.color), {}, { accessor: 'getColor', nullValue: visualChannels.color.nullValue, getAttributeValue: function getAttributeValue(config) { return function (d) { return d.properties.lineColor || config.color; }; }, // used this to get updateTriggers defaultValue: function defaultValue(config) { return config.color; } }), size: _objectSpread(_objectSpread({}, visualChannels.size), {}, { property: 'stroke', accessor: 'getWidth', condition: function condition(config) { return config.visConfig.stroked; }, nullValue: 0, getAttributeValue: function getAttributeValue() { return function (d) { return d.properties.lineWidth || defaultLineWidth; }; } }) }); } }, { key: "animationDomain", get: function get() { return this.config.animation.domain; } }, { key: "layerInfoModal", get: function get() { return { id: 'iconInfo', template: this._layerInfoModal, modalProps: { title: 'modal.tripInfo.title' } }; } }, { key: "getPositionAccessor", value: function getPositionAccessor(dataContainer) { return this.getFeature(this.config.columns)(dataContainer); } }, { 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, dataContainer) { // index for dataContainer is saved to feature.properties return dataContainer.row(object.properties.index); } }, { key: "calculateDataAttribute", value: function calculateDataAttribute(_ref3, getPosition) { var _this2 = this; var dataContainer = _ref3.dataContainer, filteredIndex = _ref3.filteredIndex; return filteredIndex.map(function (i) { return _this2.dataToFeature[i]; }).filter(function (d) { return d && d.geometry.type === 'LineString'; }); } }, { key: "formatLayerData", value: function formatLayerData(datasets, oldLayerData) { var _this3 = this; // to-do: parse segment from dataContainer var _datasets$this$config = datasets[this.config.dataId], dataContainer = _datasets$this$config.dataContainer, gpuFilter = _datasets$this$config.gpuFilter; var _this$updateData = this.updateData(datasets, oldLayerData), data = _this$updateData.data; var customFilterValueAccessor = function customFilterValueAccessor(dc, f, fieldIndex) { return dc.valueAt(f.properties.index, fieldIndex); }; var indexAccessor = function indexAccessor(f) { return f.properties.index; }; var dataAccessor = function dataAccessor(dc) { return function (d) { return { index: d.properties.index }; }; }; var accessors = this.getAttributeAccessors({ dataAccessor: dataAccessor, dataContainer: dataContainer }); return _objectSpread({ data: data, getFilterValue: gpuFilter.filterValueAccessor(dataContainer)(indexAccessor, customFilterValueAccessor), getPath: function getPath(d) { return d.geometry.coordinates; }, getTimestamps: function getTimestamps(d) { return _this3.dataToTimeStamp[d.properties.index]; } }, accessors); } }, { key: "updateAnimationDomain", value: function updateAnimationDomain(domain) { this.updateLayerConfig({ animation: _objectSpread(_objectSpread({}, this.config.animation), {}, { domain: domain }) }); } }, { key: "updateLayerMeta", value: function updateLayerMeta(dataContainer) { var getFeature = this.getPositionAccessor(dataContainer); if (getFeature === this.meta.getFeature) { // TODO: revisit this after gpu filtering return; } this.dataToFeature = (0, _geojsonUtils.getGeojsonDataMaps)(dataContainer, 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(_ref4) { var dataContainer = _ref4.dataContainer; this.updateLayerMeta(dataContainer); return this; } }, { key: "renderLayer", value: function renderLayer(opts) { var _animationConfig$doma; var data = opts.data, gpuFilter = opts.gpuFilter, mapState = opts.mapState, animationConfig = opts.animationConfig; var visConfig = this.config.visConfig; var zoomFactor = this.getZoomFactor(mapState); var isValidTime = animationConfig && Array.isArray(animationConfig.domain) && animationConfig.domain.every(Number.isFinite) && Number.isFinite(animationConfig.currentTime); if (!isValidTime) { return []; } var domain0 = (_animationConfig$doma = animationConfig.domain) === null || _animationConfig$doma === void 0 ? void 0 : _animationConfig$doma[0]; var updateTriggers = _objectSpread(_objectSpread({}, this.getVisualChannelUpdateTriggers()), {}, { getTimestamps: { columns: this.config.columns, domain0: domain0 }, getFilterValue: gpuFilter.filterValueUpdateTriggers }); var defaultLayerProps = this.getDefaultDeckLayerProps(opts); return [new _geoLayers.TripsLayer(_objectSpread(_objectSpread(_objectSpread({}, defaultLayerProps), data), {}, { getTimestamps: function getTimestamps(d) { return data.getTimestamps(d).map(function (ts) { return ts - domain0; }); }, widthScale: this.config.visConfig.thickness * zoomFactor * zoomFactorValue, rounded: true, wrapLongitude: false, parameters: { depthTest: mapState.dragRotate, depthMask: false }, trailLength: visConfig.trailLength * 1000, currentTime: animationConfig.currentTime - domain0, updateTriggers: updateTriggers }))]; } }], [{ key: "findDefaultLayerProps", value: function findDefaultLayerProps(_ref5, foundLayers) { var _this4 = this; var label = _ref5.label, _ref5$fields = _ref5.fields, fields = _ref5$fields === void 0 ? [] : _ref5$fields, dataContainer = _ref5.dataContainer, id = _ref5.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)(dataContainer, fields[col.geojson.fieldIdx]); }); if (!tripColumns.length) { return { props: [] }; } return { props: tripColumns.map(function (columns) { return { label: typeof label === 'string' && label.replace(/\.[^/.]+$/, '') || _this4.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"]); exports["default"] = TripLayer; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/trip-layer/trip-layer.js"],"names":["zoomFactorValue","defaultThickness","defaultLineWidth","tripVisConfigs","opacity","thickness","type","defaultValue","label","isRanged","range","step","group","property","colorRange","trailLength","sizeRange","geoJsonRequiredColumns","featureAccessor","geojson","dc","d","valueAt","index","fieldIdx","featureResolver","TripLayer","props","dataToFeature","dataToTimeStamp","registerVisConfig","getFeature","_layerInfoModal","TripLayerIcon","visualChannels","color","accessor","nullValue","getAttributeValue","config","properties","lineColor","size","condition","visConfig","stroked","lineWidth","animation","domain","id","template","modalProps","title","dataContainer","columns","enabled","object","row","getPosition","filteredIndex","map","i","filter","geometry","datasets","oldLayerData","dataId","gpuFilter","updateData","data","customFilterValueAccessor","f","fieldIndex","indexAccessor","dataAccessor","accessors","getAttributeAccessors","getFilterValue","filterValueAccessor","getPath","coordinates","getTimestamps","updateLayerConfig","getPositionAccessor","meta","animationDomain","updateAnimationDomain","bounds","featureTypes","updateMeta","updateLayerMeta","opts","mapState","animationConfig","zoomFactor","getZoomFactor","isValidTime","Array","isArray","every","Number","isFinite","currentTime","domain0","updateTriggers","getVisualChannelUpdateTriggers","filterValueUpdateTriggers","defaultLayerProps","getDefaultDeckLayerProps","DeckGLTripsLayer","ts","widthScale","rounded","wrapLongitude","parameters","depthTest","dragRotate","depthMask","foundLayers","fields","geojsonColumns","name","defaultColumns","GEOJSON_FIELDS","geoJsonColumns","findDefaultColumnField","tripColumns","col","length","replace","isVisible","prop","find","c","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 memoize from 'lodash.memoize';\nimport uniq from 'lodash.uniq';\nimport Layer from '../base-layer';\nimport {TripsLayer as DeckGLTripsLayer} from '@deck.gl/geo-layers';\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 {isTripGeoJsonField, parseTripGeoJsonTimestamp} from './trip-utils';\nimport TripInfoModalFactory from './trip-info-modal';\n\nconst zoomFactorValue = 8;\n\nexport const defaultThickness = 0.5;\nexport const defaultLineWidth = 1;\n\nexport const tripVisConfigs = {\n  opacity: 'opacity',\n  thickness: {\n    type: 'number',\n    defaultValue: defaultThickness,\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}) => dc => d => dc.valueAt(d.index, 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    const visualChannels = super.visualChannels;\n\n    return {\n      ...visualChannels,\n      color: {\n        ...visualChannels.color,\n        accessor: 'getColor',\n        nullValue: visualChannels.color.nullValue,\n        getAttributeValue: config => d => d.properties.lineColor || config.color,\n        // used this to get updateTriggers\n        defaultValue: config => config.color\n      },\n      size: {\n        ...visualChannels.size,\n        property: 'stroke',\n        accessor: 'getWidth',\n        condition: config => config.visConfig.stroked,\n        nullValue: 0,\n        getAttributeValue: () => d => d.properties.lineWidth || defaultLineWidth\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: 'modal.tripInfo.title'\n      }\n    };\n  }\n\n  getPositionAccessor(dataContainer) {\n    return this.getFeature(this.config.columns)(dataContainer);\n  }\n\n  static findDefaultLayerProps({label, fields = [], dataContainer, id}, foundLayers) {\n    const geojsonColumns = fields.filter(f => f.type === 'geojson').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(dataContainer, fields[col.geojson.fieldIdx])\n    );\n\n    if (!tripColumns.length) {\n      return {props: []};\n    }\n\n    return {\n      props: tripColumns.map(columns => ({\n        label: (typeof label === 'string' && label.replace(/\\.[^/.]+$/, '')) || 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, dataContainer) {\n    // index for dataContainer is saved to feature.properties\n    return dataContainer.row(object.properties.index);\n  }\n\n  calculateDataAttribute({dataContainer, filteredIndex}, getPosition) {\n    return filteredIndex\n      .map(i => this.dataToFeature[i])\n      .filter(d => d && d.geometry.type === 'LineString');\n  }\n\n  formatLayerData(datasets, oldLayerData) {\n    // to-do: parse segment from dataContainer\n    const {dataContainer, gpuFilter} = datasets[this.config.dataId];\n    const {data} = this.updateData(datasets, oldLayerData);\n\n    const customFilterValueAccessor = (dc, f, fieldIndex) => {\n      return dc.valueAt(f.properties.index, fieldIndex);\n    };\n    const indexAccessor = f => f.properties.index;\n\n    const dataAccessor = dc => d => ({index: d.properties.index});\n    const accessors = this.getAttributeAccessors({dataAccessor, dataContainer});\n\n    return {\n      data,\n      getFilterValue: gpuFilter.filterValueAccessor(dataContainer)(\n        indexAccessor,\n        customFilterValueAccessor\n      ),\n      getPath: d => d.geometry.coordinates,\n      getTimestamps: d => this.dataToTimeStamp[d.properties.index],\n      ...accessors\n    };\n  }\n\n  updateAnimationDomain(domain) {\n    this.updateLayerConfig({\n      animation: {\n        ...this.config.animation,\n        domain\n      }\n    });\n  }\n\n  updateLayerMeta(dataContainer) {\n    const getFeature = this.getPositionAccessor(dataContainer);\n    if (getFeature === this.meta.getFeature) {\n      // TODO: revisit this after gpu filtering\n      return;\n    }\n\n    this.dataToFeature = getGeojsonDataMaps(dataContainer, getFeature);\n\n    const {dataToTimeStamp, animationDomain} = 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({dataContainer}) {\n    this.updateLayerMeta(dataContainer);\n    return this;\n  }\n\n  renderLayer(opts) {\n    const {data, gpuFilter, mapState, animationConfig} = opts;\n    const {visConfig} = this.config;\n    const zoomFactor = this.getZoomFactor(mapState);\n    const isValidTime =\n      animationConfig &&\n      Array.isArray(animationConfig.domain) &&\n      animationConfig.domain.every(Number.isFinite) &&\n      Number.isFinite(animationConfig.currentTime);\n\n    if (!isValidTime) {\n      return [];\n    }\n\n    const domain0 = animationConfig.domain?.[0];\n\n    const updateTriggers = {\n      ...this.getVisualChannelUpdateTriggers(),\n      getTimestamps: {\n        columns: this.config.columns,\n        domain0\n      },\n      getFilterValue: gpuFilter.filterValueUpdateTriggers\n    };\n    const defaultLayerProps = this.getDefaultDeckLayerProps(opts);\n\n    return [\n      new DeckGLTripsLayer({\n        ...defaultLayerProps,\n        ...data,\n        getTimestamps: d => data.getTimestamps(d).map(ts => ts - domain0),\n        widthScale: this.config.visConfig.thickness * zoomFactor * zoomFactorValue,\n        rounded: true,\n        wrapLongitude: false,\n        parameters: {\n          depthTest: mapState.dragRotate,\n          depthMask: false\n        },\n        trailLength: visConfig.trailLength * 1000,\n        currentTime: animationConfig.currentTime - domain0,\n        updateTriggers\n      })\n    ];\n  }\n}\n"]}