kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
224 lines (179 loc) • 23.8 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = GeocoderPanelFactory;
var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck"));
var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass"));
var _assertThisInitialized2 = _interopRequireDefault(require("@babel/runtime/helpers/assertThisInitialized"));
var _inherits2 = _interopRequireDefault(require("@babel/runtime/helpers/inherits"));
var _possibleConstructorReturn2 = _interopRequireDefault(require("@babel/runtime/helpers/possibleConstructorReturn"));
var _getPrototypeOf2 = _interopRequireDefault(require("@babel/runtime/helpers/getPrototypeOf"));
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 _processors = _interopRequireDefault(require("../processors"));
var _core = require("@deck.gl/core");
var _schemas = _interopRequireDefault(require("../schemas"));
var _projectionUtils = require("../utils/projection-utils");
var _geocoder = _interopRequireDefault(require("./geocoder/geocoder"));
var _defaultSettings = require("../constants/default-settings");
var _templateObject;
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 _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = (0, _getPrototypeOf2["default"])(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = (0, _getPrototypeOf2["default"])(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return (0, _possibleConstructorReturn2["default"])(this, result); }; }
function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }
var ICON_LAYER = {
id: _defaultSettings.GEOCODER_LAYER_ID,
type: 'icon',
config: {
label: 'Geocoder Layer',
color: _defaultSettings.GEOCODER_ICON_COLOR,
dataId: _defaultSettings.GEOCODER_DATASET_NAME,
columns: {
lat: 'lt',
lng: 'ln',
icon: 'icon',
label: 'text'
},
isVisible: true,
hidden: true,
visConfig: {
radius: _defaultSettings.GEOCODER_ICON_SIZE
}
}
};
var PARSED_CONFIG = _schemas["default"].parseSavedConfig({
version: 'v1',
config: {
visState: {
layers: [ICON_LAYER]
}
}
});
var StyledGeocoderPanel = _styledComponents["default"].div(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n position: absolute;\n top: ", "px;\n right: ", "px;\n width: ", "px;\n box-shadow: ", ";\n z-index: 100;\n"])), function (props) {
return props.theme.geocoderTop;
}, function (props) {
return props.theme.geocoderRight;
}, function (props) {
return Number.isFinite(props.width) ? props.width : props.theme.geocoderWidth;
}, function (props) {
return props.theme.boxShadow;
});
function generateGeocoderDataset(lat, lon, text) {
return {
data: _processors["default"].processRowObject([{
lt: lat,
ln: lon,
icon: 'place',
text: text
}]),
id: _defaultSettings.GEOCODER_DATASET_NAME,
info: {
hidden: true,
id: _defaultSettings.GEOCODER_DATASET_NAME,
label: _defaultSettings.GEOCODER_DATASET_NAME
}
};
}
function isValid(key) {
return /pk\..*\..*/.test(key);
}
function GeocoderPanelFactory() {
var GeocoderPanel = /*#__PURE__*/function (_Component) {
(0, _inherits2["default"])(GeocoderPanel, _Component);
var _super = _createSuper(GeocoderPanel);
function GeocoderPanel() {
var _this;
(0, _classCallCheck2["default"])(this, GeocoderPanel);
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
args[_key] = arguments[_key];
}
_this = _super.call.apply(_super, [this].concat(args));
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "onSelected", function () {
var viewport = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : null;
var geoItem = arguments.length > 1 ? arguments[1] : undefined;
var _geoItem$center = (0, _slicedToArray2["default"])(geoItem.center, 2),
lon = _geoItem$center[0],
lat = _geoItem$center[1],
text = geoItem.text,
bbox = geoItem.bbox;
_this.removeGeocoderDataset();
_this.props.updateVisData([generateGeocoderDataset(lat, lon, text)], {
keepExistingConfig: true
}, PARSED_CONFIG);
var bounds = bbox || [lon - _defaultSettings.GEOCODER_GEO_OFFSET, lat - _defaultSettings.GEOCODER_GEO_OFFSET, lon + _defaultSettings.GEOCODER_GEO_OFFSET, lat + _defaultSettings.GEOCODER_GEO_OFFSET];
var centerAndZoom = (0, _projectionUtils.getCenterAndZoomFromBounds)(bounds, {
width: _this.props.mapState.width,
height: _this.props.mapState.height
});
if (!centerAndZoom) {
// failed to fit bounds
return;
}
_this.props.updateMap(_objectSpread(_objectSpread({
latitude: centerAndZoom.center[1],
longitude: centerAndZoom.center[0]
}, Number.isFinite(centerAndZoom.zoom) ? {
zoom: centerAndZoom.zoom
} : {}), {}, {
pitch: 0,
bearing: 0,
transitionDuration: _this.props.transitionDuration,
transitionInterpolator: new _core.FlyToInterpolator()
}));
});
(0, _defineProperty2["default"])((0, _assertThisInitialized2["default"])(_this), "removeMarker", function () {
_this.removeGeocoderDataset();
});
return _this;
}
(0, _createClass2["default"])(GeocoderPanel, [{
key: "removeGeocoderDataset",
value: function removeGeocoderDataset() {
this.props.removeDataset(_defaultSettings.GEOCODER_DATASET_NAME);
}
}, {
key: "render",
value: function render() {
var _this$props = this.props,
isGeocoderEnabled = _this$props.isGeocoderEnabled,
mapboxApiAccessToken = _this$props.mapboxApiAccessToken,
width = _this$props.width;
return /*#__PURE__*/_react["default"].createElement(StyledGeocoderPanel, {
className: "geocoder-panel",
width: width,
style: {
display: isGeocoderEnabled ? 'block' : 'none'
}
}, isValid(mapboxApiAccessToken) && /*#__PURE__*/_react["default"].createElement(_geocoder["default"], {
mapboxApiAccessToken: mapboxApiAccessToken,
onSelected: this.onSelected,
onDeleteMarker: this.removeMarker,
width: width
}));
}
}]);
return GeocoderPanel;
}(_react.Component);
(0, _defineProperty2["default"])(GeocoderPanel, "propTypes", {
isGeocoderEnabled: _propTypes["default"].bool.isRequired,
mapboxApiAccessToken: _propTypes["default"].string.isRequired,
mapState: _propTypes["default"].object.isRequired,
updateVisData: _propTypes["default"].func.isRequired,
removeDataset: _propTypes["default"].func.isRequired,
updateMap: _propTypes["default"].func.isRequired,
transitionDuration: _propTypes["default"].number,
width: _propTypes["default"].number
});
GeocoderPanel.defaultProps = {
transitionDuration: 3000
};
return GeocoderPanel;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/geocoder-panel.js"],"names":["ICON_LAYER","id","GEOCODER_LAYER_ID","type","config","label","color","GEOCODER_ICON_COLOR","dataId","GEOCODER_DATASET_NAME","columns","lat","lng","icon","isVisible","hidden","visConfig","radius","GEOCODER_ICON_SIZE","PARSED_CONFIG","KeplerGlSchema","parseSavedConfig","version","visState","layers","StyledGeocoderPanel","styled","div","props","theme","geocoderTop","geocoderRight","Number","isFinite","width","geocoderWidth","boxShadow","generateGeocoderDataset","lon","text","data","Processors","processRowObject","lt","ln","info","isValid","key","test","GeocoderPanelFactory","GeocoderPanel","viewport","geoItem","center","bbox","removeGeocoderDataset","updateVisData","keepExistingConfig","bounds","GEOCODER_GEO_OFFSET","centerAndZoom","mapState","height","updateMap","latitude","longitude","zoom","pitch","bearing","transitionDuration","transitionInterpolator","FlyToInterpolator","removeDataset","isGeocoderEnabled","mapboxApiAccessToken","display","onSelected","removeMarker","Component","PropTypes","bool","isRequired","string","object","func","number","defaultProps"],"mappings":"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","sourcesContent":["// Copyright (c) 2021 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';\nimport Processors from 'processors';\nimport {FlyToInterpolator} from '@deck.gl/core';\nimport KeplerGlSchema from 'schemas';\nimport {getCenterAndZoomFromBounds} from 'utils/projection-utils';\n\nimport Geocoder from './geocoder/geocoder';\nimport {\n  GEOCODER_DATASET_NAME,\n  GEOCODER_LAYER_ID,\n  GEOCODER_GEO_OFFSET,\n  GEOCODER_ICON_COLOR,\n  GEOCODER_ICON_SIZE\n} from 'constants/default-settings';\n\nconst ICON_LAYER = {\n  id: GEOCODER_LAYER_ID,\n  type: 'icon',\n  config: {\n    label: 'Geocoder Layer',\n    color: GEOCODER_ICON_COLOR,\n    dataId: GEOCODER_DATASET_NAME,\n    columns: {\n      lat: 'lt',\n      lng: 'ln',\n      icon: 'icon',\n      label: 'text'\n    },\n    isVisible: true,\n    hidden: true,\n    visConfig: {\n      radius: GEOCODER_ICON_SIZE\n    }\n  }\n};\n\nconst PARSED_CONFIG = KeplerGlSchema.parseSavedConfig({\n  version: 'v1',\n  config: {\n    visState: {\n      layers: [ICON_LAYER]\n    }\n  }\n});\n\nconst StyledGeocoderPanel = styled.div`\n  position: absolute;\n  top: ${props => props.theme.geocoderTop}px;\n  right: ${props => props.theme.geocoderRight}px;\n  width: ${props => (Number.isFinite(props.width) ? props.width : props.theme.geocoderWidth)}px;\n  box-shadow: ${props => props.theme.boxShadow};\n  z-index: 100;\n`;\n\nfunction generateGeocoderDataset(lat, lon, text) {\n  return {\n    data: Processors.processRowObject([\n      {\n        lt: lat,\n        ln: lon,\n        icon: 'place',\n        text\n      }\n    ]),\n    id: GEOCODER_DATASET_NAME,\n    info: {\n      hidden: true,\n      id: GEOCODER_DATASET_NAME,\n      label: GEOCODER_DATASET_NAME\n    }\n  };\n}\n\nfunction isValid(key) {\n  return /pk\\..*\\..*/.test(key);\n}\n\nexport default function GeocoderPanelFactory() {\n  class GeocoderPanel extends Component {\n    static propTypes = {\n      isGeocoderEnabled: PropTypes.bool.isRequired,\n      mapboxApiAccessToken: PropTypes.string.isRequired,\n      mapState: PropTypes.object.isRequired,\n      updateVisData: PropTypes.func.isRequired,\n      removeDataset: PropTypes.func.isRequired,\n      updateMap: PropTypes.func.isRequired,\n\n      transitionDuration: PropTypes.number,\n      width: PropTypes.number\n    };\n\n    removeGeocoderDataset() {\n      this.props.removeDataset(GEOCODER_DATASET_NAME);\n    }\n\n    onSelected = (viewport = null, geoItem) => {\n      const {\n        center: [lon, lat],\n        text,\n        bbox\n      } = geoItem;\n      this.removeGeocoderDataset();\n      this.props.updateVisData(\n        [generateGeocoderDataset(lat, lon, text)],\n        {\n          keepExistingConfig: true\n        },\n        PARSED_CONFIG\n      );\n      const bounds = bbox || [\n        lon - GEOCODER_GEO_OFFSET,\n        lat - GEOCODER_GEO_OFFSET,\n        lon + GEOCODER_GEO_OFFSET,\n        lat + GEOCODER_GEO_OFFSET\n      ];\n      const centerAndZoom = getCenterAndZoomFromBounds(bounds, {\n        width: this.props.mapState.width,\n        height: this.props.mapState.height\n      });\n\n      if (!centerAndZoom) {\n        // failed to fit bounds\n        return;\n      }\n\n      this.props.updateMap({\n        latitude: centerAndZoom.center[1],\n        longitude: centerAndZoom.center[0],\n        // For marginal or invalid bounds, zoom may be NaN. Make sure to provide a valid value in order\n        // to avoid corrupt state and potential crashes as zoom is expected to be a number\n        ...(Number.isFinite(centerAndZoom.zoom) ? {zoom: centerAndZoom.zoom} : {}),\n        pitch: 0,\n        bearing: 0,\n        transitionDuration: this.props.transitionDuration,\n        transitionInterpolator: new FlyToInterpolator()\n      });\n    };\n\n    removeMarker = () => {\n      this.removeGeocoderDataset();\n    };\n\n    render() {\n      const {isGeocoderEnabled, mapboxApiAccessToken, width} = this.props;\n      return (\n        <StyledGeocoderPanel\n          className=\"geocoder-panel\"\n          width={width}\n          style={{display: isGeocoderEnabled ? 'block' : 'none'}}\n        >\n          {isValid(mapboxApiAccessToken) && (\n            <Geocoder\n              mapboxApiAccessToken={mapboxApiAccessToken}\n              onSelected={this.onSelected}\n              onDeleteMarker={this.removeMarker}\n              width={width}\n            />\n          )}\n        </StyledGeocoderPanel>\n      );\n    }\n  }\n\n  GeocoderPanel.defaultProps = {\n    transitionDuration: 3000\n  };\n\n  return GeocoderPanel;\n}\n"]}