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