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