UNPKG

kepler.gl

Version:

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

242 lines (204 loc) 25.4 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck'); var _classCallCheck3 = _interopRequireDefault(_classCallCheck2); var _createClass2 = require('babel-runtime/helpers/createClass'); var _createClass3 = _interopRequireDefault(_createClass2); var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn'); var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2); var _inherits2 = require('babel-runtime/helpers/inherits'); var _inherits3 = _interopRequireDefault(_inherits2); var _taggedTemplateLiteral2 = require('babel-runtime/helpers/taggedTemplateLiteral'); var _taggedTemplateLiteral3 = _interopRequireDefault(_taggedTemplateLiteral2); var _class, _temp; var _templateObject = (0, _taggedTemplateLiteral3.default)(['\n display: flex;\n flex-direction: column;\n justify-content: space-around;\n width: 250px;\n\n .image-option-section {\n .image-option-section-title {\n font-weight: 500;\n font-size: 14px;\n }\n }\n\n .button-list {\n display: flex;\n flex-direction: row;\n padding: 8px 0px;\n }\n\n input {\n margin-right: 8px;\n }\n'], ['\n display: flex;\n flex-direction: column;\n justify-content: space-around;\n width: 250px;\n\n .image-option-section {\n .image-option-section-title {\n font-weight: 500;\n font-size: 14px;\n }\n }\n\n .button-list {\n display: flex;\n flex-direction: row;\n padding: 8px 0px;\n }\n\n input {\n margin-right: 8px;\n }\n']), _templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n align-items: center;\n display: flex;\n flex-direction: column;\n flex: 1;\n justify-content: center;\n padding: 30px;\n\n .dimension, .instruction {\n padding: 8px 0px;\n }\n\n .preview-image {\n background: #e2e2e2;\n border-radius: 4px;\n box-shadow: 0 8px 16px 0 rgba(0,0,0,0.18);\n width: 100%;\n padding-bottom: ', ';\n position: relative;\n }\n\n .preview-image-placeholder {\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n }\n\n .preview-image-spinner {\n position: absolute;\n left: calc(50% - 25px);\n top: calc(50% - 25px);\n }\n'], ['\n align-items: center;\n display: flex;\n flex-direction: column;\n flex: 1;\n justify-content: center;\n padding: 30px;\n\n .dimension, .instruction {\n padding: 8px 0px;\n }\n\n .preview-image {\n background: #e2e2e2;\n border-radius: 4px;\n box-shadow: 0 8px 16px 0 rgba(0,0,0,0.18);\n width: 100%;\n padding-bottom: ', ';\n position: relative;\n }\n\n .preview-image-placeholder {\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n }\n\n .preview-image-spinner {\n position: absolute;\n left: calc(50% - 25px);\n top: calc(50% - 25px);\n }\n']), _templateObject3 = (0, _taggedTemplateLiteral3.default)(['\n border-radius: 2px;\n border: 1px solid ', ';\n color: ', ';\n cursor: pointer;\n font-weight: 500;\n margin-right: 6px;\n padding: 6px 10px;\n\n :hover {\n color: ', ';\n border: 1px solid ', ';\n }\n'], ['\n border-radius: 2px;\n border: 1px solid ', ';\n color: ', ';\n cursor: pointer;\n font-weight: 500;\n margin-right: 6px;\n padding: 6px 10px;\n\n :hover {\n color: ', ';\n border: 1px solid ', ';\n }\n']); // 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 _react = require('react'); var _react2 = _interopRequireDefault(_react); var _propTypes = require('prop-types'); var _propTypes2 = _interopRequireDefault(_propTypes); var _styledComponents = require('styled-components'); var _styledComponents2 = _interopRequireDefault(_styledComponents); var _exportImageUtils = require('../../utils/export-image-utils'); var _defaultSettings = require('../../constants/default-settings'); var _loadingSpinner = require('../common/loading-spinner'); var _loadingSpinner2 = _interopRequireDefault(_loadingSpinner); var _styledComponents3 = require('../common/styled-components'); var _switch = require('../common/switch'); var _switch2 = _interopRequireDefault(_switch); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var ImageOptionList = _styledComponents2.default.div(_templateObject); var PreviewImageSection = _styledComponents2.default.div(_templateObject2, function (props) { return props.ratio === _defaultSettings.RATIOS.SCREEN ? 100 * props.height / props.width + '%' : props.ratio === _defaultSettings.RATIOS.SIXTEEN_BY_NINE ? '56.25%' : '75%'; }); var Button = _styledComponents2.default.div(_templateObject3, function (props) { return props.selected ? props.theme.primaryBtnBgd : props.theme.selectBorderColorLT; }, function (props) { return props.selected ? props.theme.primaryBtnBgd : props.theme.selectBorderColorLT; }, function (props) { return props.available && props.theme.primaryBtnBgd; }, function (props) { return props.available && props.theme.primaryBtnBgd; }); var ExportImageModal = (_temp = _class = function (_Component) { (0, _inherits3.default)(ExportImageModal, _Component); function ExportImageModal() { (0, _classCallCheck3.default)(this, ExportImageModal); return (0, _possibleConstructorReturn3.default)(this, (ExportImageModal.__proto__ || Object.getPrototypeOf(ExportImageModal)).apply(this, arguments)); } (0, _createClass3.default)(ExportImageModal, [{ key: 'render', value: function render() { var _props = this.props, height = _props.height, legend = _props.legend, ratio = _props.ratio, resolution = _props.resolution, width = _props.width, exporting = _props.exporting, imageDataUri = _props.imageDataUri, onChangeRatio = _props.onChangeRatio, onChangeResolution = _props.onChangeResolution, onToggleLegend = _props.onToggleLegend; var exportImageSize = (0, _exportImageUtils.calculateExportImageSize)({ width: width, height: height, ratio: ratio, resolution: resolution }); return _react2.default.createElement( 'div', { className: 'export-image-modal' }, _react2.default.createElement( _styledComponents3.StyledModalContent, null, _react2.default.createElement( ImageOptionList, null, _react2.default.createElement( 'div', { className: 'image-option-section' }, _react2.default.createElement( 'div', { className: 'image-option-section-title' }, 'Ratio' ), 'Choose the ratio for various usages.', _react2.default.createElement( 'div', { className: 'button-list' }, _defaultSettings.RATIO_OPTIONS.map(function (op) { return _react2.default.createElement( Button, { key: op.id, selected: ratio === op.id, onClick: function onClick() { return onChangeRatio({ ratio: op.id }); } }, op.label ); }) ) ), _react2.default.createElement( 'div', { className: 'image-option-section' }, _react2.default.createElement( 'div', { className: 'image-option-section-title' }, 'Resolution' ), 'High resolution is better for prints.', _react2.default.createElement( 'div', { className: 'button-list' }, _defaultSettings.RESOLUTION_OPTIONS.map(function (op) { return _react2.default.createElement( Button, { key: op.id, selected: resolution === op.id, onClick: function onClick() { return op.available && onChangeResolution({ resolution: op.id }); } }, op.label ); }) ) ), _react2.default.createElement( 'div', { className: 'image-option-section' }, _react2.default.createElement( 'div', { className: 'image-option-section-title' }, 'Map Legend' ), _react2.default.createElement(_switch2.default, { type: 'checkbox', id: 'add-map-legend', checked: legend, label: 'Add legend on map', onChange: onToggleLegend }) ) ), _react2.default.createElement( PreviewImageSection, { ratio: ratio, width: width, height: height }, _react2.default.createElement( 'div', { className: 'dimension' }, exportImageSize.width + ' x ' + exportImageSize.height ), _react2.default.createElement( 'div', { className: 'preview-image' }, exporting ? _react2.default.createElement( 'div', { className: 'preview-image-spinner' }, _react2.default.createElement(_loadingSpinner2.default, null) ) : _react2.default.createElement('img', { className: 'preview-image-placeholder', src: imageDataUri }) ) ) ) ); } }]); return ExportImageModal; }(_react.Component), _class.propTypes = { height: _propTypes2.default.number.isRequired, ratio: _propTypes2.default.string.isRequired, resolution: _propTypes2.default.string.isRequired, width: _propTypes2.default.number.isRequired, exporting: _propTypes2.default.bool.isRequired, imageDataUri: _propTypes2.default.string, // callbacks onChangeRatio: _propTypes2.default.func.isRequired, onChangeResolution: _propTypes2.default.func.isRequired, onToggleLegend: _propTypes2.default.func.isRequired }, _temp); var ExportImageModalFactory = function ExportImageModalFactory() { return ExportImageModal; }; exports.default = ExportImageModalFactory; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/modals/export-image-modal.js"],"names":["ImageOptionList","styled","div","PreviewImageSection","props","ratio","RATIOS","SCREEN","height","width","SIXTEEN_BY_NINE","Button","selected","theme","primaryBtnBgd","selectBorderColorLT","available","ExportImageModal","legend","resolution","exporting","imageDataUri","onChangeRatio","onChangeResolution","onToggleLegend","exportImageSize","RATIO_OPTIONS","map","op","id","label","RESOLUTION_OPTIONS","Component","propTypes","PropTypes","number","isRequired","string","bool","func","ExportImageModalFactory"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;+fAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;;;AACA;;;;AACA;;;;AAEA;;AACA;;AAKA;;;;AACA;;AACA;;;;;;AAEA,IAAMA,kBAAkBC,2BAAOC,GAAzB,iBAAN;;AAwBA,IAAMC,sBAAsBF,2BAAOC,GAA7B,mBAiBgB;AAAA,SAASE,MAAMC,KAAN,KAAgBC,wBAAOC,MAAvB,GACtB,MAAMH,MAAMI,MAAZ,GAAmBJ,MAAMK,KADH,SAExBL,MAAMC,KAAN,KAAgBC,wBAAOI,eAAvB,GAAyC,QAAzC,GAAoD,KAFrC;AAAA,CAjBhB,CAAN;;AAuCA,IAAMC,SAASV,2BAAOC,GAAhB,mBAEgB;AAAA,SAASE,MAAMQ,QAAN,GAAiBR,MAAMS,KAAN,CAAYC,aAA7B,GAA6CV,MAAMS,KAAN,CAAYE,mBAAlE;AAAA,CAFhB,EAGK;AAAA,SAASX,MAAMQ,QAAN,GAAiBR,MAAMS,KAAN,CAAYC,aAA7B,GAA6CV,MAAMS,KAAN,CAAYE,mBAAlE;AAAA,CAHL,EAUO;AAAA,SAASX,MAAMY,SAAN,IAAmBZ,MAAMS,KAAN,CAAYC,aAAxC;AAAA,CAVP,EAWkB;AAAA,SAASV,MAAMY,SAAN,IAAmBZ,MAAMS,KAAN,CAAYC,aAAxC;AAAA,CAXlB,CAAN;;IAeMG,gB;;;;;;;;;;6BAeK;AAAA,mBAaH,KAAKb,KAbF;AAAA,UAELI,MAFK,UAELA,MAFK;AAAA,UAGLU,MAHK,UAGLA,MAHK;AAAA,UAILb,KAJK,UAILA,KAJK;AAAA,UAKLc,UALK,UAKLA,UALK;AAAA,UAMLV,KANK,UAMLA,KANK;AAAA,UAOLW,SAPK,UAOLA,SAPK;AAAA,UAQLC,YARK,UAQLA,YARK;AAAA,UAULC,aAVK,UAULA,aAVK;AAAA,UAWLC,kBAXK,UAWLA,kBAXK;AAAA,UAYLC,cAZK,UAYLA,cAZK;;;AAeP,UAAMC,kBAAkB,gDAAyB;AAC/ChB,oBAD+C,EACxCD,cADwC,EAChCH,YADgC,EACzBc;AADyB,OAAzB,CAAxB;;AAIA,aACE;AAAA;AAAA,UAAK,WAAU,oBAAf;AACE;AAAC,+CAAD;AAAA;AACE;AAAC,2BAAD;AAAA;AACE;AAAA;AAAA,gBAAK,WAAU,sBAAf;AACE;AAAA;AAAA,kBAAK,WAAU,4BAAf;AAAA;AAAA,eADF;AAAA;AAGE;AAAA;AAAA,kBAAK,WAAU,aAAf;AACGO,+CAAcC,GAAd,CAAkB;AAAA,yBACjB;AAAC,0BAAD;AAAA;AACE,2BAAKC,GAAGC,EADV;AAEE,gCAAUxB,UAAUuB,GAAGC,EAFzB;AAGE,+BAAS;AAAA,+BAAMP,cAAc,EAACjB,OAAOuB,GAAGC,EAAX,EAAd,CAAN;AAAA;AAHX;AAKGD,uBAAGE;AALN,mBADiB;AAAA,iBAAlB;AADH;AAHF,aADF;AAgBE;AAAA;AAAA,gBAAK,WAAU,sBAAf;AACE;AAAA;AAAA,kBAAK,WAAU,4BAAf;AAAA;AAAA,eADF;AAAA;AAGE;AAAA;AAAA,kBAAK,WAAU,aAAf;AACGC,oDAAmBJ,GAAnB,CAAuB;AAAA,yBACtB;AAAC,0BAAD;AAAA;AACE,2BAAKC,GAAGC,EADV;AAEE,gCAAUV,eAAeS,GAAGC,EAF9B;AAGE,+BAAS;AAAA,+BAAMD,GAAGZ,SAAH,IAAgBO,mBAAmB,EAACJ,YAAYS,GAAGC,EAAhB,EAAnB,CAAtB;AAAA;AAHX;AAKGD,uBAAGE;AALN,mBADsB;AAAA,iBAAvB;AADH;AAHF,aAhBF;AA+BE;AAAA;AAAA,gBAAK,WAAU,sBAAf;AACE;AAAA;AAAA,kBAAK,WAAU,4BAAf;AAAA;AAAA,eADF;AAEE,4CAAC,gBAAD,IAAQ,MAAK,UAAb;AACQ,oBAAG,gBADX;AAEQ,yBAASZ,MAFjB;AAGQ,uBAAM,mBAHd;AAIQ,0BAAUM,cAJlB;AAFF;AA/BF,WADF;AAyCE;AAAC,+BAAD;AAAA,cAAqB,OAAOnB,KAA5B,EAAmC,OAAOI,KAA1C,EAAiD,QAAQD,MAAzD;AACE;AAAA;AAAA,gBAAK,WAAU,WAAf;AAA+BiB,8BAAgBhB,KAA/C,WAA0DgB,gBAAgBjB;AAA1E,aADF;AAEE;AAAA;AAAA,gBAAK,WAAU,eAAf;AACGY,0BACC;AAAA;AAAA,kBAAK,WAAU,uBAAf;AAAuC,8CAAC,wBAAD;AAAvC,eADD,GAEC,uCAAK,WAAU,2BAAf,EAA2C,KAAKC,YAAhD;AAHJ;AAFF;AAzCF;AADF,OADF;AAuDD;;;EAzF4BW,gB,UAEtBC,S,GAAY;AACjBzB,UAAQ0B,oBAAUC,MAAV,CAAiBC,UADR;AAEjB/B,SAAO6B,oBAAUG,MAAV,CAAiBD,UAFP;AAGjBjB,cAAYe,oBAAUG,MAAV,CAAiBD,UAHZ;AAIjB3B,SAAOyB,oBAAUC,MAAV,CAAiBC,UAJP;AAKjBhB,aAAWc,oBAAUI,IAAV,CAAeF,UALT;AAMjBf,gBAAca,oBAAUG,MANP;AAOjB;AACAf,iBAAeY,oBAAUK,IAAV,CAAeH,UARb;AASjBb,sBAAoBW,oBAAUK,IAAV,CAAeH,UATlB;AAUjBZ,kBAAgBU,oBAAUK,IAAV,CAAeH;AAVd,C;;;AA0FrB,IAAMI,0BAA0B,SAA1BA,uBAA0B;AAAA,SAAMvB,gBAAN;AAAA,CAAhC;kBACeuB,uB","file":"export-image-modal.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 React, {Component} from 'react';\nimport PropTypes from 'prop-types';\nimport styled from 'styled-components';\n\nimport {calculateExportImageSize} from 'utils/export-image-utils';\nimport {\n  RATIO_OPTIONS,\n  RATIOS,\n  RESOLUTION_OPTIONS\n} from 'constants/default-settings';\nimport LoadingSpinner from 'components/common/loading-spinner';\nimport {StyledModalContent} from 'components/common/styled-components';\nimport Switch from 'components/common/switch';\n\nconst ImageOptionList = styled.div`\n  display: flex;\n  flex-direction: column;\n  justify-content: space-around;\n  width: 250px;\n\n  .image-option-section {\n    .image-option-section-title {\n      font-weight: 500;\n      font-size: 14px;\n    }\n  }\n\n  .button-list {\n    display: flex;\n    flex-direction: row;\n    padding: 8px 0px;\n  }\n\n  input {\n    margin-right: 8px;\n  }\n`;\n\nconst PreviewImageSection = styled.div`\n  align-items: center;\n  display: flex;\n  flex-direction: column;\n  flex: 1;\n  justify-content: center;\n  padding: 30px;\n\n  .dimension, .instruction {\n    padding: 8px 0px;\n  }\n\n  .preview-image {\n    background: #e2e2e2;\n    border-radius: 4px;\n    box-shadow: 0 8px 16px 0 rgba(0,0,0,0.18);\n    width: 100%;\n    padding-bottom: ${props => props.ratio === RATIOS.SCREEN ?\n      `${100 * props.height/props.width}%`:\n      (props.ratio === RATIOS.SIXTEEN_BY_NINE ? '56.25%' : '75%')\n    };\n    position: relative;\n  }\n\n  .preview-image-placeholder {\n    position: absolute;\n    top: 0;\n    left: 0;\n    width: 100%;\n    height: 100%;\n  }\n\n  .preview-image-spinner {\n    position: absolute;\n    left: calc(50% - 25px);\n    top: calc(50% - 25px);\n  }\n`;\n\nconst Button = styled.div`\n  border-radius: 2px;\n  border: 1px solid ${props => props.selected ? props.theme.primaryBtnBgd : props.theme.selectBorderColorLT};\n  color: ${props => props.selected ? props.theme.primaryBtnBgd : props.theme.selectBorderColorLT};\n  cursor: pointer;\n  font-weight: 500;\n  margin-right: 6px;\n  padding: 6px 10px;\n\n  :hover {\n    color: ${props => props.available && props.theme.primaryBtnBgd};\n    border: 1px solid ${props => props.available && props.theme.primaryBtnBgd};\n  }\n`;\n\nclass ExportImageModal extends Component {\n\n  static propTypes = {\n    height: PropTypes.number.isRequired,\n    ratio: PropTypes.string.isRequired,\n    resolution: PropTypes.string.isRequired,\n    width: PropTypes.number.isRequired,\n    exporting: PropTypes.bool.isRequired,\n    imageDataUri: PropTypes.string,\n    // callbacks\n    onChangeRatio: PropTypes.func.isRequired,\n    onChangeResolution: PropTypes.func.isRequired,\n    onToggleLegend: PropTypes.func.isRequired\n  };\n\n  render() {\n    const {\n      height,\n      legend,\n      ratio,\n      resolution,\n      width,\n      exporting,\n      imageDataUri,\n      // callbacks:\n      onChangeRatio,\n      onChangeResolution,\n      onToggleLegend\n    } = this.props;\n\n    const exportImageSize = calculateExportImageSize({\n      width, height, ratio, resolution\n    });\n\n    return (\n      <div className=\"export-image-modal\">\n        <StyledModalContent>\n          <ImageOptionList>\n            <div className=\"image-option-section\">\n              <div className=\"image-option-section-title\">Ratio</div>\n              Choose the ratio for various usages.\n              <div className=\"button-list\">\n                {RATIO_OPTIONS.map(op => \n                  <Button\n                    key={op.id}\n                    selected={ratio === op.id}\n                    onClick={() => onChangeRatio({ratio: op.id})}\n                  >\n                    {op.label}\n                  </Button>\n                )}\n              </div>\n            </div>\n            <div className=\"image-option-section\">\n              <div className=\"image-option-section-title\">Resolution</div>\n              High resolution is better for prints.\n              <div className=\"button-list\">\n                {RESOLUTION_OPTIONS.map(op => \n                  <Button\n                    key={op.id}\n                    selected={resolution === op.id}\n                    onClick={() => op.available && onChangeResolution({resolution: op.id})}\n                  >\n                    {op.label}\n                  </Button>\n                )}\n              </div>\n            </div>\n            <div className=\"image-option-section\">\n              <div className=\"image-option-section-title\">Map Legend</div>\n              <Switch type=\"checkbox\"\n                      id=\"add-map-legend\"\n                      checked={legend}\n                      label=\"Add legend on map\"\n                      onChange={onToggleLegend}/>\n            </div>\n          </ImageOptionList>\n          <PreviewImageSection ratio={ratio} width={width} height={height}>\n            <div className=\"dimension\">{`${exportImageSize.width} x ${exportImageSize.height}`}</div>\n            <div className=\"preview-image\">\n              {exporting ?\n                <div className=\"preview-image-spinner\"><LoadingSpinner /></div> :\n                <img className=\"preview-image-placeholder\" src={imageDataUri} />\n              }\n            </div>\n          </PreviewImageSection>\n        </StyledModalContent>\n      </div>\n    );\n  }  \n}\n\nconst ExportImageModalFactory = () => ExportImageModal;\nexport default ExportImageModalFactory;\n"]}