UNPKG

kepler.gl

Version:

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

661 lines (631 loc) 92.9 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _typeof = require("@babel/runtime/helpers/typeof"); Object.defineProperty(exports, "__esModule", { value: true }); exports.getDefaultMapStyles = exports.editCustomMapStyleUpdater = exports.addCustomMapStyleUpdater = exports.INITIAL_MAP_STYLE = void 0; exports.getInitialInputStyle = getInitialInputStyle; exports.getMapStyles = getMapStyles; exports.setBackgroundColorUpdater = exports.set3dBuildingColorUpdater = exports.resetMapConfigMapStyleUpdater = exports.requestMapStylesUpdater = exports.removeCustomMapStyleUpdater = exports.receiveMapConfigUpdater = exports.mapStyleChangeUpdater = exports.mapConfigChangeUpdater = exports.loadMapStylesUpdater = exports.loadMapStyleErrUpdater = exports.loadCustomMapStyleUpdater = exports.inputMapStyleUpdater = exports.initMapStyleUpdater = void 0; var _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _tasks = _interopRequireWildcard(require("react-palm/tasks")); var _cloneDeep = _interopRequireDefault(require("lodash/cloneDeep")); var _console = _interopRequireDefault(require("global/console")); var _utils = require("@kepler.gl/utils"); var _commonUtils = require("@kepler.gl/common-utils"); var _constants = require("@kepler.gl/constants"); var _tasks2 = require("@kepler.gl/tasks"); var _d3Color = require("d3-color"); var _actions = require("@kepler.gl/actions"); function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != _typeof(e) && "function" != typeof e) return { "default": e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n["default"] = e, t && t.set(e, n), n; } function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : i + ""; } function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); } 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; } // SPDX-License-Identifier: MIT // Copyright contributors to the kepler.gl project // Utils var getDefaultMapStyles = exports.getDefaultMapStyles = function getDefaultMapStyles(cdnUrl) { return _constants.DEFAULT_MAP_STYLES.reduce(function (accu, curr) { return _objectSpread(_objectSpread({}, accu), {}, (0, _defineProperty2["default"])({}, curr.id, _objectSpread(_objectSpread({}, curr), {}, { icon: "".concat(cdnUrl, "/").concat(curr.icon) }))); }, {}); }; var getDefaultState = function getDefaultState() { var visibleLayerGroups = {}; var topLayerGroups = {}; return { styleType: _constants.DEFAULT_BASE_MAP_STYLE, visibleLayerGroups: visibleLayerGroups, topLayerGroups: topLayerGroups, mapStyles: getDefaultMapStyles((0, _utils.getApplicationConfig)().cdnUrl), // save mapbox access token mapboxApiAccessToken: null, mapboxApiUrl: _constants.DEFAULT_MAPBOX_API_URL, mapStylesReplaceDefault: false, inputStyle: getInitialInputStyle(), threeDBuildingColor: (0, _utils.hexToRgb)(_constants.DEFAULT_BLDG_COLOR), custom3DBuildingColor: false, backgroundColor: (0, _utils.hexToRgb)(_constants.DEFAULT_BACKGROUND_COLOR), isLoading: {}, bottomMapStyle: undefined, topMapStyle: undefined }; }; /** * Updaters for `mapStyle`. Can be used in your root reducer to directly modify kepler.gl's state. * Read more about [Using updaters](../advanced-usage/using-updaters.md) * @public * @example * * import keplerGlReducer, {mapStyleUpdaters} from 'kepler.gl/reducers'; * // Root Reducer * const reducers = combineReducers({ * keplerGl: keplerGlReducer, * app: appReducer * }); * * const composedReducer = (state, action) => { * switch (action.type) { * // click button to hide label from background map * case 'CLICK_BUTTON': * return { * ...state, * keplerGl: { * ...state.keplerGl, * foo: { * ...state.keplerGl.foo, * mapStyle: mapStyleUpdaters.mapConfigChangeUpdater( * mapStyle, * {payload: {visibleLayerGroups: {label: false, road: true, background: true}}} * ) * } * } * }; * } * return reducers(state, action); * }; * * export default composedReducer; */ /* eslint-disable @typescript-eslint/no-unused-vars */ // @ts-ignore var mapStyleUpdaters = null; /* eslint-enable @typescript-eslint/no-unused-vars */ /** * Default initial `mapStyle` * @memberof mapStyleUpdaters * @constant * @property styleType - Default: `'dark'` * @property visibleLayerGroups - Default: `{}` * @property topLayerGroups - Default: `{}` * @property mapStyles - mapping from style key to style object * @property mapboxApiAccessToken - Default: `null` * @Property mapboxApiUrl - Default null * @Property mapStylesReplaceDefault - Default: `false` * @property inputStyle - Default: `{}` * @property threeDBuildingColor - Default: `[r, g, b]` * @property backgroundColor - Default: `[r, g, b]` * @public */ var INITIAL_MAP_STYLE = exports.INITIAL_MAP_STYLE = getDefaultState(); /** * Create two map styles from preset map style, one for top map one for bottom * * @param {string} styleType - current map style * @param {Object} visibleLayerGroups - visible layers of bottom map * @param {Object} topLayerGroups - visible layers of top map * @param {Object} mapStyles - a dictionary of all map styles * @returns {Object} bottomMapStyle | topMapStyle | isRaster */ function getMapStyles(_ref) { var styleType = _ref.styleType, visibleLayerGroups = _ref.visibleLayerGroups, topLayerGroups = _ref.topLayerGroups, mapStyles = _ref.mapStyles; var mapStyle = mapStyles[styleType]; // style might not be loaded yet if (!mapStyle || !mapStyle.style) { return {}; } var editable = Object.keys(visibleLayerGroups).length; var bottomMapStyle = !editable ? mapStyle.style : (0, _utils.editBottomMapStyle)({ id: styleType, mapStyle: mapStyle, visibleLayerGroups: visibleLayerGroups }); var hasTopLayer = editable > 0 && Object.values(topLayerGroups).some(function (v) { return v; }); // mute top layer if not visible in bottom layer var topLayers = hasTopLayer && Object.keys(topLayerGroups).reduce(function (accu, key) { return _objectSpread(_objectSpread({}, accu), {}, (0, _defineProperty2["default"])({}, key, topLayerGroups[key] && visibleLayerGroups[key])); }, {}); var topMapStyle = hasTopLayer ? (0, _utils.editTopMapStyle)({ id: styleType, mapStyle: mapStyle, visibleLayerGroups: topLayers }) : null; return { bottomMapStyle: bottomMapStyle, topMapStyle: topMapStyle, editable: editable }; } function findLayerFillColor(layer) { return layer && layer.paint && layer.paint['background-color']; } // need to be careful because some basemap layer.paint['background-color'] values may be an interpolate array expression instead of a color string // https://docs.mapbox.com/mapbox-gl-js/style-spec/layers/#paint-background-background-color // https://docs.mapbox.com/mapbox-gl-js/style-spec/expressions/#interpolate function getPaintColor(color) { if (Array.isArray(color) && color[0] === 'interpolate') { // get color of first zoom break // ["interpolate", ["linear"], ["zoom"], 11, "hsl(35, 32%, 91%)", 13, "hsl(35, 12%, 89%)"] return color[4]; } return color; } function get3DBuildingColor(style) { // set building color to be the same as the background color. if (!style.style) { return (0, _utils.hexToRgb)(_constants.DEFAULT_BLDG_COLOR); } var backgroundLayer = (style.style.layers || []).find(function (_ref2) { var id = _ref2.id; return id === 'background'; }); var buildingLayer = (style.style.layers || []).find(function (_ref3) { var id = _ref3.id; return id.match(/building/); }); var buildingColor = findLayerFillColor(buildingLayer) || findLayerFillColor(backgroundLayer) || _constants.DEFAULT_BLDG_COLOR; // brighten or darken building based on style var operation = style.id.match(/(?=(dark|night))/) ? 'brighter' : 'darker'; var alpha = 0.2; var rgbObj = (0, _d3Color.rgb)(buildingColor)[operation]([alpha]); return [rgbObj.r, rgbObj.g, rgbObj.b]; } function getBackgroundColorFromStyleBaseLayer(style, backupBackgroundColor) { var _colorMaybeToRGB; if (!style.style) { return (0, _utils.colorMaybeToRGB)(backupBackgroundColor) || backupBackgroundColor; } // @ts-expect-error style.style not typed var baseLayer = (style.style.layers || []).find(function (_ref4) { var id = _ref4.id; return _constants.BASE_MAP_BACKGROUND_LAYER_IDS.includes(id); }); var backgroundColorOfBaseLayer = getPaintColor(findLayerFillColor(baseLayer)); var newBackgroundColor = typeof backgroundColorOfBaseLayer === 'string' ? backgroundColorOfBaseLayer : backupBackgroundColor; var newBackgroundColorAsRGBArray = (_colorMaybeToRGB = (0, _utils.colorMaybeToRGB)(newBackgroundColor) // if newBackgroundColor was in string HSL format it can introduce RGB numbers with decimals, // which may render the background-color CSS of the <StyledMap> container incorrectly when using our own color utils `rgbToHex()` // so we attempt to round to nearest integer here ) === null || _colorMaybeToRGB === void 0 ? void 0 : _colorMaybeToRGB.map(function (channelNumber) { return Math.round(channelNumber); }); return newBackgroundColorAsRGBArray || backupBackgroundColor; } // determine new backgroundColor from either previous state basemap style, previous state backgroundColor, or the DEFAULT_BACKGROUND_COLOR function getBackgroundColor(previousState, styleType) { var previousStateMapStyle = previousState.mapStyles[previousState.styleType]; var backupBackgroundColor = previousState.backgroundColor || _constants.DEFAULT_BACKGROUND_COLOR; var backgroundColor = styleType === _constants.NO_MAP_ID ? // if the style has switched to the "no basemap" style, // attempt to detect backgroundColor of the previous basemap if it was a mapbox basemap // and set it as the "no basemap" backgroundColor getBackgroundColorFromStyleBaseLayer(previousStateMapStyle, backupBackgroundColor) : // otherwise leave it alone and rely on the previous state's preexisting backgroundColor // or DEFAULT_BACKGROUND_COLOR as a last resort backupBackgroundColor; return backgroundColor; } function getLayerGroupsFromStyle(style) { return Array.isArray(style === null || style === void 0 ? void 0 : style.layers) ? _constants.DEFAULT_LAYER_GROUPS.filter(function (lg) { return style.layers.filter(lg.filter).length; }) : []; } // Updaters /** * @memberof mapStyleUpdaters * @public */ var requestMapStylesUpdater = exports.requestMapStylesUpdater = function requestMapStylesUpdater(state, _ref5) { var _ref5$payload = _ref5.payload, mapStyles = _ref5$payload.mapStyles, onSuccess = _ref5$payload.onSuccess; var toLoad = Object.keys(mapStyles).reduce(function (accu, id) { return _objectSpread(_objectSpread({}, accu), !state.isLoading[id] ? (0, _defineProperty2["default"])({}, id, mapStyles[id]) : {}); }, {}); var loadMapStyleTasks = getLoadMapStyleTasks(toLoad, state.mapboxApiAccessToken, state.mapboxApiUrl, onSuccess); var isLoading = Object.keys(toLoad).reduce(function (accu, key) { return _objectSpread(_objectSpread({}, accu), {}, (0, _defineProperty2["default"])({}, key, true)); }, {}); var nextState = _objectSpread(_objectSpread({}, state), {}, { isLoading: isLoading }); return (0, _tasks.withTask)(nextState, loadMapStyleTasks); }; /** * Propagate `mapStyle` reducer with `mapboxApiAccessToken` and `mapStylesReplaceDefault`. * if mapStylesReplaceDefault is true mapStyles is emptied; loadMapStylesUpdater() will * populate mapStyles. * * @memberof mapStyleUpdaters * @public */ var initMapStyleUpdater = exports.initMapStyleUpdater = function initMapStyleUpdater(state, _ref7) { var _ref7$payload = _ref7.payload, payload = _ref7$payload === void 0 ? {} : _ref7$payload; return _objectSpread(_objectSpread({}, state), {}, { // save mapbox access token to map style state mapboxApiAccessToken: payload.mapboxApiAccessToken || state.mapboxApiAccessToken, mapboxApiUrl: payload.mapboxApiUrl || state.mapboxApiUrl, mapStyles: !payload.mapStylesReplaceDefault ? state.mapStyles : {}, mapStylesReplaceDefault: payload.mapStylesReplaceDefault || false }); }; // }); /** * Update `visibleLayerGroups`to change layer group visibility * @memberof mapStyleUpdaters * @public */ var mapConfigChangeUpdater = exports.mapConfigChangeUpdater = function mapConfigChangeUpdater(state, action) { return _objectSpread(_objectSpread(_objectSpread({}, state), action.payload), getMapStyles(_objectSpread(_objectSpread({}, state), action.payload))); }; var hasStyleObject = function hasStyleObject(style) { return (0, _utils.isPlainObject)(style === null || style === void 0 ? void 0 : style.style); }; /** * Change to another map style. The selected style should already been loaded into `mapStyle.mapStyles` * @memberof mapStyleUpdaters * @public */ var mapStyleChangeUpdater = exports.mapStyleChangeUpdater = function mapStyleChangeUpdater(state, _ref8) { var _state$mapStyles$styl, _state$mapStyles$styl2; var _ref8$payload = _ref8.payload, styleType = _ref8$payload.styleType, onSuccess = _ref8$payload.onSuccess; if ( // we might not have received the style yet !state.mapStyles[styleType] || // or if it is a managed custom style asset // and if it has not been hydrated with URL info yet (during app first initialization) // and it does not have a style object (during adding a custom style) ((_state$mapStyles$styl = state.mapStyles[styleType]) === null || _state$mapStyles$styl === void 0 ? void 0 : _state$mapStyles$styl.custom) === 'MANAGED' && !((_state$mapStyles$styl2 = state.mapStyles[styleType]) !== null && _state$mapStyles$styl2 !== void 0 && _state$mapStyles$styl2.url) && !hasStyleObject(state.mapStyles[styleType])) { return state; } if (!hasStyleObject(state.mapStyles[styleType])) { // style hasn't loaded yet return requestMapStylesUpdater(_objectSpread(_objectSpread({}, state), {}, { styleType: styleType }), { payload: { mapStyles: (0, _defineProperty2["default"])({}, styleType, state.mapStyles[styleType]), onSuccess: onSuccess } }); } var defaultLGVisibility = (0, _utils.getDefaultLayerGroupVisibility)(state.mapStyles[styleType]); var visibleLayerGroups = (0, _utils.mergeLayerGroupVisibility)(defaultLGVisibility, state.visibleLayerGroups); var threeDBuildingColor = state.custom3DBuildingColor ? state.threeDBuildingColor : get3DBuildingColor(state.mapStyles[styleType]); // determine new backgroundColor from either previous state basemap style, previous state backgroundColor, or the DEFAULT_BACKGROUND_COLOR var backgroundColor = getBackgroundColor(state, styleType); return _objectSpread(_objectSpread({}, state), {}, { styleType: styleType, visibleLayerGroups: visibleLayerGroups, threeDBuildingColor: threeDBuildingColor, backgroundColor: backgroundColor }, getMapStyles(_objectSpread(_objectSpread({}, state), {}, { visibleLayerGroups: visibleLayerGroups, styleType: styleType }))); }; /** * Callback when load map style success * @memberof mapStyleUpdaters * @public */ var loadMapStylesUpdater = exports.loadMapStylesUpdater = function loadMapStylesUpdater(state, action) { var _ref9 = action.payload || {}, newStyles = _ref9.newStyles, onSuccess = _ref9.onSuccess; var addLayerGroups = Object.keys(newStyles).reduce(function (accu, id) { return _objectSpread(_objectSpread({}, accu), {}, (0, _defineProperty2["default"])({}, id, _objectSpread(_objectSpread({}, newStyles[id]), {}, { layerGroups: newStyles[id].layerGroups || getLayerGroupsFromStyle(newStyles[id].style) }))); }, {}); // reset isLoading var isLoading = Object.keys(state.isLoading).reduce(function (accu, key) { return _objectSpread(_objectSpread({}, accu), state.isLoading[key] && hasStyleObject(newStyles[key]) ? (0, _defineProperty2["default"])({}, key, false) : (0, _defineProperty2["default"])({}, key, state.isLoading[key])); }, {}); // add new styles to state var newState = _objectSpread(_objectSpread({}, state), {}, { isLoading: isLoading, mapStyles: _objectSpread(_objectSpread({}, state.mapStyles), addLayerGroups) }); var tasks = createActionTask(onSuccess, { styleType: state.styleType }); var nextState = newStyles[state.styleType] ? mapStyleChangeUpdater(newState, { payload: { styleType: state.styleType } }) : newState; return tasks ? (0, _tasks.withTask)(nextState, tasks) : nextState; }; function createActionTask(action, payload) { if (typeof action === 'function') { return (0, _tasks2.ACTION_TASK)().map(function () { return action(payload); }); } return null; } /** * Callback when load map style error * @memberof mapStyleUpdaters * @public */ // do nothing for now, if didn't load, skip it var loadMapStyleErrUpdater = exports.loadMapStyleErrUpdater = function loadMapStyleErrUpdater(state, _ref12) { var _ref12$payload = _ref12.payload, ids = _ref12$payload.ids, error = _ref12$payload.error; _console["default"].error(error); // reset isLoading var isLoading = Object.keys(state.isLoading).reduce(function (accu, key) { return _objectSpread(_objectSpread({}, accu), state.isLoading[key] && (ids || []).includes(key) ? (0, _defineProperty2["default"])({}, key, false) : (0, _defineProperty2["default"])({}, key, state.isLoading[key])); }, {}); return _objectSpread(_objectSpread({}, state), {}, { isLoading: isLoading }); }; /** * Load map style object when pass in saved map config * @memberof mapStyleUpdaters * @param state `mapStyle` * @param action * @param action.payload saved map config `{mapStyle, visState, mapState}` * @returns nextState or `react-pam` tasks to load map style object */ var receiveMapConfigUpdater = exports.receiveMapConfigUpdater = function receiveMapConfigUpdater(state, _ref15) { var config = _ref15.payload.config; var _ref16 = config || {}, mapStyle = _ref16.mapStyle; if (!mapStyle) { return state; } // merge default mapStyles var merged = mapStyle.mapStyles ? _objectSpread(_objectSpread({}, mapStyle), {}, { mapStyles: _objectSpread(_objectSpread({}, mapStyle.mapStyles), state.mapStyles) }) : mapStyle; // set custom3DBuildingColor: true if mapStyle contains threeDBuildingColor // @ts-expect-error merged.custom3DBuildingColor = // @ts-expect-error Boolean(mapStyle.threeDBuildingColor) || merged.custom3DBuildingColor; var newState = mapConfigChangeUpdater(state, { payload: merged }); return mapStyleChangeUpdater(newState, { payload: { styleType: newState.styleType } }); }; function getLoadMapStyleTasks(mapStyles, mapboxApiAccessToken, mapboxApiUrl, onSuccess) { return [_tasks["default"].all(Object.values(mapStyles) // @ts-expect-error .map(function (_ref17) { var id = _ref17.id, url = _ref17.url, accessToken = _ref17.accessToken; return { id: id, url: (0, _utils.getStyleDownloadUrl)(url, accessToken || mapboxApiAccessToken, mapboxApiUrl) }; }).map(_tasks2.LOAD_MAP_STYLE_TASK)).bimap( // success function (results) { return (0, _actions.loadMapStyles)(results.reduce(function (accu, _ref18) { var id = _ref18.id, style = _ref18.style; return _objectSpread(_objectSpread({}, accu), {}, (0, _defineProperty2["default"])({}, id, _objectSpread(_objectSpread({}, mapStyles[id]), {}, { style: style }))); }, {}), onSuccess); }, // error function (err) { return (0, _actions.loadMapStyleErr)(Object.keys(mapStyles), err); })]; } /** * Reset map style config to initial state * @memberof mapStyleUpdaters * @param state `mapStyle` * @returns nextState * @public */ var resetMapConfigMapStyleUpdater = exports.resetMapConfigMapStyleUpdater = function resetMapConfigMapStyleUpdater(state) { var emptyConfig = _objectSpread(_objectSpread(_objectSpread({}, INITIAL_MAP_STYLE), {}, { mapboxApiAccessToken: state.mapboxApiAccessToken, mapboxApiUrl: state.mapboxApiUrl, mapStylesReplaceDefault: state.mapStylesReplaceDefault }, state.initialState), {}, { mapStyles: state.mapStyles, initialState: state.initialState }); return mapStyleChangeUpdater(emptyConfig, { payload: { styleType: emptyConfig.styleType } }); }; /** * Callback when a custom map style object is received * @memberof mapStyleUpdaters * @public */ var loadCustomMapStyleUpdater = exports.loadCustomMapStyleUpdater = function loadCustomMapStyleUpdater(state, _ref19) { var _ref19$payload = _ref19.payload, icon = _ref19$payload.icon, style = _ref19$payload.style, error = _ref19$payload.error; return _objectSpread(_objectSpread({}, state), {}, { // @ts-expect-error inputStyle: _objectSpread(_objectSpread(_objectSpread(_objectSpread({}, state.inputStyle), style ? { id: state.inputStyle.custom === 'MANAGED' ? state.inputStyle.id // custom MANAGED type : // @ts-expect-error style.id || (0, _commonUtils.generateHashId)(), // custom LOCAL type // make a copy of the style object style: (0, _cloneDeep["default"])(style), // @ts-expect-error label: state.inputStyle.label || style.name, // gathering layer group info from style json layerGroups: getLayerGroupsFromStyle(style) } : {}), icon ? { icon: icon } : {}), error !== undefined ? { error: error } : {}) }); }; /** * Input a custom map style object * @memberof mapStyleUpdaters * @public */ var inputMapStyleUpdater = exports.inputMapStyleUpdater = function inputMapStyleUpdater(state, _ref20) { var _updated$icon, _updated$url, _updated$url2; var _ref20$payload = _ref20.payload, inputStyle = _ref20$payload.inputStyle, mapState = _ref20$payload.mapState; var updated = _objectSpread(_objectSpread({}, state.inputStyle), inputStyle); // differentiate between either a url to hosted style json that needs an icon url, // or an icon already available client-side as a data uri var isUpdatedIconDataUri = (_updated$icon = updated.icon) === null || _updated$icon === void 0 ? void 0 : _updated$icon.startsWith('data:image'); var isMapboxStyleUrl = ((_updated$url = updated.url) === null || _updated$url === void 0 ? void 0 : _updated$url.startsWith('mapbox://')) || ((_updated$url2 = updated.url) === null || _updated$url2 === void 0 ? void 0 : _updated$url2.includes('mapbox.com')); var icon = !isUpdatedIconDataUri && isMapboxStyleUrl ? // Get image icon urls only for mapbox map lib. (0, _utils.getStyleImageIcon)({ mapState: mapState, styleUrl: updated.url || '', mapboxApiAccessToken: updated.accessToken || state.mapboxApiAccessToken || '', mapboxApiUrl: state.mapboxApiUrl || _constants.DEFAULT_MAPBOX_API_URL }) : updated.icon; return _objectSpread(_objectSpread({}, state), {}, { inputStyle: _objectSpread(_objectSpread({}, updated), {}, { isValid: true, icon: icon }) }); }; /** * Add map style from user input to reducer and set it to current style * This action is called when user click confirm after putting in a valid style url in the custom map style dialog. * It should not be called from outside kepler.gl without a valid `inputStyle` in the `mapStyle` reducer. * @memberof mapStyleUpdaters */ var addCustomMapStyleUpdater = exports.addCustomMapStyleUpdater = function addCustomMapStyleUpdater(state) { var styleId = state.inputStyle.id; if (!styleId) return state; var newState = getNewStateWithCustomMapStyle(state); // set new style return mapStyleChangeUpdater(newState, { payload: { styleType: styleId } }); }; /** * Edit map style from user input to reducer. * This action is called when user clicks confirm after editing an existing custom style in the custom map style dialog. * It should not be called from outside kepler.gl without a valid `inputStyle` in the `mapStyle` reducer. * @memberof mapStyleUpdaters */ var editCustomMapStyleUpdater = exports.editCustomMapStyleUpdater = function editCustomMapStyleUpdater(state) { return getNewStateWithCustomMapStyle(state); }; function getNewStateWithCustomMapStyle(state) { var styleId = state.inputStyle.id; if (!styleId) return state; return _objectSpread(_objectSpread({}, state), {}, { // @ts-expect-error Property 'layerGroups' is missing in type 'InputStyle' but required in type 'BaseMapStyle'. Legacy case? mapStyles: _objectSpread(_objectSpread({}, state.mapStyles), {}, (0, _defineProperty2["default"])({}, styleId, _objectSpread(_objectSpread({}, state.mapStyles[styleId]), state.inputStyle))), // set to default inputStyle: getInitialInputStyle() }); } /** * Remove a custom map style from `state.mapStyle.mapStyles`. * @memberof mapStyleUpdaters */ var removeCustomMapStyleUpdater = exports.removeCustomMapStyleUpdater = function removeCustomMapStyleUpdater(state, action) { var id = action.payload.id; // eslint-disable-next-line @typescript-eslint/no-unused-vars var _state$mapStyles = state.mapStyles, _ = _state$mapStyles[id], restOfMapStyles = (0, _objectWithoutProperties2["default"])(_state$mapStyles, [id].map(_toPropertyKey)); var newState = _objectSpread(_objectSpread({}, state), {}, { mapStyles: restOfMapStyles }); if (state.styleType === id) { // if removing a custom style that is also the current active base map, // then reset to the default active base map (`mapStyle.styleType`) return mapStyleChangeUpdater(newState, { payload: { styleType: getDefaultState().styleType } }); } return newState; }; /** * Updates 3d building color * @memberof mapStyleUpdaters */ var set3dBuildingColorUpdater = exports.set3dBuildingColorUpdater = function set3dBuildingColorUpdater(state, _ref21) { var color = _ref21.payload; return _objectSpread(_objectSpread({}, state), {}, { threeDBuildingColor: color, custom3DBuildingColor: true }); }; /** * Updates background color * @memberof mapStyleUpdaters */ var setBackgroundColorUpdater = exports.setBackgroundColorUpdater = function setBackgroundColorUpdater(state, _ref22) { var color = _ref22.payload; return _objectSpread(_objectSpread({}, state), {}, { backgroundColor: color }); }; /** * Return the initial input style * @return Object */ function getInitialInputStyle() { return { id: null, accessToken: null, error: false, isValid: false, label: null, style: null, url: null, icon: null, custom: 'LOCAL', uploadedFile: null }; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfdGFza3MiLCJfaW50ZXJvcFJlcXVpcmVXaWxkY2FyZCIsInJlcXVpcmUiLCJfY2xvbmVEZWVwIiwiX2ludGVyb3BSZXF1aXJlRGVmYXVsdCIsIl9jb25zb2xlIiwiX3V0aWxzIiwiX2NvbW1vblV0aWxzIiwiX2NvbnN0YW50cyIsIl90YXNrczIiLCJfZDNDb2xvciIsIl9hY3Rpb25zIiwiX2dldFJlcXVpcmVXaWxkY2FyZENhY2hlIiwiZSIsIldlYWtNYXAiLCJyIiwidCIsIl9fZXNNb2R1bGUiLCJfdHlwZW9mIiwiaGFzIiwiZ2V0IiwibiIsIl9fcHJvdG9fXyIsImEiLCJPYmplY3QiLCJkZWZpbmVQcm9wZXJ0eSIsImdldE93blByb3BlcnR5RGVzY3JpcHRvciIsInUiLCJoYXNPd25Qcm9wZXJ0eSIsImNhbGwiLCJpIiwic2V0IiwiX3RvUHJvcGVydHlLZXkiLCJfdG9QcmltaXRpdmUiLCJTeW1ib2wiLCJ0b1ByaW1pdGl2ZSIsIlR5cGVFcnJvciIsIlN0cmluZyIsIk51bWJlciIsIm93bktleXMiLCJrZXlzIiwiZ2V0T3duUHJvcGVydHlTeW1ib2xzIiwibyIsImZpbHRlciIsImVudW1lcmFibGUiLCJwdXNoIiwiYXBwbHkiLCJfb2JqZWN0U3ByZWFkIiwiYXJndW1lbnRzIiwibGVuZ3RoIiwiZm9yRWFjaCIsIl9kZWZpbmVQcm9wZXJ0eTIiLCJnZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3JzIiwiZGVmaW5lUHJvcGVydGllcyIsImdldERlZmF1bHRNYXBTdHlsZXMiLCJleHBvcnRzIiwiY2RuVXJsIiwiREVGQVVMVF9NQVBfU1RZTEVTIiwicmVkdWNlIiwiYWNjdSIsImN1cnIiLCJpZCIsImljb24iLCJjb25jYXQiLCJnZXREZWZhdWx0U3RhdGUiLCJ2aXNpYmxlTGF5ZXJHcm91cHMiLCJ0b3BMYXllckdyb3VwcyIsInN0eWxlVHlwZSIsIkRFRkFVTFRfQkFTRV9NQVBfU1RZTEUiLCJtYXBTdHlsZXMiLCJnZXRBcHBsaWNhdGlvbkNvbmZpZyIsIm1hcGJveEFwaUFjY2Vzc1Rva2VuIiwibWFwYm94QXBpVXJsIiwiREVGQVVMVF9NQVBCT1hfQVBJX1VSTCIsIm1hcFN0eWxlc1JlcGxhY2VEZWZhdWx0IiwiaW5wdXRTdHlsZSIsImdldEluaXRpYWxJbnB1dFN0eWxlIiwidGhyZWVEQnVpbGRpbmdDb2xvciIsImhleFRvUmdiIiwiREVGQVVMVF9CTERHX0NPTE9SIiwiY3VzdG9tM0RCdWlsZGluZ0NvbG9yIiwiYmFja2dyb3VuZENvbG9yIiwiREVGQVVMVF9CQUNLR1JPVU5EX0NPTE9SIiwiaXNMb2FkaW5nIiwiYm90dG9tTWFwU3R5bGUiLCJ1bmRlZmluZWQiLCJ0b3BNYXBTdHlsZSIsIm1hcFN0eWxlVXBkYXRlcnMiLCJJTklUSUFMX01BUF9TVFlMRSIsImdldE1hcFN0eWxlcyIsIl9yZWYiLCJtYXBTdHlsZSIsInN0eWxlIiwiZWRpdGFibGUiLCJlZGl0Qm90dG9tTWFwU3R5bGUiLCJoYXNUb3BMYXllciIsInZhbHVlcyIsInNvbWUiLCJ2IiwidG9wTGF5ZXJzIiwia2V5IiwiZWRpdFRvcE1hcFN0eWxlIiwiZmluZExheWVyRmlsbENvbG9yIiwibGF5ZXIiLCJwYWludCIsImdldFBhaW50Q29sb3IiLCJjb2xvciIsIkFycmF5IiwiaXNBcnJheSIsImdldDNEQnVpbGRpbmdDb2xvciIsImJhY2tncm91bmRMYXllciIsImxheWVycyIsImZpbmQiLCJfcmVmMiIsImJ1aWxkaW5nTGF5ZXIiLCJfcmVmMyIsIm1hdGNoIiwiYnVpbGRpbmdDb2xvciIsIm9wZXJhdGlvbiIsImFscGhhIiwicmdiT2JqIiwicmdiIiwiZyIsImIiLCJnZXRCYWNrZ3JvdW5kQ29sb3JGcm9tU3R5bGVCYXNlTGF5ZXIiLCJiYWNrdXBCYWNrZ3JvdW5kQ29sb3IiLCJfY29sb3JNYXliZVRvUkdCIiwiY29sb3JNYXliZVRvUkdCIiwiYmFzZUxheWVyIiwiX3JlZjQiLCJCQVNFX01BUF9CQUNLR1JPVU5EX0xBWUVSX0lEUyIsImluY2x1ZGVzIiwiYmFja2dyb3VuZENvbG9yT2ZCYXNlTGF5ZXIiLCJuZXdCYWNrZ3JvdW5kQ29sb3IiLCJuZXdCYWNrZ3JvdW5kQ29sb3JBc1JHQkFycmF5IiwibWFwIiwiY2hhbm5lbE51bWJlciIsIk1hdGgiLCJyb3VuZCIsImdldEJhY2tncm91bmRDb2xvciIsInByZXZpb3VzU3RhdGUiLCJwcmV2aW91c1N0YXRlTWFwU3R5bGUiLCJOT19NQVBfSUQiLCJnZXRMYXllckdyb3Vwc0Zyb21TdHlsZSIsIkRFRkFVTFRfTEFZRVJfR1JPVVBTIiwibGciLCJyZXF1ZXN0TWFwU3R5bGVzVXBkYXRlciIsInN0YXRlIiwiX3JlZjUiLCJfcmVmNSRwYXlsb2FkIiwicGF5bG9hZCIsIm9uU3VjY2VzcyIsInRvTG9hZCIsImxvYWRNYXBTdHlsZVRhc2tzIiwiZ2V0TG9hZE1hcFN0eWxlVGFza3MiLCJuZXh0U3RhdGUiLCJ3aXRoVGFzayIsImluaXRNYXBTdHlsZVVwZGF0ZXIiLCJfcmVmNyIsIl9yZWY3JHBheWxvYWQiLCJtYXBDb25maWdDaGFuZ2VVcGRhdGVyIiwiYWN0aW9uIiwiaGFzU3R5bGVPYmplY3QiLCJpc1BsYWluT2JqZWN0IiwibWFwU3R5bGVDaGFuZ2VVcGRhdGVyIiwiX3JlZjgiLCJfc3RhdGUkbWFwU3R5bGVzJHN0eWwiLCJfc3RhdGUkbWFwU3R5bGVzJHN0eWwyIiwiX3JlZjgkcGF5bG9hZCIsImN1c3RvbSIsInVybCIsImRlZmF1bHRMR1Zpc2liaWxpdHkiLCJnZXREZWZhdWx0TGF5ZXJHcm91cFZpc2liaWxpdHkiLCJtZXJnZUxheWVyR3JvdXBWaXNpYmlsaXR5IiwibG9hZE1hcFN0eWxlc1VwZGF0ZXIiLCJfcmVmOSIsIm5ld1N0eWxlcyIsImFkZExheWVyR3JvdXBzIiwibGF5ZXJHcm91cHMiLCJuZXdTdGF0ZSIsInRhc2tzIiwiY3JlYXRlQWN0aW9uVGFzayIsIkFDVElPTl9UQVNLIiwibG9hZE1hcFN0eWxlRXJyVXBkYXRlciIsIl9yZWYxMiIsIl9yZWYxMiRwYXlsb2FkIiwiaWRzIiwiZXJyb3IiLCJDb25zb2xlIiwicmVjZWl2ZU1hcENvbmZpZ1VwZGF0ZXIiLCJfcmVmMTUiLCJjb25maWciLCJfcmVmMTYiLCJtZXJnZWQiLCJCb29sZWFuIiwiVGFzayIsImFsbCIsIl9yZWYxNyIsImFjY2Vzc1Rva2VuIiwiZ2V0U3R5bGVEb3dubG9hZFVybCIsIkxPQURfTUFQX1NUWUxFX1RBU0siLCJiaW1hcCIsInJlc3VsdHMiLCJsb2FkTWFwU3R5bGVzIiwiX3JlZjE4IiwiZXJyIiwibG9hZE1hcFN0eWxlRXJyIiwicmVzZXRNYXBDb25maWdNYXBTdHlsZVVwZGF0ZXIiLCJlbXB0eUNvbmZpZyIsImluaXRpYWxTdGF0ZSIsImxvYWRDdXN0b21NYXBTdHlsZVVwZGF0ZXIiLCJfcmVmMTkiLCJfcmVmMTkkcGF5bG9hZCIsImdlbmVyYXRlSGFzaElkIiwiY2xvbmVEZWVwIiwibGFiZWwiLCJuYW1lIiwiaW5wdXRNYXBTdHlsZVVwZGF0ZXIiLCJfcmVmMjAiLCJfdXBkYXRlZCRpY29uIiwiX3VwZGF0ZWQkdXJsIiwiX3VwZGF0ZWQkdXJsMiIsIl9yZWYyMCRwYXlsb2FkIiwibWFwU3RhdGUiLCJ1cGRhdGVkIiwiaXNVcGRhdGVkSWNvbkRhdGFVcmkiLCJzdGFydHNXaXRoIiwiaXNNYXBib3hTdHlsZVVybCIsImdldFN0eWxlSW1hZ2VJY29uIiwic3R5bGVVcmwiLCJpc1ZhbGlkIiwiYWRkQ3VzdG9tTWFwU3R5bGVVcGRhdGVyIiwic3R5bGVJZCIsImdldE5ld1N0YXRlV2l0aEN1c3RvbU1hcFN0eWxlIiwiZWRpdEN1c3RvbU1hcFN0eWxlVXBkYXRlciIsInJlbW92ZUN1c3RvbU1hcFN0eWxlVXBkYXRlciIsIl9zdGF0ZSRtYXBTdHlsZXMiLCJfIiwicmVzdE9mTWFwU3R5bGVzIiwiX29iamVjdFdpdGhvdXRQcm9wZXJ0aWVzMiIsInNldDNkQnVpbGRpbmdDb2xvclVwZGF0ZXIiLCJfcmVmMjEiLCJzZXRCYWNrZ3JvdW5kQ29sb3JVcGRhdGVyIiwiX3JlZjIyIiwidXBsb2FkZWRGaWxlIl0sInNvdXJjZXMiOlsiLi4vc3JjL21hcC1zdHlsZS11cGRhdGVycy50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogTUlUXG4vLyBDb3B5cmlnaHQgY29udHJpYnV0b3JzIHRvIHRoZSBrZXBsZXIuZ2wgcHJvamVjdFxuXG5pbXBvcnQgVGFzaywge3dpdGhUYXNrfSBmcm9tICdyZWFjdC1wYWxtL3Rhc2tzJztcbmltcG9ydCBjbG9uZURlZXAgZnJvbSAnbG9kYXNoL2Nsb25lRGVlcCc7XG5pbXBvcnQgQ29uc29sZSBmcm9tICdnbG9iYWwvY29uc29sZSc7XG5cbi8vIFV0aWxzXG5pbXBvcnQge1xuICBnZXREZWZhdWx0TGF5ZXJHcm91cFZpc2liaWxpdHksXG4gIGdldFN0eWxlRG93bmxvYWRVcmwsXG4gIG1lcmdlTGF5ZXJHcm91cFZpc2liaWxpdHksXG4gIGVkaXRUb3BNYXBTdHlsZSxcbiAgZWRpdEJvdHRvbU1hcFN0eWxlLFxuICBnZXRTdHlsZUltYWdlSWNvbixcbiAgaXNQbGFpbk9iamVjdCxcbiAgaGV4VG9SZ2IsXG4gIGNvbG9yTWF5YmVUb1JHQixcbiAgZ2V0QXBwbGljYXRpb25Db25maWdcbn0gZnJvbSAnQGtlcGxlci5nbC91dGlscyc7XG5pbXBvcnQge2dlbmVyYXRlSGFzaElkfSBmcm9tICdAa2VwbGVyLmdsL2NvbW1vbi11dGlscyc7XG5pbXBvcnQge1xuICBERUZBVUxUX01BUF9TVFlMRVMsXG4gIERFRkFVTFRfTEFZRVJfR1JPVVBTLFxuICBERUZBVUxUX01BUEJPWF9BUElfVVJMLFxuICBOT19NQVBfSUQsXG4gIERFRkFVTFRfQkxER19DT0xPUixcbiAgREVGQVVMVF9CQUNLR1JPVU5EX0NPTE9SLFxuICBCQVNFX01BUF9CQUNLR1JPVU5EX0xBWUVSX0lEUyxcbiAgREVGQVVMVF9CQVNFX01BUF9TVFlMRVxufSBmcm9tICdAa2VwbGVyLmdsL2NvbnN0YW50cyc7XG5pbXBvcnQge0FDVElPTl9UQVNLLCBMT0FEX01BUF9TVFlMRV9UQVNLfSBmcm9tICdAa2VwbGVyLmdsL3Rhc2tzJztcbmltcG9ydCB7cmdifSBmcm9tICdkMy1jb2xvcic7XG5cbmltcG9ydCB7XG4gIFJHQkNvbG9yLFxuICBMYXllckdyb3VwLFxuICBCYXNlTWFwU3R5bGUsXG4gIE1hcFN0eWxlcyxcbiAgSW5wdXRTdHlsZSxcbiAgVmlzaWJsZUxheWVyR3JvdXBzXG59IGZyb20gJ0BrZXBsZXIuZ2wvdHlwZXMnO1xuaW1wb3J0IHtcbiAgQWN0aW9uVHlwZXMsXG4gIFJlY2VpdmVNYXBDb25maWdQYXlsb2FkLFxuICBLZXBsZXJHbEluaXRQYXlsb2FkLFxuICBNYXBTdHlsZUFjdGlvbnMsXG4gIGxvYWRNYXBTdHlsZXMsXG4gIGxvYWRNYXBTdHlsZUVyclxufSBmcm9tICdAa2VwbGVyLmdsL2FjdGlvbnMnO1xuXG5leHBvcnQgdHlwZSBNYXBib3hTdHlsZVVybCA9IHN0cmluZztcblxuZXhwb3J0IHR5cGUgTWFwU3R5bGUgPSB7XG4gIHN0eWxlVHlwZTogc3RyaW5nO1xuICB2aXNpYmxlTGF5ZXJHcm91cHM6IFZpc2libGVMYXllckdyb3VwcztcbiAgdG9wTGF5ZXJHcm91cHM6IFZpc2libGVMYXllckdyb3VwcztcbiAgbWFwU3R5bGVzOiBNYXBTdHlsZXM7XG4gIC8vIHNhdmUgbWFwYm94IGFjY2VzcyB0b2tlblxuICBtYXBib3hBcGlBY2Nlc3NUb2tlbjogc3RyaW5nIHwgbnVsbDtcbiAgbWFwYm94QXBpVXJsOiBzdHJpbmc7XG4gIG1hcFN0eWxlc1JlcGxhY2VEZWZhdWx0OiBib29sZWFuO1xuICBpbnB1dFN0eWxlOiBJbnB1dFN0eWxlO1xuICB0aHJlZURCdWlsZGluZ0NvbG9yOiBSR0JDb2xvcjtcbiAgYmFja2dyb3VuZENvbG9yOiBSR0JDb2xvcjtcbiAgY3VzdG9tM0RCdWlsZGluZ0NvbG9yOiBib29sZWFuO1xuICBib3R0b21NYXBTdHlsZTogYW55O1xuICB0b3BNYXBTdHlsZTogYW55O1xuICBpbml0aWFsU3RhdGU/OiBNYXBTdHlsZTtcbiAgaXNMb2FkaW5nOiB7XG4gICAgW2tleTogc3RyaW5nXTogYm9vbGVhbjtcbiAgfTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXREZWZhdWx0TWFwU3R5bGVzID0gKGNkblVybDogc3RyaW5nKSA9PiB7XG4gIHJldHVybiBERUZBVUxUX01BUF9TVFlMRVMucmVkdWNlKFxuICAgIChhY2N1LCBjdXJyKSA9PiAoe1xuICAgICAgLi4uYWNjdSxcbiAgICAgIFtjdXJyLmlkXToge1xuICAgICAgICAuLi5jdXJyLFxuICAgICAgICBpY29uOiBgJHtjZG5Vcmx9LyR7Y3Vyci5pY29ufWBcbiAgICAgIH1cbiAgICB9KSxcbiAgICB7fVxuICApO1xufTtcblxuY29uc3QgZ2V0RGVmYXVsdFN0YXRlID0gKCk6IE1hcFN0eWxlID0+IHtcbiAgY29uc3QgdmlzaWJsZUxheWVyR3JvdXBzID0ge307XG4gIGNvbnN0IHRvcExheWVyR3JvdXBzID0ge307XG5cbiAgcmV0dXJuIHtcbiAgICBzdHlsZVR5cGU6IERFRkFVTFRfQkFTRV9NQVBfU1RZTEUsXG4gICAgdmlzaWJsZUxheWVyR3JvdXBzLFxuICAgIHRvcExheWVyR3JvdXBzLFxuICAgIG1hcFN0eWxlczogZ2V0RGVmYXVsdE1hcFN0eWxlcyhnZXRBcHBsaWNhdGlvbkNvbmZpZygpLmNkblVybCksXG4gICAgLy8gc2F2ZSBtYXBib3ggYWNjZXNzIHRva2VuXG4gICAgbWFwYm94QXBpQWNjZXNzVG9rZW46IG51bGwsXG4gICAgbWFwYm94QXBpVXJsOiBERUZBVUxUX01BUEJPWF9BUElfVVJMLFxuICAgIG1hcFN0eWxlc1JlcGxhY2VEZWZhdWx0OiBmYWxzZSxcbiAgICBpbnB1dFN0eWxlOiBnZXRJbml0aWFsSW5wdXRTdHlsZSgpLFxuICAgIHRocmVlREJ1aWxkaW5nQ29sb3I6IGhleFRvUmdiKERFRkFVTFRfQkxER19DT0xPUiksXG4gICAgY3VzdG9tM0RCdWlsZGluZ0NvbG9yOiBmYWxzZSxcbiAgICBiYWNrZ3JvdW5kQ29sb3I6IGhleFRvUmdiKERFRkFVTFRfQkFDS0dST1VORF9DT0xPUiksXG4gICAgaXNMb2FkaW5nOiB7fSxcbiAgICBib3R0b21NYXBTdHlsZTogdW5kZWZpbmVkLFxuICAgIHRvcE1hcFN0eWxlOiB1bmRlZmluZWRcbiAgfTtcbn07XG5cbi8qKlxuICogVXBkYXRlcnMgZm9yIGBtYXBTdHlsZWAuIENhbiBiZSB1c2VkIGluIHlvdXIgcm9vdCByZWR1Y2VyIHRvIGRpcmVjdGx5IG1vZGlmeSBrZXBsZXIuZ2wncyBzdGF0ZS5cbiAqIFJlYWQgbW9yZSBhYm91dCBbVXNpbmcgdXBkYXRlcnNdKC4uL2FkdmFuY2VkLXVzYWdlL3VzaW5nLXVwZGF0ZXJzLm1kKVxuICogQHB1YmxpY1xuICogQGV4YW1wbGVcbiAqXG4gKiBpbXBvcnQga2VwbGVyR2xSZWR1Y2VyLCB7bWFwU3R5bGVVcGRhdGVyc30gZnJvbSAna2VwbGVyLmdsL3JlZHVjZXJzJztcbiAqIC8vIFJvb3QgUmVkdWNlclxuICogY29uc3QgcmVkdWNlcnMgPSBjb21iaW5lUmVkdWNlcnMoe1xuICogIGtlcGxlckdsOiBrZXBsZXJHbFJlZHVjZXIsXG4gKiAgYXBwOiBhcHBSZWR1Y2VyXG4gKiB9KTtcbiAqXG4gKiBjb25zdCBjb21wb3NlZFJlZHVjZXIgPSAoc3RhdGUsIGFjdGlvbikgPT4ge1xuICogIHN3aXRjaCAoYWN0aW9uLnR5cGUpIHtcbiAqICAgIC8vIGNsaWNrIGJ1dHRvbiB0byBoaWRlIGxhYmVsIGZyb20gYmFja2dyb3VuZCBtYXBcbiAqICAgIGNhc2UgJ0NMSUNLX0JVVFRPTic6XG4gKiAgICAgIHJldHVybiB7XG4gKiAgICAgICAgLi4uc3RhdGUsXG4gKiAgICAgICAga2VwbGVyR2w6IHtcbiAqICAgICAgICAgIC4uLnN0YXRlLmtlcGxlckdsLFxuICogICAgICAgICAgZm9vOiB7XG4gKiAgICAgICAgICAgICAuLi5zdGF0ZS5rZXBsZXJHbC5mb28sXG4gKiAgICAgICAgICAgICBtYXBTdHlsZTogbWFwU3R5bGVVcGRhdGVycy5tYXBDb25maWdDaGFuZ2VVcGRhdGVyKFxuICogICAgICAgICAgICAgICBtYXBTdHlsZSxcbiAqICAgICAgICAgICAgICAge3BheWxvYWQ6IHt2aXNpYmxlTGF5ZXJHcm91cHM6IHtsYWJlbDogZmFsc2UsIHJvYWQ6IHRydWUsIGJhY2tncm91bmQ6IHRydWV9fX1cbiAqICAgICAgICAgICAgIClcbiAqICAgICAgICAgIH1cbiAqICAgICAgICB9XG4gKiAgICAgIH07XG4gKiAgfVxuICogIHJldHVybiByZWR1Y2VycyhzdGF0ZSwgYWN0aW9uKTtcbiAqIH07XG4gKlxuICogZXhwb3J0IGRlZmF1bHQgY29tcG9zZWRSZWR1Y2VyO1xuICovXG5cbi8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFycyAqL1xuLy8gQHRzLWlnbm9yZVxuY29uc3QgbWFwU3R5bGVVcGRhdGVycyA9IG51bGw7XG4vKiBlc2xpbnQtZW5hYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFycyAqL1xuLyoqXG4gKiBEZWZhdWx0IGluaXRpYWwgYG1hcFN0eWxlYFxuICogQG1lbWJlcm9mIG1hcFN0eWxlVXBkYXRlcnNcbiAqIEBjb25zdGFudFxuICogQHByb3BlcnR5IHN0eWxlVHlwZSAtIERlZmF1bHQ6IGAnZGFyaydgXG4gKiBAcHJvcGVydHkgdmlzaWJsZUxheWVyR3JvdXBzIC0gRGVmYXVsdDogYHt9YFxuICogQHByb3BlcnR5IHRvcExheWVyR3JvdXBzIC0gRGVmYXVsdDogYHt9YFxuICogQHByb3BlcnR5IG1hcFN0eWxlcyAtIG1hcHBpbmcgZnJvbSBzdHlsZSBrZXkgdG8gc3R5bGUgb2JqZWN0XG4gKiBAcHJvcGVydHkgbWFwYm94QXBpQWNjZXNzVG9rZW4gLSBEZWZhdWx0OiBgbnVsbGBcbiAqIEBQcm9wZXJ0eSBtYXBib3hBcGlVcmwgLSBEZWZhdWx0IG51bGxcbiAqIEBQcm9wZXJ0eSBtYXBTdHlsZXNSZXBsYWNlRGVmYXVsdCAtIERlZmF1bHQ6IGBmYWxzZWBcbiAqIEBwcm9wZXJ0eSBpbnB1dFN0eWxlIC0gRGVmYXVsdDogYHt9YFxuICogQHByb3BlcnR5IHRocmVlREJ1aWxkaW5nQ29sb3IgLSBEZWZhdWx0OiBgW3IsIGcsIGJdYFxuICogQHByb3BlcnR5IGJhY2tncm91bmRDb2xvciAtIERlZmF1bHQ6IGBbciwgZywgYl1gXG4gKiBAcHVibGljXG4gKi9cbmV4cG9ydCBjb25zdCBJTklUSUFMX01BUF9TVFlMRTogTWFwU3R5bGUgPSBnZXREZWZhdWx0U3RhdGUoKTtcblxuaW50ZXJmYWNlIEdldE1hcFN0eWxlc1BhcmFtIHtcbiAgc3R5bGVUeXBlOiBzdHJpbmc7XG4gIHZpc2libGVMYXllckdyb3Vwczoge1tpZDogc3RyaW5nXTogTGF5ZXJHcm91cCB8IGJvb2xlYW59O1xuICB0b3BMYXllckdyb3Vwczoge1tpZDogc3RyaW5nXTogTGF5ZXJHcm91cCB8IGJvb2xlYW59O1xuICBtYXBTdHlsZXM6IHtbaWQ6IHN0cmluZ106IGFueX07XG59XG5cbi8qKlxuICogQ3JlYXRlIHR3byBtYXAgc3R5bGVzIGZyb20gcHJlc2V0IG1hcCBzdHlsZSwgb25lIGZvciB0b3AgbWFwIG9uZSBmb3IgYm90dG9tXG4gKlxuICogQHBhcmFtIHtzdHJpbmd9IHN0eWxlVHlwZSAtIGN1cnJlbnQgbWFwIHN0eWxlXG4gKiBAcGFyYW0ge09iamVjdH0gdmlzaWJsZUxheWVyR3JvdXBzIC0gdmlzaWJsZSBsYXllcnMgb2YgYm90dG9tIG1hcFxuICogQHBhcmFtIHtPYmplY3R9IHRvcExheWVyR3JvdXBzIC0gdmlzaWJsZSBsYXllcnMgb2YgdG9wIG1hcFxuICogQHBhcmFtIHtPYmplY3R9IG1hcFN0eWxlcyAtIGEgZGljdGlvbmFyeSBvZiBhbGwgbWFwIHN0eWxlc1xuICogQHJldHVybnMge09iamVjdH0gYm90dG9tTWFwU3R5bGUgfCB0b3BNYXBTdHlsZSB8IGlzUmFzdGVyXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRNYXBTdHlsZXMoe1xuICBzdHlsZVR5cGUsXG4gIHZpc2libGVMYXllckdyb3VwcyxcbiAgdG9wTGF5ZXJHcm91cHMsXG4gIG1hcFN0eWxlc1xufTogR2V0TWFwU3R5bGVzUGFyYW0pIHtcbiAgY29uc3QgbWFwU3R5bGUgPSBtYXBTdHlsZXNbc3R5bGVUeXBlXTtcblxuICAvLyBzdHlsZSBtaWdodCBub3QgYmUgbG9hZGVkIHlldFxuICBpZiAoIW1hcFN0eWxlIHx8ICFtYXBTdHlsZS5zdHlsZSkge1xuICAgIHJldHVybiB7fTtcbiAgfVxuXG4gIGNvbnN0IGVkaXRhYmxlID0gT2JqZWN0LmtleXModmlzaWJsZUxheWVyR3JvdXBzKS5sZW5ndGg7XG5cbiAgY29uc3QgYm90dG9tTWFwU3R5bGUgPSAhZWRpdGFibGVcbiAgICA/IG1hcFN0eWxlLnN0eWxlXG4gICAgOiBlZGl0Qm90dG9tTWFwU3R5bGUoe1xuICAgICAgICBpZDogc3R5bGVUeXBlLFxuICAgICAgICBtYXBTdHlsZSxcbiAgICAgICAgdmlzaWJsZUxheWVyR3JvdXBzXG4gICAgICB9KTtcblxuICBjb25zdCBoYXNUb3BMYXllciA9IGVkaXRhYmxlID4gMCAmJiBPYmplY3QudmFsdWVzKHRvcExheWVyR3JvdXBzKS5zb21lKHYgPT4gdik7XG5cbiAgLy8gbXV0ZSB0b3AgbGF5ZXIgaWYgbm90IHZpc2libGUgaW4gYm90dG9tIGxheWVyXG4gIGNvbnN0IHRvcExheWVycyA9XG4gICAgaGFzVG9wTGF5ZXIgJiZcbiAgICBPYmplY3Qua2V5cyh0b3BMYXllckdyb3VwcykucmVkdWNlKFxuICAgICAgKGFjY3UsIGtleSkgPT4gKHtcbiAgICAgICAgLi4uYWNjdSxcbiAgICAgICAgW2tleV06IHRvcExheWVyR3JvdXBzW2tleV0gJiYgdmlzaWJsZUxheWVyR3JvdXBzW2tleV1cbiAgICAgIH0pLFxuICAgICAge30gYXMge1tpZDogc3RyaW5nXTogTGF5ZXJHcm91cCB8IGJvb2xlYW59XG4gICAgKTtcblxuICBjb25zdCB0b3BNYXBTdHlsZSA9IGhhc1RvcExheWVyXG4gICAgPyBlZGl0VG9wTWFwU3R5bGUoe1xuICAgICAgICBpZDogc3R5bGVUeXBlLFxuICAgICAgICBtYXBTdHlsZSxcbiAgICAgICAgdmlzaWJsZUxheWVyR3JvdXBzOiB0b3BMYXllcnNcbiAgICAgIH0pXG4gICAgOiBudWxsO1xuXG4gIHJldHVybiB7Ym90dG9tTWFwU3R5bGUsIHRvcE1hcFN0eWxlLCBlZGl0YWJsZX07XG59XG5cbmZ1bmN0aW9uIGZpbmRMYXllckZpbGxDb2xvcihsYXllcikge1xuICByZXR1cm4gbGF5ZXIgJiYgbGF5ZXIucGFpbnQgJiYgbGF5ZXIucGFpbnRbJ2JhY2tncm91bmQtY29sb3InXTtcbn1cblxuLy8gbmVlZCB0byBiZSBjYXJlZnVsIGJlY2F1c2Ugc29tZSBiYXNlbWFwIGxheWVyLnBhaW50WydiYWNrZ3JvdW5kLWNvbG9yJ10gdmFsdWVzIG1heSBiZSBhbiBpbnRlcnBvbGF0ZSBhcnJheSBleHByZXNzaW9uIGluc3RlYWQgb2YgYSBjb2xvciBzdHJpbmdcbi8vIGh0dHBzOi8vZG9jcy5tYXBib3guY29tL21hcGJveC1nbC1qcy9zdHlsZS1zcGVjL2xheWVycy8jcGFpbnQtYmFja2dyb3VuZC1iYWNrZ3JvdW5kLWNvbG9yXG4vLyBodHRwczovL2RvY3MubWFwYm94LmNvbS9tYXBib3gtZ2wtanMvc3R5bGUtc3BlYy9leHByZXNzaW9ucy8jaW50ZXJwb2xhdGVcbmZ1bmN0aW9uIGdldFBhaW50Q29sb3IoY29sb3IpIHtcbiAgaWYgKEFycmF5LmlzQXJyYXkoY29sb3IpICYmIGNvbG9yWzBdID09PSAnaW50ZXJwb2xhdGUnKSB7XG4gICAgLy8gZ2V0IGNvbG9yIG9mIGZpcnN0IHpvb20gYnJlYWtcbiAgICAvLyBbXCJpbnRlcnBvbGF0ZVwiLCBbXCJsaW5lYXJcIl0sIFtcInpvb21cIl0sIDExLCBcImhzbCgzNSwgMzIlLCA5MSUpXCIsIDEzLCBcImhzbCgzNSwgMTIlLCA4OSUpXCJdXG4gICAgcmV0dXJuIGNvbG9yWzRdO1xuICB9XG4gIHJldHVybiBjb2xvcjtcbn1cblxuZnVuY3Rpb24gZ2V0M0RCdWlsZGluZ0NvbG9yKHN0eWxlKTogUkdCQ29sb3Ige1xuICAvLyBzZXQgYnVpbGRpbmcgY29sb3IgdG8gYmUgdGhlIHNhbWUgYXMgdGhlIGJhY2tncm91bmQgY29sb3IuXG4gIGlmICghc3R5bGUuc3R5bGUpIHtcbiAgICByZXR1cm4gaGV4VG9SZ2IoREVGQVVMVF9CTERHX0NPTE9SKTtcbiAgfVxuXG4gIGNvbnN0IGJhY2tncm91bmRMYXllciA9IChzdHlsZS5zdHlsZS5sYXllcnMgfHwgW10pLmZpbmQoKHtpZH0pID0+IGlkID09PSAnYmFja2dyb3VuZCcpO1xuXG4gIGNvbnN0IGJ1aWxkaW5nTGF5ZXIgPSAoc3R5bGUuc3R5bGUubGF5ZXJzIHx8IFtdKS5maW5kKCh7aWR9KSA9PiBpZC5tYXRjaCgvYnVpbGRpbmcvKSk7XG5cbiAgY29uc3QgYnVpbGRpbmdDb2xvciA9XG4gICAgZmluZExheWVyRmlsbENvbG9yKGJ1aWxkaW5nTGF5ZXIpIHx8IGZpbmRMYXllckZpbGxDb2xvcihiYWNrZ3JvdW5kTGF5ZXIpIHx8IERFRkFVTFRfQkxER19DT0xPUjtcblxuICAvLyBicmlnaHRlbiBvciBkYXJrZW4gYnVpbGRpbmcgYmFzZWQgb24gc3R5bGVcbiAgY29uc3Qgb3BlcmF0aW9uID0gc3R5bGUuaWQubWF0Y2goLyg/PShkYXJrfG5pZ2h0KSkvKSA/ICdicmlnaHRlcicgOiAnZGFya2VyJztcblxuICBjb25zdCBhbHBoYSA9IDAuMjtcbiAgY29uc3QgcmdiT2JqID0gcmdiKGJ1aWxkaW5nQ29sb3IpW29wZXJhdGlvbl0oW2FscGhhXSk7XG4gIHJldHVybiBbcmdiT2JqLnIsIHJnYk9iai5nLCByZ2JPYmouYl07XG59XG5cbmZ1bmN0aW9uIGdldEJhY2tncm91bmRDb2xvckZyb21TdHlsZUJhc2VMYXllcihcbiAgc3R5bGU6IEJhc2VNYXBTdHlsZSxcbiAgYmFja3VwQmFja2dyb3VuZENvbG9yOiBSR0JDb2xvclxuKTogUkdCQ29sb3Ige1xuICBpZiAoIXN0eWxlLnN0eWxlKSB7XG4gICAgcmV0dXJuIGNvbG9yTWF5YmVUb1JHQihiYWNrdXBCYWNrZ3JvdW5kQ29sb3IpIHx8IGJhY2t1cEJhY2tncm91bmRDb2xvcjtcbiAgfVxuXG4gIC8vIEB0cy1leHBlY3QtZXJyb3Igc3R5bGUuc3R5bGUgbm90IHR5cGVkXG4gIGNvbnN0IGJhc2VMYXllciA9IChzdHlsZS5zdHlsZS5sYXllcnMgfHwgW10pLmZpbmQoKHtpZH0pID0+XG4gICAgQkFTRV9NQVBfQkFDS0dST1VORF9MQVlFUl9JRFMuaW5jbHVkZXMoaWQpXG4gICk7XG5cbiAgY29uc3QgYmFja2dyb3VuZENvbG9yT2ZCYXNlTGF5ZXIgPSBnZXRQYWludENvbG9yKGZpbmRMYXllckZpbGxDb2xvcihiYXNlTGF5ZXIpKTtcblxuICBjb25zdCBuZXdCYWNrZ3JvdW5kQ29sb3IgPVxuICAgIHR5cGVvZiBiYWNrZ3JvdW5kQ29sb3JPZkJhc2VMYXllciA9PT0gJ3N0cmluZydcbiAgICAgID8gYmFja2dyb3VuZENvbG9yT2ZCYXNlTGF5ZXJcbiAgICAgIDogYmFja3VwQmFja2dyb3VuZENvbG9yO1xuXG4gIGNvbnN0IG5ld0JhY2tncm91bmRDb2xvckFzUkdCQXJyYXkgPSBjb2xvck1heWJlVG9SR0IobmV3QmFja2dyb3VuZENvbG9yKVxuICAgIC8vIGlmIG5ld0JhY2tncm91bmRDb2xvciB3YXMgaW4gc3RyaW5nIEhTTCBmb3JtYXQgaXQgY2FuIGludHJvZHVjZSBSR0IgbnVtYmVycyB3aXRoIGRlY2ltYWxzLFxuICAgIC8vIHdoaWNoIG1heSByZW5kZXIgdGhlIGJhY2tncm91bmQtY29sb3IgQ1NTIG9mIHRoZSA8U3R5bGVkTWFwPiBjb250YWluZXIgaW5jb3JyZWN0bHkgd2hlbiB1c2luZyBvdXIgb3duIGNvbG9yIHV0aWxzIGByZ2JUb0hleCgpYFxuICAgIC8vIHNvIHdlIGF0dGVtcHQgdG8gcm91bmQgdG8gbmVhcmVzdCBpbnRlZ2VyIGhlcmVcbiAgICA/Lm1hcChjaGFubmVsTnVtYmVyID0+IE1hdGgucm91bmQoY2hhbm5lbE51bWJlcikpIGFzIFJHQkNvbG9yIHwgbnVsbDtcblxuICByZXR1cm4gbmV3QmFja2dyb3VuZENvbG9yQXNSR0JBcnJheSB8fCBiYWNrdXBCYWNrZ3JvdW5kQ29sb3I7XG59XG5cbi8vIGRldGVybWluZSBuZXcgYmFja2dyb3VuZENvbG9yIGZyb20gZWl0aGVyIHByZXZpb3VzIHN0YXRlIGJhc2VtYXAgc3R5bGUsIHByZXZpb3VzIHN0YXRlIGJhY2tncm91bmRDb2xvciwgb3IgdGhlIERFRkFVTFRfQkFDS0dST1VORF9DT0xPUlxuZnVuY3Rpb24gZ2V0QmFja2dyb3VuZENvbG9yKHByZXZpb3VzU3RhdGU6IE1hcFN0eWxlLCBzdHlsZVR5cGU6IHN0cmluZykge1xuICBjb25zdCBwcmV2aW91c1N0YXRlTWFwU3R5bGUgPSBwcmV2aW91c1N0YXRlLm1hcFN0eWxlc1twcmV2aW91c1N0YXRlLnN0eWxlVHlwZV07XG4gIGNvbnN0IGJhY2t1cEJhY2tncm91bmRDb2xvciA9IHByZXZpb3VzU3RhdGUuYmFja2dyb3VuZENvbG9yIHx8IERFRkFVTFRfQkFDS0dST1VORF9DT0xPUjtcbiAgY29uc3QgYmFja2dyb3VuZENvbG9yID1cbiAgICBzdHlsZVR5cGUgPT09IE5PX01BUF9JRFxuICAgICAgPyAvLyBpZiB0aGUgc3R5bGUgaGFzIHN3aXRjaGVkIHRvIHRoZSBcIm5vIGJhc2VtYXBcIiBzdHlsZSxcbiAgICAgICAgLy8gYXR0ZW1wdCB0byBkZXRlY3QgYmFja2dyb3VuZENvbG9yIG9mIHRoZSBwcmV2aW91cyBiYXNlbWFwIGlmIGl0IHdhcyBhIG1hcGJveCBiYXNlbWFwXG4gICAgICAgIC8vIGFuZCBzZXQgaXQgYXMgdGhlIFwibm8gYmFzZW1hcFwiIGJhY2tncm91bmRDb2xvclxuICAgICAgICBnZXRCYWNrZ3JvdW5kQ29sb3JGcm9tU3R5bGVCYXNlTGF5ZXIocHJldmlvdXNTdGF0ZU1hcFN0eWxlLCBiYWNrdXBCYWNrZ3JvdW5kQ29sb3IpXG4gICAgICA6IC8vIG90aGVyd2lzZSBsZWF2ZSBpdCBhbG9uZSBhbmQgcmVseSBvbiB0aGUgcHJldmlvdXMgc3RhdGUncyBwcmVleGlzdGluZyBiYWNrZ3JvdW5kQ29sb3JcbiAgICAgICAgLy8gb3IgREVGQVVMVF9CQUNLR1JPVU5EX0NPTE9SIGFzIGEgbGFzdCByZXNvcnRcbiAgICAgICAgYmFja3VwQmFja2dyb3VuZENvbG9yO1xuXG4gIHJldHVybiBiYWNrZ3JvdW5kQ29sb3I7XG59XG5cbmZ1bmN0aW9uIGdldExheWVyR3JvdXBzRnJvbVN0eWxlKHN0eWxlKSB7XG4gIHJldHVybiBBcnJheS5pc0FycmF5KHN0eWxlPy5sYXllcnMpXG4gICAgPyBERUZBVUxUX0xBWUVSX0dST1VQUy5maWx0ZXIobGcgPT4gc3R5bGUubGF5ZXJzLmZpbHRlcihsZy5maWx0ZXIpLmxlbmd0aClcbiAgICA6IFtdO1xufVxuXG4vLyBVcGRhdGVyc1xuXG4vKipcbiAqIEBtZW1iZXJvZiBtYXBTdHlsZVVwZGF0ZXJzXG4gKiBAcHVibGljXG4gKi9cbmV4cG9ydCBjb25zdCByZXF1ZXN0TWFwU3R5bGVzVXBkYXRlciA9IChcbiAgc3RhdGU6IE1hcFN0eWxlLFxuICB7cGF5bG9hZDoge21hcFN0eWxlcywgb25TdWNjZXNzfX06IE1hcFN0eWxlQWN0aW9ucy5SZXF1ZXN0TWFwU3R5bGVzVXBkYXRlckFjdGlvblxuKTogTWFwU3R5bGUgPT4ge1xuICBjb25zdCB0b0xvYWQgPSBPYmplY3Qua2V5cyhtYXBTdHlsZXMpLnJlZHVjZShcbiAgICAoYWNjdSwgaWQpID0+ICh7XG4gICAgICAuLi5hY2N1LFxuICAgICAgLi4uKCFzdGF0ZS5pc0xvYWRpbmdbaWRdID8ge1tpZF06IG1hcFN0eWxlc1tpZF19IDoge30pXG4gICAgfSksXG4gICAge31cbiAgKTtcbiAgY29uc3QgbG9hZE1hcFN0eWxlVGFza3MgPSBnZXRMb2FkTWFwU3R5bGVUYXNrcyhcbiAgICB0b0xvYWQsXG4gICAgc3RhdGUubWFwYm94QXBpQWNjZXNzVG9rZW4sXG4gICAgc3RhdGUubWFwYm94QXBpVXJsLFxuICAgIG9uU3VjY2Vzc1xuICApO1xuXG4gIGNvbnN0IGlzTG9hZGluZyA9IE9iamVjdC5rZXlzKHRvTG9hZCkucmVkdWNlKFxuICAgIChhY2N1LCBrZXkpID0+ICh7XG4gICAgICAuLi5hY2N1LFxuICAgICAgW2tleV06IHRydWVcbiAgICB9KSxcbiAgICB7fVxuICApO1xuICBjb25zdCBuZXh0U3RhdGUgPSB7XG4gICAgLi4uc3RhdGUsXG4gICAgaXNMb2FkaW5nXG4gIH07XG4gIHJldHVybiB3aXRoVGFzayhuZXh0U3RhdGUsIGxvYWRNYXBTdHlsZVRhc2tzKTtcbn07XG5cbi8qKlxuICogUHJvcGFnYXRlIGBtYXBTdHlsZWAgcmVkdWNlciB3aXRoIGBtYXBib3hBcGlBY2Nlc3NUb2tlbmAgYW5kIGBtYXBTdHlsZXNSZXBsYWNlRGVmYXVsdGAuXG4gKiBpZiBtYXBTdHlsZXNSZXBsYWNlRGVmYXVsdCBpcyB0cnVlIG1hcFN0eWxlcyBpcyBlbXB0aWVkOyBsb2FkTWFwU3R5bGVzVXBkYXRlcigpIHdpbGxcbiAqIHBvcHVsYXRlIG1hcFN0eWxlcy5cbiAqXG4gKiBAbWVtYmVyb2YgbWFwU3R5bGVVcGRhdGVyc1xuICogQHB1YmxpY1xuICovXG5leHBvcnQgY29uc3QgaW5pdE1hcFN0eWxlVXBkYXRlciA9IChcbiAgc3RhdGU6IE1hcFN0eWxlLFxuICB7XG4gICAgcGF5bG9hZCA9IHt9XG4gIH06IHtcbiAgICB0eXBlPzogdHlwZW9mIEFjdGlvblR5cGVzLklOSVQ7XG4gICAgcGF5bG9hZDogS2VwbGVyR2xJbml0UGF5bG9hZDtcbiAgfVxuKTogTWFwU3R5bGUgPT4gKHtcbiAgLi4uc3RhdGUsXG4gIC8vIHNhdmUgbWFwYm94IGFjY2VzcyB0b2tlbiB0byBtYXAgc3R5bGUgc3RhdGVcbiAgbWFwYm94QXBpQWNjZXNzVG9rZW46IHBheWxvYWQubWFwYm94QXBpQWNjZXNzVG9rZW4gfHwgc3RhdGUubWFwYm94QXBpQWNjZXNzVG9rZW4sXG4gIG1hcGJveEFwaVVybDogcGF5bG9hZC5tYXBib3hBcGlVcmwgfHwgc3RhdGUubWFwYm94QXBpVXJsLFxuICBtYXBTdHlsZXM6ICFwYXlsb2FkLm1hcFN0eWxlc1JlcGxhY2VEZWZhdWx0ID8gc3RhdGUubWFwU3R5bGVzIDoge30sXG4gIG1hcFN0eWxlc1JlcGxhY2VEZWZhdWx0OiBwYXlsb2FkLm1hcFN0eWxlc1JlcGxhY2VEZWZhdWx0IHx8IGZhbHNlXG59KTtcbi8vIH0pO1xuXG4vKipcbiAqIFVwZGF0ZSBgdmlzaWJsZUxheWVyR3JvdXBzYHRvIGNoYW5nZSBsYXllciBncm91cCB2aXNpYmlsaXR5XG4gKiBAbWVtYmVyb2YgbWFwU3R5bGVVcGRhdGVyc1xuICogQHB1YmxpY1xuICovXG5leHBvcnQgY29uc3QgbWFwQ29uZmlnQ2hhbmdlVXBkYXRlciA9IChcbiAgc3RhdGU6IE1hcFN0eWxlLFxuICBhY3Rpb246IE1hcFN0eWxlQWN0aW9ucy5NYXBDb25maWdDaGFuZ2VVcGRhdGVyQWN0aW9uXG4pOiBNYXBTdHlsZSA9PiB7XG4gIHJldHVybiB7XG4gICAgLi4uc3RhdGUsXG4gICAgLi4uYWN0aW9uLnBheWxvYWQsXG4gICAgLi4uZ2V0TWFwU3R5bGVzKHtcbiAgICAgIC4uLnN0YXRlLFxuICAgICAgLi4uYWN0aW9uLnBheWxvYWRcbiAgICB9KVxuICB9O1xufTtcblxuY29uc3QgaGFzU3R5bGVPYmplY3QgPSBzdHlsZSA9PiBpc1BsYWluT2JqZWN0KHN0eWxlPy5zdHlsZSk7XG5cbi8qKlxuICogQ2hhbmdlIHRvIGFub3RoZXIgbWFwIHN0eWxlLiBUaGUgc2VsZWN0ZWQgc3R5bGUgc2hvdWxkIGFscmVhZHkgYmVlbiBsb2FkZWQgaW50byBgbWFwU3R5bGUubWFwU3R5bGVzYFxuICogQG1lbWJlcm9mIG1hcFN0eWxlVXBkYXRlcnNcbiAqIEBwdWJsaWNcbiAqL1xuZXhwb3J0IGNvbnN0IG1hcFN0eWxlQ2hhbmdlVXBkYXRlciA9IChcbiAgc3RhdGU6IE1hcFN0eWxlLFxuIC