UNPKG

kepler.gl

Version:

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

327 lines (271 loc) 29.5 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = undefined; 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, _temp2; var _templateObject = (0, _taggedTemplateLiteral3.default)(['\n margin-top: 10px;\n color: ', ';\n'], ['\n margin-top: 10px;\n color: ', ';\n']), _templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n color: ', ';\n'], ['\n color: ', ';\n']), _templateObject3 = (0, _taggedTemplateLiteral3.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'], ['\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']), _templateObject4 = (0, _taggedTemplateLiteral3.default)(['\n color: ', ';\n font-size: 20px;\n height: 36px;\n'], ['\n color: ', ';\n font-size: 20px;\n height: 36px;\n']), _templateObject5 = (0, _taggedTemplateLiteral3.default)(['\n color: ', ';\n margin-bottom: 60px;\n\n .file-type-row {\n margin-bottom: 26px;\n }\n'], ['\n color: ', ';\n margin-bottom: 60px;\n\n .file-type-row {\n margin-bottom: 26px;\n }\n']), _templateObject6 = (0, _taggedTemplateLiteral3.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'], ['\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']), _templateObject7 = (0, _taggedTemplateLiteral3.default)(['\n display: flex;\n justify-content: center;\n align-items: center;\n'], ['\n display: flex;\n justify-content: center;\n align-items: center;\n']), _templateObject8 = (0, _taggedTemplateLiteral3.default)(['\n position: absolute;\n bottom: 0;\n padding: 10px 30px;\n'], ['\n position: absolute;\n bottom: 0;\n padding: 10px 30px;\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 _uploadButton = require('./upload-button'); var _uploadButton2 = _interopRequireDefault(_uploadButton); var _icons = require('../icons'); var _loadingSpinner = require('../loading-spinner'); var _loadingSpinner2 = _interopRequireDefault(_loadingSpinner); var _utils = require('../../../utils/utils'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } 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 = _styledComponents2.default.span(_templateObject, function (props) { return props.theme.errorColor; }); var PositiveMsg = _styledComponents2.default.span(_templateObject2, function (props) { return props.theme.primaryBtnActBgd; }); var StyledFileDrop = _styledComponents2.default.div(_templateObject3, function (props) { return props.theme.subtextColorLT; }, function (props) { return props.theme.linkBtnColor; }); var MsgWrapper = _styledComponents2.default.div(_templateObject4, function (props) { return props.theme.modalTitleColor; }); var StyledDragNDropIcon = _styledComponents2.default.div(_templateObject5, fileIconColor); var StyledFileUpload = _styledComponents2.default.div(_templateObject6, function (props) { return props.theme.textColorLT; }); var StyledMessage = _styledComponents2.default.div(_templateObject7); var StyledDisclaimer = StyledMessage.extend(_templateObject8); var FileUpload = (_temp2 = _class = function (_Component) { (0, _inherits3.default)(FileUpload, _Component); function FileUpload() { var _ref; var _temp, _this, _ret; (0, _classCallCheck3.default)(this, FileUpload); for (var _len = arguments.length, args = Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } return _ret = (_temp = (_this = (0, _possibleConstructorReturn3.default)(this, (_ref = FileUpload.__proto__ || Object.getPrototypeOf(FileUpload)).call.apply(_ref, [this].concat(args))), _this), _this.state = { dragOver: false, files: [], errorFiles: [] }, _this._isValidFileType = function (filename) { var validFileExt = _this.props.validFileExt; var fileExt = validFileExt.find(function (ext) { return filename.endsWith(ext); }); return Boolean(fileExt); }, _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; }); }, _this._toggleDragState = function (newState) { _this.setState({ dragOver: newState }); }, _temp), (0, _possibleConstructorReturn3.default)(_this, _ret); } (0, _createClass3.default)(FileUpload, [{ key: '_renderMessage', value: function _renderMessage() { var _state = this.state, errorFiles = _state.errorFiles, files = _state.files; if (errorFiles.length) { return _react2.default.createElement( WarningMsg, null, 'File ' + errorFiles.join(', ') + ' is not supported.' ); } if (!files.length) { return null; } return _react2.default.createElement( StyledMessage, { className: 'file-uploader__message' }, _react2.default.createElement( 'div', null, 'Uploading...' ), _react2.default.createElement( PositiveMsg, null, files.map(function (f) { return f.name; }).join(' and ') + '...' ), _react2.default.createElement(_loadingSpinner2.default, { size: 20 }) ); } }, { key: 'render', value: function render() { var _this2 = this; var _state2 = this.state, dragOver = _state2.dragOver, files = _state2.files; var validFileExt = this.props.validFileExt; return _react2.default.createElement( StyledFileUpload, { className: 'file-uploader', innerRef: function innerRef(cmp) { return _this2.frame = cmp; } }, _react2.default.createElement('input', { className: 'filter-upload__input', type: 'file', onChange: this._onChange }), FileDrop ? _react2.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 }, _react2.default.createElement( 'div', { className: 'file-upload__message' }, CONFIG_UPLOAD_MESSAGE ), _react2.default.createElement( StyledFileDrop, { dragOver: dragOver }, _react2.default.createElement( 'div', { style: { opacity: dragOver ? 0.5 : 1 } }, _react2.default.createElement( StyledDragNDropIcon, null, _react2.default.createElement( 'div', { className: 'file-type-row' }, validFileExt.map(function (ext) { return _react2.default.createElement(_icons.FileType, { key: ext, ext: ext, height: '50px', fontSize: '9px' }); }) ), _react2.default.createElement(_icons.DragNDrop, { height: '44px' }) ), _react2.default.createElement( 'div', null, this._renderMessage() ) ), !files.length ? _react2.default.createElement( 'div', null, _react2.default.createElement( MsgWrapper, null, MESSAGE ), _react2.default.createElement( 'span', { className: 'file-upload-or' }, 'or' ), _react2.default.createElement( _uploadButton2.default, { onUpload: this._handleFileDrop }, 'browse your files' ) ) : null, _react2.default.createElement( StyledDisclaimer, null, DISCLAIMER ) ) ) : null, _react2.default.createElement( WarningMsg, null, (0, _utils.isChrome)() ? CHROME_MSG : '' ) ); } }]); return FileUpload; }(_react.Component), _class.defaultProps = { validFileExt: defaultValidFileExt }, _class.propTypes = { onFileUpload: _propTypes2.default.func.isRequired, validFileExt: _propTypes2.default.arrayOf(_propTypes2.default.string) }, _temp2); exports.default = FileUpload; //# 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","state","dragOver","files","errorFiles","_isValidFileType","validFileExt","fileExt","find","filename","endsWith","ext","Boolean","_handleFileDrop","e","stopPropagation","nextState","i","length","file","name","push","setState","onFileUpload","_toggleDragState","newState","join","map","f","frame","cmp","_onChange","opacity","_renderMessage","Component","defaultProps","propTypes","PropTypes","func","isRequired","arrayOf","string"],"mappings":"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","file":"file-upload.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 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 = '*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 = '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(\n      nextState,\n      () =>\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 key={ext} ext={ext} height=\"50px\" fontSize=\"9px\"/>\n                    ))}\n                  </div>\n                  <DragNDrop height=\"44px\" />\n                </StyledDragNDropIcon>\n                <div>{this._renderMessage()}</div>\n              </div>\n              {!files.length ? <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> : null}\n              <StyledDisclaimer>{DISCLAIMER}</StyledDisclaimer>\n            </StyledFileDrop>\n          </FileDrop>\n        ) : null}\n\n        <WarningMsg>{isChrome() ? CHROME_MSG : ''}</WarningMsg>\n      </StyledFileUpload>\n    );\n  }\n}\n"]}