UNPKG

kepler.gl

Version:

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

331 lines (279 loc) 32.8 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); var _extends2 = require('babel-runtime/helpers/extends'); var _extends3 = _interopRequireDefault(_extends2); 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 display: flex;\n flex-direction: column;\n justify-content: space-around;\n font-size: 12px;\n .modal-section {\n margin-bottom: 32px;\n }\n .modal-section:first-child {\n margin-top: 24px;\n }\n \n .modal-section {\n .modal-section-title {\n font-weight: 500;\n }\n .modal-section-subtitle {\n color: ', ';\n }\n \n input {\n margin-top: 8px;\n }\n }\n\n input {\n margin-right: 8px;\n }\n'], ['\n display: flex;\n flex-direction: column;\n justify-content: space-around;\n font-size: 12px;\n .modal-section {\n margin-bottom: 32px;\n }\n .modal-section:first-child {\n margin-top: 24px;\n }\n \n .modal-section {\n .modal-section-title {\n font-weight: 500;\n }\n .modal-section-subtitle {\n color: ', ';\n }\n \n input {\n margin-top: 8px;\n }\n }\n\n input {\n margin-right: 8px;\n }\n']), _templateObject2 = (0, _taggedTemplateLiteral3.default)(['\n align-items: center;\n display: flex;\n flex-direction: column;\n justify-content: center;\n margin-left: 116px;\n flex-shrink: 0;\n width: ', 'px;\n\n .preview-title {\n font-weight: 500;\n font-size: 10px;\n padding: 8px 0px;\n }\n \n .preview-title.error {\n color: ', ';\n }\n\n .preview-image {\n background: ', ';\n border-radius: 4px;\n box-shadow: 0 8px 16px 0 rgba(0,0,0,0.18);\n width: ', 'px;\n height: ', 'px;\n position: relative;\n }\n\n .preview-image-placeholder {\n position: absolute;\n top: 0;\n left: 0;\n }\n\n .preview-image-spinner {\n position: absolute;\n left: calc(50% - 25px);\n top: calc(50% - 25px);\n }\n'], ['\n align-items: center;\n display: flex;\n flex-direction: column;\n justify-content: center;\n margin-left: 116px;\n flex-shrink: 0;\n width: ', 'px;\n\n .preview-title {\n font-weight: 500;\n font-size: 10px;\n padding: 8px 0px;\n }\n \n .preview-title.error {\n color: ', ';\n }\n\n .preview-image {\n background: ', ';\n border-radius: 4px;\n box-shadow: 0 8px 16px 0 rgba(0,0,0,0.18);\n width: ', 'px;\n height: ', 'px;\n position: relative;\n }\n\n .preview-image-placeholder {\n position: absolute;\n top: 0;\n left: 0;\n }\n\n .preview-image-spinner {\n position: absolute;\n left: calc(50% - 25px);\n top: calc(50% - 25px);\n }\n']), _templateObject3 = (0, _taggedTemplateLiteral3.default)(['\n font-weight: 500;\n \n :hover {\n cursor: pointer;\n }\n'], ['\n font-weight: 500;\n \n :hover {\n cursor: pointer;\n }\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. // Utils var _react = require('react'); var _react2 = _interopRequireDefault(_react); var _propTypes = require('prop-types'); var _propTypes2 = _interopRequireDefault(_propTypes); var _classnames = require('classnames'); var _classnames2 = _interopRequireDefault(_classnames); var _styledComponents = require('styled-components'); var _styledComponents2 = _interopRequireDefault(_styledComponents); var _reactMapGl = require('react-map-gl'); var _reactMapGl2 = _interopRequireDefault(_reactMapGl); var _reactDom = require('react-dom'); var _styledComponents3 = require('../common/styled-components'); var _mapboxUtils = require('../../utils/map-style-utils/mapbox-utils'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } var MapH = 190; var MapW = 264; var ErrorMsg = { styleError: 'Failed to load map style, make sure it is published. For private style, paste in your access token.' }; var InstructionPanel = _styledComponents2.default.div(_templateObject, function (props) { return props.theme.subtextColorLT; }); var PreviewMap = _styledComponents2.default.div(_templateObject2, MapW, function (props) { return props.theme.errorColor; }, function (props) { return props.theme.modalImagePlaceHolder; }, MapW, MapH); var InlineLink = _styledComponents2.default.a(_templateObject3); var AddMapStyleModal = (_temp2 = _class = function (_Component) { (0, _inherits3.default)(AddMapStyleModal, _Component); function AddMapStyleModal() { var _ref; var _temp, _this, _ret; (0, _classCallCheck3.default)(this, AddMapStyleModal); 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 = AddMapStyleModal.__proto__ || Object.getPrototypeOf(AddMapStyleModal)).call.apply(_ref, [this].concat(args))), _this), _this.state = { reRenderKey: 0 }, _this.loadMapStyleJson = function (style) { _this.props.loadCustomMapStyle({ style: style, error: false }); }, _this.loadMapStyleIcon = function () { if (_this.mapRef) { var canvas = (0, _reactDom.findDOMNode)(_this.mapRef).querySelector('.mapboxgl-canvas'); var dataUri = canvas.toDataURL(); _this.props.loadCustomMapStyle({ icon: dataUri }); } }, _this.loadMaoStyleError = function () { _this.props.loadCustomMapStyle({ error: true }); }, _temp), (0, _possibleConstructorReturn3.default)(_this, _ret); } (0, _createClass3.default)(AddMapStyleModal, [{ key: 'componentWillReceiveProps', value: function componentWillReceiveProps(nextProps) { if (this.props.inputStyle.accessToken !== nextProps.inputStyle.accessToken) { // toke has changed // ReactMapGl doesn't re-create map when token has changed // here we force the map to update this.setState({ reRenderKey: this.state.reRenderKey + 1 }); } } }, { key: 'componentDidUpdate', value: function componentDidUpdate() { var _this2 = this; var map = this.mapRef && this.mapRef.getMap(); if (map && this._map !== map) { this._map = map; map.on('style.load', function () { var style = map.getStyle(); _this2.loadMapStyleJson(style); }); map.on('render', function () { if (map.isStyleLoaded()) { _this2.loadMapStyleIcon(); } }); map.on('error', function () { _this2.loadMaoStyleError(); }); } } }, { key: 'render', value: function render() { var _this3 = this; var _props = this.props, inputStyle = _props.inputStyle, mapState = _props.mapState; var mapProps = (0, _extends3.default)({}, mapState, { preserveDrawingBuffer: true, mapboxApiAccessToken: inputStyle.accessToken || this.props.mapboxApiAccessToken, transformRequest: _mapboxUtils.transformRequest }); return _react2.default.createElement( 'div', { className: 'add-map-style-modal' }, _react2.default.createElement( _styledComponents3.StyledModalContent, null, _react2.default.createElement( InstructionPanel, null, _react2.default.createElement( 'div', { className: 'modal-section' }, _react2.default.createElement( 'div', { className: 'modal-section-title' }, '1. Publish your style at mapbox or provide access token' ), _react2.default.createElement( 'div', { className: 'modal-section-subtitle' }, 'You can create your own map style at', _react2.default.createElement( InlineLink, { target: '_blank', href: 'https://www.mapbox.com/studio/styles/' }, ' mapbox' ), ' and', _react2.default.createElement( InlineLink, { target: '_blank', href: 'https://www.mapbox.com/help/studio-manual-publish/' }, ' publish' ), ' it.' ), _react2.default.createElement( 'div', { className: 'modal-section-subtitle' }, 'To use private style, paste your', _react2.default.createElement( InlineLink, { target: '_blank', href: 'https://www.mapbox.com/help/how-access-tokens-work/' }, ' access token' ), ' here. *kepler.gl is a client-side application, data stays in your browser..' ), _react2.default.createElement(_styledComponents3.InputLight, { type: 'text', value: inputStyle.accessToken || '', onChange: function onChange(_ref2) { var value = _ref2.target.value; return _this3.props.inputMapStyle((0, _extends3.default)({}, inputStyle, { accessToken: value })); }, placeholder: 'e.g. pk.abcdefg.xxxxxx' }) ), _react2.default.createElement( 'div', { className: 'modal-section' }, _react2.default.createElement( 'div', { className: 'modal-section-title' }, '2. Paste style url' ), _react2.default.createElement( 'div', { className: 'modal-section-subtitle' }, 'What is a', _react2.default.createElement( InlineLink, { target: '_blank', href: 'https://www.mapbox.com/help/studio-manual-publish/#style-url' }, ' style URL' ) ), _react2.default.createElement(_styledComponents3.InputLight, { type: 'text', value: inputStyle.url || '', onChange: function onChange(_ref3) { var value = _ref3.target.value; return _this3.props.inputMapStyle((0, _extends3.default)({}, inputStyle, { url: value })); }, placeholder: 'e.g. mapbox://styles/uberdataviz/abcdefghijklmnopq' }) ), _react2.default.createElement( 'div', { className: 'modal-section' }, _react2.default.createElement( 'div', { className: 'modal-section-title' }, '3. Name your style' ), _react2.default.createElement(_styledComponents3.InputLight, { type: 'text', value: inputStyle.label || '', onChange: function onChange(_ref4) { var value = _ref4.target.value; return _this3.props.inputMapStyle((0, _extends3.default)({}, inputStyle, { label: value })); } }) ) ), _react2.default.createElement( PreviewMap, null, _react2.default.createElement( 'div', { className: (0, _classnames2.default)('preview-title', { error: inputStyle.error }) }, inputStyle.error ? ErrorMsg.styleError : inputStyle.style && inputStyle.style.name || '' ), _react2.default.createElement( 'div', { className: 'preview-image' }, !inputStyle.isValid ? _react2.default.createElement('div', { className: 'preview-image-spinner' }) : _react2.default.createElement( _styledComponents3.StyledMapContainer, null, _react2.default.createElement(_reactMapGl2.default, (0, _extends3.default)({}, mapProps, { ref: function ref(el) { _this3.mapRef = el; }, key: this.state.reRenderKey, width: MapW, height: MapH, mapStyle: inputStyle.url })) ) ) ) ) ); } }]); return AddMapStyleModal; }(_react.Component), _class.propTypes = { mapState: _propTypes2.default.object.isRequired, inputMapStyle: _propTypes2.default.func.isRequired, loadCustomMapStyle: _propTypes2.default.func.isRequired, inputStyle: _propTypes2.default.object.isRequired }, _temp2); var AddMapStyleModalFactory = function AddMapStyleModalFactory() { return AddMapStyleModal; }; exports.default = AddMapStyleModalFactory; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/modals/add-map-style-modal.js"],"names":["MapH","MapW","ErrorMsg","styleError","InstructionPanel","styled","div","props","theme","subtextColorLT","PreviewMap","errorColor","modalImagePlaceHolder","InlineLink","a","AddMapStyleModal","state","reRenderKey","loadMapStyleJson","style","loadCustomMapStyle","error","loadMapStyleIcon","mapRef","canvas","querySelector","dataUri","toDataURL","icon","loadMaoStyleError","nextProps","inputStyle","accessToken","setState","map","getMap","_map","on","getStyle","isStyleLoaded","mapState","mapProps","preserveDrawingBuffer","mapboxApiAccessToken","transformRequest","value","target","inputMapStyle","url","label","name","isValid","el","Component","propTypes","PropTypes","object","isRequired","func","AddMapStyleModalFactory"],"mappings":"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","file":"add-map-style-modal.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 classnames from 'classnames';\nimport styled from 'styled-components';\nimport MapboxGLMap from 'react-map-gl';\nimport {findDOMNode} from 'react-dom';\nimport {StyledModalContent, InputLight, StyledMapContainer} from 'components/common/styled-components';\n\n// Utils\nimport {transformRequest} from 'utils/map-style-utils/mapbox-utils';\n\nconst MapH = 190;\nconst MapW = 264;\nconst ErrorMsg = {\n  styleError : 'Failed to load map style, make sure it is published. For private style, paste in your access token.'\n};\n\nconst InstructionPanel = styled.div`\n  display: flex;\n  flex-direction: column;\n  justify-content: space-around;\n  font-size: 12px;\n  .modal-section {\n    margin-bottom: 32px;\n  }\n  .modal-section:first-child {\n    margin-top: 24px;\n  }\n  \n  .modal-section {\n    .modal-section-title {\n      font-weight: 500;\n    }\n    .modal-section-subtitle {\n      color: ${props => props.theme.subtextColorLT};\n    }\n    \n    input {\n      margin-top: 8px;\n    }\n  }\n\n  input {\n    margin-right: 8px;\n  }\n`;\n\nconst PreviewMap = styled.div`\n  align-items: center;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  margin-left: 116px;\n  flex-shrink: 0;\n  width: ${MapW}px;\n\n  .preview-title {\n    font-weight: 500;\n    font-size: 10px;\n    padding: 8px 0px;\n  }\n  \n  .preview-title.error {\n    color: ${props => props.theme.errorColor};\n  }\n\n  .preview-image {\n    background: ${props => props.theme.modalImagePlaceHolder};\n    border-radius: 4px;\n    box-shadow: 0 8px 16px 0 rgba(0,0,0,0.18);\n    width: ${MapW}px;\n    height: ${MapH}px;\n    position: relative;\n  }\n\n  .preview-image-placeholder {\n    position: absolute;\n    top: 0;\n    left: 0;\n  }\n\n  .preview-image-spinner {\n    position: absolute;\n    left: calc(50% - 25px);\n    top: calc(50% - 25px);\n  }\n`;\n\nconst InlineLink = styled.a`\n  font-weight: 500;\n  \n  :hover {\n    cursor: pointer;\n  }\n`;\n\nclass AddMapStyleModal extends Component {\n  static propTypes = {\n    mapState: PropTypes.object.isRequired,\n    inputMapStyle: PropTypes.func.isRequired,\n    loadCustomMapStyle: PropTypes.func.isRequired,\n    inputStyle: PropTypes.object.isRequired\n  };\n\n  state = {\n    reRenderKey: 0\n  };\n\n  componentWillReceiveProps(nextProps) {\n    if (this.props.inputStyle.accessToken !== nextProps.inputStyle.accessToken) {\n      // toke has changed\n      // ReactMapGl doesn't re-create map when token has changed\n      // here we force the map to update\n      this.setState({\n        reRenderKey: this.state.reRenderKey + 1\n      });\n    }\n  }\n\n  componentDidUpdate() {\n    const map = this.mapRef && this.mapRef.getMap();\n    if (map && this._map !== map) {\n      this._map = map;\n\n      map.on('style.load', () => {\n        const style = map.getStyle();\n        this.loadMapStyleJson(style);\n      });\n\n      map.on('render', () => {\n        if (map.isStyleLoaded()) {\n          this.loadMapStyleIcon();\n        }\n      });\n\n      map.on('error', () => {\n        this.loadMaoStyleError();\n      })\n    }\n  }\n\n  loadMapStyleJson = (style) => {\n    this.props.loadCustomMapStyle({style, error: false});\n  };\n\n  loadMapStyleIcon = () => {\n    if (this.mapRef) {\n      const canvas = findDOMNode(this.mapRef).querySelector('.mapboxgl-canvas');\n      const dataUri = canvas.toDataURL();\n      this.props.loadCustomMapStyle({\n        icon: dataUri\n      });\n    }\n  };\n\n  loadMaoStyleError = () => {\n    this.props.loadCustomMapStyle({error: true});\n  };\n\n  render() {\n    const {inputStyle, mapState} = this.props;\n\n  const mapProps = {\n    ...mapState,\n    preserveDrawingBuffer: true,\n    mapboxApiAccessToken: inputStyle.accessToken || this.props.mapboxApiAccessToken,\n    transformRequest\n  };\n\n    return (\n      <div className=\"add-map-style-modal\">\n        <StyledModalContent>\n          <InstructionPanel>\n            <div className=\"modal-section\">\n              <div className=\"modal-section-title\">1. Publish your style at mapbox or provide access token</div>\n              <div className=\"modal-section-subtitle\">\n                You can create your own map style at\n                <InlineLink target=\"_blank\" href=\"https://www.mapbox.com/studio/styles/\"> mapbox</InlineLink> and\n                <InlineLink target=\"_blank\" href=\"https://www.mapbox.com/help/studio-manual-publish/\"> publish</InlineLink> it.\n              </div>\n              <div className=\"modal-section-subtitle\">\n                To use private style, paste your\n                <InlineLink target=\"_blank\" href=\"https://www.mapbox.com/help/how-access-tokens-work/\"> access token</InlineLink> here. *kepler.gl is a client-side application, data stays in your browser..\n              </div>\n              <InputLight\n                type=\"text\"\n                value={inputStyle.accessToken || ''}\n                onChange={({target: {value}}) => this.props.inputMapStyle({...inputStyle, accessToken: value})}\n                placeholder=\"e.g. pk.abcdefg.xxxxxx\"\n              />\n            </div>\n            <div className=\"modal-section\">\n              <div className=\"modal-section-title\">2. Paste style url</div>\n              <div className=\"modal-section-subtitle\">\n                What is a\n                <InlineLink target=\"_blank\" href=\"https://www.mapbox.com/help/studio-manual-publish/#style-url\"> style URL</InlineLink>\n              </div>\n              <InputLight\n                type=\"text\"\n                value={inputStyle.url || ''}\n                onChange={({target: {value}}) => this.props.inputMapStyle({...inputStyle, url: value})}\n                placeholder=\"e.g. mapbox://styles/uberdataviz/abcdefghijklmnopq\"\n              />\n            </div>\n            <div className=\"modal-section\">\n              <div className=\"modal-section-title\">3. Name your style</div>\n              <InputLight\n                type=\"text\"\n                value={inputStyle.label || ''}\n                onChange={({target: {value}}) => this.props.inputMapStyle({...inputStyle, label: value})}\n              />\n            </div>\n          </InstructionPanel>\n          <PreviewMap>\n            <div className={classnames('preview-title', {error: inputStyle.error})}>\n              {inputStyle.error ? ErrorMsg.styleError :\n                (inputStyle.style && inputStyle.style.name) || ''}</div>\n            <div className=\"preview-image\">\n              {!inputStyle.isValid ?\n                <div className=\"preview-image-spinner\"/> :\n                <StyledMapContainer>\n                  <MapboxGLMap\n                    {...mapProps}\n                    ref={el => {\n                      this.mapRef = el;\n                    }}\n                    key={this.state.reRenderKey}\n                    width={MapW}\n                    height={MapH}\n                    mapStyle={inputStyle.url}/>\n                </StyledMapContainer>\n              }\n            </div>\n          </PreviewMap>\n        </StyledModalContent>\n      </div>\n    );\n  }\n}\n\nconst AddMapStyleModalFactory = () => AddMapStyleModal;\nexport default AddMapStyleModalFactory;\n"]}