UNPKG

kepler.gl

Version:

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

276 lines (226 loc) 29.4 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.addCustomMapStyleUpdater = exports.inputMapStyleUpdater = exports.loadCustomMapStyleUpdater = exports.resetMapConfigMapStyleUpdater = exports.receiveMapConfigUpdater = exports.loadMapStyleErrUpdater = exports.loadMapStylesUpdater = exports.mapStyleChangeUpdater = exports.mapConfigChangeUpdater = exports.initMapStyleUpdater = exports.INITIAL_MAP_STYLE = undefined; var _defineProperty2 = require('babel-runtime/helpers/defineProperty'); var _defineProperty3 = _interopRequireDefault(_defineProperty2); var _extends6 = require('babel-runtime/helpers/extends'); var _extends7 = _interopRequireDefault(_extends6); exports.getInitialInputStyle = getInitialInputStyle; var _immutable = require('immutable'); var _immutable2 = _interopRequireDefault(_immutable); var _tasks = require('react-palm/tasks'); var _tasks2 = _interopRequireDefault(_tasks); var _mapboxGlStyleEditor = require('../utils/map-style-utils/mapbox-gl-style-editor'); var _defaultSettings = require('../constants/default-settings'); var _utils = require('../utils/utils'); var _tasks3 = require('../tasks/tasks'); var _mapStyleActions = require('../actions/map-style-actions'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var getDefaultState = function getDefaultState() { var visibleLayerGroups = {}; var styleType = 'dark'; var topLayerGroups = {}; return { styleType: styleType, visibleLayerGroups: visibleLayerGroups, topLayerGroups: topLayerGroups, mapStyles: _defaultSettings.DEFAULT_MAP_STYLES.reduce(function (accu, curr) { return (0, _extends7.default)({}, accu, (0, _defineProperty3.default)({}, curr.id, curr)); }, {}), // save mapbox access token mapboxApiAccessToken: null, inputStyle: getInitialInputStyle() }; }; // Utils // Copyright (c) 2018 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. 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 ? _immutable2.default.fromJS(mapStyle.style) : (0, _mapboxGlStyleEditor.editBottomMapStyle)({ id: styleType, mapStyle: mapStyle, visibleLayerGroups: visibleLayerGroups }); var hasTopLayer = editable && 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 (0, _extends7.default)({}, accu, (0, _defineProperty3.default)({}, key, topLayerGroups[key] && visibleLayerGroups[key])); }, {}); var topMapStyle = hasTopLayer ? (0, _mapboxGlStyleEditor.editTopMapStyle)({ id: styleType, mapStyle: mapStyle, visibleLayerGroups: topLayers }) : null; return { bottomMapStyle: bottomMapStyle, topMapStyle: topMapStyle, editable: editable }; } function getLayerGroupsFromStyle(style) { return _defaultSettings.DEFAULT_LAYER_GROUPS.filter(function (lg) { return style.layers.filter(lg.filter).length; }); } // Updaters var initMapStyleUpdater = exports.initMapStyleUpdater = function initMapStyleUpdater(state, action) { return (0, _extends7.default)({}, state, { // save mapbox access token to map style state mapboxApiAccessToken: (action.payload || {}).mapboxApiAccessToken }); }; var mapConfigChangeUpdater = exports.mapConfigChangeUpdater = function mapConfigChangeUpdater(state, action) { return (0, _extends7.default)({}, state, action.payload, getMapStyles((0, _extends7.default)({}, state, action.payload))); }; var mapStyleChangeUpdater = exports.mapStyleChangeUpdater = function mapStyleChangeUpdater(state, _ref2) { var styleType = _ref2.payload; if (!state.mapStyles[styleType]) { // we might not have received the style yet return state; } var defaultLGVisibility = (0, _mapboxGlStyleEditor.getDefaultLayerGroupVisibility)(state.mapStyles[styleType]); var visibleLayerGroups = (0, _mapboxGlStyleEditor.mergeLayerGroupVisibility)(defaultLGVisibility, state.visibleLayerGroups); return (0, _extends7.default)({}, state, { styleType: styleType, visibleLayerGroups: visibleLayerGroups }, getMapStyles((0, _extends7.default)({}, state, { visibleLayerGroups: visibleLayerGroups, styleType: styleType }))); }; var loadMapStylesUpdater = exports.loadMapStylesUpdater = function loadMapStylesUpdater(state, action) { var newStyles = action.payload; // add new styles to state var newState = (0, _extends7.default)({}, state, { mapStyles: (0, _extends7.default)({}, state.mapStyles, newStyles) }); return newStyles[state.styleType] ? mapStyleChangeUpdater(newState, { payload: state.styleType }) : newState; }; // do nothing for now, if didn't load, skip it var loadMapStyleErrUpdater = exports.loadMapStyleErrUpdater = function loadMapStyleErrUpdater(state, action) { return state; }; var receiveMapConfigUpdater = exports.receiveMapConfigUpdater = function receiveMapConfigUpdater(state, _ref3) { var mapStyle = _ref3.payload.mapStyle; if (!mapStyle) { return state; } // if saved custom mapStyles load the style object var loadMapStyleTasks = mapStyle.mapStyles ? [_tasks2.default.all(Object.values(mapStyle.mapStyles).map(function (_ref4) { var id = _ref4.id, url = _ref4.url, accessToken = _ref4.accessToken; return { id: id, url: (0, _mapboxGlStyleEditor.getStyleDownloadUrl)(url, accessToken || state.mapboxApiAccessToken) }; }).map(_tasks3.LOAD_MAP_STYLE_TASK)).bimap( // success function (results) { return (0, _mapStyleActions.loadMapStyles)(results.reduce(function (accu, _ref5) { var id = _ref5.id, style = _ref5.style; return (0, _extends7.default)({}, accu, (0, _defineProperty3.default)({}, id, (0, _extends7.default)({}, mapStyle.mapStyles[id], { layerGroups: getLayerGroupsFromStyle(style), style: style }))); }, {})); }, // error function (error) { return (0, _mapStyleActions.loadMapStyleErr)(error); })] : null; var newState = mapConfigChangeUpdater(state, { payload: mapStyle }); return loadMapStyleTasks ? (0, _tasks.withTask)(newState, loadMapStyleTasks) : newState; }; var resetMapConfigMapStyleUpdater = exports.resetMapConfigMapStyleUpdater = function resetMapConfigMapStyleUpdater(state) { var emptyConfig = (0, _extends7.default)({}, INITIAL_MAP_STYLE, { mapboxApiAccessToken: state.mapboxApiAccessToken }, state.initialState, { mapStyles: state.mapStyles, initialState: state.initialState }); return mapStyleChangeUpdater(emptyConfig, { payload: emptyConfig.styleType }); }; var loadCustomMapStyleUpdater = exports.loadCustomMapStyleUpdater = function loadCustomMapStyleUpdater(state, _ref6) { var _ref6$payload = _ref6.payload, icon = _ref6$payload.icon, style = _ref6$payload.style, error = _ref6$payload.error; return (0, _extends7.default)({}, state, { inputStyle: (0, _extends7.default)({}, state.inputStyle, style ? { id: style.id || (0, _utils.generateHashId)(), // make a copy of the style object style: JSON.parse(JSON.stringify(style)), label: style.name, // gathering layer group info from style json layerGroups: getLayerGroupsFromStyle(style) } : {}, icon ? { icon: icon } : {}, error !== undefined ? { error: error } : {}) }); }; var inputMapStyleUpdater = exports.inputMapStyleUpdater = function inputMapStyleUpdater(state, _ref7) { var inputStyle = _ref7.payload; return (0, _extends7.default)({}, state, { inputStyle: (0, _extends7.default)({}, inputStyle, { isValid: (0, _mapboxGlStyleEditor.isValidStyleUrl)(inputStyle.url) }) }); }; var addCustomMapStyleUpdater = exports.addCustomMapStyleUpdater = function addCustomMapStyleUpdater(state, action) { var styleId = state.inputStyle.id; var newState = (0, _extends7.default)({}, state, { mapStyles: (0, _extends7.default)({}, state.mapStyles, (0, _defineProperty3.default)({}, styleId, state.inputStyle)), // set to default inputStyle: getInitialInputStyle() }); // set new style return mapStyleChangeUpdater(newState, { payload: styleId }); }; function getInitialInputStyle() { return { accessToken: null, error: false, isValid: false, label: null, style: null, url: null, custom: true }; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/reducers/map-style-updaters.js"],"names":["getInitialInputStyle","getDefaultState","visibleLayerGroups","styleType","topLayerGroups","mapStyles","DEFAULT_MAP_STYLES","reduce","accu","curr","id","mapboxApiAccessToken","inputStyle","INITIAL_MAP_STYLE","getMapStyles","mapStyle","style","editable","Object","keys","length","bottomMapStyle","Immutable","fromJS","hasTopLayer","values","some","v","topLayers","key","topMapStyle","getLayerGroupsFromStyle","DEFAULT_LAYER_GROUPS","filter","layers","lg","initMapStyleUpdater","state","action","payload","mapConfigChangeUpdater","mapStyleChangeUpdater","defaultLGVisibility","loadMapStylesUpdater","newStyles","newState","loadMapStyleErrUpdater","receiveMapConfigUpdater","loadMapStyleTasks","Task","all","map","url","accessToken","LOAD_MAP_STYLE_TASK","bimap","results","layerGroups","error","resetMapConfigMapStyleUpdater","emptyConfig","initialState","loadCustomMapStyleUpdater","icon","JSON","parse","stringify","label","name","undefined","inputMapStyleUpdater","isValid","addCustomMapStyleUpdater","styleId","custom"],"mappings":"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","file":"map-style-updaters.js","sourcesContent":["// Copyright (c) 2018 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 Immutable from 'immutable';\nimport Task, {withTask} from 'react-palm/tasks';\n\n// Utils\nimport {\n  getDefaultLayerGroupVisibility,\n  isValidStyleUrl,\n  getStyleDownloadUrl,\n  mergeLayerGroupVisibility,\n  editTopMapStyle,\n  editBottomMapStyle\n} from 'utils/map-style-utils/mapbox-gl-style-editor';\nimport {DEFAULT_MAP_STYLES, DEFAULT_LAYER_GROUPS} from 'constants/default-settings';\nimport {generateHashId} from 'utils/utils';\nimport {LOAD_MAP_STYLE_TASK} from 'tasks/tasks';\nimport {loadMapStyles, loadMapStyleErr} from 'actions/map-style-actions';\n\nconst getDefaultState = () => {\n  const visibleLayerGroups = {};\n  const styleType = 'dark';\n  const topLayerGroups = {};\n\n  return {\n    styleType,\n    visibleLayerGroups,\n    topLayerGroups,\n    mapStyles: DEFAULT_MAP_STYLES.reduce((accu, curr) => ({\n      ...accu,\n      [curr.id]: curr\n    }), {}),\n    // save mapbox access token\n    mapboxApiAccessToken: null,\n    inputStyle: getInitialInputStyle()\n  };\n};\n\nexport const INITIAL_MAP_STYLE = getDefaultState();\n\n/**\n * Create two map styles from preset map style, one for top map one for bottom\n *\n * @param {string} styleType - current map style\n * @param {object} visibleLayerGroups - visible layers of bottom map\n * @param {object} topLayerGroups - visible layers of top map\n * @param {object} mapStyles - a dictionary of all map styles\n * @returns {object} bottomMapStyle | topMapStyle | isRaster\n */\nfunction getMapStyles({\n  styleType,\n  visibleLayerGroups,\n  topLayerGroups,\n  mapStyles\n}) {\n  const mapStyle = mapStyles[styleType];\n\n  // style might not be loaded yet\n  if (!mapStyle || !mapStyle.style) {\n    return {};\n  }\n\n  const editable = Object.keys(visibleLayerGroups).length;\n\n  const bottomMapStyle = !editable\n    ? Immutable.fromJS(mapStyle.style)\n    : editBottomMapStyle({\n        id: styleType,\n        mapStyle,\n        visibleLayerGroups\n      });\n\n  const hasTopLayer = editable && Object.values(topLayerGroups).some(v => v);\n\n  // mute top layer if not visible in bottom layer\n  const topLayers =\n    hasTopLayer &&\n    Object.keys(topLayerGroups).reduce(\n      (accu, key) => ({\n        ...accu,\n        [key]: topLayerGroups[key] && visibleLayerGroups[key]\n      }),\n      {}\n    );\n\n  const topMapStyle = hasTopLayer\n    ? editTopMapStyle({\n        id: styleType,\n        mapStyle,\n        visibleLayerGroups: topLayers\n      })\n    : null;\n\n  return {bottomMapStyle, topMapStyle, editable};\n}\n\nfunction getLayerGroupsFromStyle(style) {\n  return DEFAULT_LAYER_GROUPS.filter(lg => style.layers.filter(lg.filter).length);\n}\n\n// Updaters\nexport const initMapStyleUpdater = (state, action) => ({\n  ...state,\n  // save mapbox access token to map style state\n  mapboxApiAccessToken: (action.payload || {}).mapboxApiAccessToken\n});\n\nexport const mapConfigChangeUpdater = (state, action) => ({\n  ...state,\n  ...action.payload,\n  ...getMapStyles({\n    ...state,\n    ...action.payload\n  })\n});\n\nexport const mapStyleChangeUpdater = (state, {payload: styleType}) => {\n  if (!state.mapStyles[styleType]) {\n    // we might not have received the style yet\n    return state;\n  }\n  const defaultLGVisibility = getDefaultLayerGroupVisibility(\n    state.mapStyles[styleType]\n  );\n\n  const visibleLayerGroups = mergeLayerGroupVisibility(defaultLGVisibility, state.visibleLayerGroups);\n\n  return {\n    ...state,\n    styleType,\n    visibleLayerGroups,\n    ...getMapStyles({\n      ...state,\n      visibleLayerGroups,\n      styleType\n    })\n  };\n};\n\nexport const loadMapStylesUpdater = (state, action) => {\n  const newStyles = action.payload;\n\n  // add new styles to state\n  const newState = {\n    ...state,\n    mapStyles: {\n      ...state.mapStyles,\n      ...newStyles\n    }\n  };\n\n  return newStyles[state.styleType]\n    ? mapStyleChangeUpdater(newState, {payload: state.styleType})\n    : newState;\n};\n\n// do nothing for now, if didn't load, skip it\nexport const loadMapStyleErrUpdater = (state, action) => state;\nexport const receiveMapConfigUpdater = (state, {payload: {mapStyle}}) => {\n  if (!mapStyle) {\n    return state;\n  }\n\n  // if saved custom mapStyles load the style object\n  const loadMapStyleTasks = mapStyle.mapStyles ? [\n    Task.all(\n      Object.values(mapStyle.mapStyles)\n      .map(({id, url, accessToken}) => ({\n        id, url: getStyleDownloadUrl(url, accessToken || state.mapboxApiAccessToken)\n      }))\n      .map(LOAD_MAP_STYLE_TASK))\n      .bimap(\n        // success\n        results => (\n          loadMapStyles(\n            results.reduce((accu, {id, style}) => ({\n              ...accu,\n              [id]: {\n                ...mapStyle.mapStyles[id],\n                layerGroups: getLayerGroupsFromStyle(style),\n                style\n              }\n            }), {})\n          )\n        ),\n        // error\n        error => loadMapStyleErr(error)\n      )\n  ] : null;\n\n  const newState = mapConfigChangeUpdater(state, {payload: mapStyle});\n\n  return loadMapStyleTasks ? withTask(\n    newState,\n    loadMapStyleTasks\n  ) : newState;\n};\n\nexport const resetMapConfigMapStyleUpdater = (state) => {\n  const emptyConfig = {\n    ...INITIAL_MAP_STYLE,\n    mapboxApiAccessToken: state.mapboxApiAccessToken,\n    ...state.initialState,\n    mapStyles: state.mapStyles,\n    initialState: state.initialState\n  };\n\n  return mapStyleChangeUpdater(emptyConfig, {payload: emptyConfig.styleType});\n};\n\nexport const loadCustomMapStyleUpdater = (state, {payload: {icon, style, error}}) => ({\n  ...state,\n  inputStyle: {\n    ...state.inputStyle,\n    // style json and icon will load asynchronously\n    ...(style ? {\n      id: style.id || generateHashId(),\n      // make a copy of the style object\n      style: JSON.parse(JSON.stringify(style)),\n      label: style.name,\n      // gathering layer group info from style json\n      layerGroups: getLayerGroupsFromStyle(style)\n    } : {}),\n    ...(icon ? {icon} : {}),\n    ...(error !== undefined ? {error} : {})\n  }\n});\n\nexport const inputMapStyleUpdater = (state, {payload: inputStyle}) => ({\n  ...state,\n  inputStyle: {\n    ...inputStyle,\n    isValid: isValidStyleUrl(inputStyle.url)\n  }\n});\n\nexport const addCustomMapStyleUpdater = (state, action) => {\n  const styleId = state.inputStyle.id;\n  const newState = {\n    ...state,\n    mapStyles: {\n      ...state.mapStyles,\n      [styleId]: state.inputStyle\n    },\n    // set to default\n    inputStyle: getInitialInputStyle()\n  };\n  // set new style\n  return mapStyleChangeUpdater(newState, {payload: styleId});\n};\n\nexport function getInitialInputStyle() {\n  return {\n    accessToken: null,\n    error: false,\n    isValid: false,\n    label: null,\n    style: null,\n    url: null,\n    custom: true\n  };\n}\n"]}