| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  | import { ImageUploader } from "./ImageUploader" | 
					
						
							| 
									
										
										
										
											2024-09-28 22:42:56 +02:00
										 |  |  | import { AuthorizedPanoramax, ImageData, Panoramax, PanoramaxXYZ } from "panoramax-js/dist" | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  | import ExifReader from "exifreader" | 
					
						
							|  |  |  | import ImageProvider, { ProvidedImage } from "./ImageProvider" | 
					
						
							|  |  |  | import BaseUIElement from "../../UI/BaseUIElement" | 
					
						
							|  |  |  | import { LicenseInfo } from "./LicenseInfo" | 
					
						
							|  |  |  | import { GeoOperations } from "../GeoOperations" | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  | import Constants from "../../Models/Constants" | 
					
						
							|  |  |  | import { Store, Stores, UIEventSource } from "../UIEventSource" | 
					
						
							| 
									
										
										
										
											2024-09-30 01:08:07 +02:00
										 |  |  | import SvelteUIElement from "../../UI/Base/SvelteUIElement" | 
					
						
							|  |  |  | import Panoramax_bw from "../../assets/svg/Panoramax_bw.svelte" | 
					
						
							|  |  |  | import Link from "../../UI/Base/Link" | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default class PanoramaxImageProvider extends ImageProvider { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public static readonly singleton = new PanoramaxImageProvider() | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |     private static readonly xyz = new PanoramaxXYZ() | 
					
						
							|  |  |  |     private static defaultPanoramax = new AuthorizedPanoramax(Constants.panoramax.url, Constants.panoramax.token) | 
					
						
							| 
									
										
										
										
											2024-09-30 01:08:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-27 03:26:17 +02:00
										 |  |  |     public defaultKeyPrefixes: string[] = ["panoramax"] | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |     public readonly name: string = "panoramax" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |     private static knownMeta: Record<string, { data: ImageData, time: Date }> = {} | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-30 01:08:07 +02:00
										 |  |  |     public SourceIcon(img?: { id: string, url: string, host?: string }, location?: { lon: number; lat: number; }): BaseUIElement { | 
					
						
							|  |  |  |         const p = new Panoramax(img.host) | 
					
						
							|  |  |  |         return new Link(new SvelteUIElement(Panoramax_bw), p.createViewLink({ | 
					
						
							|  |  |  |             imageId: img?.id, | 
					
						
							|  |  |  |             location | 
					
						
							|  |  |  |         }), true) | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |     public addKnownMeta(meta: ImageData) { | 
					
						
							|  |  |  |         PanoramaxImageProvider.knownMeta[meta.id] = { data: meta, time: new Date() } | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Tries to get the entry from the mapcomplete-panoramax instance. Might return undefined | 
					
						
							|  |  |  |      * @param id | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private async getInfoFromMapComplete(id: string): Promise<{ data: ImageData, url: string }> { | 
					
						
							|  |  |  |         const sequence = "6e702976-580b-419c-8fb3-cf7bd364e6f8" // We always reuse this sequence
 | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         const url = `https://panoramax.mapcomplete.org/` | 
					
						
							| 
									
										
										
										
											2024-09-30 01:08:07 +02:00
										 |  |  |         const data = await PanoramaxImageProvider.defaultPanoramax.imageInfo(id, sequence) | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         return { url, data } | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private async getInfoFromXYZ(imageId: string): Promise<{ data: ImageData, url: string }> { | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         const data = await PanoramaxImageProvider.xyz.imageInfo(imageId) | 
					
						
							|  |  |  |         return { data, url: "https://api.panoramax.xyz/" } | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Reads a geovisio-somewhat-looking-like-geojson object and converts it to a provided image | 
					
						
							|  |  |  |      * @param meta | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |     private featureToImage(info: { data: ImageData, url: string }) { | 
					
						
							|  |  |  |         const meta = info?.data | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |         if (!meta) { | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         const url = info.url | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         function makeAbsolute(s: string) { | 
					
						
							|  |  |  |             if (!s.startsWith("https://") && !s.startsWith("http://")) { | 
					
						
							|  |  |  |                 const parsed = new URL(url) | 
					
						
							|  |  |  |                 return parsed.protocol + "//" + parsed.host + s | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |             return s | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const [lon, lat] = GeoOperations.centerpointCoordinates(meta) | 
					
						
							|  |  |  |         return <ProvidedImage>{ | 
					
						
							|  |  |  |             id: meta.id, | 
					
						
							|  |  |  |             url: makeAbsolute(meta.assets.sd.href), | 
					
						
							|  |  |  |             url_hd: makeAbsolute(meta.assets.hd.href), | 
					
						
							| 
									
										
										
										
											2024-09-30 01:08:07 +02:00
										 |  |  |             host: meta["links"].find(l => l.rel === "root")?.href, | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |             lon, lat, | 
					
						
							|  |  |  |             key: "panoramax", | 
					
						
							|  |  |  |             provider: this, | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |             status: meta.properties["geovisio:status"], | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |             rotation: Number(meta.properties["view:azimuth"]), | 
					
						
							|  |  |  |             date: new Date(meta.properties.datetime), | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private async getInfoFor(id: string): Promise<{ data: ImageData, url: string }> { | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         if (!id.match(/^[a-zA-Z0-9-]+$/)) { | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const cached = PanoramaxImageProvider.knownMeta[id] | 
					
						
							|  |  |  |         if (cached) { | 
					
						
							| 
									
										
										
										
											2024-09-28 22:42:56 +02:00
										 |  |  |             if (new Date().getTime() - cached.time.getTime() < 1000) { | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-28 22:42:56 +02:00
										 |  |  |                 return { data: cached.data, url: undefined } | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |             return await this.getInfoFromMapComplete(id) | 
					
						
							|  |  |  |         } catch (e) { | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |             console.debug(e) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         try { | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |             return await this.getInfoFromXYZ(id) | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         } catch (e) { | 
					
						
							| 
									
										
										
										
											2024-09-28 22:42:56 +02:00
										 |  |  |             console.debug(e) | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         return undefined | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-30 01:08:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |     public async ExtractUrls(key: string, value: string): Promise<ProvidedImage[]> { | 
					
						
							| 
									
										
										
										
											2024-09-30 01:08:07 +02:00
										 |  |  |         if (!Panoramax.isId(value)) { | 
					
						
							| 
									
										
										
										
											2024-09-28 22:42:56 +02:00
										 |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         return [await this.getInfoFor(value).then(r => this.featureToImage(<any>r))] | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |     getRelevantUrls(tags: Record<string, string>, prefixes: string[]): Store<ProvidedImage[]> { | 
					
						
							|  |  |  |         const source = UIEventSource.FromPromise(super.getRelevantUrlsFor(tags, prefixes)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         function hasLoading(data: ProvidedImage[]) { | 
					
						
							| 
									
										
										
										
											2024-09-28 22:42:56 +02:00
										 |  |  |             if (data === undefined) { | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |                 return true | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return data?.some(img => img?.status !== undefined && img?.status !== "ready" && img?.status !== "broken") | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Stores.Chronic(1500, () => | 
					
						
							|  |  |  |             hasLoading(source.data), | 
					
						
							|  |  |  |         ).addCallback(_ => { | 
					
						
							|  |  |  |             console.log("UPdating... ") | 
					
						
							|  |  |  |             super.getRelevantUrlsFor(tags, prefixes).then(data => { | 
					
						
							|  |  |  |                 console.log("New panoramax data is", data, hasLoading(data)) | 
					
						
							|  |  |  |                 source.set(data) | 
					
						
							|  |  |  |                 return !hasLoading(data) | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return source | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public async DownloadAttribution(providedImage: { url: string; id: string; }): Promise<LicenseInfo> { | 
					
						
							|  |  |  |         const meta = await this.getInfoFor(providedImage.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             artist: meta.data.providers.at(-1).name, // We take the last provider, as that one probably contain the username of the uploader
 | 
					
						
							|  |  |  |             date: new Date(meta.data.properties["datetime"]), | 
					
						
							|  |  |  |             licenseShortName: meta.data.properties["geovisio:license"], | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public apiUrls(): string[] { | 
					
						
							|  |  |  |         return ["https://panoramax.mapcomplete.org", "https://panoramax.xyz"] | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export class PanoramaxUploader implements ImageUploader { | 
					
						
							|  |  |  |     private readonly _panoramax: AuthorizedPanoramax | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     constructor(url: string, token: string) { | 
					
						
							|  |  |  |         this._panoramax = new AuthorizedPanoramax(url, token) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     async uploadImage(blob: File, currentGps: [number, number], author: string): Promise<{ | 
					
						
							|  |  |  |         key: string; | 
					
						
							|  |  |  |         value: string; | 
					
						
							| 
									
										
										
										
											2024-09-27 03:26:17 +02:00
										 |  |  |         absoluteUrl: string | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |     }> { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const tags = await ExifReader.load(blob) | 
					
						
							|  |  |  |         const hasDate = tags.DateTime !== undefined | 
					
						
							|  |  |  |         const hasGPS = tags.GPSLatitude !== undefined && tags.GPSLongitude !== undefined | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const [lon, lat] = currentGps | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const p = this._panoramax | 
					
						
							|  |  |  |         const defaultSequence = (await p.mySequences())[0] | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |         const img = <ImageData>await p.addImage(blob, defaultSequence, { | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |             lat: !hasGPS ? lat : undefined, | 
					
						
							|  |  |  |             lon: !hasGPS ? lon : undefined, | 
					
						
							|  |  |  |             datetime: !hasDate ? new Date().toISOString() : undefined, | 
					
						
							|  |  |  |             exifOverride: { | 
					
						
							|  |  |  |                 Artist: author, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |         PanoramaxImageProvider.singleton.addKnownMeta(img) | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             key: "panoramax", | 
					
						
							|  |  |  |             value: img.id, | 
					
						
							| 
									
										
										
										
											2024-09-28 02:04:14 +02:00
										 |  |  |             absoluteUrl: img.assets.hd.href, | 
					
						
							| 
									
										
										
										
											2024-09-26 19:15:20 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |