UNPKG

kepler.gl.geoiq

Version:

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

377 lines (319 loc) 40.4 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.pointVisConfigs = exports.pointOptionalColumns = exports.pointRequiredColumns = exports.pointLabelResolver = exports.pointLabelAccessor = exports.pointPosResolver = exports.pointPosAccessor = 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 _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 _baseLayer = _interopRequireDefault(require("../base-layer")); var _lodash = _interopRequireDefault(require("lodash.memoize")); var _deck = require("deck.gl"); var _scatterplotBrushingLayer = _interopRequireDefault(require("../../deckgl-layers/scatterplot-brushing-layer/scatterplot-brushing-layer")); var _lodash2 = _interopRequireDefault(require("lodash.uniq")); var _colorUtils = require("../../utils/color-utils"); var _pointLayerIcon = _interopRequireDefault(require("./point-layer-icon")); var _defaultSettings = require("../../constants/default-settings"); 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; } var pointPosAccessor = function pointPosAccessor(_ref) { var lat = _ref.lat, lng = _ref.lng, altitude = _ref.altitude; return function (d) { return [d.data[lng.fieldIdx], d.data[lat.fieldIdx], altitude && altitude.fieldIdx > -1 ? d.data[altitude.fieldIdx] : 0]; }; }; exports.pointPosAccessor = pointPosAccessor; var pointPosResolver = function pointPosResolver(_ref2) { var lat = _ref2.lat, lng = _ref2.lng, altitude = _ref2.altitude; return "".concat(lat.fieldIdx, "-").concat(lng.fieldIdx, "-").concat(altitude ? altitude.fieldIdx : 'z'); }; exports.pointPosResolver = pointPosResolver; var pointLabelAccessor = function pointLabelAccessor(textLabel) { return function (d) { return String(d.data[textLabel.field.tableFieldIndex - 1]); }; }; exports.pointLabelAccessor = pointLabelAccessor; var pointLabelResolver = function pointLabelResolver(textLabel) { return textLabel.field && textLabel.field.tableFieldIndex; }; exports.pointLabelResolver = pointLabelResolver; var pointRequiredColumns = ['lat', 'lng']; exports.pointRequiredColumns = pointRequiredColumns; var pointOptionalColumns = ['altitude']; exports.pointOptionalColumns = pointOptionalColumns; var pointVisConfigs = { radius: 'radius', fixedRadius: 'fixedRadius', opacity: 'opacity', outline: 'outline', thickness: 'thickness', colorRange: 'colorRange', radiusRange: 'radiusRange', 'hi-precision': 'hi-precision' }; exports.pointVisConfigs = pointVisConfigs; var PointLayer = /*#__PURE__*/ function (_Layer) { (0, _inherits2["default"])(PointLayer, _Layer); function PointLayer(props) { var _this; (0, _classCallCheck2["default"])(this, PointLayer); _this = (0, _possibleConstructorReturn2["default"])(this, (0, _getPrototypeOf2["default"])(PointLayer).call(this, props)); _this.registerVisConfig(pointVisConfigs); _this.getPosition = (0, _lodash["default"])(pointPosAccessor, pointPosResolver); _this.getText = (0, _lodash["default"])(pointLabelAccessor, pointLabelResolver); return _this; } (0, _createClass2["default"])(PointLayer, [{ key: "formatLayerData", // TODO: fix complexity /* eslint-disable complexity */ value: function formatLayerData(_, allData, filteredIndex, oldLayerData) { var _this2 = this; var opt = arguments.length > 4 && arguments[4] !== undefined ? arguments[4] : {}; var _this$config = this.config, colorScale = _this$config.colorScale, colorDomain = _this$config.colorDomain, colorField = _this$config.colorField, color = _this$config.color, columns = _this$config.columns, sizeField = _this$config.sizeField, sizeScale = _this$config.sizeScale, sizeDomain = _this$config.sizeDomain, textLabel = _this$config.textLabel, _this$config$visConfi = _this$config.visConfig, radiusRange = _this$config$visConfi.radiusRange, fixedRadius = _this$config$visConfi.fixedRadius, colorRange = _this$config$visConfi.colorRange; // point color var cScale = colorField && this.getVisChannelScale(colorScale, colorDomain, colorRange.colors.map(_colorUtils.hexToRgb)); // point radius var rScale = sizeField && this.getVisChannelScale(sizeScale, sizeDomain, radiusRange, fixedRadius); var getPosition = this.getPosition(columns); if (!oldLayerData || oldLayerData.getPosition !== getPosition) { this.updateLayerMeta(allData, getPosition); } var data; if (oldLayerData && oldLayerData.data && opt.sameData && oldLayerData.getPosition === getPosition) { data = oldLayerData.data; } else { data = filteredIndex.reduce(function (accu, index) { var pos = getPosition({ data: allData[index] }); // 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)) { return accu; } accu.push({ data: allData[index] }); return accu; }, []); } // get all distinct characters in the text labels var getText = this.getText(textLabel); var labelCharacterSet; if (oldLayerData && oldLayerData.labelCharacterSet && opt.sameData && oldLayerData.getText === getText) { labelCharacterSet = oldLayerData.labelCharacterSet; } else { var textLabels = textLabel.field ? data.map(getText) : []; labelCharacterSet = (0, _lodash2["default"])(textLabels.join('')); } var getRadius = rScale ? function (d) { return _this2.getEncodedChannelValue(rScale, d.data, sizeField); } : 1; var getColor = cScale ? function (d) { return _this2.getEncodedChannelValue(cScale, d.data, colorField); } : color; return { data: data, labelCharacterSet: labelCharacterSet, getPosition: getPosition, getColor: getColor, getRadius: getRadius, getText: getText }; } /* eslint-enable complexity */ }, { key: "updateLayerMeta", value: function updateLayerMeta(allData, getPosition) { var bounds = this.getPointsBounds(allData, function (d) { return getPosition({ data: d }); }); this.updateMeta({ bounds: bounds }); } }, { key: "renderLayer", value: function renderLayer(_ref3) { var _this3 = this; var data = _ref3.data, idx = _ref3.idx, layerInteraction = _ref3.layerInteraction, objectHovered = _ref3.objectHovered, mapState = _ref3.mapState, interactionConfig = _ref3.interactionConfig; var enableBrushing = interactionConfig.brush.enabled; var layerProps = _objectSpread({ outline: this.config.visConfig.outline, radiusMinPixels: 1, fp64: this.config.visConfig['hi-precision'], strokeWidth: this.config.visConfig.thickness, radiusScale: this.getRadiusScaleByZoom(mapState) }, this.config.visConfig.fixedRadius ? {} : { radiusMaxPixels: 500 }); var interaction = { autoHighlight: !enableBrushing, enableBrushing: enableBrushing, brushRadius: interactionConfig.brush.config.size * 1000, highlightColor: this.config.highlightColor }; return [new _scatterplotBrushingLayer["default"](_objectSpread({}, layerProps, {}, layerInteraction, {}, data, {}, interaction, { idx: idx, id: this.id, opacity: this.config.visConfig.opacity, pickable: true, parameters: { // circles will be flat on the map when the altitude column is not used depthTest: this.config.columns.altitude.fieldIdx > -1 }, updateTriggers: { getRadius: { sizeField: this.config.sizeField, radiusRange: this.config.visConfig.radiusRange, fixedRadius: this.config.visConfig.fixedRadius, sizeScale: this.config.sizeScale }, getColor: { color: this.config.color, colorField: this.config.colorField, colorRange: this.config.visConfig.colorRange, colorScale: this.config.colorScale } } }))].concat((0, _toConsumableArray2["default"])(this.config.textLabel.field ? [new _deck.TextLayer({ id: "".concat(this.id, "-label"), data: data.data, getPosition: data.getPosition, getPixelOffset: this.config.textLabel.offset, getSize: this.config.textLabel.size, getTextAnchor: this.config.textLabel.anchor, getText: data.getText, getColor: function getColor(d) { return _this3.config.textLabel.color; }, fp64: this.config.visConfig['hi-precision'], parameters: { // text will always show on top of all layers depthTest: false }, characterSet: data.labelCharacterSet, updateTriggers: { getPosition: data.getPosition, getPixelOffset: this.config.textLabel.offset, getText: this.config.textLabel.field, getTextAnchor: this.config.textLabel.anchor, getSize: this.config.textLabel.size, getColor: this.config.textLabel.color } })] : [])); } }, { key: "type", get: function get() { return 'point'; } }, { key: "isAggregated", get: function get() { return false; } }, { key: "layerIcon", get: function get() { return _pointLayerIcon["default"]; } }, { key: "requiredLayerColumns", get: function get() { return pointRequiredColumns; } }, { key: "optionalColumns", get: function get() { return pointOptionalColumns; } }, { key: "columnPairs", get: function get() { return this.defaultPointColumnPairs; } }, { key: "noneLayerDataAffectingProps", get: function get() { return [].concat((0, _toConsumableArray2["default"])((0, _get2["default"])((0, _getPrototypeOf2["default"])(PointLayer.prototype), "noneLayerDataAffectingProps", this)), ['radius']); } }, { key: "visualChannels", get: function get() { return _objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(PointLayer.prototype), "visualChannels", this), { size: _objectSpread({}, (0, _get2["default"])((0, _getPrototypeOf2["default"])(PointLayer.prototype), "visualChannels", this).size, { range: 'radiusRange', property: 'radius', channelScaleType: 'radius' }) }); } }], [{ key: "findDefaultLayerProps", value: function findDefaultLayerProps(_ref4) { var _ref4$fieldPairs = _ref4.fieldPairs, fieldPairs = _ref4$fieldPairs === void 0 ? [] : _ref4$fieldPairs; var props = []; // Make layer for each pair fieldPairs.forEach(function (pair) { // find fields for tableFieldIndex var latField = pair.pair.lat; var lngField = pair.pair.lng; var layerName = pair.defaultName; var prop = { label: layerName.length ? layerName : 'Point' }; // default layer color for begintrip and dropoff point if (latField.value in _defaultSettings.DEFAULT_LAYER_COLOR) { prop.color = (0, _colorUtils.hexToRgb)(_defaultSettings.DEFAULT_LAYER_COLOR[latField.value]); } // set the first layer to be visible if (props.length === 0) { prop.isVisible = true; } // const newLayer = new KeplerGlLayers.PointLayer(prop); prop.columns = { lat: latField, lng: lngField, altitude: { value: null, fieldIdx: -1, optional: true } }; props.push(prop); }); return props; } }]); return PointLayer; }(_baseLayer["default"]); exports["default"] = PointLayer; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/layers/point-layer/point-layer.js"],"names":["pointPosAccessor","lat","lng","altitude","d","data","fieldIdx","pointPosResolver","pointLabelAccessor","textLabel","String","field","tableFieldIndex","pointLabelResolver","pointRequiredColumns","pointOptionalColumns","pointVisConfigs","radius","fixedRadius","opacity","outline","thickness","colorRange","radiusRange","PointLayer","props","registerVisConfig","getPosition","getText","_","allData","filteredIndex","oldLayerData","opt","config","colorScale","colorDomain","colorField","color","columns","sizeField","sizeScale","sizeDomain","visConfig","cScale","getVisChannelScale","colors","map","hexToRgb","rScale","updateLayerMeta","sameData","reduce","accu","index","pos","every","Number","isFinite","push","labelCharacterSet","textLabels","join","getRadius","getEncodedChannelValue","getColor","bounds","getPointsBounds","updateMeta","idx","layerInteraction","objectHovered","mapState","interactionConfig","enableBrushing","brush","enabled","layerProps","radiusMinPixels","fp64","strokeWidth","radiusScale","getRadiusScaleByZoom","radiusMaxPixels","interaction","autoHighlight","brushRadius","size","highlightColor","ScatterplotBrushingLayer","id","pickable","parameters","depthTest","updateTriggers","TextLayer","getPixelOffset","offset","getSize","getTextAnchor","anchor","characterSet","PointLayerIcon","defaultPointColumnPairs","range","property","channelScaleType","fieldPairs","forEach","pair","latField","lngField","layerName","defaultName","prop","label","length","value","DEFAULT_LAYER_COLOR","isVisible","optional","Layer"],"mappings":"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","sourcesContent":["// Copyright (c) 2019 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 Layer from '../base-layer';\nimport memoize from 'lodash.memoize';\nimport {TextLayer} from 'deck.gl';\nimport ScatterplotBrushingLayer from 'deckgl-layers/scatterplot-brushing-layer/scatterplot-brushing-layer';\nimport uniq from 'lodash.uniq';\nimport {hexToRgb} from 'utils/color-utils';\nimport PointLayerIcon from './point-layer-icon';\nimport {DEFAULT_LAYER_COLOR} from 'constants/default-settings';\n\nexport const pointPosAccessor = ({lat, lng, altitude}) => d => [\n  d.data[lng.fieldIdx],\n  d.data[lat.fieldIdx],\n  altitude && altitude.fieldIdx > -1 ? d.data[altitude.fieldIdx] : 0\n];\n\nexport const pointPosResolver = ({lat, lng, altitude}) =>\n  `${lat.fieldIdx}-${lng.fieldIdx}-${altitude ? altitude.fieldIdx : 'z'}`;\n\nexport const pointLabelAccessor = textLabel => d =>\n  String(d.data[textLabel.field.tableFieldIndex - 1]);\nexport const pointLabelResolver = textLabel =>\n  textLabel.field && textLabel.field.tableFieldIndex;\n\nexport const pointRequiredColumns = ['lat', 'lng'];\nexport const pointOptionalColumns = ['altitude'];\n\nexport const pointVisConfigs = {\n  radius: 'radius',\n  fixedRadius: 'fixedRadius',\n  opacity: 'opacity',\n  outline: 'outline',\n  thickness: 'thickness',\n  colorRange: 'colorRange',\n  radiusRange: 'radiusRange',\n  'hi-precision': 'hi-precision'\n};\n\nexport default class PointLayer extends Layer {\n  constructor(props) {\n    super(props);\n\n    this.registerVisConfig(pointVisConfigs);\n    this.getPosition = memoize(pointPosAccessor, pointPosResolver);\n    this.getText = memoize(pointLabelAccessor, pointLabelResolver);\n  }\n\n  get type() {\n    return 'point';\n  }\n\n  get isAggregated() {\n    return false;\n  }\n\n  get layerIcon() {\n    return PointLayerIcon;\n  }\n  get requiredLayerColumns() {\n    return pointRequiredColumns;\n  }\n\n  get optionalColumns() {\n    return pointOptionalColumns;\n  }\n\n  get columnPairs() {\n    return this.defaultPointColumnPairs;\n  }\n\n  get noneLayerDataAffectingProps() {\n    return [...super.noneLayerDataAffectingProps, 'radius'];\n  }\n\n  get visualChannels() {\n    return {\n      ...super.visualChannels,\n      size: {\n        ...super.visualChannels.size,\n        range: 'radiusRange',\n        property: 'radius',\n        channelScaleType: 'radius'\n      }\n    };\n  }\n\n  static findDefaultLayerProps({fieldPairs = []}) {\n    const props = [];\n\n    // Make layer for each pair\n    fieldPairs.forEach(pair => {\n      // find fields for tableFieldIndex\n      const latField = pair.pair.lat;\n      const lngField = pair.pair.lng;\n      const layerName = pair.defaultName;\n\n      const prop = {\n        label: layerName.length ? layerName : 'Point'\n      };\n\n      // default layer color for begintrip and dropoff point\n      if (latField.value in DEFAULT_LAYER_COLOR) {\n        prop.color = hexToRgb(DEFAULT_LAYER_COLOR[latField.value]);\n      }\n\n      // set the first layer to be visible\n      if (props.length === 0) {\n        prop.isVisible = true;\n      }\n\n      // const newLayer = new KeplerGlLayers.PointLayer(prop);\n      prop.columns = {\n        lat: latField,\n        lng: lngField,\n        altitude: {value: null, fieldIdx: -1, optional: true}\n      };\n\n      props.push(prop);\n    });\n\n    return props;\n  }\n\n  // TODO: fix complexity\n  /* eslint-disable complexity */\n  formatLayerData(_, allData, filteredIndex, oldLayerData, opt = {}) {\n    const {\n      colorScale,\n      colorDomain,\n      colorField,\n      color,\n      columns,\n      sizeField,\n      sizeScale,\n      sizeDomain,\n      textLabel,\n      visConfig: {radiusRange, fixedRadius, colorRange}\n    } = this.config;\n\n    // point color\n    const cScale =\n      colorField &&\n      this.getVisChannelScale(\n        colorScale,\n        colorDomain,\n        colorRange.colors.map(hexToRgb)\n      );\n\n    // point radius\n    const rScale =\n      sizeField &&\n      this.getVisChannelScale(sizeScale, sizeDomain, radiusRange, fixedRadius);\n\n    const getPosition = this.getPosition(columns);\n    if (!oldLayerData || oldLayerData.getPosition !== getPosition) {\n      this.updateLayerMeta(allData, getPosition);\n    }\n\n    let data;\n    if (\n      oldLayerData &&\n      oldLayerData.data &&\n      opt.sameData &&\n      oldLayerData.getPosition === getPosition\n    ) {\n      data = oldLayerData.data;\n    } else {\n      data = filteredIndex.reduce((accu, index) => {\n        const pos = getPosition({data: allData[index]});\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)) {\n          return accu;\n        }\n\n        accu.push({\n          data: allData[index]\n        });\n\n        return accu;\n      }, []);\n    }\n\n    // get all distinct characters in the text labels\n    const getText = this.getText(textLabel);\n    let labelCharacterSet;\n    if (\n      oldLayerData &&\n      oldLayerData.labelCharacterSet &&\n      opt.sameData &&\n      oldLayerData.getText === getText\n    ) {\n      labelCharacterSet = oldLayerData.labelCharacterSet;\n    } else {\n      const textLabels = textLabel.field ? data.map(getText) : [];\n      labelCharacterSet = uniq(textLabels.join(''));\n    }\n\n    const getRadius = rScale\n      ? d => this.getEncodedChannelValue(rScale, d.data, sizeField)\n      : 1;\n\n    const getColor = cScale\n      ? d => this.getEncodedChannelValue(cScale, d.data, colorField)\n      : color;\n\n    return {\n      data,\n      labelCharacterSet,\n      getPosition,\n      getColor,\n      getRadius,\n      getText\n    };\n  }\n  /* eslint-enable complexity */\n\n  updateLayerMeta(allData, getPosition) {\n    const bounds = this.getPointsBounds(allData, d => getPosition({data: d}));\n    this.updateMeta({bounds});\n  }\n\n  renderLayer({\n    data,\n    idx,\n    layerInteraction,\n    objectHovered,\n    mapState,\n    interactionConfig\n  }) {\n    const enableBrushing = interactionConfig.brush.enabled;\n\n    const layerProps = {\n      outline: this.config.visConfig.outline,\n      radiusMinPixels: 1,\n      fp64: this.config.visConfig['hi-precision'],\n      strokeWidth: this.config.visConfig.thickness,\n      radiusScale: this.getRadiusScaleByZoom(mapState),\n      ...(this.config.visConfig.fixedRadius ? {} : {radiusMaxPixels: 500})\n    };\n\n    const interaction = {\n      autoHighlight: !enableBrushing,\n      enableBrushing,\n      brushRadius: interactionConfig.brush.config.size * 1000,\n      highlightColor: this.config.highlightColor\n    };\n\n    return [\n      new ScatterplotBrushingLayer({\n        ...layerProps,\n        ...layerInteraction,\n        ...data,\n        ...interaction,\n        idx,\n        id: this.id,\n        opacity: this.config.visConfig.opacity,\n        pickable: true,\n        parameters: {\n          // circles will be flat on the map when the altitude column is not used\n          depthTest: this.config.columns.altitude.fieldIdx > -1\n        },\n\n        updateTriggers: {\n          getRadius: {\n            sizeField: this.config.sizeField,\n            radiusRange: this.config.visConfig.radiusRange,\n            fixedRadius: this.config.visConfig.fixedRadius,\n            sizeScale: this.config.sizeScale\n          },\n          getColor: {\n            color: this.config.color,\n            colorField: this.config.colorField,\n            colorRange: this.config.visConfig.colorRange,\n            colorScale: this.config.colorScale\n          }\n        }\n      }),\n      // text label layer\n      ...(this.config.textLabel.field\n        ? [\n            new TextLayer({\n              id: `${this.id}-label`,\n              data: data.data,\n              getPosition: data.getPosition,\n              getPixelOffset: this.config.textLabel.offset,\n              getSize: this.config.textLabel.size,\n              getTextAnchor: this.config.textLabel.anchor,\n              getText: data.getText,\n              getColor: d => this.config.textLabel.color,\n              fp64: this.config.visConfig['hi-precision'],\n              parameters: {\n                // text will always show on top of all layers\n                depthTest: false\n              },\n              characterSet: data.labelCharacterSet,\n              updateTriggers: {\n                getPosition: data.getPosition,\n                getPixelOffset: this.config.textLabel.offset,\n                getText: this.config.textLabel.field,\n                getTextAnchor: this.config.textLabel.anchor,\n                getSize: this.config.textLabel.size,\n                getColor: this.config.textLabel.color\n              }\n            })\n          ]\n        : [])\n    ];\n  }\n}\n"]}