| 
									
										
										
										
											2020-10-02 19:00:24 +02:00
										 |  |  | import * as L from "leaflet"; | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  | import {UIEventSource} from "../UIEventSource"; | 
					
						
							|  |  |  | import {Utils} from "../../Utils"; | 
					
						
							| 
									
										
										
										
											2020-11-05 12:28:02 +01:00
										 |  |  | import Svg from "../../Svg"; | 
					
						
							| 
									
										
										
										
											2021-01-03 00:19:42 +01:00
										 |  |  | import Img from "../../UI/Base/Img"; | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  | import {LocalStorageSource} from "../Web/LocalStorageSource"; | 
					
						
							| 
									
										
										
										
											2021-05-27 18:55:37 +02:00
										 |  |  | import LayoutConfig from "../../Customizations/JSON/LayoutConfig"; | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  | import {VariableUiElement} from "../../UI/Base/VariableUIElement"; | 
					
						
							| 
									
										
										
										
											2020-06-28 02:42:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-28 00:45:49 +02:00
										 |  |  | export default class GeoLocationHandler extends VariableUiElement { | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Wether or not the geolocation is active, aka the user requested the current location | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private readonly _isActive: UIEventSource<boolean>; | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * 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<boolean>; | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * The callback over the permission API | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private readonly _permission: UIEventSource<string>; | 
					
						
							|  |  |  |     /*** | 
					
						
							|  |  |  |      * The marker on the map, in order to update it | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private _marker: L.Marker; | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Literally: _currentGPSLocation.data != undefined | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private readonly _hasLocation: UIEventSource<boolean>; | 
					
						
							|  |  |  |     private readonly _currentGPSLocation: UIEventSource<{ | 
					
						
							|  |  |  |         latlng: any; | 
					
						
							|  |  |  |         accuracy: number; | 
					
						
							|  |  |  |     }>; | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Kept in order to update the marker | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private readonly _leafletMap: UIEventSource<L.Map>; | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 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<string>; | 
					
						
							|  |  |  |     private readonly _layoutToUse: UIEventSource<LayoutConfig>; | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |     constructor( | 
					
						
							|  |  |  |         currentGPSLocation: UIEventSource<{ latlng: any; accuracy: number }>, | 
					
						
							|  |  |  |         leafletMap: UIEventSource<L.Map>, | 
					
						
							|  |  |  |         layoutToUse: UIEventSource<LayoutConfig> | 
					
						
							|  |  |  |     ) { | 
					
						
							|  |  |  |         const hasLocation = currentGPSLocation.map( | 
					
						
							|  |  |  |             (location) => location !== undefined | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         const previousLocationGrant = LocalStorageSource.Get( | 
					
						
							|  |  |  |             "geolocation-permissions" | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         const isActive = new UIEventSource<boolean>(false); | 
					
						
							|  |  |  |         const isLocked = new UIEventSource<boolean>(false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         super( | 
					
						
							|  |  |  |             hasLocation.map( | 
					
						
							|  |  |  |                 (hasLocationData) => { | 
					
						
							|  |  |  |                     if (isLocked.data) { | 
					
						
							|  |  |  |                         return Svg.crosshair_locked_ui(); | 
					
						
							|  |  |  |                     } else if (hasLocationData) { | 
					
						
							|  |  |  |                         return Svg.crosshair_blue_ui(); | 
					
						
							|  |  |  |                     } else if (isActive.data) { | 
					
						
							|  |  |  |                         return Svg.crosshair_blue_center_ui(); | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         return Svg.crosshair_ui(); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 [isActive, isLocked] | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |         this._isActive = isActive; | 
					
						
							|  |  |  |         this._isLocked = isLocked; | 
					
						
							|  |  |  |         this._permission = new UIEventSource<string>(""); | 
					
						
							|  |  |  |         this._previousLocationGrant = previousLocationGrant; | 
					
						
							|  |  |  |         this._currentGPSLocation = currentGPSLocation; | 
					
						
							|  |  |  |         this._leafletMap = leafletMap; | 
					
						
							|  |  |  |         this._layoutToUse = layoutToUse; | 
					
						
							|  |  |  |         this._hasLocation = hasLocation; | 
					
						
							|  |  |  |         const self = this; | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         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); | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         this.onClick(() => { | 
					
						
							|  |  |  |             if (self._hasLocation.data) { | 
					
						
							|  |  |  |                 self._isLocked.setData(!self._isLocked.data); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.init(true); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         this.init(false); | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         this._currentGPSLocation.addCallback((location) => { | 
					
						
							|  |  |  |             self._previousLocationGrant.setData("granted"); | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |             const timeSinceRequest = | 
					
						
							|  |  |  |                 (new Date().getTime() - (self._lastUserRequest?.getTime() ?? 0)) / 1000; | 
					
						
							|  |  |  |             if (timeSinceRequest < 30) { | 
					
						
							|  |  |  |                 self.MoveToCurrentLoction(16); | 
					
						
							|  |  |  |             } else if (self._isLocked.data) { | 
					
						
							|  |  |  |                 self.MoveToCurrentLoction(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |             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
 | 
					
						
							|  |  |  |             }); | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |             const map = self._leafletMap.data; | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |             const newMarker = L.marker(location.latlng, {icon: icon}); | 
					
						
							|  |  |  |             newMarker.addTo(map); | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |             if (self._marker !== undefined) { | 
					
						
							|  |  |  |                 map.removeLayer(self._marker); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self._marker = newMarker; | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |     private init(askPermission: boolean) { | 
					
						
							|  |  |  |         const self = this; | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         if (self._isActive.data) { | 
					
						
							|  |  |  |             self.MoveToCurrentLoction(16); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         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); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-18 19:48:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         if (askPermission) { | 
					
						
							|  |  |  |             self.StartGeolocating(true); | 
					
						
							|  |  |  |         } else if (this._previousLocationGrant.data === "granted") { | 
					
						
							|  |  |  |             this._previousLocationGrant.setData(""); | 
					
						
							| 
									
										
										
										
											2021-05-27 18:55:37 +02:00
										 |  |  |             self.StartGeolocating(false); | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-28 02:42:22 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |     private MoveToCurrentLoction(targetZoom?: number) { | 
					
						
							|  |  |  |         const location = this._currentGPSLocation.data; | 
					
						
							|  |  |  |         this._lastUserRequest = undefined; | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         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; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         // 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); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-29 20:30:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |     private StartGeolocating(zoomToGPS = true) { | 
					
						
							|  |  |  |         const self = this; | 
					
						
							|  |  |  |         console.log("Starting geolocation"); | 
					
						
							| 
									
										
										
										
											2020-06-28 02:42:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         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); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-02 21:03:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         console.log("Searching location using GPS"); | 
					
						
							| 
									
										
										
										
											2021-01-02 21:03:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         if (self._isActive.data) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self._isActive.setData(true); | 
					
						
							| 
									
										
										
										
											2020-07-31 01:45:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-26 09:46:54 +02:00
										 |  |  |         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 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							| 
									
										
										
										
											2020-06-28 02:42:22 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-13 14:39:50 +02:00
										 |  |  | } |