import {Store, UIEventSource} from "../UIEventSource"; import Svg from "../../Svg"; import {LocalStorageSource} from "../Web/LocalStorageSource"; import {VariableUiElement} from "../../UI/Base/VariableUIElement"; import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig"; import {QueryParameters} from "../Web/QueryParameters"; import {BBox} from "../BBox"; import Constants from "../../Models/Constants"; import SimpleFeatureSource from "../FeatureSource/Sources/SimpleFeatureSource"; export interface GeoLocationPointProperties { id: "gps", "user:location": "yes", "date": string, "latitude": number "longitude": number, "speed": number, "accuracy": number "heading": number "altitude": number } export default class GeoLocationHandler extends VariableUiElement { private readonly currentLocation?: SimpleFeatureSource /** * Wether or not the geolocation is active, aka the user requested the current location */ private readonly _isActive: UIEventSource; /** * Wether or not the geolocation is locked, aka the user requested the current location and wants the crosshair to follow the user */ private readonly _isLocked: UIEventSource; /** * The callback over the permission API * @private */ private readonly _permission: UIEventSource; /** * Literally: _currentGPSLocation.data != undefined * @private */ private readonly _hasLocation: Store; private readonly _currentGPSLocation: UIEventSource; /** * Kept in order to update the marker * @private */ private readonly _leafletMap: UIEventSource; /** * The date when the user requested the geolocation. If we have a location, it'll autozoom to it the first 30 secs */ private _lastUserRequest: UIEventSource; /** * A small flag on localstorage. If the user previously granted the geolocation, it will be set. * On firefox, the permissions api is broken (probably fingerprint resistiance) and "granted + don't ask again" doesn't stick between sessions. * * Instead, we set this flag. If this flag is set upon loading the page, we start geolocating immediately. * If the user denies the geolocation this time, we unset this flag * @private */ private readonly _previousLocationGrant: UIEventSource; private readonly _layoutToUse: LayoutConfig; constructor( state: { selectedElement: UIEventSource; currentUserLocation?: SimpleFeatureSource, leafletMap: UIEventSource, layoutToUse: LayoutConfig, featureSwitchGeolocation: UIEventSource } ) { const currentGPSLocation = new UIEventSource(undefined, "GPS-coordinate") const leafletMap = state.leafletMap const initedAt = new Date() let autozoomDone = false; const hasLocation = currentGPSLocation.map( (location) => location !== undefined ); const previousLocationGrant = LocalStorageSource.Get( "geolocation-permissions" ); const isActive = new UIEventSource(false); const isLocked = new UIEventSource(false); const permission = new UIEventSource(""); const lastClick = new UIEventSource(undefined); const lastClickWithinThreeSecs = lastClick.map(lastClick => { if (lastClick === undefined) { return false; } const timeDiff = (new Date().getTime() - lastClick.getTime()) / 1000 return timeDiff <= 3 }) const latLonGiven = QueryParameters.wasInitialized("lat") && QueryParameters.wasInitialized("lon") const willFocus = lastClick.map(lastUserRequest => { const timeDiffInited = (new Date().getTime() - initedAt.getTime()) / 1000 if (!latLonGiven && !autozoomDone && timeDiffInited < Constants.zoomToLocationTimeout) { return true } if (lastUserRequest === undefined) { return false; } const timeDiff = (new Date().getTime() - lastUserRequest.getTime()) / 1000 return timeDiff <= Constants.zoomToLocationTimeout }) lastClick.addCallbackAndRunD(_ => { window.setTimeout(() => { if (lastClickWithinThreeSecs.data || willFocus.data) { lastClick.ping() } }, 500) }) super( hasLocation.map( (hasLocationData) => { if (permission.data === "denied") { return Svg.location_refused_svg(); } if (!isActive.data) { return Svg.location_empty_svg() } if (!hasLocationData) { // Position not yet found but we are active: we spin to indicate activity // If will focus is active too, we indicate this differently const icon = willFocus.data ? Svg.location_svg() : Svg.location_empty_svg() icon.SetStyle("animation: spin 4s linear infinite;") return icon; } if (isLocked.data) { return Svg.location_locked_svg() } if (lastClickWithinThreeSecs.data) { return Svg.location_unlocked_svg() } // We have a location, so we show a dot in the center return Svg.location_svg(); }, [isActive, isLocked, permission, lastClickWithinThreeSecs, willFocus] ) ); this.SetClass("mapcontrol") this._isActive = isActive; this._isLocked = isLocked; this._permission = permission this._previousLocationGrant = previousLocationGrant; this._currentGPSLocation = currentGPSLocation; this._leafletMap = leafletMap; this._layoutToUse = state.layoutToUse; this._hasLocation = hasLocation; this._lastUserRequest = lastClick const self = this; const currentPointer = this._isActive.map( (isActive) => { if (isActive && !self._hasLocation.data) { return "cursor-wait"; } return "cursor-pointer"; }, [this._hasLocation] ); currentPointer.addCallbackAndRun((pointerClass) => { self.RemoveClass("cursor-wait") self.RemoveClass("cursor-pointer") self.SetClass(pointerClass); }); this.onClick(() => { /* * If the previous click was within 3 seconds (and we have an active location), then we lock to the location */ if (self._hasLocation.data) { if (isLocked.data) { isLocked.setData(false) } else if (lastClick.data !== undefined) { const timeDiff = (new Date().getTime() - lastClick.data.getTime()) / 1000 if (timeDiff <= 3) { isLocked.setData(true) lastClick.setData(undefined) } else { lastClick.setData(new Date()) } } else { lastClick.setData(new Date()) } } self.init(true, true); }); const doAutoZoomToLocation = !latLonGiven && state.featureSwitchGeolocation.data && state.selectedElement.data !== undefined this.init(false, doAutoZoomToLocation); isLocked.addCallbackAndRunD(isLocked => { if (isLocked) { leafletMap.data?.dragging?.disable() } else { leafletMap.data?.dragging?.enable() } }) this.currentLocation = state.currentUserLocation this._currentGPSLocation.addCallback((location) => { self._previousLocationGrant.setData("granted"); const feature = { "type": "Feature", properties: { id: "gps", "user:location": "yes", "date": new Date().toISOString(), "latitude": location.latitude, "longitude": location.longitude, "speed": location.speed, "accuracy": location.accuracy, "heading": location.heading, "altitude": location.altitude }, geometry: { type: "Point", coordinates: [location.longitude, location.latitude], } } self.currentLocation?.features?.setData([{feature, freshness: new Date()}]) if (willFocus.data) { console.log("Zooming to user location: willFocus is set") lastClick.setData(undefined); autozoomDone = true; self.MoveToCurrentLocation(16); } else if (self._isLocked.data) { self.MoveToCurrentLocation(); } }); } private init(askPermission: boolean, zoomToLocation: boolean) { const self = this; if (self._isActive.data) { self.MoveToCurrentLocation(16); return; } if (typeof navigator === "undefined") { return } try { navigator?.permissions ?.query({name: "geolocation"}) ?.then(function (status) { console.log("Geolocation permission is ", status.state); if (status.state === "granted") { self.StartGeolocating(zoomToLocation); } self._permission.setData(status.state); status.onchange = function () { self._permission.setData(status.state); }; }); } catch (e) { console.error(e); } if (askPermission) { self.StartGeolocating(zoomToLocation); } else if (this._previousLocationGrant.data === "granted") { this._previousLocationGrant.setData(""); self.StartGeolocating(zoomToLocation); } } /** * Moves to the currently loaded location. * * // Should move to any location * let resultingLocation = undefined * let resultingzoom = 1 * const state = { * selectedElement: new UIEventSource(undefined); * currentUserLocation: undefined , * leafletMap: new UIEventSource({getZoom: () => resultingzoom; setView: (loc, zoom) => {resultingLocation = loc; resultingzoom = zoom}), * layoutToUse: new LayoutConfig({ * id: 'test', * title: {"en":"test"} * description: "A testing theme", * layers: [] * }), * featureSwitchGeolocation : new UIEventSource(true) * } * const handler = new GeoLocationHandler(state) * handler._currentGPSLocation.setData( {latitude : 51.3, longitude: 4.1}) * handler.MoveToCurrentLocation() * resultingLocation // => [51.3, 4.1] * handler._currentGPSLocation.setData( {latitude : 60, longitude: 60) // out of bounds * handler.MoveToCurrentLocation() * resultingLocation // => [60, 60] * * // should refuse to move if out of bounds * let resultingLocation = undefined * let resultingzoom = 1 * const state = { * selectedElement: new UIEventSource(undefined); * currentUserLocation: undefined , * leafletMap: new UIEventSource({getZoom: () => resultingzoom; setView: (loc, zoom) => {resultingLocation = loc; resultingzoom = zoom}), * layoutToUse: new LayoutConfig({ * id: 'test', * title: {"en":"test"} * "lockLocation": [ [ 2.1, 50.4], [6.4, 51.54 ]], * description: "A testing theme", * layers: [] * }), * featureSwitchGeolocation : new UIEventSource(true) * } * const handler = new GeoLocationHandler(state) * handler._currentGPSLocation.setData( {latitude : 51.3, longitude: 4.1}) * handler.MoveToCurrentLocation() * resultingLocation // => [51.3, 4.1] * handler._currentGPSLocation.setData( {latitude : 60, longitude: 60) // out of bounds * handler.MoveToCurrentLocation() * resultingLocation // => [51.3, 4.1] */ private MoveToCurrentLocation(targetZoom?: number) { const location = this._currentGPSLocation.data; this._lastUserRequest.setData(undefined); if ( this._currentGPSLocation.data.latitude === 0 && this._currentGPSLocation.data.longitude === 0 ) { console.debug("Not moving to GPS-location: it is null island"); return; } // We check that the GPS location is not out of bounds const b = this._layoutToUse.lockLocation; let inRange = true; if (b) { if (b !== true) { // B is an array with our locklocation inRange = new BBox(b).contains([location.longitude, location.latitude]) } } if (!inRange) { console.log("Not zooming to GPS location: out of bounds", b, location); } else { const currentZoom = this._leafletMap.data.getZoom() this._leafletMap.data.setView([location.latitude, location.longitude], Math.max(targetZoom ?? 0, currentZoom)); } } private StartGeolocating(zoomToGPS = true) { const self = this; this._lastUserRequest.setData(zoomToGPS ? new Date() : new Date(0)) if (self._permission.data === "denied") { self._previousLocationGrant.setData(""); self._isActive.setData(false) return ""; } if (this._currentGPSLocation.data !== undefined) { this.MoveToCurrentLocation(16); } if (self._isActive.data) { return; } self._isActive.setData(true); navigator.geolocation.watchPosition( function (position) { self._currentGPSLocation.setData(position.coords); }, function () { console.warn("Could not get location with navigator.geolocation"); }, { enableHighAccuracy: true } ); } }