kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
275 lines (220 loc) • 33.9 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
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 _assertThisInitialized2 = _interopRequireDefault(require("@babel/runtime/helpers/assertThisInitialized"));
var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
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 _mapsLayout = _interopRequireDefault(require("./maps-layout"));
var _exportUtils = require("../utils/export-utils");
var _mapboxGlStyleEditor = require("../utils/map-style-utils/mapbox-gl-style-editor");
var _dataUtils = require("../utils/data-utils");
var _projectionUtils = require("../utils/projection-utils");
var _defaultSettings = require("../constants/default-settings");
var _templateObject, _templateObject2;
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 _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = (0, _getPrototypeOf2["default"])(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = (0, _getPrototypeOf2["default"])(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return (0, _possibleConstructorReturn2["default"])(this, result); }; }
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
var CLASS_FILTER = ['mapboxgl-control-container', 'attrition-link', 'attrition-logo'];
var DOM_FILTER_FUNC = function DOM_FILTER_FUNC(node) {
return !CLASS_FILTER.includes(node.className);
};
var OUT_OF_SCREEN_POSITION = -9999;
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,
setExportImageSetting: _propTypes["default"].object.isRequired,
setExportImageDataUri: _propTypes["default"].func.isRequired,
setExportImageError: _propTypes["default"].func.isRequired,
splitMaps: _propTypes["default"].arrayOf(_propTypes["default"].object)
};
PlotContainerFactory.deps = [_mapContainer["default"], _mapsLayout["default"]]; // Remove mapbox logo in exported map, because it contains non-ascii characters
var StyledPlotContainer = _styledComponents["default"].div(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n .mapboxgl-ctrl-bottom-left,\n .mapboxgl-ctrl-bottom-right,\n .mapbox-attribution-container {\n display: none;\n }\n\n position: absolute;\n top: ", "px;\n left: ", "px;\n"])), OUT_OF_SCREEN_POSITION, OUT_OF_SCREEN_POSITION);
var StyledMapContainer = _styledComponents["default"].div(_templateObject2 || (_templateObject2 = (0, _taggedTemplateLiteral2["default"])(["\n width: ", "px;\n height: ", "px;\n display: flex;\n"])), function (props) {
return props.width;
}, function (props) {
return props.height;
});
var deckGlProps = {
glOptions: {
preserveDrawingBuffer: true,
useDevicePixels: false
}
};
function PlotContainerFactory(MapContainer, MapsLayout) {
var PlotContainer = /*#__PURE__*/function (_Component) {
(0, _inherits2["default"])(PlotContainer, _Component);
var _super = _createSuper(PlotContainer);
function PlotContainer(_props) {
var _this;
(0, _classCallCheck2["default"])(this, PlotContainer);
_this = _super.call(this, _props);
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "plottingAreaRef", /*#__PURE__*/(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(_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) {
(0, _exportUtils.convertToPng)(_this.plottingAreaRef.current, {
filter: DOM_FILTER_FUNC
}).then(_this.props.setExportImageDataUri)["catch"](function (err) {
_this.props.setExportImageError(err);
if (_this.props.enableErrorNotification) {
_this.props.addNotification((0, _notificationsUtils.exportImageError)({
err: err
}));
}
});
}
});
_this._onMapRender = (0, _lodash["default"])(_this._onMapRender, 500);
_this._retrieveNewScreenshot = (0, _lodash["default"])(_this._retrieveNewScreenshot, 500);
return _this;
}
(0, _createClass2["default"])(PlotContainer, [{
key: "componentDidMount",
value: function componentDidMount() {
this.props.setExportImageSetting({
processing: true
});
}
}, {
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.props.setExportImageSetting({
processing: true
});
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 mapState = mapFields.mapState;
var isSplit = splitMaps && splitMaps.length > 1;
var size = {
width: imageSize.imageW || 1,
height: imageSize.imageH || 1
};
var width = size.width / (isSplit ? 2 : 1);
var height = size.height;
var scale = this.mapScaleSelector(this.props);
var newMapState = _objectSpread(_objectSpread({}, mapState), {}, {
width: width,
height: height,
zoom: mapState.zoom + (Math.log2(scale) || 0)
}); // center and all layer bounds
if (exportImageSetting.center) {
var renderedLayers = mapFields.layers.filter(function (layer, idx) {
return layer.id !== _defaultSettings.GEOCODER_LAYER_ID && layer.shouldRenderLayer(mapFields.layerData[idx]);
});
var bounds = (0, _dataUtils.findMapBounds)(renderedLayers);
var centerAndZoom = (0, _projectionUtils.getCenterAndZoomFromBounds)(bounds, {
width: width,
height: height
});
if (centerAndZoom) {
var zoom = Number.isFinite(centerAndZoom.zoom) ? centerAndZoom.zoom : mapState.zoom;
newMapState.longitude = centerAndZoom.center[0];
newMapState.latitude = centerAndZoom.center[1];
newMapState.zoom = zoom + Number(Math.log2(scale) || 0);
}
}
var mapProps = _objectSpread(_objectSpread({}, mapFields), {}, {
mapStyle: this.scaledMapStyleSelector(this.props),
// override viewport based on export settings
mapState: newMapState,
mapControls: {
// override map legend visibility
mapLegend: {
show: legend,
active: true
}
},
MapComponent: _reactMapGl.StaticMap,
onMapRender: this._onMapRender,
isExport: true,
deckGlProps: deckGlProps
});
var mapContainers = !isSplit ? /*#__PURE__*/_react["default"].createElement(MapContainer, (0, _extends2["default"])({
index: 0,
primary: true
}, mapProps)) : /*#__PURE__*/_react["default"].createElement(MapsLayout, null, splitMaps.map(function (settings, index) {
return /*#__PURE__*/_react["default"].createElement(MapContainer, (0, _extends2["default"])({
key: index,
index: index,
primary: index === 1
}, mapProps, {
mapLayers: splitMaps[index].layers
}));
}));
return /*#__PURE__*/_react["default"].createElement(StyledPlotContainer, {
className: "export-map-instance"
}, /*#__PURE__*/_react["default"].createElement(StyledMapContainer, {
ref: this.plottingAreaRef,
width: size.width,
height: size.height
}, 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":["CLASS_FILTER","DOM_FILTER_FUNC","node","includes","className","OUT_OF_SCREEN_POSITION","propTypes","width","PropTypes","number","isRequired","height","exportImageSetting","object","addNotification","func","mapFields","setExportImageSetting","setExportImageDataUri","setExportImageError","splitMaps","arrayOf","PlotContainerFactory","deps","MapContainerFactory","MapsLayoutFactory","StyledPlotContainer","styled","div","StyledMapContainer","props","deckGlProps","glOptions","preserveDrawingBuffer","useDevicePixels","MapContainer","MapsLayout","PlotContainer","mapStyle","imageSize","mapState","scale","imageW","imageH","isSplit","mapStyleSelector","mapScaleSelector","bottomMapStyle","topMapStyle","map","isStyleLoaded","_retrieveNewScreenshot","plottingAreaRef","current","filter","then","err","enableErrorNotification","_onMapRender","processing","prevProps","checks","shouldRetrieveScreenshot","some","item","legend","length","size","newMapState","zoom","Math","log2","center","renderedLayers","layers","layer","idx","id","GEOCODER_LAYER_ID","shouldRenderLayer","layerData","bounds","centerAndZoom","Number","isFinite","longitude","latitude","mapProps","scaledMapStyleSelector","mapControls","mapLegend","show","active","MapComponent","StaticMap","onMapRender","isExport","mapContainers","settings","index","Component","propsTypes"],"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\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 MapsLayoutFactory from './maps-layout';\nimport {convertToPng, getScaleFromImageSize} from 'utils/export-utils';\nimport {scaleMapStyleByResolution} from 'utils/map-style-utils/mapbox-gl-style-editor';\n\nimport {findMapBounds} from 'utils/data-utils';\nimport {getCenterAndZoomFromBounds} from 'utils/projection-utils';\nimport {GEOCODER_LAYER_ID} from 'constants/default-settings';\n\nconst CLASS_FILTER = ['mapboxgl-control-container', 'attrition-link', 'attrition-logo'];\nconst DOM_FILTER_FUNC = node => !CLASS_FILTER.includes(node.className);\nconst OUT_OF_SCREEN_POSITION = -9999;\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  setExportImageSetting: PropTypes.object.isRequired,\n  setExportImageDataUri: PropTypes.func.isRequired,\n  setExportImageError: PropTypes.func.isRequired,\n  splitMaps: PropTypes.arrayOf(PropTypes.object)\n};\n\nPlotContainerFactory.deps = [MapContainerFactory, MapsLayoutFactory];\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  .mapbox-attribution-container {\n    display: none;\n  }\n\n  position: absolute;\n  top: ${OUT_OF_SCREEN_POSITION}px;\n  left: ${OUT_OF_SCREEN_POSITION}px;\n`;\n\nconst StyledMapContainer = styled.div`\n  width: ${props => props.width}px;\n  height: ${props => props.height}px;\n  display: flex;\n`;\n\nconst deckGlProps = {\n  glOptions: {\n    preserveDrawingBuffer: true,\n    useDevicePixels: false\n  }\n};\n\nexport default function PlotContainerFactory(MapContainer, MapsLayout) {\n  class PlotContainer extends Component {\n    constructor(props) {\n      super(props);\n      this._onMapRender = debounce(this._onMapRender, 500);\n      this._retrieveNewScreenshot = debounce(this._retrieveNewScreenshot, 500);\n    }\n\n    componentDidMount() {\n      this.props.setExportImageSetting({processing: true});\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.props.setExportImageSetting({processing: true});\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        convertToPng(this.plottingAreaRef.current, {filter: DOM_FILTER_FUNC})\n          .then(this.props.setExportImageDataUri)\n          .catch(err => {\n            this.props.setExportImageError(err);\n            if (this.props.enableErrorNotification) {\n              this.props.addNotification(exportImageError({err}));\n            }\n          });\n      }\n    };\n\n    render() {\n      const {exportImageSetting, mapFields, splitMaps} = this.props;\n      const {imageSize = {}, legend} = exportImageSetting;\n      const {mapState} = mapFields;\n      const isSplit = splitMaps && splitMaps.length > 1;\n\n      const size = {\n        width: imageSize.imageW || 1,\n        height: imageSize.imageH || 1\n      };\n      const width = size.width / (isSplit ? 2 : 1);\n      const height = size.height;\n      const scale = this.mapScaleSelector(this.props);\n      const newMapState = {\n        ...mapState,\n        width,\n        height,\n        zoom: mapState.zoom + (Math.log2(scale) || 0)\n      };\n\n      // center and all layer bounds\n      if (exportImageSetting.center) {\n        const renderedLayers = mapFields.layers.filter(\n          (layer, idx) =>\n            layer.id !== GEOCODER_LAYER_ID && layer.shouldRenderLayer(mapFields.layerData[idx])\n        );\n        const bounds = findMapBounds(renderedLayers);\n        const centerAndZoom = getCenterAndZoomFromBounds(bounds, {width, height});\n        if (centerAndZoom) {\n          const zoom = Number.isFinite(centerAndZoom.zoom) ? centerAndZoom.zoom : mapState.zoom;\n\n          newMapState.longitude = centerAndZoom.center[0];\n          newMapState.latitude = centerAndZoom.center[1];\n          newMapState.zoom = zoom + Number(Math.log2(scale) || 0);\n        }\n      }\n\n      const mapProps = {\n        ...mapFields,\n        mapStyle: this.scaledMapStyleSelector(this.props),\n\n        // override viewport based on export settings\n        mapState: newMapState,\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} primary={true} {...mapProps} />\n      ) : (\n        <MapsLayout>\n          {splitMaps.map((settings, index) => (\n            <MapContainer\n              key={index}\n              index={index}\n              primary={index === 1}\n              {...mapProps}\n              mapLayers={splitMaps[index].layers}\n            />\n          ))}\n        </MapsLayout>\n      );\n      return (\n        <StyledPlotContainer className=\"export-map-instance\">\n          <StyledMapContainer ref={this.plottingAreaRef} width={size.width} height={size.height}>\n            {mapContainers}\n          </StyledMapContainer>\n        </StyledPlotContainer>\n      );\n    }\n  }\n\n  PlotContainer.propsTypes = propTypes;\n  return PlotContainer;\n}\n"]}