import * as L from "leaflet"; import {UIEventSource} from "../UIEventSource"; import {UIElement} from "../../UI/UIElement"; import {Utils} from "../../Utils"; import Svg from "../../Svg"; import Img from "../../UI/Base/Img"; import {LocalStorageSource} from "../Web/LocalStorageSource"; import LayoutConfig from "../../Customizations/JSON/LayoutConfig"; import BaseUIElement from "../../UI/BaseUIElement"; import {VariableUiElement} from "../../UI/Base/VariableUIElement"; export default class GeoLocationHandler extends UIElement { /** * Wether or not the geolocation is active, aka the user requested the current location * @private */ private readonly _isActive: UIEventSource = new UIEventSource(false); /** * The callback over the permission API * @private */ private readonly _permission: UIEventSource = new 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 = LocalStorageSource.Get("geolocation-permissions"); private readonly _layoutToUse: UIEventSource; private readonly _element: BaseUIElement; constructor(currentGPSLocation: UIEventSource<{ latlng: any; accuracy: number }>, leafletMap: UIEventSource, layoutToUse: UIEventSource) { super(); this._currentGPSLocation = currentGPSLocation; this._leafletMap = leafletMap; this._layoutToUse = layoutToUse; this._hasLocation = currentGPSLocation.map((location) => location !== undefined); 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._element = new VariableUiElement( this._hasLocation.map(hasLocation => { if (hasLocation) { return Svg.crosshair_blue_ui() } if (self._isActive.data) { return Svg.crosshair_blue_center_ui(); } return Svg.crosshair_ui(); }, [this._isActive]) ); this.onClick(() => self.init(true)) self.init(false) } protected InnerRender(): string | BaseUIElement { return this._element } private init(askPermission: boolean) { const self = this; const map = this._leafletMap.data; this._currentGPSLocation.addCallback((location) => { self._previousLocationGrant.setData("granted"); const timeSinceRequest = (new Date().getTime() - (self._lastUserRequest?.getTime() ?? 0)) / 1000; if (timeSinceRequest < 30) { self.MoveToCurrentLoction(16) } 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.crosshair.replace(/#000000/g, color)), iconSize: [40, 40], // size of the icon iconAnchor: [20, 20], // point of the icon which will correspond to marker's location }) const newMarker = L.marker(location.latlng, {icon: icon}); newMarker.addTo(map); if (self._marker !== undefined) { map.removeLayer(self._marker); } self._marker = newMarker; }); try { navigator?.permissions?.query({name: 'geolocation'}) ?.then(function (status) { console.log("Geolocation is already", status) if (status.state === "granted") { self.StartGeolocating(false); } self._permission.setData(status.state); status.onchange = function () { self._permission.setData(status.state); } }); } catch (e) { console.error(e) } if (askPermission) { self.StartGeolocating(true); } else if (this._previousLocationGrant.data === "granted") { this._previousLocationGrant.setData(""); self.StartGeolocating(false); } } private locate() { const self = this; const map: any = this._leafletMap.data; if (navigator.geolocation) { navigator.geolocation.getCurrentPosition(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") }); return; } else { map.findAccuratePosition({ maxWait: 10000, // defaults to 10000 desiredAccuracy: 50 // defaults to 20 }); } } private MoveToCurrentLoction(targetZoom = 16) { 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; console.log("Starting geolocation") this._lastUserRequest = zoomToGPS ? new Date() : new Date(0); if (self._permission.data === "denied") { self._previousLocationGrant.setData(""); return ""; } if (this._currentGPSLocation.data !== undefined) { this.MoveToCurrentLoction(16) } console.log("Searching location using GPS") this.locate(); if (!self._isActive.data) { self._isActive.setData(true); Utils.DoEvery(60000, () => { if (document.visibilityState !== "visible") { console.log("Not starting gps: document not visible") return; } this.locate(); }) } } }