UNPKG

kepler.gl

Version:

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

233 lines (189 loc) 25.9 kB
"use strict"; var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = PlotContainerFactory; var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends")); var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck")); var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass")); var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn")); var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf")); var _assertThisInitialized2 = _interopRequireDefault(require("@babel/runtime/helpers/assertThisInitialized")); var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral")); var _react = _interopRequireWildcard(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _reselect = require("reselect"); var _styledComponents = _interopRequireDefault(require("styled-components")); var _reactMapGl = require("react-map-gl"); var _lodash = _interopRequireDefault(require("lodash.debounce")); var _notificationsUtils = require("../utils/notifications-utils"); var _mapContainer = _interopRequireDefault(require("./map-container")); var _exportUtils = require("../utils/export-utils"); var _mapboxGlStyleEditor = require("../utils/map-style-utils/mapbox-gl-style-editor"); 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; } function _templateObject() { var data = (0, _taggedTemplateLiteral2["default"])(["\n .mapboxgl-ctrl-bottom-left,\n .mapboxgl-ctrl-bottom-right {\n display: none;\n }\n\n position: absolute;\n"]); _templateObject = function _templateObject() { return data; }; return data; } var propTypes = { width: _propTypes["default"].number.isRequired, height: _propTypes["default"].number.isRequired, exportImageSetting: _propTypes["default"].object.isRequired, addNotification: _propTypes["default"].func.isRequired, mapFields: _propTypes["default"].object.isRequired }; PlotContainerFactory.deps = [_mapContainer["default"]]; // Remove mapbox logo in exported map, because it contains non-ascii characters var StyledPlotContainer = _styledComponents["default"].div(_templateObject()); var deckGlProps = { glOptions: { preserveDrawingBuffer: true, useDevicePixels: false } }; function PlotContainerFactory(MapContainer) { var PlotContainer = /*#__PURE__*/ function (_Component) { (0, _inherits2["default"])(PlotContainer, _Component); function PlotContainer(_props) { var _this; (0, _classCallCheck2["default"])(this, PlotContainer); _this = (0, _possibleConstructorReturn2["default"])(this, (0, _getPrototypeOf2["default"])(PlotContainer).call(this, _props)); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "plottingAreaRef", (0, _react.createRef)()); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "mapStyleSelector", function (props) { return props.mapFields.mapStyle; }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "mapScaleSelector", function (props) { var imageSize = props.exportImageSetting.imageSize; var mapState = props.mapFields.mapState; if (imageSize.scale) { return imageSize.scale; } var scale = (0, _exportUtils.getScaleFromImageSize)(imageSize.imageW, imageSize.imageH, mapState.width * (mapState.isSplit ? 2 : 1), mapState.height); return scale > 0 ? scale : 1; }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "scaledMapStyleSelector", (0, _reselect.createSelector)(_this.mapStyleSelector, _this.mapScaleSelector, function (mapStyle, scale) { return _objectSpread({}, mapStyle, { bottomMapStyle: (0, _mapboxGlStyleEditor.scaleMapStyleByResolution)(mapStyle.bottomMapStyle, scale), topMapStyle: (0, _mapboxGlStyleEditor.scaleMapStyleByResolution)(mapStyle.topMapStyle, scale) }); })); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_onMapRender", function (map) { if (map.isStyleLoaded()) { _this._retrieveNewScreenshot(); } }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_retrieveNewScreenshot", function () { if (_this.plottingAreaRef.current) { _this.props.startExportingImage(); var filter = function filter(node) { return node.className !== 'mapboxgl-control-container'; }; (0, _exportUtils.convertToPng)(_this.plottingAreaRef.current, { filter: filter }).then(_this.props.setExportImageDataUri)["catch"](function (err) { _this.props.setExportImageError(err); _this.props.addNotification((0, _notificationsUtils.exportImageError)({ err: err })); }); } }); _this._onMapRender = (0, _lodash["default"])(_this._onMapRender, 500); return _this; } (0, _createClass2["default"])(PlotContainer, [{ key: "componentDidMount", value: function componentDidMount() { this.props.startExportingImage(); } }, { key: "componentDidUpdate", value: function componentDidUpdate(prevProps) { var _this2 = this; // re-fetch the new screenshot only when ratio legend or resolution changes var checks = ['ratio', 'resolution', 'legend']; var shouldRetrieveScreenshot = checks.some(function (item) { return _this2.props.exportImageSetting[item] !== prevProps.exportImageSetting[item]; }); if (shouldRetrieveScreenshot) { this._retrieveNewScreenshot(); } } }, { key: "render", value: function render() { var _this$props = this.props, exportImageSetting = _this$props.exportImageSetting, mapFields = _this$props.mapFields, splitMaps = _this$props.splitMaps; var _exportImageSetting$i = exportImageSetting.imageSize, imageSize = _exportImageSetting$i === void 0 ? {} : _exportImageSetting$i, legend = exportImageSetting.legend; var isSplit = splitMaps && splitMaps.length > 1; var size = { width: imageSize.imageW || 1, height: imageSize.imageH || 1 }; var scale = this.mapScaleSelector(this.props); var mapProps = _objectSpread({}, mapFields, { mapStyle: this.scaledMapStyleSelector(this.props), // override viewport based on export settings mapState: _objectSpread({}, mapFields.mapState, { width: size.width / (isSplit ? 2 : 1), height: size.height, zoom: mapFields.mapState.zoom + (Math.log2(scale) || 0) }), mapControls: { // override map legend visibility mapLegend: { show: legend, active: true } }, MapComponent: _reactMapGl.StaticMap, onMapRender: this._onMapRender, isExport: true, deckGlProps: deckGlProps }); var mapContainers = !isSplit ? _react["default"].createElement(MapContainer, (0, _extends2["default"])({ index: 0 }, mapProps)) : splitMaps.map(function (settings, index) { return _react["default"].createElement(MapContainer, (0, _extends2["default"])({ key: index, index: index }, mapProps, { mapLayers: splitMaps[index].layers })); }); return _react["default"].createElement(StyledPlotContainer, { style: { position: 'absolute', top: -9999, left: -9999 }, className: "export-map-instance" }, _react["default"].createElement("div", { ref: this.plottingAreaRef, style: { width: "".concat(size.width, "px"), height: "".concat(size.height, "px"), display: 'flex' } }, mapContainers)); } }]); return PlotContainer; }(_react.Component); PlotContainer.propsTypes = propTypes; return PlotContainer; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/plot-container.js"],"names":["propTypes","width","PropTypes","number","isRequired","height","exportImageSetting","object","addNotification","func","mapFields","PlotContainerFactory","deps","MapContainerFactory","StyledPlotContainer","styled","div","deckGlProps","glOptions","preserveDrawingBuffer","useDevicePixels","MapContainer","PlotContainer","props","mapStyle","imageSize","mapState","scale","imageW","imageH","isSplit","mapStyleSelector","mapScaleSelector","bottomMapStyle","topMapStyle","map","isStyleLoaded","_retrieveNewScreenshot","plottingAreaRef","current","startExportingImage","filter","node","className","then","setExportImageDataUri","err","setExportImageError","_onMapRender","prevProps","checks","shouldRetrieveScreenshot","some","item","splitMaps","legend","length","size","mapProps","scaledMapStyleSelector","zoom","Math","log2","mapControls","mapLegend","show","active","MapComponent","StaticMap","onMapRender","isExport","mapContainers","settings","index","layers","position","top","left","display","Component","propsTypes"],"mappings":"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","sourcesContent":["// Copyright (c) 2020 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\n// libraries\nimport React, {Component, createRef} from 'react';\nimport PropTypes from 'prop-types';\nimport {createSelector} from 'reselect';\nimport styled from 'styled-components';\nimport {StaticMap} from 'react-map-gl';\nimport debounce from 'lodash.debounce';\nimport {exportImageError} from 'utils/notifications-utils';\nimport MapContainerFactory from './map-container';\nimport {convertToPng} from 'utils/export-utils';\nimport {scaleMapStyleByResolution} from 'utils/map-style-utils/mapbox-gl-style-editor';\nimport {getScaleFromImageSize} from 'utils/export-utils';\n\nconst propTypes = {\n  width: PropTypes.number.isRequired,\n  height: PropTypes.number.isRequired,\n  exportImageSetting: PropTypes.object.isRequired,\n  addNotification: PropTypes.func.isRequired,\n  mapFields: PropTypes.object.isRequired\n};\n\nPlotContainerFactory.deps = [MapContainerFactory];\n\n// Remove mapbox logo in exported map, because it contains non-ascii characters\nconst StyledPlotContainer = styled.div`\n  .mapboxgl-ctrl-bottom-left,\n  .mapboxgl-ctrl-bottom-right {\n    display: none;\n  }\n\n  position: absolute;\n`;\n\nconst deckGlProps = {\n  glOptions: {\n    preserveDrawingBuffer: true,\n    useDevicePixels: false\n  }\n};\n\nexport default function PlotContainerFactory(MapContainer) {\n  class PlotContainer extends Component {\n    constructor(props) {\n      super(props);\n      this._onMapRender = debounce(this._onMapRender, 500);\n    }\n\n    componentDidMount() {\n      this.props.startExportingImage();\n    }\n\n    componentDidUpdate(prevProps) {\n      // re-fetch the new screenshot only when ratio legend or resolution changes\n      const checks = ['ratio', 'resolution', 'legend'];\n      const shouldRetrieveScreenshot = checks.some(\n        item => this.props.exportImageSetting[item] !== prevProps.exportImageSetting[item]\n      );\n      if (shouldRetrieveScreenshot) {\n        this._retrieveNewScreenshot();\n      }\n    }\n\n    plottingAreaRef = createRef();\n\n    mapStyleSelector = props => props.mapFields.mapStyle;\n    mapScaleSelector = props => {\n      const {imageSize} = props.exportImageSetting;\n      const {mapState} = props.mapFields;\n      if (imageSize.scale) {\n        return imageSize.scale;\n      }\n\n      const scale = getScaleFromImageSize(\n        imageSize.imageW,\n        imageSize.imageH,\n        mapState.width * (mapState.isSplit ? 2 : 1),\n        mapState.height\n      );\n\n      return scale > 0 ? scale : 1;\n    };\n\n    scaledMapStyleSelector = createSelector(\n      this.mapStyleSelector,\n      this.mapScaleSelector,\n      (mapStyle, scale) => ({\n        ...mapStyle,\n        bottomMapStyle: scaleMapStyleByResolution(mapStyle.bottomMapStyle, scale),\n        topMapStyle: scaleMapStyleByResolution(mapStyle.topMapStyle, scale)\n      })\n    );\n\n    _onMapRender = map => {\n      if (map.isStyleLoaded()) {\n        this._retrieveNewScreenshot();\n      }\n    };\n\n    _retrieveNewScreenshot = () => {\n      if (this.plottingAreaRef.current) {\n        this.props.startExportingImage();\n        const filter = node => node.className !== 'mapboxgl-control-container';\n\n        convertToPng(this.plottingAreaRef.current, {filter})\n          .then(this.props.setExportImageDataUri)\n          .catch(err => {\n            this.props.setExportImageError(err);\n            this.props.addNotification(exportImageError({err}));\n          });\n      }\n    };\n\n    render() {\n      const {exportImageSetting, mapFields, splitMaps} = this.props;\n      const {imageSize = {}, legend} = exportImageSetting;\n      const isSplit = splitMaps && splitMaps.length > 1;\n\n      const size = {\n        width: imageSize.imageW || 1,\n        height: imageSize.imageH || 1\n      };\n      const scale = this.mapScaleSelector(this.props);\n      const mapProps = {\n        ...mapFields,\n        mapStyle: this.scaledMapStyleSelector(this.props),\n\n        // override viewport based on export settings\n        mapState: {\n          ...mapFields.mapState,\n          width: size.width / (isSplit ? 2 : 1),\n          height: size.height,\n          zoom: mapFields.mapState.zoom + (Math.log2(scale) || 0)\n        },\n        mapControls: {\n          // override map legend visibility\n          mapLegend: {\n            show: legend,\n            active: true\n          }\n        },\n        MapComponent: StaticMap,\n        onMapRender: this._onMapRender,\n        isExport: true,\n        deckGlProps\n      };\n\n      const mapContainers = !isSplit ? (\n        <MapContainer index={0} {...mapProps} />\n      ) : (\n        splitMaps.map((settings, index) => (\n          <MapContainer\n            key={index}\n            index={index}\n            {...mapProps}\n            mapLayers={splitMaps[index].layers}\n          />\n        ))\n      );\n\n      return (\n        <StyledPlotContainer\n          style={{position: 'absolute', top: -9999, left: -9999}}\n          className=\"export-map-instance\"\n        >\n          <div\n            ref={this.plottingAreaRef}\n            style={{\n              width: `${size.width}px`,\n              height: `${size.height}px`,\n              display: 'flex'\n            }}\n          >\n            {mapContainers}\n          </div>\n        </StyledPlotContainer>\n      );\n    }\n  }\n\n  PlotContainer.propsTypes = propTypes;\n  return PlotContainer;\n}\n"]}