kepler.gl.geoiq
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
289 lines (237 loc) • 31.3 kB
JavaScript
;
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 _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
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 _classnames = _interopRequireDefault(require("classnames"));
var _styledComponents = _interopRequireDefault(require("styled-components"));
var _reactMapGl = _interopRequireDefault(require("react-map-gl"));
var _reactDom = require("react-dom");
var _styledComponents2 = require("../common/styled-components");
var _mapboxUtils = require("../../utils/map-style-utils/mapbox-utils");
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2["default"])(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
function _templateObject3() {
var data = (0, _taggedTemplateLiteral2["default"])(["\n font-weight: 500;\n \n :hover {\n cursor: pointer;\n }\n"]);
_templateObject3 = function _templateObject3() {
return data;
};
return data;
}
function _templateObject2() {
var data = (0, _taggedTemplateLiteral2["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"]);
_templateObject2 = function _templateObject2() {
return data;
};
return data;
}
function _templateObject() {
var data = (0, _taggedTemplateLiteral2["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"]);
_templateObject = function _templateObject() {
return data;
};
return data;
}
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 = _styledComponents["default"].div(_templateObject(), function (props) {
return props.theme.subtextColorLT;
});
var PreviewMap = _styledComponents["default"].div(_templateObject2(), MapW, function (props) {
return props.theme.errorColor;
}, function (props) {
return props.theme.modalImagePlaceHolder;
}, MapW, MapH);
var InlineLink = _styledComponents["default"].a(_templateObject3());
var AddMapStyleModal =
/*#__PURE__*/
function (_Component) {
(0, _inherits2["default"])(AddMapStyleModal, _Component);
function AddMapStyleModal() {
var _getPrototypeOf2;
var _this;
(0, _classCallCheck2["default"])(this, AddMapStyleModal);
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"])(AddMapStyleModal)).call.apply(_getPrototypeOf2, [this].concat(args)));
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "state", {
reRenderKey: 0
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "loadMapStyleJson", function (style) {
_this.props.loadCustomMapStyle({
style: style,
error: false
});
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_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
});
}
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "loadMaoStyleError", function () {
_this.props.loadCustomMapStyle({
error: true
});
});
return _this;
}
(0, _createClass2["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 _this$props = this.props,
inputStyle = _this$props.inputStyle,
mapState = _this$props.mapState;
var mapProps = _objectSpread({}, mapState, {
preserveDrawingBuffer: true,
mapboxApiAccessToken: inputStyle.accessToken || this.props.mapboxApiAccessToken,
transformRequest: _mapboxUtils.transformRequest
});
return _react["default"].createElement("div", {
className: "add-map-style-modal"
}, _react["default"].createElement(_styledComponents2.StyledModalContent, null, _react["default"].createElement(InstructionPanel, null, _react["default"].createElement("div", {
className: "modal-section"
}, _react["default"].createElement("div", {
className: "modal-section-title"
}, "1. Publish your style at mapbox or provide access token"), _react["default"].createElement("div", {
className: "modal-section-subtitle"
}, "You can create your own map style at", _react["default"].createElement(InlineLink, {
target: "_blank",
href: "https://www.mapbox.com/studio/styles/"
}, " mapbox"), " and", _react["default"].createElement(InlineLink, {
target: "_blank",
href: "https://www.mapbox.com/help/studio-manual-publish/"
}, " publish"), " it."), _react["default"].createElement("div", {
className: "modal-section-subtitle"
}, "To use private style, paste your", _react["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.."), _react["default"].createElement(_styledComponents2.InputLight, {
type: "text",
value: inputStyle.accessToken || '',
onChange: function onChange(_ref) {
var value = _ref.target.value;
return _this3.props.inputMapStyle(_objectSpread({}, inputStyle, {
accessToken: value
}));
},
placeholder: "e.g. pk.abcdefg.xxxxxx"
})), _react["default"].createElement("div", {
className: "modal-section"
}, _react["default"].createElement("div", {
className: "modal-section-title"
}, "2. Paste style url"), _react["default"].createElement("div", {
className: "modal-section-subtitle"
}, "What is a", _react["default"].createElement(InlineLink, {
target: "_blank",
href: "https://www.mapbox.com/help/studio-manual-publish/#style-url"
}, " style URL")), _react["default"].createElement(_styledComponents2.InputLight, {
type: "text",
value: inputStyle.url || '',
onChange: function onChange(_ref2) {
var value = _ref2.target.value;
return _this3.props.inputMapStyle(_objectSpread({}, inputStyle, {
url: value
}));
},
placeholder: "e.g. mapbox://styles/uberdataviz/abcdefghijklmnopq"
})), _react["default"].createElement("div", {
className: "modal-section"
}, _react["default"].createElement("div", {
className: "modal-section-title"
}, "3. Name your style"), _react["default"].createElement(_styledComponents2.InputLight, {
type: "text",
value: inputStyle.label || '',
onChange: function onChange(_ref3) {
var value = _ref3.target.value;
return _this3.props.inputMapStyle(_objectSpread({}, inputStyle, {
label: value
}));
}
}))), _react["default"].createElement(PreviewMap, null, _react["default"].createElement("div", {
className: (0, _classnames["default"])('preview-title', {
error: inputStyle.error
})
}, inputStyle.error ? ErrorMsg.styleError : inputStyle.style && inputStyle.style.name || ''), _react["default"].createElement("div", {
className: "preview-image"
}, !inputStyle.isValid ? _react["default"].createElement("div", {
className: "preview-image-spinner"
}) : _react["default"].createElement(_styledComponents2.StyledMapContainer, null, _react["default"].createElement(_reactMapGl["default"], (0, _extends2["default"])({}, mapProps, {
ref: function ref(el) {
_this3.mapRef = el;
},
key: this.state.reRenderKey,
width: MapW,
height: MapH,
mapStyle: inputStyle.url
})))))));
}
}]);
return AddMapStyleModal;
}(_react.Component);
(0, _defineProperty2["default"])(AddMapStyleModal, "propTypes", {
mapState: _propTypes["default"].object.isRequired,
inputMapStyle: _propTypes["default"].func.isRequired,
loadCustomMapStyle: _propTypes["default"].func.isRequired,
inputStyle: _propTypes["default"].object.isRequired
});
var AddMapStyleModalFactory = function AddMapStyleModalFactory() {
return AddMapStyleModal;
};
var _default = AddMapStyleModalFactory;
exports["default"] = _default;
//# 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","reRenderKey","style","loadCustomMapStyle","error","mapRef","canvas","querySelector","dataUri","toDataURL","icon","nextProps","inputStyle","accessToken","setState","state","map","getMap","_map","on","getStyle","loadMapStyleJson","isStyleLoaded","loadMapStyleIcon","loadMaoStyleError","mapState","mapProps","preserveDrawingBuffer","mapboxApiAccessToken","transformRequest","value","target","inputMapStyle","url","label","name","isValid","el","Component","PropTypes","object","isRequired","func","AddMapStyleModalFactory"],"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 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"]}