UNPKG

kepler.gl

Version:

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

178 lines (133 loc) 20.6 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.injector = injector; exports.flattenDeps = flattenDeps; exports.provideRecipesToInjector = provideRecipesToInjector; exports.typeCheckRecipe = typeCheckRecipe; exports.withState = withState; exports.ERROR_MSG = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties")); var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray")); var _toConsumableArray2 = _interopRequireDefault(require("@babel/runtime/helpers/toConsumableArray")); var _react = _interopRequireDefault(require("react")); var _reactRedux = require("react-redux"); var _redux = require("redux"); var _window = require("global/window"); var _context = _interopRequireDefault(require("./context")); 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 MissingComp = function MissingComp() { return /*#__PURE__*/_react["default"].createElement("div", null); }; var ERROR_MSG = { wrongRecipeType: "injectComponents takes an array of factories replacement pairs as input, " + "each pair be a array as [originalFactory, replacement].", noDep: function noDep(fac, parent) { return "".concat(fac.name, " is required as a dependency of ").concat(parent.name, ", ") + "but is not provided to injectComponents. It will not be rendered."; }, notFunc: 'factory and its replacement should be a function' }; exports.ERROR_MSG = ERROR_MSG; function injector() { var map = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : new Map(); var cache = new Map(); // map<factory, factory -> ?> var get = function get(fac, parent) { var factory = map.get(fac); // factory is not injected if (!factory) { _window.console.error(ERROR_MSG.noDep(fac, parent)); return MissingComp; } // check if custom factory deps is declared var instances = cache.get(factory) || factory.apply(void 0, (0, _toConsumableArray2["default"])(factory.deps ? factory.deps.map(function (dep) { return get(dep, factory); }) : [])); cache.set(fac, instances); return instances; }; // if you have two functions that happen to have the exactly same text // it will be override: 2018-02-05 return { provide: function provide(factory, replacement) { if (!typeCheckRecipe([factory, replacement])) { return injector(map); } return injector(new Map(map).set(factory, replacement)); }, get: get }; } // entryPoint function flattenDeps(allDeps, factory) { var addToDeps = allDeps.concat([factory]); return Array.isArray(factory.deps) && factory.deps.length ? factory.deps.reduce(function (accu, dep) { return flattenDeps(accu, dep); }, addToDeps) : addToDeps; } function provideRecipesToInjector(recipes, appInjector) { var provided = new Map(); return recipes.reduce(function (inj, recipe) { var _inj; if (!typeCheckRecipe(recipe)) { return inj; } // collect dependencies of custom factories, if there is any. // Add them to the appInjector var customDependencies = flattenDeps([], recipe[1]); inj = customDependencies.reduce(function (ij, factory) { if (provided.get(factory)) { _window.console.warn("".concat(factory.name, " already injected from ").concat(provided.get(factory).name, ", injecting ").concat(recipe[0].name, " after ").concat(provided.get(factory).name, " will override it")); } return ij.provide(factory, factory); }, inj); provided.set(recipe[0], recipe[1]); return (_inj = inj).provide.apply(_inj, (0, _toConsumableArray2["default"])(recipe)); }, appInjector); } function typeCheckRecipe(recipe) { if (!Array.isArray(recipe) || recipe.length < 2) { _window.console.error('Error injecting [factory, replacement]', recipe); _window.console.error(ERROR_MSG.wrongRecipeType); return false; } var _recipe = (0, _slicedToArray2["default"])(recipe, 2), factory = _recipe[0], replacement = _recipe[1]; if (typeof factory !== 'function') { _window.console.error('Error injecting factory: ', factory); _window.console.error(ERROR_MSG.notFunc); return false; } else if (typeof replacement !== 'function') { _window.console.error('Error injecting replacement for: ', factory); _window.console.error(ERROR_MSG.notFunc); return false; } return true; } var identity = function identity(state) { return state; }; // Helper to add reducer state to custom component function withState() { var lenses = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : []; var mapStateToProps = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : identity; var actions = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : {}; return function (Component) { var WrappedComponent = function WrappedComponent(_ref) { var state = _ref.state, props = (0, _objectWithoutProperties2["default"])(_ref, ["state"]); return /*#__PURE__*/_react["default"].createElement(_context["default"].Consumer, null, function (context) { return /*#__PURE__*/_react["default"].createElement(Component, lenses.reduce(function (totalState, lens) { return _objectSpread(_objectSpread({}, totalState), lens(context.selector(state))); }, props)); }); }; return (0, _reactRedux.connect)(function (state) { return _objectSpread(_objectSpread({}, mapStateToProps(state)), {}, { state: state }); }, function (dispatch) { return Object.keys(actions).reduce(function (accu, key) { return _objectSpread(_objectSpread({}, accu), {}, (0, _defineProperty2["default"])({}, key, (0, _redux.bindActionCreators)(actions[key], dispatch))); }, {}); })(WrappedComponent); }; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/injector.js"],"names":["MissingComp","ERROR_MSG","wrongRecipeType","noDep","fac","parent","name","notFunc","injector","map","Map","cache","get","factory","Console","error","instances","deps","dep","set","provide","replacement","typeCheckRecipe","flattenDeps","allDeps","addToDeps","concat","Array","isArray","length","reduce","accu","provideRecipesToInjector","recipes","appInjector","provided","inj","recipe","customDependencies","ij","warn","identity","state","withState","lenses","mapStateToProps","actions","Component","WrappedComponent","props","context","totalState","lens","selector","dispatch","Object","keys","key"],"mappings":"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","sourcesContent":["// Copyright (c) 2021 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 React from 'react';\nimport {connect} from 'react-redux';\nimport {bindActionCreators} from 'redux';\nimport {console as Console} from 'global/window';\nimport KeplerGlContext from 'components/context';\n\nconst MissingComp = () => <div />;\n\nexport const ERROR_MSG = {\n  wrongRecipeType:\n    `injectComponents takes an array of factories replacement pairs as input, ` +\n    `each pair be a array as [originalFactory, replacement].`,\n\n  noDep: (fac, parent) =>\n    `${fac.name} is required as a dependency of ${parent.name}, ` +\n    `but is not provided to injectComponents. It will not be rendered.`,\n\n  notFunc: 'factory and its replacement should be a function'\n};\n\nexport function injector(map = new Map()) {\n  const cache = new Map(); // map<factory, factory -> ?>\n  const get = (fac, parent) => {\n    const factory = map.get(fac);\n    // factory is not injected\n    if (!factory) {\n      Console.error(ERROR_MSG.noDep(fac, parent));\n      return MissingComp;\n    }\n\n    // check if custom factory deps is declared\n    const instances =\n      cache.get(factory) ||\n      factory(...(factory.deps ? factory.deps.map(dep => get(dep, factory)) : []));\n\n    cache.set(fac, instances);\n    return instances;\n  };\n\n  // if you have two functions that happen to have the exactly same text\n  // it will be override: 2018-02-05\n  return {\n    provide: (factory, replacement) => {\n      if (!typeCheckRecipe([factory, replacement])) {\n        return injector(map);\n      }\n      return injector(new Map(map).set(factory, replacement));\n    },\n    get\n  };\n}\n\n// entryPoint\nexport function flattenDeps(allDeps, factory) {\n  const addToDeps = allDeps.concat([factory]);\n  return Array.isArray(factory.deps) && factory.deps.length\n    ? factory.deps.reduce((accu, dep) => flattenDeps(accu, dep), addToDeps)\n    : addToDeps;\n}\n\nexport function provideRecipesToInjector(recipes, appInjector) {\n  const provided = new Map();\n\n  return recipes.reduce((inj, recipe) => {\n    if (!typeCheckRecipe(recipe)) {\n      return inj;\n    }\n\n    // collect dependencies of custom factories, if there is any.\n    // Add them to the appInjector\n    const customDependencies = flattenDeps([], recipe[1]);\n    inj = customDependencies.reduce((ij, factory) => {\n      if (provided.get(factory)) {\n        Console.warn(\n          `${factory.name} already injected from ${provided.get(factory).name}, injecting ${\n            recipe[0].name\n          } after ${provided.get(factory).name} will override it`\n        );\n      }\n      return ij.provide(factory, factory);\n    }, inj);\n\n    provided.set(recipe[0], recipe[1]);\n    return inj.provide(...recipe);\n  }, appInjector);\n}\n\nexport function typeCheckRecipe(recipe) {\n  if (!Array.isArray(recipe) || recipe.length < 2) {\n    Console.error('Error injecting [factory, replacement]', recipe);\n    Console.error(ERROR_MSG.wrongRecipeType);\n    return false;\n  }\n\n  const [factory, replacement] = recipe;\n  if (typeof factory !== 'function') {\n    Console.error('Error injecting factory: ', factory);\n    Console.error(ERROR_MSG.notFunc);\n    return false;\n  } else if (typeof replacement !== 'function') {\n    Console.error('Error injecting replacement for: ', factory);\n    Console.error(ERROR_MSG.notFunc);\n    return false;\n  }\n\n  return true;\n}\n\nconst identity = state => state;\n// Helper to add reducer state to custom component\nexport function withState(lenses = [], mapStateToProps = identity, actions = {}) {\n  return Component => {\n    const WrappedComponent = ({state, ...props}) => (\n      <KeplerGlContext.Consumer>\n        {context => (\n          <Component\n            {...lenses.reduce(\n              (totalState, lens) => ({\n                ...totalState,\n                ...lens(context.selector(state))\n              }),\n              props\n            )}\n          />\n        )}\n      </KeplerGlContext.Consumer>\n    );\n\n    return connect(\n      state => ({...mapStateToProps(state), state}),\n      dispatch =>\n        Object.keys(actions).reduce(\n          (accu, key) => ({\n            ...accu,\n            [key]: bindActionCreators(actions[key], dispatch)\n          }),\n          {}\n        )\n    )(WrappedComponent);\n  };\n}\n"]}