| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  | import { ImageUploader } from "./ImageUploader" | 
					
						
							|  |  |  | import LinkImageAction from "../Osm/Actions/LinkImageAction" | 
					
						
							|  |  |  | import FeaturePropertiesStore from "../FeatureSource/Actors/FeaturePropertiesStore" | 
					
						
							|  |  |  | import { OsmId, OsmTags } from "../../Models/OsmFeature" | 
					
						
							|  |  |  | import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig" | 
					
						
							|  |  |  | import { Store, UIEventSource } from "../UIEventSource" | 
					
						
							|  |  |  | import { OsmConnection } from "../Osm/OsmConnection" | 
					
						
							|  |  |  | import { Changes } from "../Osm/Changes" | 
					
						
							|  |  |  | import Translations from "../../UI/i18n/Translations" | 
					
						
							| 
									
										
										
										
											2023-12-26 22:30:27 +01:00
										 |  |  | import NoteCommentElement from "../../UI/Popup/Notes/NoteCommentElement" | 
					
						
							| 
									
										
										
										
											2024-05-28 01:25:43 +02:00
										 |  |  | import { Translation } from "../../UI/i18n/Translation" | 
					
						
							| 
									
										
										
										
											2023-09-25 02:13:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * The ImageUploadManager has a | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | export class ImageUploadManager { | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |     private readonly _uploader: ImageUploader | 
					
						
							|  |  |  |     private readonly _featureProperties: FeaturePropertiesStore | 
					
						
							|  |  |  |     private readonly _layout: LayoutConfig | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private readonly _uploadStarted: Map<string, UIEventSource<number>> = new Map() | 
					
						
							|  |  |  |     private readonly _uploadFinished: Map<string, UIEventSource<number>> = new Map() | 
					
						
							|  |  |  |     private readonly _uploadFailed: Map<string, UIEventSource<number>> = new Map() | 
					
						
							|  |  |  |     private readonly _uploadRetried: Map<string, UIEventSource<number>> = new Map() | 
					
						
							|  |  |  |     private readonly _uploadRetriedSuccess: Map<string, UIEventSource<number>> = new Map() | 
					
						
							|  |  |  |     private readonly _osmConnection: OsmConnection | 
					
						
							|  |  |  |     private readonly _changes: Changes | 
					
						
							| 
									
										
										
										
											2024-03-28 15:55:12 +01:00
										 |  |  |     public readonly isUploading: Store<boolean> | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     constructor( | 
					
						
							|  |  |  |         layout: LayoutConfig, | 
					
						
							|  |  |  |         uploader: ImageUploader, | 
					
						
							|  |  |  |         featureProperties: FeaturePropertiesStore, | 
					
						
							|  |  |  |         osmConnection: OsmConnection, | 
					
						
							|  |  |  |         changes: Changes | 
					
						
							|  |  |  |     ) { | 
					
						
							|  |  |  |         this._uploader = uploader | 
					
						
							|  |  |  |         this._featureProperties = featureProperties | 
					
						
							|  |  |  |         this._layout = layout | 
					
						
							|  |  |  |         this._osmConnection = osmConnection | 
					
						
							|  |  |  |         this._changes = changes | 
					
						
							| 
									
										
										
										
											2024-03-28 15:55:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const failed = this.getCounterFor(this._uploadFailed, "*") | 
					
						
							|  |  |  |         const done = this.getCounterFor(this._uploadFinished, "*") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-13 02:40:21 +02:00
										 |  |  |         this.isUploading = this.getCounterFor(this._uploadStarted, "*").map( | 
					
						
							|  |  |  |             (startedCount) => { | 
					
						
							|  |  |  |                 return startedCount > failed.data + done.data | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             [failed, done] | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2023-09-25 02:55:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-25 02:13:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Gets various counters. | 
					
						
							|  |  |  |      * Note that counters can only increase | 
					
						
							|  |  |  |      * If a retry was a success, both 'retrySuccess' _and_ 'uploadFinished' will be increased | 
					
						
							|  |  |  |      * @param featureId: the id of the feature you want information for. '*' has a global counter | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public getCountsFor(featureId: string | "*"): { | 
					
						
							|  |  |  |         retried: Store<number> | 
					
						
							|  |  |  |         uploadStarted: Store<number> | 
					
						
							|  |  |  |         retrySuccess: Store<number> | 
					
						
							|  |  |  |         failed: Store<number> | 
					
						
							|  |  |  |         uploadFinished: Store<number> | 
					
						
							|  |  |  |     } { | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             uploadStarted: this.getCounterFor(this._uploadStarted, featureId), | 
					
						
							|  |  |  |             uploadFinished: this.getCounterFor(this._uploadFinished, featureId), | 
					
						
							|  |  |  |             retried: this.getCounterFor(this._uploadRetried, featureId), | 
					
						
							|  |  |  |             failed: this.getCounterFor(this._uploadFailed, featureId), | 
					
						
							|  |  |  |             retrySuccess: this.getCounterFor(this._uploadRetriedSuccess, featureId), | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-25 02:55:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-25 02:13:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-28 01:25:43 +02:00
										 |  |  |     public canBeUploaded(file: File): true | {error: Translation} { | 
					
						
							|  |  |  |         const sizeInBytes = file.size | 
					
						
							|  |  |  |         const self = this | 
					
						
							|  |  |  |         if (sizeInBytes > this._uploader.maxFileSizeInMegabytes * 1000000) { | 
					
						
							|  |  |  |              const error = Translations.t.image.toBig.Subs({ | 
					
						
							|  |  |  |                 actual_size: Math.floor(sizeInBytes / 1000000) + "MB", | 
					
						
							|  |  |  |                 max_size: self._uploader.maxFileSizeInMegabytes + "MB", | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |             return {error} | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Uploads the given image, applies the correct title and license for the known user. | 
					
						
							|  |  |  |      * Will then add this image to the OSM-feature or the OSM-note | 
					
						
							| 
									
										
										
										
											2023-10-22 00:51:43 +02:00
										 |  |  |      * @param file a jpg file to upload | 
					
						
							|  |  |  |      * @param tagsStore The tags of the feature | 
					
						
							|  |  |  |      * @param targetKey Use this key to save the attribute under. Default: 'image' | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-10-30 13:44:27 +01:00
										 |  |  |     public async uploadImageAndApply( | 
					
						
							|  |  |  |         file: File, | 
					
						
							|  |  |  |         tagsStore: UIEventSource<OsmTags>, | 
					
						
							|  |  |  |         targetKey?: string | 
					
						
							|  |  |  |     ): Promise<void> { | 
					
						
							| 
									
										
										
										
											2024-05-28 01:25:43 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const canBeUploaded = this.canBeUploaded(file) | 
					
						
							|  |  |  |         if(canBeUploaded !== true){ | 
					
						
							|  |  |  |             throw canBeUploaded.error | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-28 01:25:43 +02:00
										 |  |  |         const tags = tagsStore.data | 
					
						
							|  |  |  |         const featureId = <OsmId>tags.id | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |         const licenseStore = this._osmConnection?.GetPreference("pictures-license", "CC0") | 
					
						
							|  |  |  |         const license = licenseStore?.data ?? "CC0" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const matchingLayer = this._layout?.getMatchingLayer(tags) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const title = | 
					
						
							|  |  |  |             matchingLayer?.title?.GetRenderValue(tags)?.Subs(tags)?.textFor("en") ?? | 
					
						
							|  |  |  |             tags.name ?? | 
					
						
							|  |  |  |             "https//osm.org/" + tags.id | 
					
						
							|  |  |  |         const description = [ | 
					
						
							|  |  |  |             "author:" + this._osmConnection.userDetails.data.name, | 
					
						
							|  |  |  |             "license:" + license, | 
					
						
							|  |  |  |             "osmid:" + tags.id, | 
					
						
							|  |  |  |         ].join("\n") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-30 13:44:27 +01:00
										 |  |  |         const action = await this.uploadImageWithLicense( | 
					
						
							|  |  |  |             featureId, | 
					
						
							|  |  |  |             title, | 
					
						
							|  |  |  |             description, | 
					
						
							|  |  |  |             file, | 
					
						
							| 
									
										
										
										
											2023-12-03 04:49:28 +01:00
										 |  |  |             targetKey, | 
					
						
							| 
									
										
										
										
											2023-12-06 17:27:30 +01:00
										 |  |  |             tags?.data?.["_orig_theme"] | 
					
						
							| 
									
										
										
										
											2023-10-30 13:44:27 +01:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-03-11 16:35:15 +01:00
										 |  |  |         if (!action) { | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |         if (!isNaN(Number(featureId))) { | 
					
						
							| 
									
										
										
										
											2023-10-16 13:17:30 +02:00
										 |  |  |             // This is a map note
 | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |             const url = action._url | 
					
						
							|  |  |  |             await this._osmConnection.addCommentToNote(featureId, url) | 
					
						
							|  |  |  |             NoteCommentElement.addCommentTo(url, <UIEventSource<any>>tagsStore, { | 
					
						
							|  |  |  |                 osmConnection: this._osmConnection, | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         await this._changes.applyAction(action) | 
					
						
							| 
									
										
										
										
											2023-09-25 02:13:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |     private async uploadImageWithLicense( | 
					
						
							|  |  |  |         featureId: OsmId, | 
					
						
							|  |  |  |         title: string, | 
					
						
							|  |  |  |         description: string, | 
					
						
							| 
									
										
										
										
											2023-10-22 00:51:43 +02:00
										 |  |  |         blob: File, | 
					
						
							| 
									
										
										
										
											2023-12-03 04:49:28 +01:00
										 |  |  |         targetKey: string | undefined, | 
					
						
							|  |  |  |         theme?: string | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |     ): Promise<LinkImageAction> { | 
					
						
							|  |  |  |         this.increaseCountFor(this._uploadStarted, featureId) | 
					
						
							|  |  |  |         const properties = this._featureProperties.getStore(featureId) | 
					
						
							|  |  |  |         let key: string | 
					
						
							|  |  |  |         let value: string | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |             ;({ key, value } = await this._uploader.uploadImage(title, description, blob)) | 
					
						
							|  |  |  |         } catch (e) { | 
					
						
							|  |  |  |             this.increaseCountFor(this._uploadRetried, featureId) | 
					
						
							|  |  |  |             console.error("Could not upload image, trying again:", e) | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |                 ;({ key, value } = await this._uploader.uploadImage(title, description, blob)) | 
					
						
							|  |  |  |                 this.increaseCountFor(this._uploadRetriedSuccess, featureId) | 
					
						
							|  |  |  |             } catch (e) { | 
					
						
							|  |  |  |                 console.error("Could again not upload image due to", e) | 
					
						
							|  |  |  |                 this.increaseCountFor(this._uploadFailed, featureId) | 
					
						
							| 
									
										
										
										
											2024-03-11 16:35:15 +01:00
										 |  |  |                 return undefined | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         console.log("Uploading done, creating action for", featureId) | 
					
						
							| 
									
										
										
										
											2023-10-22 00:51:43 +02:00
										 |  |  |         key = targetKey ?? key | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |         const action = new LinkImageAction(featureId, key, value, properties, { | 
					
						
							| 
									
										
										
										
											2023-12-03 04:49:28 +01:00
										 |  |  |             theme: theme ?? this._layout.id, | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |             changeType: "add-image", | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |         this.increaseCountFor(this._uploadFinished, featureId) | 
					
						
							|  |  |  |         return action | 
					
						
							| 
									
										
										
										
											2023-09-25 02:13:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |     private getCounterFor(collection: Map<string, UIEventSource<number>>, key: string | "*") { | 
					
						
							|  |  |  |         if (this._featureProperties.aliases.has(key)) { | 
					
						
							|  |  |  |             key = this._featureProperties.aliases.get(key) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!collection.has(key)) { | 
					
						
							|  |  |  |             collection.set(key, new UIEventSource<number>(0)) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return collection.get(key) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-25 02:13:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |     private increaseCountFor(collection: Map<string, UIEventSource<number>>, key: string | "*") { | 
					
						
							| 
									
										
										
										
											2023-10-16 13:17:30 +02:00
										 |  |  |         { | 
					
						
							|  |  |  |             const counter = this.getCounterFor(collection, key) | 
					
						
							|  |  |  |             counter.setData(counter.data + 1) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             const global = this.getCounterFor(collection, "*") | 
					
						
							|  |  |  |             global.setData(global.data + 1) | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-28 23:50:27 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-25 02:13:24 +02:00
										 |  |  | } |