Source: geo-position.js

  1. import { logger } from "./shims";
  2. const initialGeoTimeoutSeconds = 1;
  3. const defaultCoords = {
  4. latitude: 1,
  5. longitude: 1
  6. };
  7. // for an initial rapid, low-accuracy position
  8. const fastGeolocationPositionOptions = {
  9. enableHighAccuracy: false,
  10. timeout: initialGeoTimeoutSeconds
  11. };
  12. // subsequent position monitoring should be high-accuracy
  13. const accurateGeolocationPositionOptions = {
  14. enableHighAccuracy: true
  15. };
  16. /** Responsible for tracking the user's position, when geo listening is enabled and the browser is capable
  17. * @property {Boolean} geoListenEnabled - whether or not the geo positioning system is enabled and available
  18. * @see https://developer.mozilla.org/en-US/docs/Web/API/Geolocation/Using_geolocation **/
  19. export class GeoPosition {
  20. /** Create a new GeoPosition.
  21. * @param {Object} navigator - provides access to geolocation system
  22. * @param {Object} options - parameters for initializing this GeoPosition
  23. * @param {Boolean} [options.geoListenEnabled = false] - whether or not to attempt to use geolocation **/
  24. constructor(navigator,options = {}) {
  25. this._navigator = navigator;
  26. this._initialGeolocationPromise = Promise.resolve(defaultCoords);
  27. if (this._navigator.geolocation && options.geoListenEnabled) {
  28. this.geoListenEnabled = true;
  29. } else {
  30. this.geoListenEnabled = false;
  31. }
  32. }
  33. /** @return {String} Human-readable representation of this GeoPosition **/
  34. toString() {
  35. return `GeoPosition (enabled: ${this.geoListenEnabled})`;
  36. }
  37. /** Attempts to get an initial rough geographic location for the listener, then sets up a callback
  38. * to update the position.
  39. * @param {Function} geoUpdateCallback - object that should receive geolocation coordinate updates
  40. * @see geoListenEnabled **/
  41. connect(geoUpdateCallback = () => {}) {
  42. if (!this.geoListenEnabled) {
  43. logger.info("Geolocation disabled");
  44. this._initialGeolocationPromise = Promise.resolve({});
  45. return;
  46. }
  47. logger.info("Initializing geolocation system");
  48. this._initialGeolocationPromise = new Promise((resolve,reject) => {
  49. this._navigator.geolocation.getCurrentPosition((initialPosition) => {
  50. let coords = initialPosition.coords;
  51. logger.info("Received initial geolocation",coords);
  52. geoUpdateCallback(coords);
  53. let geoWatchId = this._navigator.geolocation.watchPosition((updatedPosition) => {
  54. let newCoords = updatedPosition.coords;
  55. geoUpdateCallback(newCoords);
  56. },(error) => {
  57. logger.warn(`Unable to watch position: ${error.message} (code #${error.code})`);
  58. },accurateGeolocationPositionOptions);
  59. logger.info(`Monitoring geoposition updates (watch ID ${geoWatchId})`);
  60. resolve(coords);
  61. },function initialGeoError(error) {
  62. logger.warn(`Unable to get initial geolocation: ${error.message} (code #${error.code})`);
  63. resolve(defaultCoords);
  64. },fastGeolocationPositionOptions);
  65. });
  66. }
  67. /** Allows you to wait on the progress of the .connect() behavior, attempting to get an initial
  68. * estimate of the user's position. Note that this promise will never fail - if we cannot get an
  69. * accurate estimate, we fall back to default coordinates (currently latitude 1, longitude 1)
  70. * @return {Promise} Represents the attempt to get an initial estimate of the user's position **/
  71. waitForInitialGeolocation() {
  72. return this._initialGeolocationPromise;
  73. }
  74. }