UNPKG

kepler.gl.geoiq

Version:

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

317 lines (249 loc) 28.7 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"] = 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 _uploadButton = _interopRequireDefault(require("./upload-button")); var _icons = require("../icons"); var _loadingSpinner = _interopRequireDefault(require("../loading-spinner")); var _utils = require("../../../utils/utils"); function _templateObject8() { var data = (0, _taggedTemplateLiteral2["default"])(["\n position: absolute;\n bottom: 0;\n padding: 10px 30px;\n"]); _templateObject8 = function _templateObject8() { return data; }; return data; } function _templateObject7() { var data = (0, _taggedTemplateLiteral2["default"])(["\n display: flex;\n justify-content: center;\n align-items: center;\n"]); _templateObject7 = function _templateObject7() { return data; }; return data; } function _templateObject6() { var data = (0, _taggedTemplateLiteral2["default"])(["\n .filter-upload__input {\n visibility: hidden;\n height: 0;\n position: absolute;\n }\n\n .file-drop {\n position: relative;\n }\n\n .file-upload__message {\n color: ", ";\n font-size: 14px;\n margin-bottom: 12px;\n }\n"]); _templateObject6 = function _templateObject6() { return data; }; return data; } function _templateObject5() { var data = (0, _taggedTemplateLiteral2["default"])(["\n color: ", ";\n margin-bottom: 60px;\n\n .file-type-row {\n margin-bottom: 26px;\n }\n"]); _templateObject5 = function _templateObject5() { return data; }; return data; } function _templateObject4() { var data = (0, _taggedTemplateLiteral2["default"])(["\n color: ", ";\n font-size: 20px;\n height: 36px;\n"]); _templateObject4 = function _templateObject4() { return data; }; return data; } function _templateObject3() { var data = (0, _taggedTemplateLiteral2["default"])(["\n background-color: white;\n border-radius: 4px;\n border-style: dashed;\n border-width: 1px;\n border-color: ", ";\n height: 414px;\n padding-top: 60px;\n text-align: center;\n width: 100%;\n\n .file-upload-or {\n color: ", ";\n padding-right: 4px;\n }\n"]); _templateObject3 = function _templateObject3() { return data; }; return data; } function _templateObject2() { var data = (0, _taggedTemplateLiteral2["default"])(["\n color: ", ";\n"]); _templateObject2 = function _templateObject2() { return data; }; return data; } function _templateObject() { var data = (0, _taggedTemplateLiteral2["default"])(["\n margin-top: 10px;\n color: ", ";\n"]); _templateObject = function _templateObject() { return data; }; return data; } var FileDrop = typeof document !== 'undefined' ? require('react-file-drop') : null; // File.type is not reliable if the OS does not have a // registered mapping for the extension. // NOTE: Shapefiles must be in a compressed format since // it requires multiple files to be present. var defaultValidFileExt = ['csv', // 'tar.gz', // 'tgz', // 'zip', // 'gpx', // 'kml', 'json', 'geojson']; var MESSAGE = ' Drag & Drop Your File(s) Here'; var CHROME_MSG = '*Chrome user: Limit file size to 250mb, if need to upload larger file, try Safari'; var DISCLAIMER = '*Kepler.gl is a client-side application with no server backend. Data lives only on your machine/browser. ' + 'No information or map data is sent to any server.'; var CONFIG_UPLOAD_MESSAGE = 'Upload data files or upload a saved map via previously exported single Json of both config and data'; var fileIconColor = '#D3D8E0'; var WarningMsg = _styledComponents["default"].span(_templateObject(), function (props) { return props.theme.errorColor; }); var PositiveMsg = _styledComponents["default"].span(_templateObject2(), function (props) { return props.theme.primaryBtnActBgd; }); var StyledFileDrop = _styledComponents["default"].div(_templateObject3(), function (props) { return props.theme.subtextColorLT; }, function (props) { return props.theme.linkBtnColor; }); var MsgWrapper = _styledComponents["default"].div(_templateObject4(), function (props) { return props.theme.modalTitleColor; }); var StyledDragNDropIcon = _styledComponents["default"].div(_templateObject5(), fileIconColor); var StyledFileUpload = _styledComponents["default"].div(_templateObject6(), function (props) { return props.theme.textColorLT; }); var StyledMessage = _styledComponents["default"].div(_templateObject7()); var StyledDisclaimer = StyledMessage.extend(_templateObject8()); var FileUpload = /*#__PURE__*/ function (_Component) { (0, _inherits2["default"])(FileUpload, _Component); function FileUpload() { var _getPrototypeOf2; var _this; (0, _classCallCheck2["default"])(this, FileUpload); 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"])(FileUpload)).call.apply(_getPrototypeOf2, [this].concat(args))); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "state", { dragOver: false, files: [], errorFiles: [] }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_isValidFileType", function (filename) { var validFileExt = _this.props.validFileExt; var fileExt = validFileExt.find(function (ext) { return filename.endsWith(ext); }); return Boolean(fileExt); }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_handleFileDrop", function (files, e) { if (e) { e.stopPropagation(); } var nextState = { files: [], errorFiles: [], dragOver: false }; for (var i = 0; i < files.length; i++) { var file = files[i]; if (file && _this._isValidFileType(file.name)) { nextState.files.push(file); } else { nextState.errorFiles.push(file.name); } } _this.setState(nextState, function () { return nextState.files.length ? _this.props.onFileUpload(nextState.files) : null; }); }); (0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "_toggleDragState", function (newState) { _this.setState({ dragOver: newState }); }); return _this; } (0, _createClass2["default"])(FileUpload, [{ key: "_renderMessage", value: function _renderMessage() { var _this$state = this.state, errorFiles = _this$state.errorFiles, files = _this$state.files; if (errorFiles.length) { return _react["default"].createElement(WarningMsg, null, "File ".concat(errorFiles.join(', '), " is not supported.")); } if (!files.length) { return null; } return _react["default"].createElement(StyledMessage, { className: "file-uploader__message" }, _react["default"].createElement("div", null, "Uploading..."), _react["default"].createElement(PositiveMsg, null, "".concat(files.map(function (f) { return f.name; }).join(' and '), "...")), _react["default"].createElement(_loadingSpinner["default"], { size: 20 })); } }, { key: "render", value: function render() { var _this2 = this; var _this$state2 = this.state, dragOver = _this$state2.dragOver, files = _this$state2.files; var validFileExt = this.props.validFileExt; return _react["default"].createElement(StyledFileUpload, { className: "file-uploader", innerRef: function innerRef(cmp) { return _this2.frame = cmp; } }, _react["default"].createElement("input", { className: "filter-upload__input", type: "file", onChange: this._onChange }), FileDrop ? _react["default"].createElement(FileDrop, { frame: this.frame, targetAlwaysVisible: true, onDragOver: function onDragOver() { return _this2._toggleDragState(true); }, onDragLeave: function onDragLeave() { return _this2._toggleDragState(false); }, onDrop: this._handleFileDrop }, _react["default"].createElement("div", { className: "file-upload__message" }, CONFIG_UPLOAD_MESSAGE), _react["default"].createElement(StyledFileDrop, { dragOver: dragOver }, _react["default"].createElement("div", { style: { opacity: dragOver ? 0.5 : 1 } }, _react["default"].createElement(StyledDragNDropIcon, null, _react["default"].createElement("div", { className: "file-type-row" }, validFileExt.map(function (ext) { return _react["default"].createElement(_icons.FileType, { key: ext, ext: ext, height: "50px", fontSize: "9px" }); })), _react["default"].createElement(_icons.DragNDrop, { height: "44px" })), _react["default"].createElement("div", null, this._renderMessage())), !files.length ? _react["default"].createElement("div", null, _react["default"].createElement(MsgWrapper, null, MESSAGE), _react["default"].createElement("span", { className: "file-upload-or" }, "or"), _react["default"].createElement(_uploadButton["default"], { onUpload: this._handleFileDrop }, "browse your files")) : null, _react["default"].createElement(StyledDisclaimer, null, DISCLAIMER))) : null, _react["default"].createElement(WarningMsg, null, (0, _utils.isChrome)() ? CHROME_MSG : '')); } }]); return FileUpload; }(_react.Component); exports["default"] = FileUpload; (0, _defineProperty2["default"])(FileUpload, "defaultProps", { validFileExt: defaultValidFileExt }); (0, _defineProperty2["default"])(FileUpload, "propTypes", { onFileUpload: _propTypes["default"].func.isRequired, validFileExt: _propTypes["default"].arrayOf(_propTypes["default"].string) }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/common/file-uploader/file-upload.js"],"names":["FileDrop","document","require","defaultValidFileExt","MESSAGE","CHROME_MSG","DISCLAIMER","CONFIG_UPLOAD_MESSAGE","fileIconColor","WarningMsg","styled","span","props","theme","errorColor","PositiveMsg","primaryBtnActBgd","StyledFileDrop","div","subtextColorLT","linkBtnColor","MsgWrapper","modalTitleColor","StyledDragNDropIcon","StyledFileUpload","textColorLT","StyledMessage","StyledDisclaimer","extend","FileUpload","dragOver","files","errorFiles","filename","validFileExt","fileExt","find","ext","endsWith","Boolean","e","stopPropagation","nextState","i","length","file","_isValidFileType","name","push","setState","onFileUpload","newState","state","join","map","f","cmp","frame","_onChange","_toggleDragState","_handleFileDrop","opacity","_renderMessage","Component","PropTypes","func","isRequired","arrayOf","string"],"mappings":"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","sourcesContent":["// Copyright (c) 2019 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 UploadButton from './upload-button';\nimport {FileType, DragNDrop} from 'components/common/icons';\nimport LoadingSpinner from 'components/common/loading-spinner';\nimport {isChrome} from 'utils/utils';\n\nconst FileDrop =\n  typeof document !== 'undefined' ? require('react-file-drop') : null;\n\n// File.type is not reliable if the OS does not have a\n// registered mapping for the extension.\n// NOTE: Shapefiles must be in a compressed format since\n// it requires multiple files to be present.\nconst defaultValidFileExt = [\n  'csv',\n  // 'tar.gz',\n  // 'tgz',\n  // 'zip',\n  // 'gpx',\n  // 'kml',\n  'json',\n  'geojson'\n];\n\nconst MESSAGE = ' Drag & Drop Your File(s) Here';\nconst CHROME_MSG =\n  '*Chrome user: Limit file size to 250mb, if need to upload larger file, try Safari';\nconst DISCLAIMER =\n  '*Kepler.gl is a client-side application with no server backend. Data lives only on your machine/browser. ' +\n  'No information or map data is sent to any server.';\nconst CONFIG_UPLOAD_MESSAGE =\n  'Upload data files or upload a saved map via previously exported single Json of both config and data';\n\nconst fileIconColor = '#D3D8E0';\n\nconst WarningMsg = styled.span`\n  margin-top: 10px;\n  color: ${props => props.theme.errorColor};\n`;\n\nconst PositiveMsg = styled.span`\n  color: ${props => props.theme.primaryBtnActBgd};\n`;\n\nconst StyledFileDrop = styled.div`\n  background-color: white;\n  border-radius: 4px;\n  border-style: dashed;\n  border-width: 1px;\n  border-color: ${props => props.theme.subtextColorLT};\n  height: 414px;\n  padding-top: 60px;\n  text-align: center;\n  width: 100%;\n\n  .file-upload-or {\n    color: ${props => props.theme.linkBtnColor};\n    padding-right: 4px;\n  }\n`;\n\nconst MsgWrapper = styled.div`\n  color: ${props => props.theme.modalTitleColor};\n  font-size: 20px;\n  height: 36px;\n`;\n\nconst StyledDragNDropIcon = styled.div`\n  color: ${fileIconColor};\n  margin-bottom: 60px;\n\n  .file-type-row {\n    margin-bottom: 26px;\n  }\n`;\n\nconst StyledFileUpload = styled.div`\n  .filter-upload__input {\n    visibility: hidden;\n    height: 0;\n    position: absolute;\n  }\n\n  .file-drop {\n    position: relative;\n  }\n\n  .file-upload__message {\n    color: ${props => props.theme.textColorLT};\n    font-size: 14px;\n    margin-bottom: 12px;\n  }\n`;\n\nconst StyledMessage = styled.div`\n  display: flex;\n  justify-content: center;\n  align-items: center;\n`;\n\nconst StyledDisclaimer = StyledMessage.extend`\n  position: absolute;\n  bottom: 0;\n  padding: 10px 30px;\n`;\n\nexport default class FileUpload extends Component {\n  static defaultProps = {\n    validFileExt: defaultValidFileExt\n  };\n\n  static propTypes = {\n    onFileUpload: PropTypes.func.isRequired,\n    validFileExt: PropTypes.arrayOf(PropTypes.string)\n  };\n\n  state = {\n    dragOver: false,\n    files: [],\n    errorFiles: []\n  };\n\n  _isValidFileType = filename => {\n    const {validFileExt} = this.props;\n    const fileExt = validFileExt.find(ext => filename.endsWith(ext));\n\n    return Boolean(fileExt);\n  };\n\n  _handleFileDrop = (files, e) => {\n    if (e) {\n      e.stopPropagation();\n    }\n\n    const nextState = {files: [], errorFiles: [], dragOver: false};\n    for (let i = 0; i < files.length; i++) {\n      const file = files[i];\n\n      if (file && this._isValidFileType(file.name)) {\n        nextState.files.push(file);\n      } else {\n        nextState.errorFiles.push(file.name);\n      }\n    }\n\n    this.setState(nextState, () =>\n      nextState.files.length ? this.props.onFileUpload(nextState.files) : null\n    );\n  };\n\n  _toggleDragState = newState => {\n    this.setState({dragOver: newState});\n  };\n\n  _renderMessage() {\n    const {errorFiles, files} = this.state;\n\n    if (errorFiles.length) {\n      return (\n        <WarningMsg>\n          {`File ${errorFiles.join(', ')} is not supported.`}\n        </WarningMsg>\n      );\n    }\n\n    if (!files.length) {\n      return null;\n    }\n\n    return (\n      <StyledMessage className=\"file-uploader__message\">\n        <div>Uploading...</div>\n        <PositiveMsg>\n          {`${files.map(f => f.name).join(' and ')}...`}\n        </PositiveMsg>\n        <LoadingSpinner size={20} />\n      </StyledMessage>\n    );\n  }\n\n  render() {\n    const {dragOver, files} = this.state;\n    const {validFileExt} = this.props;\n    return (\n      <StyledFileUpload\n        className=\"file-uploader\"\n        innerRef={cmp => (this.frame = cmp)}\n      >\n        <input\n          className=\"filter-upload__input\"\n          type=\"file\"\n          onChange={this._onChange}\n        />\n        {FileDrop ? (\n          <FileDrop\n            frame={this.frame}\n            targetAlwaysVisible\n            onDragOver={() => this._toggleDragState(true)}\n            onDragLeave={() => this._toggleDragState(false)}\n            onDrop={this._handleFileDrop}\n          >\n            <div className=\"file-upload__message\">{CONFIG_UPLOAD_MESSAGE}</div>\n            <StyledFileDrop dragOver={dragOver}>\n              <div style={{opacity: dragOver ? 0.5 : 1}}>\n                <StyledDragNDropIcon>\n                  <div className=\"file-type-row\">\n                    {validFileExt.map(ext => (\n                      <FileType\n                        key={ext}\n                        ext={ext}\n                        height=\"50px\"\n                        fontSize=\"9px\"\n                      />\n                    ))}\n                  </div>\n                  <DragNDrop height=\"44px\" />\n                </StyledDragNDropIcon>\n                <div>{this._renderMessage()}</div>\n              </div>\n              {!files.length ? (\n                <div>\n                  <MsgWrapper>{MESSAGE}</MsgWrapper>\n                  <span className=\"file-upload-or\">or</span>\n                  <UploadButton onUpload={this._handleFileDrop}>\n                    browse your files\n                  </UploadButton>\n                </div>\n              ) : null}\n              <StyledDisclaimer>{DISCLAIMER}</StyledDisclaimer>\n            </StyledFileDrop>\n          </FileDrop>\n        ) : null}\n\n        <WarningMsg>{isChrome() ? CHROME_MSG : ''}</WarningMsg>\n      </StyledFileUpload>\n    );\n  }\n}\n"]}