UNPKG

kepler.gl

Version:

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

207 lines (182 loc) 27.4 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.MapInfoPanel = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral")); var _react = _interopRequireDefault(require("react")); 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 _localization = require("../../localization"); var _templateObject; /** @typedef {import('./save-map-modal').SaveMapModalProps} SaveMapModalProps */ var StyledSaveMapModal = _styledComponents["default"].div.attrs({ className: 'save-map-modal' })(_templateObject || (_templateObject = (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"]))); 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 /*#__PURE__*/_react["default"].createElement("div", { className: "selection map-info-panel" }, /*#__PURE__*/_react["default"].createElement(_styledComponents2.StyledModalSection, { className: "save-map-modal-name" }, /*#__PURE__*/_react["default"].createElement("div", { className: "modal-section-title" }, "Name*"), /*#__PURE__*/_react["default"].createElement("div", null, /*#__PURE__*/_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" }))), /*#__PURE__*/_react["default"].createElement(_styledComponents2.StyledModalSection, null, /*#__PURE__*/_react["default"].createElement("div", { className: "save-map-modal-description", style: { display: 'flex' } }, /*#__PURE__*/_react["default"].createElement("div", { className: "modal-section-title" }, "Description"), /*#__PURE__*/_react["default"].createElement("div", { className: "modal-section-subtitle" }, "(optional)")), /*#__PURE__*/_react["default"].createElement("div", null, /*#__PURE__*/_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" })), /*#__PURE__*/_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() { /** * @type {React.FunctionComponent<SaveMapModalProps>} */ var SaveMapModal = function SaveMapModal(_ref2) { var mapInfo = _ref2.mapInfo, exportImage = _ref2.exportImage, _ref2$characterLimits = _ref2.characterLimits, characterLimits = _ref2$characterLimits === void 0 ? {} : _ref2$characterLimits, cloudProviders = _ref2.cloudProviders, isProviderLoading = _ref2.isProviderLoading, currentProvider = _ref2.currentProvider, providerError = _ref2.providerError, onSetCloudProvider = _ref2.onSetCloudProvider, onUpdateImageSetting = _ref2.onUpdateImageSetting, cleanupExportImage = _ref2.cleanupExportImage, onSetMapInfo = _ref2.onSetMapInfo; var onChangeInput = function onChangeInput(key, _ref3) { var value = _ref3.target.value; onSetMapInfo((0, _defineProperty2["default"])({}, key, value)); }; var provider = currentProvider ? cloudProviders.find(function (p) { return p.name === currentProvider; }) : null; return /*#__PURE__*/_react["default"].createElement(_providerModalContainer["default"], { onSetCloudProvider: onSetCloudProvider, cloudProviders: cloudProviders, currentProvider: currentProvider }, /*#__PURE__*/_react["default"].createElement(_imageModalContainer["default"], { currentProvider: currentProvider, cloudProviders: cloudProviders, onUpdateImageSetting: onUpdateImageSetting, cleanupExportImage: cleanupExportImage }, /*#__PURE__*/_react["default"].createElement(StyledSaveMapModal, null, /*#__PURE__*/_react["default"].createElement(_styledComponents2.StyledModalContent, { className: "save-map-modal-content" }, /*#__PURE__*/_react["default"].createElement(_styledComponents2.StyledExportSection, { disabled: isProviderLoading }, /*#__PURE__*/_react["default"].createElement("div", { className: "description" }, /*#__PURE__*/_react["default"].createElement("div", { className: "title" }, /*#__PURE__*/_react["default"].createElement(_localization.FormattedMessage, { id: 'modal.saveMap.title' })), /*#__PURE__*/_react["default"].createElement("div", { className: "subtitle" }, /*#__PURE__*/_react["default"].createElement(_localization.FormattedMessage, { id: 'modal.saveMap.subtitle' }))), /*#__PURE__*/_react["default"].createElement("div", { className: "selection" }, cloudProviders.map(function (cloudProvider) { return /*#__PURE__*/_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 && /*#__PURE__*/_react["default"].createElement(_styledComponents2.StyledExportSection, { style: { margin: '2px 0' } }, /*#__PURE__*/_react["default"].createElement("div", { className: "description" }), /*#__PURE__*/_react["default"].createElement("div", { className: "selection" }, /*#__PURE__*/_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"))), /*#__PURE__*/_react["default"].createElement(_styledComponents2.StyledExportSection, null, /*#__PURE__*/_react["default"].createElement("div", { className: "description image-preview-panel" }, /*#__PURE__*/_react["default"].createElement(_imagePreview["default"], { exportImage: exportImage, width: _defaultSettings.MAP_THUMBNAIL_DIMENSION.width, showDimension: false })), isProviderLoading ? /*#__PURE__*/_react["default"].createElement("div", { className: "selection map-saving-animation" }, /*#__PURE__*/_react["default"].createElement(_statusPanel.UploadAnimation, { icon: provider && provider.icon })) : /*#__PURE__*/_react["default"].createElement(MapInfoPanel, { mapInfo: mapInfo, characterLimits: characterLimits, onChangeInput: onChangeInput })), providerError ? /*#__PURE__*/_react["default"].createElement(_statusPanel["default"], { isLoading: false, error: providerError, providerIcon: provider && provider.icon }) : null)))); }; SaveMapModal.defaultProps = { characterLimits: _defaultSettings.MAP_INFO_CHARACTER, cloudProviders: [], 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","exportImage","cloudProviders","isProviderLoading","currentProvider","providerError","onSetCloudProvider","onUpdateImageSetting","cleanupExportImage","onSetMapInfo","key","value","target","provider","find","p","name","map","cloudProvider","Boolean","getAccessToken","getManagementUrl","margin","textDecoration","displayName","MAP_THUMBNAIL_DIMENSION","width","icon","defaultProps"],"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\nimport React from 'react';\nimport styled from 'styled-components';\nimport CloudTile from './cloud-tile';\nimport ImageModalContainer from './image-modal-container';\nimport ProviderModalContainer from './provider-modal-container';\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 'localization';\n\n/** @typedef {import('./save-map-modal').SaveMapModalProps} SaveMapModalProps */\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  /**\n   * @type {React.FunctionComponent<SaveMapModalProps>}\n   */\n  const SaveMapModal = ({\n    mapInfo,\n    exportImage,\n    characterLimits = {},\n    cloudProviders,\n    isProviderLoading,\n    currentProvider,\n    providerError,\n    onSetCloudProvider,\n    onUpdateImageSetting,\n    cleanupExportImage,\n    onSetMapInfo\n  }) => {\n    const onChangeInput = (key, {target: {value}}) => {\n      onSetMapInfo({[key]: value});\n    };\n    const provider = currentProvider ? cloudProviders.find(p => p.name === currentProvider) : 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          cleanupExportImage={cleanupExportImage}\n        >\n          <StyledSaveMapModal>\n            <StyledModalContent className=\"save-map-modal-content\">\n              <StyledExportSection disabled={isProviderLoading}>\n                <div className=\"description\">\n                  <div className=\"title\">\n                    <FormattedMessage id={'modal.saveMap.title'} />\n                  </div>\n                  <div className=\"subtitle\">\n                    <FormattedMessage id={'modal.saveMap.subtitle'} />\n                  </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={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  SaveMapModal.defaultProps = {\n    characterLimits: MAP_INFO_CHARACTER,\n    cloudProviders: [],\n    providerError: null,\n    isProviderLoading: false,\n    onSetCloudProvider: nop,\n    onUpdateImageSetting: nop\n  };\n\n  return SaveMapModal;\n}\n\nexport default SaveMapModalFactory;\n"]}