| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  | import { DesugaringStep } from "./Conversion" | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  | import { ThemeConfigJson } from "../Json/ThemeConfigJson" | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  | import { AvailableRasterLayers } from "../../RasterLayers" | 
					
						
							|  |  |  | import { ExtractImages } from "./FixImages" | 
					
						
							|  |  |  | import { ConversionContext } from "./ConversionContext" | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  | import ThemeConfig from "../ThemeConfig" | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  | import { Utils } from "../../../Utils" | 
					
						
							|  |  |  | import { DetectDuplicatePresets, DoesImageExist, ValidateLanguageCompleteness } from "./Validation" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  | export class ValidateTheme extends DesugaringStep<ThemeConfigJson> { | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * The paths where this layer is originally saved. Triggers some extra checks | 
					
						
							|  |  |  |      * @private | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     private readonly _path?: string | 
					
						
							|  |  |  |     private readonly _isBuiltin: boolean | 
					
						
							|  |  |  |     //private readonly _sharedTagRenderings: Map<string, any>
 | 
					
						
							|  |  |  |     private readonly _validateImage: DesugaringStep<string> | 
					
						
							|  |  |  |     private readonly _extractImages: ExtractImages = undefined | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     constructor( | 
					
						
							|  |  |  |         doesImageExist: DoesImageExist, | 
					
						
							|  |  |  |         path: string, | 
					
						
							|  |  |  |         isBuiltin: boolean, | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |         sharedTagRenderings?: Set<string> | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |     ) { | 
					
						
							|  |  |  |         super("Doesn't change anything, but emits warnings and errors", [], "ValidateTheme") | 
					
						
							|  |  |  |         this._validateImage = doesImageExist | 
					
						
							|  |  |  |         this._path = path | 
					
						
							|  |  |  |         this._isBuiltin = isBuiltin | 
					
						
							|  |  |  |         if (sharedTagRenderings) { | 
					
						
							|  |  |  |             this._extractImages = new ExtractImages(this._isBuiltin, sharedTagRenderings) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |     convert(json: ThemeConfigJson, context: ConversionContext): ThemeConfigJson { | 
					
						
							|  |  |  |         const theme = new ThemeConfig(json, this._isBuiltin) | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |         { | 
					
						
							|  |  |  |             // Legacy format checks
 | 
					
						
							|  |  |  |             if (this._isBuiltin) { | 
					
						
							|  |  |  |                 if (json["units"] !== undefined) { | 
					
						
							|  |  |  |                     context.err( | 
					
						
							|  |  |  |                         "The theme " + | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                             json.id + | 
					
						
							|  |  |  |                             " has units defined - these should be defined on the layer instead. (Hint: use overrideAll: { '+units': ... }) " | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                     ) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (json["roamingRenderings"] !== undefined) { | 
					
						
							|  |  |  |                     context.err( | 
					
						
							|  |  |  |                         "Theme " + | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                             json.id + | 
					
						
							|  |  |  |                             " contains an old 'roamingRenderings'. Use an 'overrideAll' instead" | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                     ) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!json.title) { | 
					
						
							|  |  |  |             context.enter("title").err(`The theme ${json.id} does not have a title defined.`) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!json.icon) { | 
					
						
							|  |  |  |             context.enter("icon").err("A theme should have an icon") | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (this._isBuiltin && this._extractImages !== undefined) { | 
					
						
							|  |  |  |             // Check images: are they local, are the licenses there, is the theme icon square, ...
 | 
					
						
							|  |  |  |             const images = this._extractImages.convert(json, context.inOperation("ValidateTheme")) | 
					
						
							|  |  |  |             const remoteImages = images.filter((img) => img.path.indexOf("http") == 0) | 
					
						
							|  |  |  |             for (const remoteImage of remoteImages) { | 
					
						
							|  |  |  |                 context.err( | 
					
						
							|  |  |  |                     "Found a remote image: " + | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                         remoteImage.path + | 
					
						
							|  |  |  |                         " in theme " + | 
					
						
							|  |  |  |                         json.id + | 
					
						
							|  |  |  |                         ", please download it." | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                 ) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             for (const image of images) { | 
					
						
							|  |  |  |                 this._validateImage.convert(image.path, context.enters(image.context)) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |             if (this._isBuiltin) { | 
					
						
							|  |  |  |                 if (theme.id !== theme.id.toLowerCase()) { | 
					
						
							|  |  |  |                     context.err("Theme ids should be in lowercase, but it is " + theme.id) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 const filename = this._path.substring( | 
					
						
							|  |  |  |                     this._path.lastIndexOf("/") + 1, | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                     this._path.length - 5 | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                 ) | 
					
						
							|  |  |  |                 if (theme.id !== filename) { | 
					
						
							|  |  |  |                     context.err( | 
					
						
							|  |  |  |                         "Theme ids should be the same as the name.json, but we got id: " + | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                             theme.id + | 
					
						
							|  |  |  |                             " and filename " + | 
					
						
							|  |  |  |                             filename + | 
					
						
							|  |  |  |                             " (" + | 
					
						
							|  |  |  |                             this._path + | 
					
						
							|  |  |  |                             ")" | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                     ) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 this._validateImage.convert(theme.icon, context.enter("icon")) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const dups = Utils.Duplicates(json.layers.map((layer) => layer["id"])) | 
					
						
							|  |  |  |             if (dups.length > 0) { | 
					
						
							|  |  |  |                 context.err( | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                     `The theme ${json.id} defines multiple layers with id ${dups.join(", ")}` | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                 ) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (json["mustHaveLanguage"] !== undefined) { | 
					
						
							|  |  |  |                 new ValidateLanguageCompleteness(...json["mustHaveLanguage"]).convert( | 
					
						
							|  |  |  |                     theme, | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                     context | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                 ) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (!json.hideFromOverview && theme.id !== "personal" && this._isBuiltin) { | 
					
						
							|  |  |  |                 // The first key in the the title-field must be english, otherwise the title in the loading page will be the different language
 | 
					
						
							|  |  |  |                 const targetLanguage = theme.title.SupportedLanguages()[0] | 
					
						
							|  |  |  |                 if (targetLanguage !== "en") { | 
					
						
							|  |  |  |                     context.err( | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                         `TargetLanguage is not 'en' for public theme ${theme.id}, it is ${targetLanguage}. Move 'en' up in the title of the theme and set it as the first key` | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                     ) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // Official, public themes must have a full english translation
 | 
					
						
							|  |  |  |                 new ValidateLanguageCompleteness("en").convert(theme, context) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } catch (e) { | 
					
						
							|  |  |  |             console.error(e) | 
					
						
							|  |  |  |             context.err("Could not validate the theme due to: " + e) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (theme.id !== "personal") { | 
					
						
							|  |  |  |             new DetectDuplicatePresets().convert(theme, context) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!theme.title) { | 
					
						
							|  |  |  |             context.enter("title").err("A theme must have a title") | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!theme.description) { | 
					
						
							|  |  |  |             context.enter("description").err("A theme must have a description") | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (theme.overpassUrl && typeof theme.overpassUrl === "string") { | 
					
						
							|  |  |  |             context | 
					
						
							|  |  |  |                 .enter("overpassUrl") | 
					
						
							|  |  |  |                 .err("The overpassURL is a string, use a list of strings instead. Wrap it with [ ]") | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (json.defaultBackgroundId) { | 
					
						
							|  |  |  |             const backgroundId = json.defaultBackgroundId | 
					
						
							|  |  |  |             const isCategory = | 
					
						
							|  |  |  |                 backgroundId === "photo" || backgroundId === "map" || backgroundId === "osmbasedmap" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-03 23:48:35 +02:00
										 |  |  |             const knownIds = Array.from(AvailableRasterLayers.allAvailableGlobalLayers).map( | 
					
						
							|  |  |  |                 (l) => l.properties.id | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2025-04-17 02:33:49 +02:00
										 |  |  |             const available = new Set(knownIds) | 
					
						
							|  |  |  |             if (!isCategory && !available.has(backgroundId)) { | 
					
						
							|  |  |  |                 const nearby = Utils.sortedByLevenshteinDistance(backgroundId, knownIds, (t) => t) | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                 context | 
					
						
							|  |  |  |                     .enter("defaultBackgroundId") | 
					
						
							|  |  |  |                     .err( | 
					
						
							|  |  |  |                         `This layer ID is not known: ${backgroundId}. Perhaps you meant one of ${nearby | 
					
						
							|  |  |  |                             .slice(0, 5) | 
					
						
							| 
									
										
										
										
											2025-05-03 23:48:35 +02:00
										 |  |  |                             .join(", ")}`
 | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2025-04-17 02:33:49 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (let i = 0; i < theme.layers.length; i++) { | 
					
						
							|  |  |  |             const layer = theme.layers[i] | 
					
						
							|  |  |  |             if (!layer.id.match("[a-z][a-z0-9_]*")) { | 
					
						
							|  |  |  |                 context | 
					
						
							|  |  |  |                     .enters("layers", i, "id") | 
					
						
							|  |  |  |                     .err("Invalid ID:" + layer.id + "should match [a-z][a-z0-9_]*") | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return json | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |