import * as L from "leaflet"; import {UIEventSource} from "../UIEventSource"; import Svg from "../../Svg"; import Img from "../../UI/Base/Img"; import {LocalStorageSource} from "../Web/LocalStorageSource"; import {VariableUiElement} from "../../UI/Base/VariableUIElement"; import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig"; import {QueryParameters} from "../Web/QueryParameters"; export default class GeoLocationHandler extends VariableUiElement { /** * Wether or not the geolocation is active, aka the user requested the current location * @private */ 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 */ private readonly _isLocked: UIEventSource; /** * The callback over the permission API * @private */ private readonly _permission: UIEventSource; /*** * The marker on the map, in order to update it * @private */ private _marker: L.Marker; /** * Literally: _currentGPSLocation.data != undefined * @private */ private readonly _hasLocation: UIEventSource; private readonly _currentGPSLocation: UIEventSource<{ latlng: any; accuracy: number; }>; /** * 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 */ private _lastUserRequest: Date; /** * 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: UIEventSource; constructor( currentGPSLocation: UIEventSource<{ latlng: any; accuracy: number }>, leafletMap: UIEventSource, layoutToUse: UIEventSource ) { 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 }) lastClick.addCallbackAndRunD(_ => { window.setTimeout(() => { if (lastClickWithinThreeSecs.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 const icon = 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] ) ); this.SetClass("mapcontrol") this._isActive = isActive; this._isLocked = isLocked; this._permission = permission this._previousLocationGrant = previousLocationGrant; this._currentGPSLocation = currentGPSLocation; this._leafletMap = leafletMap; this._layoutToUse = layoutToUse; this._hasLocation = hasLocation; 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.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 latLonGiven = QueryParameters.wasInitialized("lat") && QueryParameters.wasInitialized("lon") this.init(false, !latLonGiven); isLocked.addCallbackAndRunD(isLocked => { if (isLocked) { leafletMap.data?.dragging?.disable() } else { leafletMap.data?.dragging?.enable() } }) this._currentGPSLocation.addCallback((location) => { self._previousLocationGrant.setData("granted"); const timeSinceRequest = (new Date().getTime() - (self._lastUserRequest?.getTime() ?? 0)) / 1000; if (timeSinceRequest < 30) { self.MoveToCurrentLoction(16); } else if (self._isLocked.data) { self.MoveToCurrentLoction(); } let color = "#1111cc"; try { color = getComputedStyle(document.body).getPropertyValue( "--catch-detail-color" ); } catch (e) { console.error(e); } const icon = L.icon({ iconUrl: Img.AsData(Svg.location.replace(/#000000/g, color).replace(/#000/g, color)), iconSize: [40, 40], // size of the icon iconAnchor: [20, 20], // point of the icon which will correspond to marker's location }); const map = self._leafletMap.data; const newMarker = L.marker(location.latlng, {icon: icon}); newMarker.addTo(map); if (self._marker !== undefined) { map.removeLayer(self._marker); } self._marker = newMarker; }); } private init(askPermission: boolean, forceZoom: boolean) { const self = this; if (self._isActive.data) { self.MoveToCurrentLoction(16); return; } try { navigator?.permissions ?.query({name: "geolocation"}) ?.then(function (status) { console.log("Geolocation permission is ", status.state); if (status.state === "granted") { self.StartGeolocating(forceZoom); } self._permission.setData(status.state); status.onchange = function () { self._permission.setData(status.state); }; }); } catch (e) { console.error(e); } if (askPermission) { self.StartGeolocating(forceZoom); } else if (this._previousLocationGrant.data === "granted") { this._previousLocationGrant.setData(""); self.StartGeolocating(forceZoom); } } private MoveToCurrentLoction(targetZoom?: number) { const location = this._currentGPSLocation.data; this._lastUserRequest = undefined; if ( this._currentGPSLocation.data.latlng[0] === 0 && this._currentGPSLocation.data.latlng[1] === 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.data.lockLocation; let inRange = true; if (b) { if (b !== true) { // B is an array with our locklocation inRange = b[0][0] <= location.latlng[0] && location.latlng[0] <= b[1][0] && b[0][1] <= location.latlng[1] && location.latlng[1] <= b[1][1]; } } if (!inRange) { console.log( "Not zooming to GPS location: out of bounds", b, location.latlng ); } else { this._leafletMap.data.setView(location.latlng, targetZoom); } } private StartGeolocating(zoomToGPS = true) { const self = this; this._lastUserRequest = 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.MoveToCurrentLoction(16); } if (self._isActive.data) { return; } self._isActive.setData(true); navigator.geolocation.watchPosition( function (position) { self._currentGPSLocation.setData({ latlng: [position.coords.latitude, position.coords.longitude], accuracy: position.coords.accuracy, }); }, function () { console.warn("Could not get location with navigator.geolocation"); }, { enableHighAccuracy: true } ); } }