kepler.gl.geoiq
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
207 lines (165 loc) • 21.5 kB
JavaScript
;
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 _window = _interopRequireDefault(require("global/window"));
var _mapContainer = _interopRequireDefault(require("./map-container"));
var _exportImageUtils = require("../utils/export-image-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"]);
_templateObject = function _templateObject() {
return data;
};
return data;
}
var propTypes = {
width: _propTypes["default"].number.isRequired,
height: _propTypes["default"].number.isRequired,
exportImageSetting: _propTypes["default"].object.isRequired,
mapFields: _propTypes["default"].object.isRequired
};
PlotContainerFactory.deps = [_mapContainer["default"]];
var StyledPlotContainer = _styledComponents["default"].div(_templateObject());
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), "mapStyleSelector", function (props) {
return props.mapFields.mapStyle;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "resolutionSelector", function (props) {
return props.exportImageSetting.resolution;
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "scaledMapStyleSelector", (0, _reselect.createSelector)(_this.mapStyleSelector, _this.resolutionSelector, function (mapStyle, resolution) {
return {
bottomMapStyle: (0, _mapboxGlStyleEditor.scaleMapStyleByResolution)(mapStyle.bottomMapStyle, resolution),
topMapStyle: (0, _mapboxGlStyleEditor.scaleMapStyleByResolution)(mapStyle.topMapStyle, resolution)
};
}));
(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) {
// setting windowDevicePixelRatio to 1
// so that large mapbox base map will load in full
var savedDevicePixelRatio = _window["default"].devicePixelRatio;
_window["default"].devicePixelRatio = 1;
_this.props.startExportingImage();
(0, _exportImageUtils.convertToPng)(_this.plottingAreaRef).then(function (dataUri) {
_this.props.setExportImageDataUri({
dataUri: dataUri
});
_window["default"].devicePixelRatio = savedDevicePixelRatio;
});
}
});
_this._onMapRender = (0, _lodash["default"])(_this._onMapRender, 500);
return _this;
}
(0, _createClass2["default"])(PlotContainer, [{
key: "componentWillMount",
value: function componentWillMount() {
this.props.startExportingImage();
}
}, {
key: "componentWillReceiveProps",
value: function componentWillReceiveProps(newProps) {
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] !== newProps.exportImageSetting[item];
});
if (shouldRetrieveScreenshot) {
this._retrieveNewScreenshot();
}
}
}, {
key: "render",
value: function render() {
var _this3 = this;
var _this$props = this.props,
width = _this$props.width,
height = _this$props.height,
exportImageSetting = _this$props.exportImageSetting,
mapFields = _this$props.mapFields;
var ratio = exportImageSetting.ratio,
resolution = exportImageSetting.resolution,
legend = exportImageSetting.legend;
var exportImageSize = (0, _exportImageUtils.calculateExportImageSize)({
width: width,
height: height,
ratio: ratio,
resolution: resolution
});
var mapProps = _objectSpread({}, mapFields, {
mapStyle: this.scaledMapStyleSelector(this.props),
// override viewport based on export settings
mapState: _objectSpread({}, mapFields.mapState, {}, exportImageSize, {
zoom: mapFields.mapState.zoom + exportImageSize.zoomOffset
}),
mapControls: {
// override map legend visibility
mapLegend: {
show: legend,
active: true
}
},
MapComponent: _reactMapGl.StaticMap
});
return _react["default"].createElement(StyledPlotContainer, {
style: {
position: 'absolute',
top: -9999,
left: -9999
}
}, _react["default"].createElement("div", {
ref: function ref(element) {
_this3.plottingAreaRef = element;
},
style: {
width: exportImageSize.width,
height: exportImageSize.height
}
}, _react["default"].createElement(MapContainer, (0, _extends2["default"])({
index: 0,
onMapRender: this._onMapRender,
isExport: true
}, mapProps))));
}
}]);
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","mapFields","PlotContainerFactory","deps","MapContainerFactory","StyledPlotContainer","styled","div","MapContainer","PlotContainer","props","mapStyle","resolution","mapStyleSelector","resolutionSelector","bottomMapStyle","topMapStyle","map","isStyleLoaded","_retrieveNewScreenshot","plottingAreaRef","savedDevicePixelRatio","window","devicePixelRatio","startExportingImage","then","dataUri","setExportImageDataUri","_onMapRender","newProps","checks","shouldRetrieveScreenshot","some","item","ratio","legend","exportImageSize","mapProps","scaledMapStyleSelector","mapState","zoom","zoomOffset","mapControls","mapLegend","show","active","MapComponent","StaticMap","position","top","left","element","Component","propsTypes"],"mappings":"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","sourcesContent":["// Copyright (c) 2019 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} 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 window from 'global/window';\n\nimport MapContainerFactory from './map-container';\nimport {calculateExportImageSize, convertToPng} from 'utils/export-image-utils';\nimport {scaleMapStyleByResolution} from 'utils/map-style-utils/mapbox-gl-style-editor';\nconst propTypes = {\n  width: PropTypes.number.isRequired,\n  height: PropTypes.number.isRequired,\n  exportImageSetting: PropTypes.object.isRequired,\n  mapFields: PropTypes.object.isRequired\n};\n\nPlotContainerFactory.deps = [MapContainerFactory];\n\nconst StyledPlotContainer = styled.div`\n  .mapboxgl-ctrl-bottom-left,\n  .mapboxgl-ctrl-bottom-right {\n    display: none;\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    componentWillMount() {\n      this.props.startExportingImage();\n    }\n\n    componentWillReceiveProps(newProps) {\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 =>\n          this.props.exportImageSetting[item] !==\n          newProps.exportImageSetting[item]\n      );\n      if (shouldRetrieveScreenshot) {\n        this._retrieveNewScreenshot();\n      }\n    }\n\n    mapStyleSelector = props => props.mapFields.mapStyle;\n    resolutionSelector = props => props.exportImageSetting.resolution;\n    scaledMapStyleSelector = createSelector(\n      this.mapStyleSelector,\n      this.resolutionSelector,\n      (mapStyle, resolution) => ({\n        bottomMapStyle: scaleMapStyleByResolution(\n          mapStyle.bottomMapStyle,\n          resolution\n        ),\n        topMapStyle: scaleMapStyleByResolution(mapStyle.topMapStyle, resolution)\n      })\n    );\n\n    _onMapRender = map => {\n      if (map.isStyleLoaded()) {\n        this._retrieveNewScreenshot();\n      }\n    };\n\n    _retrieveNewScreenshot = () => {\n      if (this.plottingAreaRef) {\n      // setting windowDevicePixelRatio to 1\n      // so that large mapbox base map will load in full\n        const savedDevicePixelRatio = window.devicePixelRatio;\n        window.devicePixelRatio = 1;\n\n        this.props.startExportingImage();\n        convertToPng(this.plottingAreaRef).then(dataUri => {\n          this.props.setExportImageDataUri({dataUri});\n          window.devicePixelRatio = savedDevicePixelRatio;\n        });\n      }\n    };\n\n    render() {\n      const {width, height, exportImageSetting, mapFields} = this.props;\n      const {ratio, resolution, legend} = exportImageSetting;\n      const exportImageSize = calculateExportImageSize({\n        width,\n        height,\n        ratio,\n        resolution\n      });\n\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          ...exportImageSize,\n          zoom: mapFields.mapState.zoom + exportImageSize.zoomOffset\n        },\n        mapControls: {\n          // override map legend visibility\n          mapLegend: {\n            show: legend,\n            active: true\n          }\n        },\n        MapComponent: StaticMap\n      };\n\n      return (\n        <StyledPlotContainer\n          style={{position: 'absolute', top: -9999, left: -9999}}\n        >\n          <div\n            ref={element => {\n              this.plottingAreaRef = element;\n            }}\n            style={{\n              width: exportImageSize.width,\n              height: exportImageSize.height\n            }}\n          >\n            <MapContainer\n              index={0}\n              onMapRender={this._onMapRender}\n              isExport\n              {...mapProps}\n            />\n          </div>\n        </StyledPlotContainer>\n      );\n    }\n  }\n\n  PlotContainer.propsTypes = propTypes;\n  return PlotContainer;\n}\n"]}