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