UNPKG

kepler.gl

Version:

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

315 lines (264 loc) 35.1 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard"); Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = exports.testForCoordinates = void 0; var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator")); var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray")); var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/helpers/taggedTemplateLiteral")); var _react = _interopRequireWildcard(require("react")); var _styledComponents = _interopRequireDefault(require("styled-components")); var _classnames = _interopRequireDefault(require("classnames")); var _mapbox = _interopRequireDefault(require("mapbox")); var _reactIntl = require("react-intl"); var _viewportMercatorProject = require("viewport-mercator-project"); var _keyevent = _interopRequireDefault(require("../../constants/keyevent")); var _styledComponents2 = require("../common/styled-components"); var _icons = require("../common/icons"); 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; } // matches only valid coordinates var COORDINATE_REGEX_STRING = '^[-+]?([1-8]?\\d(\\.\\d+)?|90(\\.0+)?),\\s*[-+]?(180(\\.0+)?|((1[0-7]\\d)|([1-9]?\\d))(\\.\\d+)?)'; var COORDINATE_REGEX = RegExp(COORDINATE_REGEX_STRING); var PLACEHOLDER = 'Enter an address or coordinates, ex 37.79,-122.40'; var debounceTimeout = null; var testForCoordinates = function testForCoordinates(query) { var isValid = COORDINATE_REGEX.test(query.trim()); if (!isValid) { return [isValid, query]; } var tokens = query.trim().split(','); return [isValid, Number(tokens[0]), Number(tokens[1])]; }; exports.testForCoordinates = testForCoordinates; var StyledContainer = _styledComponents["default"].div(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n position: relative;\n color: ", ";\n\n .geocoder-input {\n box-shadow: ", ";\n\n .geocoder-input__search {\n position: absolute;\n height: ", "px;\n width: 30px;\n padding-left: 6px;\n display: flex;\n align-items: center;\n justify-content: center;\n color: ", ";\n }\n\n input {\n padding: 4px 36px;\n height: ", "px;\n caret-color: unset;\n }\n }\n\n .geocoder-results {\n box-shadow: ", ";\n background-color: ", ";\n position: absolute;\n width: ", "px;\n margin-top: ", "px;\n }\n\n .geocoder-item {\n ", ";\n ", ";\n\n &.active {\n background-color: ", ";\n }\n }\n\n .remove-result {\n position: absolute;\n right: 16px;\n top: 0px;\n height: ", "px;\n display: flex;\n align-items: center;\n\n :hover {\n cursor: pointer;\n color: ", ";\n }\n }\n"])), function (props) { return props.theme.textColor; }, function (props) { return props.theme.boxShadow; }, function (props) { return props.theme.geocoderInputHeight; }, function (props) { return props.theme.subtextColor; }, function (props) { return props.theme.geocoderInputHeight; }, function (props) { return props.theme.boxShadow; }, function (props) { return props.theme.panelBackground; }, function (props) { return Number.isFinite(props.width) ? props.width : props.theme.geocoderWidth; }, function (props) { return props.theme.dropdownWapperMargin; }, function (props) { return props.theme.dropdownListItem; }, function (props) { return props.theme.textTruncate; }, function (props) { return props.theme.dropdownListHighlightBg; }, function (props) { return props.theme.geocoderInputHeight; }, function (props) { return props.theme.textColorHl; }); /** @type {import('./geocoder').GeocoderComponent} */ var GeoCoder = function GeoCoder(_ref) { var mapboxApiAccessToken = _ref.mapboxApiAccessToken, _ref$className = _ref.className, className = _ref$className === void 0 ? '' : _ref$className, _ref$limit = _ref.limit, limit = _ref$limit === void 0 ? 5 : _ref$limit, _ref$timeout = _ref.timeout, timeout = _ref$timeout === void 0 ? 300 : _ref$timeout, _ref$formatItem = _ref.formatItem, formatItem = _ref$formatItem === void 0 ? function (item) { return item.place_name; } : _ref$formatItem, viewport = _ref.viewport, onSelected = _ref.onSelected, onDeleteMarker = _ref.onDeleteMarker, transitionDuration = _ref.transitionDuration, pointZoom = _ref.pointZoom, width = _ref.width, intl = _ref.intl; var _useState = (0, _react.useState)(''), _useState2 = (0, _slicedToArray2["default"])(_useState, 2), inputValue = _useState2[0], setInputValue = _useState2[1]; var _useState3 = (0, _react.useState)(false), _useState4 = (0, _slicedToArray2["default"])(_useState3, 2), showResults = _useState4[0], setShowResults = _useState4[1]; var _useState5 = (0, _react.useState)(false), _useState6 = (0, _slicedToArray2["default"])(_useState5, 2), showDelete = _useState6[0], setShowDelete = _useState6[1]; /** @type {import('./geocoder').Results} */ var initialResults = []; var _useState7 = (0, _react.useState)(initialResults), _useState8 = (0, _slicedToArray2["default"])(_useState7, 2), results = _useState8[0], setResults = _useState8[1]; var _useState9 = (0, _react.useState)(0), _useState10 = (0, _slicedToArray2["default"])(_useState9, 2), selectedIndex = _useState10[0], setSelectedIndex = _useState10[1]; var client = (0, _react.useMemo)(function () { return new _mapbox["default"](mapboxApiAccessToken); }, [mapboxApiAccessToken]); var onChange = (0, _react.useCallback)(function (event) { var queryString = event.target.value; setInputValue(queryString); var _testForCoordinates = testForCoordinates(queryString), _testForCoordinates2 = (0, _slicedToArray2["default"])(_testForCoordinates, 3), hasValidCoordinates = _testForCoordinates2[0], longitude = _testForCoordinates2[1], latitude = _testForCoordinates2[2]; if (hasValidCoordinates) { setResults([{ center: [latitude, longitude], place_name: queryString }]); } else { clearTimeout(debounceTimeout); debounceTimeout = setTimeout( /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee() { var response; return _regenerator["default"].wrap(function _callee$(_context) { while (1) { switch (_context.prev = _context.next) { case 0: if (!(limit > 0 && Boolean(queryString))) { _context.next = 11; break; } _context.prev = 1; _context.next = 4; return client.geocodeForward(queryString, { limit: limit }); case 4: response = _context.sent; if (response.entity.features) { setShowResults(true); setResults(response.entity.features); } _context.next = 11; break; case 8: _context.prev = 8; _context.t0 = _context["catch"](1); // TODO: show geocode error // eslint-disable-next-line no-console console.log(_context.t0); case 11: case "end": return _context.stop(); } } }, _callee, null, [[1, 8]]); })), timeout); } }, [client, limit, timeout, setResults, setShowResults]); var onBlur = (0, _react.useCallback)(function () { setTimeout(function () { setShowResults(false); }, timeout); }, [setShowResults, timeout]); var onFocus = (0, _react.useCallback)(function () { return setShowResults(true); }, [setShowResults]); var onItemSelected = (0, _react.useCallback)(function (item) { var newViewport = new _viewportMercatorProject.WebMercatorViewport(viewport); var bbox = item.bbox, center = item.center; newViewport = bbox ? newViewport.fitBounds([[bbox[0], bbox[1]], [bbox[2], bbox[3]]]) : { longitude: center[0], latitude: center[1], zoom: pointZoom }; var _newViewport = newViewport, longitude = _newViewport.longitude, latitude = _newViewport.latitude, zoom = _newViewport.zoom; onSelected(_objectSpread(_objectSpread({}, viewport), { longitude: longitude, latitude: latitude, zoom: zoom, transitionDuration: transitionDuration }), item); setShowResults(false); setInputValue(formatItem(item)); setShowDelete(true); }, [viewport, onSelected, transitionDuration, pointZoom, formatItem]); var onMarkDeleted = (0, _react.useCallback)(function () { setShowDelete(false); setInputValue(''); onDeleteMarker(); }, [onDeleteMarker]); var onKeyDown = (0, _react.useCallback)(function (e) { if (!results || results.length === 0) { return; } switch (e.keyCode) { case _keyevent["default"].DOM_VK_UP: setSelectedIndex(selectedIndex > 0 ? selectedIndex - 1 : selectedIndex); break; case _keyevent["default"].DOM_VK_DOWN: setSelectedIndex(selectedIndex < results.length - 1 ? selectedIndex + 1 : selectedIndex); break; case _keyevent["default"].DOM_VK_ENTER: case _keyevent["default"].DOM_VK_RETURN: if (results[selectedIndex]) { onItemSelected(results[selectedIndex]); } break; default: break; } }, [results, selectedIndex, setSelectedIndex, onItemSelected]); return /*#__PURE__*/_react["default"].createElement(StyledContainer, { className: className, width: width }, /*#__PURE__*/_react["default"].createElement("div", { className: "geocoder-input" }, /*#__PURE__*/_react["default"].createElement("div", { className: "geocoder-input__search" }, /*#__PURE__*/_react["default"].createElement(_icons.Search, { height: "20px" })), /*#__PURE__*/_react["default"].createElement(_styledComponents2.Input, { type: "text", onChange: onChange, onBlur: onBlur, onFocus: onFocus, onKeyDown: onKeyDown, value: inputValue, placeholder: intl ? intl.formatMessage({ id: 'geocoder.title', defaultMessage: PLACEHOLDER }) : PLACEHOLDER }), showDelete ? /*#__PURE__*/_react["default"].createElement("div", { className: "remove-result" }, /*#__PURE__*/_react["default"].createElement(_icons.Delete, { height: "12px", onClick: onMarkDeleted })) : null), showResults ? /*#__PURE__*/_react["default"].createElement("div", { className: "geocoder-results" }, results.map(function (item, index) { return /*#__PURE__*/_react["default"].createElement("div", { key: index, className: (0, _classnames["default"])('geocoder-item', { active: selectedIndex === index }), onClick: function onClick() { return onItemSelected(item); } }, formatItem(item)); })) : null); }; var _default = (0, _reactIntl.injectIntl)(GeoCoder); exports["default"] = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/components/geocoder/geocoder.js"],"names":["COORDINATE_REGEX_STRING","COORDINATE_REGEX","RegExp","PLACEHOLDER","debounceTimeout","testForCoordinates","query","isValid","test","trim","tokens","split","Number","StyledContainer","styled","div","props","theme","textColor","boxShadow","geocoderInputHeight","subtextColor","panelBackground","isFinite","width","geocoderWidth","dropdownWapperMargin","dropdownListItem","textTruncate","dropdownListHighlightBg","textColorHl","GeoCoder","mapboxApiAccessToken","className","limit","timeout","formatItem","item","place_name","viewport","onSelected","onDeleteMarker","transitionDuration","pointZoom","intl","inputValue","setInputValue","showResults","setShowResults","showDelete","setShowDelete","initialResults","results","setResults","selectedIndex","setSelectedIndex","client","MapboxClient","onChange","event","queryString","target","value","hasValidCoordinates","longitude","latitude","center","clearTimeout","setTimeout","Boolean","geocodeForward","response","entity","features","console","log","onBlur","onFocus","onItemSelected","newViewport","WebMercatorViewport","bbox","fitBounds","zoom","onMarkDeleted","onKeyDown","e","length","keyCode","KeyEvent","DOM_VK_UP","DOM_VK_DOWN","DOM_VK_ENTER","DOM_VK_RETURN","formatMessage","id","defaultMessage","map","index","active"],"mappings":";;;;;;;;;;;;;;;;;;;;;AAoBA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;;;;;AAEA;AACA,IAAMA,uBAAuB,GAC3B,mGADF;AAEA,IAAMC,gBAAgB,GAAGC,MAAM,CAACF,uBAAD,CAA/B;AAEA,IAAMG,WAAW,GAAG,mDAApB;AAEA,IAAIC,eAAe,GAAG,IAAtB;;AAEO,IAAMC,kBAAkB,GAAG,SAArBA,kBAAqB,CAAAC,KAAK,EAAI;AACzC,MAAMC,OAAO,GAAGN,gBAAgB,CAACO,IAAjB,CAAsBF,KAAK,CAACG,IAAN,EAAtB,CAAhB;;AAEA,MAAI,CAACF,OAAL,EAAc;AACZ,WAAO,CAACA,OAAD,EAAUD,KAAV,CAAP;AACD;;AAED,MAAMI,MAAM,GAAGJ,KAAK,CAACG,IAAN,GAAaE,KAAb,CAAmB,GAAnB,CAAf;AAEA,SAAO,CAACJ,OAAD,EAAUK,MAAM,CAACF,MAAM,CAAC,CAAD,CAAP,CAAhB,EAA6BE,MAAM,CAACF,MAAM,CAAC,CAAD,CAAP,CAAnC,CAAP;AACD,CAVM;;;;AAYP,IAAMG,eAAe,GAAGC,6BAAOC,GAAV,+9BAEV,UAAAC,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYC,SAAhB;AAAA,CAFK,EAKH,UAAAF,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYE,SAAhB;AAAA,CALF,EASL,UAAAH,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYG,mBAAhB;AAAA,CATA,EAeN,UAAAJ,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYI,YAAhB;AAAA,CAfC,EAoBL,UAAAL,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYG,mBAAhB;AAAA,CApBA,EA0BH,UAAAJ,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYE,SAAhB;AAAA,CA1BF,EA2BG,UAAAH,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYK,eAAhB;AAAA,CA3BR,EA6BR,UAAAN,KAAK;AAAA,SAAKJ,MAAM,CAACW,QAAP,CAAgBP,KAAK,CAACQ,KAAtB,IAA+BR,KAAK,CAACQ,KAArC,GAA6CR,KAAK,CAACC,KAAN,CAAYQ,aAA9D;AAAA,CA7BG,EA8BH,UAAAT,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYS,oBAAhB;AAAA,CA9BF,EAkCf,UAAAV,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYU,gBAAhB;AAAA,CAlCU,EAmCf,UAAAX,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYW,YAAhB;AAAA,CAnCU,EAsCK,UAAAZ,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYY,uBAAhB;AAAA,CAtCV,EA8CP,UAAAb,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYG,mBAAhB;AAAA,CA9CE,EAoDN,UAAAJ,KAAK;AAAA,SAAIA,KAAK,CAACC,KAAN,CAAYa,WAAhB;AAAA,CApDC,CAArB;AAyDA;;;AACA,IAAMC,QAAQ,GAAG,SAAXA,QAAW,OAaX;AAAA,MAZJC,oBAYI,QAZJA,oBAYI;AAAA,4BAXJC,SAWI;AAAA,MAXJA,SAWI,+BAXQ,EAWR;AAAA,wBAVJC,KAUI;AAAA,MAVJA,KAUI,2BAVI,CAUJ;AAAA,0BATJC,OASI;AAAA,MATJA,OASI,6BATM,GASN;AAAA,6BARJC,UAQI;AAAA,MARJA,UAQI,gCARS,UAAAC,IAAI;AAAA,WAAIA,IAAI,CAACC,UAAT;AAAA,GAQb;AAAA,MAPJC,QAOI,QAPJA,QAOI;AAAA,MANJC,UAMI,QANJA,UAMI;AAAA,MALJC,cAKI,QALJA,cAKI;AAAA,MAJJC,kBAII,QAJJA,kBAII;AAAA,MAHJC,SAGI,QAHJA,SAGI;AAAA,MAFJnB,KAEI,QAFJA,KAEI;AAAA,MADJoB,IACI,QADJA,IACI;;AAAA,kBACgC,qBAAS,EAAT,CADhC;AAAA;AAAA,MACGC,UADH;AAAA,MACeC,aADf;;AAAA,mBAEkC,qBAAS,KAAT,CAFlC;AAAA;AAAA,MAEGC,WAFH;AAAA,MAEgBC,cAFhB;;AAAA,mBAGgC,qBAAS,KAAT,CAHhC;AAAA;AAAA,MAGGC,UAHH;AAAA,MAGeC,aAHf;AAIJ;;;AACA,MAAMC,cAAc,GAAG,EAAvB;;AALI,mBAM0B,qBAASA,cAAT,CAN1B;AAAA;AAAA,MAMGC,OANH;AAAA,MAMYC,UANZ;;AAAA,mBAOsC,qBAAS,CAAT,CAPtC;AAAA;AAAA,MAOGC,aAPH;AAAA,MAOkBC,gBAPlB;;AASJ,MAAMC,MAAM,GAAG,oBAAQ;AAAA,WAAM,IAAIC,kBAAJ,CAAiBzB,oBAAjB,CAAN;AAAA,GAAR,EAAsD,CAACA,oBAAD,CAAtD,CAAf;AAEA,MAAM0B,QAAQ,GAAG,wBACf,UAAAC,KAAK,EAAI;AACP,QAAMC,WAAW,GAAGD,KAAK,CAACE,MAAN,CAAaC,KAAjC;AACAhB,IAAAA,aAAa,CAACc,WAAD,CAAb;;AAFO,8BAG4CvD,kBAAkB,CAACuD,WAAD,CAH9D;AAAA;AAAA,QAGAG,mBAHA;AAAA,QAGqBC,SAHrB;AAAA,QAGgCC,QAHhC;;AAIP,QAAIF,mBAAJ,EAAyB;AACvBV,MAAAA,UAAU,CAAC,CAAC;AAACa,QAAAA,MAAM,EAAE,CAACD,QAAD,EAAWD,SAAX,CAAT;AAAgC1B,QAAAA,UAAU,EAAEsB;AAA5C,OAAD,CAAD,CAAV;AACD,KAFD,MAEO;AACLO,MAAAA,YAAY,CAAC/D,eAAD,CAAZ;AACAA,MAAAA,eAAe,GAAGgE,UAAU,6FAAC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,sBACvBlC,KAAK,GAAG,CAAR,IAAamC,OAAO,CAACT,WAAD,CADG;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA,uBAGAJ,MAAM,CAACc,cAAP,CAAsBV,WAAtB,EAAmC;AAAC1B,kBAAAA,KAAK,EAALA;AAAD,iBAAnC,CAHA;;AAAA;AAGjBqC,gBAAAA,QAHiB;;AAIvB,oBAAIA,QAAQ,CAACC,MAAT,CAAgBC,QAApB,EAA8B;AAC5BzB,kBAAAA,cAAc,CAAC,IAAD,CAAd;AACAK,kBAAAA,UAAU,CAACkB,QAAQ,CAACC,MAAT,CAAgBC,QAAjB,CAAV;AACD;;AAPsB;AAAA;;AAAA;AAAA;AAAA;AASvB;AACA;AACAC,gBAAAA,OAAO,CAACC,GAAR;;AAXuB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,OAAD,IAczBxC,OAdyB,CAA5B;AAeD;AACF,GAzBc,EA0Bf,CAACqB,MAAD,EAAStB,KAAT,EAAgBC,OAAhB,EAAyBkB,UAAzB,EAAqCL,cAArC,CA1Be,CAAjB;AA6BA,MAAM4B,MAAM,GAAG,wBAAY,YAAM;AAC/BR,IAAAA,UAAU,CAAC,YAAM;AACfpB,MAAAA,cAAc,CAAC,KAAD,CAAd;AACD,KAFS,EAEPb,OAFO,CAAV;AAGD,GAJc,EAIZ,CAACa,cAAD,EAAiBb,OAAjB,CAJY,CAAf;AAMA,MAAM0C,OAAO,GAAG,wBAAY;AAAA,WAAM7B,cAAc,CAAC,IAAD,CAApB;AAAA,GAAZ,EAAwC,CAACA,cAAD,CAAxC,CAAhB;AAEA,MAAM8B,cAAc,GAAG,wBACrB,UAAAzC,IAAI,EAAI;AACN,QAAI0C,WAAW,GAAG,IAAIC,4CAAJ,CAAwBzC,QAAxB,CAAlB;AADM,QAEC0C,IAFD,GAEiB5C,IAFjB,CAEC4C,IAFD;AAAA,QAEOf,MAFP,GAEiB7B,IAFjB,CAEO6B,MAFP;AAINa,IAAAA,WAAW,GAAGE,IAAI,GACdF,WAAW,CAACG,SAAZ,CAAsB,CACpB,CAACD,IAAI,CAAC,CAAD,CAAL,EAAUA,IAAI,CAAC,CAAD,CAAd,CADoB,EAEpB,CAACA,IAAI,CAAC,CAAD,CAAL,EAAUA,IAAI,CAAC,CAAD,CAAd,CAFoB,CAAtB,CADc,GAKd;AACEjB,MAAAA,SAAS,EAAEE,MAAM,CAAC,CAAD,CADnB;AAEED,MAAAA,QAAQ,EAAEC,MAAM,CAAC,CAAD,CAFlB;AAGEiB,MAAAA,IAAI,EAAExC;AAHR,KALJ;AAJM,uBAe8BoC,WAf9B;AAAA,QAeCf,SAfD,gBAeCA,SAfD;AAAA,QAeYC,QAfZ,gBAeYA,QAfZ;AAAA,QAesBkB,IAftB,gBAesBA,IAftB;AAiBN3C,IAAAA,UAAU,iCAAKD,QAAL,GAAkB;AAACyB,MAAAA,SAAS,EAATA,SAAD;AAAYC,MAAAA,QAAQ,EAARA,QAAZ;AAAsBkB,MAAAA,IAAI,EAAJA,IAAtB;AAA4BzC,MAAAA,kBAAkB,EAAlBA;AAA5B,KAAlB,GAAoEL,IAApE,CAAV;AAEAW,IAAAA,cAAc,CAAC,KAAD,CAAd;AACAF,IAAAA,aAAa,CAACV,UAAU,CAACC,IAAD,CAAX,CAAb;AACAa,IAAAA,aAAa,CAAC,IAAD,CAAb;AACD,GAvBoB,EAwBrB,CAACX,QAAD,EAAWC,UAAX,EAAuBE,kBAAvB,EAA2CC,SAA3C,EAAsDP,UAAtD,CAxBqB,CAAvB;AA2BA,MAAMgD,aAAa,GAAG,wBAAY,YAAM;AACtClC,IAAAA,aAAa,CAAC,KAAD,CAAb;AACAJ,IAAAA,aAAa,CAAC,EAAD,CAAb;AACAL,IAAAA,cAAc;AACf,GAJqB,EAInB,CAACA,cAAD,CAJmB,CAAtB;AAMA,MAAM4C,SAAS,GAAG,wBAChB,UAAAC,CAAC,EAAI;AACH,QAAI,CAAClC,OAAD,IAAYA,OAAO,CAACmC,MAAR,KAAmB,CAAnC,EAAsC;AACpC;AACD;;AACD,YAAQD,CAAC,CAACE,OAAV;AACE,WAAKC,qBAASC,SAAd;AACEnC,QAAAA,gBAAgB,CAACD,aAAa,GAAG,CAAhB,GAAoBA,aAAa,GAAG,CAApC,GAAwCA,aAAzC,CAAhB;AACA;;AACF,WAAKmC,qBAASE,WAAd;AACEpC,QAAAA,gBAAgB,CAACD,aAAa,GAAGF,OAAO,CAACmC,MAAR,GAAiB,CAAjC,GAAqCjC,aAAa,GAAG,CAArD,GAAyDA,aAA1D,CAAhB;AACA;;AACF,WAAKmC,qBAASG,YAAd;AACA,WAAKH,qBAASI,aAAd;AACE,YAAIzC,OAAO,CAACE,aAAD,CAAX,EAA4B;AAC1BwB,UAAAA,cAAc,CAAC1B,OAAO,CAACE,aAAD,CAAR,CAAd;AACD;;AACD;;AACF;AACE;AAdJ;AAgBD,GArBe,EAsBhB,CAACF,OAAD,EAAUE,aAAV,EAAyBC,gBAAzB,EAA2CuB,cAA3C,CAtBgB,CAAlB;AAyBA,sBACE,gCAAC,eAAD;AAAiB,IAAA,SAAS,EAAE7C,SAA5B;AAAuC,IAAA,KAAK,EAAET;AAA9C,kBACE;AAAK,IAAA,SAAS,EAAC;AAAf,kBACE;AAAK,IAAA,SAAS,EAAC;AAAf,kBACE,gCAAC,aAAD;AAAQ,IAAA,MAAM,EAAC;AAAf,IADF,CADF,eAIE,gCAAC,wBAAD;AACE,IAAA,IAAI,EAAC,MADP;AAEE,IAAA,QAAQ,EAAEkC,QAFZ;AAGE,IAAA,MAAM,EAAEkB,MAHV;AAIE,IAAA,OAAO,EAAEC,OAJX;AAKE,IAAA,SAAS,EAAEQ,SALb;AAME,IAAA,KAAK,EAAExC,UANT;AAOE,IAAA,WAAW,EACTD,IAAI,GACAA,IAAI,CAACkD,aAAL,CAAmB;AAACC,MAAAA,EAAE,EAAE,gBAAL;AAAuBC,MAAAA,cAAc,EAAE7F;AAAvC,KAAnB,CADA,GAEAA;AAVR,IAJF,EAiBG8C,UAAU,gBACT;AAAK,IAAA,SAAS,EAAC;AAAf,kBACE,gCAAC,aAAD;AAAQ,IAAA,MAAM,EAAC,MAAf;AAAsB,IAAA,OAAO,EAAEmC;AAA/B,IADF,CADS,GAIP,IArBN,CADF,EAyBGrC,WAAW,gBACV;AAAK,IAAA,SAAS,EAAC;AAAf,KACGK,OAAO,CAAC6C,GAAR,CAAY,UAAC5D,IAAD,EAAO6D,KAAP;AAAA,wBACX;AACE,MAAA,GAAG,EAAEA,KADP;AAEE,MAAA,SAAS,EAAE,4BAAW,eAAX,EAA4B;AAACC,QAAAA,MAAM,EAAE7C,aAAa,KAAK4C;AAA3B,OAA5B,CAFb;AAGE,MAAA,OAAO,EAAE;AAAA,eAAMpB,cAAc,CAACzC,IAAD,CAApB;AAAA;AAHX,OAKGD,UAAU,CAACC,IAAD,CALb,CADW;AAAA,GAAZ,CADH,CADU,GAYR,IArCN,CADF;AAyCD,CAhKD;;eAkKe,2BAAWN,QAAX,C","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, {useCallback, useMemo, useState} from 'react';\nimport styled from 'styled-components';\nimport classnames from 'classnames';\nimport MapboxClient from 'mapbox';\nimport {injectIntl} from 'react-intl';\nimport {WebMercatorViewport} from 'viewport-mercator-project';\nimport KeyEvent from 'constants/keyevent';\nimport {Input} from 'components/common/styled-components';\nimport {Search, Delete} from 'components/common/icons';\n\n// matches only valid coordinates\nconst COORDINATE_REGEX_STRING =\n  '^[-+]?([1-8]?\\\\d(\\\\.\\\\d+)?|90(\\\\.0+)?),\\\\s*[-+]?(180(\\\\.0+)?|((1[0-7]\\\\d)|([1-9]?\\\\d))(\\\\.\\\\d+)?)';\nconst COORDINATE_REGEX = RegExp(COORDINATE_REGEX_STRING);\n\nconst PLACEHOLDER = 'Enter an address or coordinates, ex 37.79,-122.40';\n\nlet debounceTimeout = null;\n\nexport const testForCoordinates = query => {\n  const isValid = COORDINATE_REGEX.test(query.trim());\n\n  if (!isValid) {\n    return [isValid, query];\n  }\n\n  const tokens = query.trim().split(',');\n\n  return [isValid, Number(tokens[0]), Number(tokens[1])];\n};\n\nconst StyledContainer = styled.div`\n  position: relative;\n  color: ${props => props.theme.textColor};\n\n  .geocoder-input {\n    box-shadow: ${props => props.theme.boxShadow};\n\n    .geocoder-input__search {\n      position: absolute;\n      height: ${props => props.theme.geocoderInputHeight}px;\n      width: 30px;\n      padding-left: 6px;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      color: ${props => props.theme.subtextColor};\n    }\n\n    input {\n      padding: 4px 36px;\n      height: ${props => props.theme.geocoderInputHeight}px;\n      caret-color: unset;\n    }\n  }\n\n  .geocoder-results {\n    box-shadow: ${props => props.theme.boxShadow};\n    background-color: ${props => props.theme.panelBackground};\n    position: absolute;\n    width: ${props => (Number.isFinite(props.width) ? props.width : props.theme.geocoderWidth)}px;\n    margin-top: ${props => props.theme.dropdownWapperMargin}px;\n  }\n\n  .geocoder-item {\n    ${props => props.theme.dropdownListItem};\n    ${props => props.theme.textTruncate};\n\n    &.active {\n      background-color: ${props => props.theme.dropdownListHighlightBg};\n    }\n  }\n\n  .remove-result {\n    position: absolute;\n    right: 16px;\n    top: 0px;\n    height: ${props => props.theme.geocoderInputHeight}px;\n    display: flex;\n    align-items: center;\n\n    :hover {\n      cursor: pointer;\n      color: ${props => props.theme.textColorHl};\n    }\n  }\n`;\n\n/** @type {import('./geocoder').GeocoderComponent} */\nconst GeoCoder = ({\n  mapboxApiAccessToken,\n  className = '',\n  limit = 5,\n  timeout = 300,\n  formatItem = item => item.place_name,\n  viewport,\n  onSelected,\n  onDeleteMarker,\n  transitionDuration,\n  pointZoom,\n  width,\n  intl\n}) => {\n  const [inputValue, setInputValue] = useState('');\n  const [showResults, setShowResults] = useState(false);\n  const [showDelete, setShowDelete] = useState(false);\n  /** @type {import('./geocoder').Results} */\n  const initialResults = [];\n  const [results, setResults] = useState(initialResults);\n  const [selectedIndex, setSelectedIndex] = useState(0);\n\n  const client = useMemo(() => new MapboxClient(mapboxApiAccessToken), [mapboxApiAccessToken]);\n\n  const onChange = useCallback(\n    event => {\n      const queryString = event.target.value;\n      setInputValue(queryString);\n      const [hasValidCoordinates, longitude, latitude] = testForCoordinates(queryString);\n      if (hasValidCoordinates) {\n        setResults([{center: [latitude, longitude], place_name: queryString}]);\n      } else {\n        clearTimeout(debounceTimeout);\n        debounceTimeout = setTimeout(async () => {\n          if (limit > 0 && Boolean(queryString)) {\n            try {\n              const response = await client.geocodeForward(queryString, {limit});\n              if (response.entity.features) {\n                setShowResults(true);\n                setResults(response.entity.features);\n              }\n            } catch (e) {\n              // TODO: show geocode error\n              // eslint-disable-next-line no-console\n              console.log(e);\n            }\n          }\n        }, timeout);\n      }\n    },\n    [client, limit, timeout, setResults, setShowResults]\n  );\n\n  const onBlur = useCallback(() => {\n    setTimeout(() => {\n      setShowResults(false);\n    }, timeout);\n  }, [setShowResults, timeout]);\n\n  const onFocus = useCallback(() => setShowResults(true), [setShowResults]);\n\n  const onItemSelected = useCallback(\n    item => {\n      let newViewport = new WebMercatorViewport(viewport);\n      const {bbox, center} = item;\n\n      newViewport = bbox\n        ? newViewport.fitBounds([\n            [bbox[0], bbox[1]],\n            [bbox[2], bbox[3]]\n          ])\n        : {\n            longitude: center[0],\n            latitude: center[1],\n            zoom: pointZoom\n          };\n\n      const {longitude, latitude, zoom} = newViewport;\n\n      onSelected({...viewport, ...{longitude, latitude, zoom, transitionDuration}}, item);\n\n      setShowResults(false);\n      setInputValue(formatItem(item));\n      setShowDelete(true);\n    },\n    [viewport, onSelected, transitionDuration, pointZoom, formatItem]\n  );\n\n  const onMarkDeleted = useCallback(() => {\n    setShowDelete(false);\n    setInputValue('');\n    onDeleteMarker();\n  }, [onDeleteMarker]);\n\n  const onKeyDown = useCallback(\n    e => {\n      if (!results || results.length === 0) {\n        return;\n      }\n      switch (e.keyCode) {\n        case KeyEvent.DOM_VK_UP:\n          setSelectedIndex(selectedIndex > 0 ? selectedIndex - 1 : selectedIndex);\n          break;\n        case KeyEvent.DOM_VK_DOWN:\n          setSelectedIndex(selectedIndex < results.length - 1 ? selectedIndex + 1 : selectedIndex);\n          break;\n        case KeyEvent.DOM_VK_ENTER:\n        case KeyEvent.DOM_VK_RETURN:\n          if (results[selectedIndex]) {\n            onItemSelected(results[selectedIndex]);\n          }\n          break;\n        default:\n          break;\n      }\n    },\n    [results, selectedIndex, setSelectedIndex, onItemSelected]\n  );\n\n  return (\n    <StyledContainer className={className} width={width}>\n      <div className=\"geocoder-input\">\n        <div className=\"geocoder-input__search\">\n          <Search height=\"20px\" />\n        </div>\n        <Input\n          type=\"text\"\n          onChange={onChange}\n          onBlur={onBlur}\n          onFocus={onFocus}\n          onKeyDown={onKeyDown}\n          value={inputValue}\n          placeholder={\n            intl\n              ? intl.formatMessage({id: 'geocoder.title', defaultMessage: PLACEHOLDER})\n              : PLACEHOLDER\n          }\n        />\n        {showDelete ? (\n          <div className=\"remove-result\">\n            <Delete height=\"12px\" onClick={onMarkDeleted} />\n          </div>\n        ) : null}\n      </div>\n\n      {showResults ? (\n        <div className=\"geocoder-results\">\n          {results.map((item, index) => (\n            <div\n              key={index}\n              className={classnames('geocoder-item', {active: selectedIndex === index})}\n              onClick={() => onItemSelected(item)}\n            >\n              {formatItem(item)}\n            </div>\n          ))}\n        </div>\n      ) : null}\n    </StyledContainer>\n  );\n};\n\nexport default injectIntl(GeoCoder);\n"]}