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
JavaScript
'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":"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","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"]}