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
JavaScript
"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"]}