UNPKG

kepler.gl

Version:

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

261 lines (222 loc) 31.2 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"] = exports.MapInfoPanel = void 0; 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 _getPrototypeOf3 = _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 _styledComponents = _interopRequireDefault(require("styled-components")); var _cloudTile = _interopRequireDefault(require("./cloud-tile")); var _imageModalContainer = _interopRequireDefault(require("./image-modal-container")); var _providerModalContainer = _interopRequireDefault(require("./provider-modal-container")); var _statusPanel = _interopRequireWildcard(require("./status-panel")); var _defaultSettings = require("../../constants/default-settings"); var _styledComponents2 = require("../common/styled-components"); var _imagePreview = _interopRequireDefault(require("../common/image-preview")); var _reactIntl = require("react-intl"); function _templateObject() { var data = (0, _taggedTemplateLiteral2["default"])(["\n .save-map-modal-content {\n min-height: 400px;\n flex-direction: column;\n }\n\n .description {\n width: 300px;\n }\n\n .image-preview-panel {\n width: 300px;\n\n .image-preview {\n padding: 0;\n }\n }\n\n .map-info-panel {\n flex-direction: column;\n }\n\n .save-map-modal-description {\n .modal-section-subtitle {\n margin-left: 6px;\n }\n }\n"]); _templateObject = function _templateObject() { return data; }; return data; } var StyledSaveMapModal = _styledComponents["default"].div.attrs({ className: 'save-map-modal' })(_templateObject()); var nop = function nop() {}; var MapInfoPanel = function MapInfoPanel(_ref) { var _ref$mapInfo = _ref.mapInfo, mapInfo = _ref$mapInfo === void 0 ? { description: '', title: '' } : _ref$mapInfo, characterLimits = _ref.characterLimits, onChangeInput = _ref.onChangeInput; return _react["default"].createElement("div", { className: "selection map-info-panel" }, _react["default"].createElement(_styledComponents2.StyledModalSection, { className: "save-map-modal-name" }, _react["default"].createElement("div", { className: "modal-section-title" }, "Name*"), _react["default"].createElement("div", null, _react["default"].createElement(_styledComponents2.InputLight, { id: "map-title", type: "text", value: mapInfo.title, onChange: function onChange(e) { return onChangeInput('title', e); }, placeholder: "Type map title" }))), _react["default"].createElement(_styledComponents2.StyledModalSection, null, _react["default"].createElement("div", { className: "save-map-modal-description", style: { display: 'flex' } }, _react["default"].createElement("div", { className: "modal-section-title" }, "Description"), _react["default"].createElement("div", { className: "modal-section-subtitle" }, "(optional)")), _react["default"].createElement("div", null, _react["default"].createElement(_styledComponents2.TextAreaLight, { rows: "3", id: "map-description", style: { resize: 'none' }, value: mapInfo.description, onChange: function onChange(e) { return onChangeInput('description', e); }, placeholder: "Type map description" })), _react["default"].createElement(_styledComponents2.StyledModalInputFootnote, { className: "save-map-modal-description__footnote", error: characterLimits.description && mapInfo.description.length > characterLimits.description }, mapInfo.description.length, "/", characterLimits.description || _defaultSettings.MAP_INFO_CHARACTER.description, ' ', "characters"))); }; exports.MapInfoPanel = MapInfoPanel; function SaveMapModalFactory() { var SaveMapModal = /*#__PURE__*/ function (_Component) { (0, _inherits2["default"])(SaveMapModal, _Component); function SaveMapModal() { var _getPrototypeOf2; var _this; (0, _classCallCheck2["default"])(this, SaveMapModal); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = (0, _possibleConstructorReturn2["default"])(this, (_getPrototypeOf2 = (0, _getPrototypeOf3["default"])(SaveMapModal)).call.apply(_getPrototypeOf2, [this].concat(args))); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_onChangeInput", function (key, e) { var value = e.target.value; _this.props.onSetMapInfo((0, _defineProperty2["default"])({}, key, value)); }); return _this; } (0, _createClass2["default"])(SaveMapModal, [{ key: "render", value: function render() { var _this$props = this.props, mapInfo = _this$props.mapInfo, exportImage = _this$props.exportImage, _this$props$character = _this$props.characterLimits, characterLimits = _this$props$character === void 0 ? {} : _this$props$character, cloudProviders = _this$props.cloudProviders, isProviderLoading = _this$props.isProviderLoading, currentProvider = _this$props.currentProvider, providerError = _this$props.providerError, onSetCloudProvider = _this$props.onSetCloudProvider, onUpdateImageSetting = _this$props.onUpdateImageSetting; var provider = currentProvider ? cloudProviders.find(function (p) { return p.name === currentProvider; }) : null; return _react["default"].createElement(_providerModalContainer["default"], { onSetCloudProvider: onSetCloudProvider, cloudProviders: cloudProviders, currentProvider: currentProvider }, _react["default"].createElement(_imageModalContainer["default"], { currentProvider: currentProvider, cloudProviders: cloudProviders, onUpdateImageSetting: onUpdateImageSetting }, _react["default"].createElement(StyledSaveMapModal, null, _react["default"].createElement(_styledComponents2.StyledModalContent, { className: "save-map-modal-content" }, _react["default"].createElement(_styledComponents2.StyledExportSection, { disabled: isProviderLoading }, _react["default"].createElement("div", { className: "description" }, _react["default"].createElement("div", { className: "title" }, _react["default"].createElement(_reactIntl.FormattedMessage, { id: 'modal.saveMap.title' })), _react["default"].createElement("div", { className: "subtitle" }, _react["default"].createElement(_reactIntl.FormattedMessage, { id: 'modal.saveMap.subtitle' }))), _react["default"].createElement("div", { className: "selection" }, cloudProviders.map(function (cloudProvider) { return _react["default"].createElement(_cloudTile["default"], { key: cloudProvider.name, onSelect: function onSelect() { return onSetCloudProvider(cloudProvider.name); }, onSetCloudProvider: onSetCloudProvider, cloudProvider: cloudProvider, isSelected: cloudProvider.name === currentProvider, isConnected: Boolean(cloudProvider.getAccessToken && cloudProvider.getAccessToken()) }); }))), provider && provider.getManagementUrl && _react["default"].createElement(_styledComponents2.StyledExportSection, { style: { margin: '2px 0' } }, _react["default"].createElement("div", { className: "description" }), _react["default"].createElement("div", { className: "selection" }, _react["default"].createElement("a", { key: 1, href: provider.getManagementUrl(), target: "_blank", rel: "noopener noreferrer", style: { textDecoration: 'underline' } }, "Go to your Kepler.gl ", provider.displayName, " page"))), _react["default"].createElement(_styledComponents2.StyledExportSection, null, _react["default"].createElement("div", { className: "description image-preview-panel" }, _react["default"].createElement(_imagePreview["default"], { exportImage: exportImage, width: _defaultSettings.MAP_THUMBNAIL_DIMENSION.width, showDimension: false })), isProviderLoading ? _react["default"].createElement("div", { className: "selection map-saving-animation" }, _react["default"].createElement(_statusPanel.UploadAnimation, { icon: provider && provider.icon })) : _react["default"].createElement(MapInfoPanel, { mapInfo: mapInfo, characterLimits: characterLimits, onChangeInput: this._onChangeInput })), providerError ? _react["default"].createElement(_statusPanel["default"], { isLoading: false, error: providerError, providerIcon: provider && provider.icon }) : null)))); } }]); return SaveMapModal; }(_react.Component); (0, _defineProperty2["default"])(SaveMapModal, "propTypes", { exportImage: _propTypes["default"].object.isRequired, mapInfo: _propTypes["default"].object.isRequired, isProviderLoading: _propTypes["default"].bool.isRequired, thumbWidth: _propTypes["default"].number, thumbHeight: _propTypes["default"].number, characterLimits: _propTypes["default"].object, cloudProviders: _propTypes["default"].arrayOf(_propTypes["default"].object), currentProvider: _propTypes["default"].string, onSetMapInfo: _propTypes["default"].func.isRequired, onSetCloudProvider: _propTypes["default"].func.isRequired, onUpdateImageSetting: _propTypes["default"].func.isRequired }); (0, _defineProperty2["default"])(SaveMapModal, "defaultProps", { characterLimits: _defaultSettings.MAP_INFO_CHARACTER, cloudProviders: [], currentProvider: null, providerError: null, isProviderLoading: false, onSetCloudProvider: nop, onUpdateImageSetting: nop }); return SaveMapModal; } var _default = SaveMapModalFactory; exports["default"] = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/modals/save-map-modal.js"],"names":["StyledSaveMapModal","styled","div","attrs","className","nop","MapInfoPanel","mapInfo","description","title","characterLimits","onChangeInput","e","display","resize","length","MAP_INFO_CHARACTER","SaveMapModalFactory","SaveMapModal","key","value","target","props","onSetMapInfo","exportImage","cloudProviders","isProviderLoading","currentProvider","providerError","onSetCloudProvider","onUpdateImageSetting","provider","find","p","name","map","cloudProvider","Boolean","getAccessToken","getManagementUrl","margin","textDecoration","displayName","MAP_THUMBNAIL_DIMENSION","width","icon","_onChangeInput","Component","PropTypes","object","isRequired","bool","thumbWidth","number","thumbHeight","arrayOf","string","func"],"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\nimport React, {Component} from 'react';\nimport PropTypes from 'prop-types';\nimport styled from 'styled-components';\nimport CloudTile from './cloud-tile';\nimport ImageModalContainer from './image-modal-container';\nimport ProviderModalContainer from './provider-modal-container';\n\nimport StatusPanel, {UploadAnimation} from './status-panel';\n\nimport {MAP_THUMBNAIL_DIMENSION, MAP_INFO_CHARACTER} from 'constants/default-settings';\n\nimport {\n  StyledModalContent,\n  InputLight,\n  TextAreaLight,\n  StyledExportSection,\n  StyledModalSection,\n  StyledModalInputFootnote\n} from 'components/common/styled-components';\nimport ImagePreview from 'components/common/image-preview';\nimport { FormattedMessage } from \"react-intl\";\n\nconst StyledSaveMapModal = styled.div.attrs({\n  className: 'save-map-modal'\n})`\n  .save-map-modal-content {\n    min-height: 400px;\n    flex-direction: column;\n  }\n\n  .description {\n    width: 300px;\n  }\n\n  .image-preview-panel {\n    width: 300px;\n\n    .image-preview {\n      padding: 0;\n    }\n  }\n\n  .map-info-panel {\n    flex-direction: column;\n  }\n\n  .save-map-modal-description {\n    .modal-section-subtitle {\n      margin-left: 6px;\n    }\n  }\n`;\n\nconst nop = () => {};\n\nexport const MapInfoPanel = ({\n  mapInfo = {description: '', title: ''},\n  characterLimits,\n  onChangeInput\n}) => (\n  <div className=\"selection map-info-panel\">\n    <StyledModalSection className=\"save-map-modal-name\">\n      <div className=\"modal-section-title\">Name*</div>\n      <div>\n        <InputLight\n          id=\"map-title\"\n          type=\"text\"\n          value={mapInfo.title}\n          onChange={e => onChangeInput('title', e)}\n          placeholder=\"Type map title\"\n        />\n      </div>\n    </StyledModalSection>\n    <StyledModalSection>\n      <div className=\"save-map-modal-description\" style={{display: 'flex'}}>\n        <div className=\"modal-section-title\">Description</div>\n        <div className=\"modal-section-subtitle\">(optional)</div>\n      </div>\n      <div>\n        <TextAreaLight\n          rows=\"3\"\n          id=\"map-description\"\n          style={{resize: 'none'}}\n          value={mapInfo.description}\n          onChange={e => onChangeInput('description', e)}\n          placeholder=\"Type map description\"\n        />\n      </div>\n      <StyledModalInputFootnote\n        className=\"save-map-modal-description__footnote\"\n        error={\n          characterLimits.description && mapInfo.description.length > characterLimits.description\n        }\n      >\n        {mapInfo.description.length}/{characterLimits.description || MAP_INFO_CHARACTER.description}{' '}\n        characters\n      </StyledModalInputFootnote>\n    </StyledModalSection>\n  </div>\n);\n\nfunction SaveMapModalFactory() {\n  class SaveMapModal extends Component {\n    static propTypes = {\n      exportImage: PropTypes.object.isRequired,\n      mapInfo: PropTypes.object.isRequired,\n      isProviderLoading: PropTypes.bool.isRequired,\n      thumbWidth: PropTypes.number,\n      thumbHeight: PropTypes.number,\n      characterLimits: PropTypes.object,\n      cloudProviders: PropTypes.arrayOf(PropTypes.object),\n      currentProvider: PropTypes.string,\n      onSetMapInfo: PropTypes.func.isRequired,\n      onSetCloudProvider: PropTypes.func.isRequired,\n      onUpdateImageSetting: PropTypes.func.isRequired\n    };\n\n    static defaultProps = {\n      characterLimits: MAP_INFO_CHARACTER,\n      cloudProviders: [],\n      currentProvider: null,\n      providerError: null,\n      isProviderLoading: false,\n      onSetCloudProvider: nop,\n      onUpdateImageSetting: nop\n    };\n\n    _onChangeInput = (key, e) => {\n      const {\n        target: {value}\n      } = e;\n      this.props.onSetMapInfo({[key]: value});\n    };\n\n    render() {\n      const {\n        mapInfo,\n        exportImage,\n        characterLimits = {},\n        cloudProviders,\n        isProviderLoading,\n        currentProvider,\n        providerError,\n        onSetCloudProvider,\n        onUpdateImageSetting\n      } = this.props;\n      const provider = currentProvider\n        ? cloudProviders.find(p => p.name === currentProvider)\n        : null;\n\n      return (\n        <ProviderModalContainer\n          onSetCloudProvider={onSetCloudProvider}\n          cloudProviders={cloudProviders}\n          currentProvider={currentProvider}\n        >\n          <ImageModalContainer\n            currentProvider={currentProvider}\n            cloudProviders={cloudProviders}\n            onUpdateImageSetting={onUpdateImageSetting}\n          >\n            <StyledSaveMapModal>\n              <StyledModalContent className=\"save-map-modal-content\">\n                <StyledExportSection disabled={isProviderLoading}>\n                  <div className=\"description\">\n                    <div className=\"title\"><FormattedMessage id={'modal.saveMap.title'} /></div>\n                    <div className=\"subtitle\"><FormattedMessage id={'modal.saveMap.subtitle'} /></div>\n                  </div>\n                  <div className=\"selection\">\n                    {cloudProviders.map(cloudProvider => (\n                      <CloudTile\n                        key={cloudProvider.name}\n                        onSelect={() => onSetCloudProvider(cloudProvider.name)}\n                        onSetCloudProvider={onSetCloudProvider}\n                        cloudProvider={cloudProvider}\n                        isSelected={cloudProvider.name === currentProvider}\n                        isConnected={Boolean(\n                          cloudProvider.getAccessToken && cloudProvider.getAccessToken()\n                        )}\n                      />\n                    ))}\n                  </div>\n                </StyledExportSection>\n                {provider && provider.getManagementUrl && (\n                  <StyledExportSection style={{margin: '2px 0'}}>\n                    <div className=\"description\" />\n                    <div className=\"selection\">\n                      <a\n                        key={1}\n                        href={provider.getManagementUrl()}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{textDecoration: 'underline'}}\n                      >\n                        Go to your Kepler.gl {provider.displayName} page\n                      </a>\n                    </div>\n                  </StyledExportSection>\n                )}\n                <StyledExportSection>\n                  <div className=\"description image-preview-panel\">\n                    <ImagePreview\n                      exportImage={exportImage}\n                      width={MAP_THUMBNAIL_DIMENSION.width}\n                      showDimension={false}\n                    />\n                  </div>\n                  {isProviderLoading ? (\n                    <div className=\"selection map-saving-animation\">\n                      <UploadAnimation icon={provider && provider.icon} />\n                    </div>\n                  ) : (\n                    <MapInfoPanel\n                      mapInfo={mapInfo}\n                      characterLimits={characterLimits}\n                      onChangeInput={this._onChangeInput}\n                    />\n                  )}\n                </StyledExportSection>\n                {providerError ? (\n                  <StatusPanel\n                    isLoading={false}\n                    error={providerError}\n                    providerIcon={provider && provider.icon}\n                  />\n                ) : null}\n              </StyledModalContent>\n            </StyledSaveMapModal>\n          </ImageModalContainer>\n        </ProviderModalContainer>\n      );\n    }\n  }\n  return SaveMapModal;\n}\n\nexport default SaveMapModalFactory;\n"]}