| 
									
										
										
										
											2025-01-11 01:18:56 +01:00
										 |  |  | import ThemeConfig, { MinimalThemeInformation } from "../Models/ThemeConfig/ThemeConfig" | 
					
						
							| 
									
										
										
										
											2023-07-28 01:12:42 +02:00
										 |  |  | import { QueryParameters } from "./Web/QueryParameters" | 
					
						
							|  |  |  | import { FixedUiElement } from "../UI/Base/FixedUiElement" | 
					
						
							|  |  |  | import { Utils } from "../Utils" | 
					
						
							|  |  |  | import { FixLegacyTheme } from "../Models/ThemeConfig/Conversion/LegacyJsonConvert" | 
					
						
							|  |  |  | import { LayerConfigJson } from "../Models/ThemeConfig/Json/LayerConfigJson" | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  | import known_layers from "../assets/generated/known_layers.json" | 
					
						
							| 
									
										
										
										
											2023-07-28 01:12:42 +02:00
										 |  |  | import { PrepareTheme } from "../Models/ThemeConfig/Conversion/PrepareTheme" | 
					
						
							| 
									
										
										
										
											2023-02-08 01:14:21 +01:00
										 |  |  | import licenses from "../assets/generated/license_info.json" | 
					
						
							| 
									
										
										
										
											2022-02-08 00:56:47 +01:00
										 |  |  | import TagRenderingConfig from "../Models/ThemeConfig/TagRenderingConfig" | 
					
						
							| 
									
										
										
										
											2023-07-28 01:12:42 +02:00
										 |  |  | import { FixImages } from "../Models/ThemeConfig/Conversion/FixImages" | 
					
						
							| 
									
										
										
										
											2023-07-15 18:04:30 +02:00
										 |  |  | import questions from "../assets/generated/layers/questions.json" | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  | import { DoesImageExist, PrevalidateTheme } from "../Models/ThemeConfig/Conversion/Validation" | 
					
						
							| 
									
										
										
										
											2023-07-28 01:12:42 +02:00
										 |  |  | import { DesugaringContext } from "../Models/ThemeConfig/Conversion/Conversion" | 
					
						
							| 
									
										
										
										
											2023-10-11 04:16:52 +02:00
										 |  |  | import { TagRenderingConfigJson } from "../Models/ThemeConfig/Json/TagRenderingConfigJson" | 
					
						
							|  |  |  | import { QuestionableTagRenderingConfigJson } from "../Models/ThemeConfig/Json/QuestionableTagRenderingConfigJson" | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  | import { ThemeConfigJson } from "../Models/ThemeConfig/Json/ThemeConfigJson" | 
					
						
							| 
									
										
										
										
											2024-08-11 12:03:24 +02:00
										 |  |  | import { ValidateThemeAndLayers } from "../Models/ThemeConfig/Conversion/ValidateThemeAndLayers" | 
					
						
							| 
									
										
										
										
											2025-01-11 01:18:56 +01:00
										 |  |  | import * as theme_overview from "../assets/generated/theme_overview.json" | 
					
						
							| 
									
										
										
										
											2022-02-08 00:56:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  | export default class DetermineTheme { | 
					
						
							| 
									
										
										
										
											2022-02-04 15:48:26 +01:00
										 |  |  |     private static readonly _knownImages = new Set(Array.from(licenses).map((l) => l.path)) | 
					
						
							| 
									
										
										
										
											2023-07-08 02:53:45 +02:00
										 |  |  |     private static readonly loadCustomThemeParam = QueryParameters.GetQueryParameter( | 
					
						
							|  |  |  |         "userlayout", | 
					
						
							|  |  |  |         "false", | 
					
						
							| 
									
										
										
										
											2024-10-17 04:13:58 +02:00
										 |  |  |         "If the parameter is an URL, it should point to a .json of a theme which will be loaded and used" | 
					
						
							| 
									
										
										
										
											2023-07-08 02:53:45 +02:00
										 |  |  |     ) | 
					
						
							| 
									
										
										
										
											2024-04-23 15:35:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-08 02:53:45 +02:00
										 |  |  |     public static getCustomDefinition(): string { | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |         const layoutFromBase64 = decodeURIComponent(DetermineTheme.loadCustomThemeParam.data) | 
					
						
							| 
									
										
										
										
											2023-07-08 02:53:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (layoutFromBase64.startsWith("http")) { | 
					
						
							|  |  |  |             return layoutFromBase64 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return undefined | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-15 05:20:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-16 16:06:26 +02:00
										 |  |  |     private static async expandRemoteLayers( | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |         layoutConfig: ThemeConfigJson | 
					
						
							|  |  |  |     ): Promise<ThemeConfigJson> { | 
					
						
							| 
									
										
										
										
											2024-06-16 16:06:26 +02:00
										 |  |  |         if (!layoutConfig.layers) { | 
					
						
							| 
									
										
										
										
											2024-04-23 21:31:58 +02:00
										 |  |  |             // This is probably a layer in 'layer-only-mode'
 | 
					
						
							|  |  |  |             return layoutConfig | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-04-23 15:35:18 +02:00
										 |  |  |         for (let i = 0; i < layoutConfig.layers.length; i++) { | 
					
						
							|  |  |  |             const l = layoutConfig.layers[i] | 
					
						
							|  |  |  |             if (typeof l !== "string") { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |                 new URL(l) | 
					
						
							|  |  |  |                 console.log("Downloading remote layer " + l) | 
					
						
							| 
									
										
										
										
											2024-10-17 04:13:58 +02:00
										 |  |  |                 layoutConfig.layers[i] = <LayerConfigJson>await Utils.downloadJson(l) | 
					
						
							| 
									
										
										
										
											2024-04-23 15:35:18 +02:00
										 |  |  |             } catch (_) { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return layoutConfig | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-22 01:24:21 +01:00
										 |  |  |     private static createConversionContext(): DesugaringContext { | 
					
						
							|  |  |  |         const knownLayersDict = new Map<string, LayerConfigJson>() | 
					
						
							|  |  |  |         for (const key in known_layers["layers"]) { | 
					
						
							|  |  |  |             const layer = known_layers["layers"][key] | 
					
						
							|  |  |  |             knownLayersDict.set(layer.id, <LayerConfigJson>layer) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const convertState: DesugaringContext = { | 
					
						
							|  |  |  |             tagRenderings: DetermineTheme.getSharedTagRenderings(), | 
					
						
							|  |  |  |             tagRenderingOrder: DetermineTheme.getSharedTagRenderingOrder(), | 
					
						
							|  |  |  |             sharedLayers: knownLayersDict, | 
					
						
							| 
									
										
										
										
											2025-01-28 15:42:34 +01:00
										 |  |  |             publicLayers: new Set<string>(), | 
					
						
							| 
									
										
										
										
											2025-01-22 01:24:21 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         return convertState | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-15 05:20:02 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Gets the correct layout for this website | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |     public static async getTheme(): Promise<ThemeConfig | undefined> { | 
					
						
							|  |  |  |         const layoutFromBase64 = decodeURIComponent(DetermineTheme.loadCustomThemeParam.data) | 
					
						
							| 
									
										
										
										
											2021-10-15 05:20:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (layoutFromBase64.startsWith("http")) { | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |             return await DetermineTheme.LoadRemoteTheme(layoutFromBase64) | 
					
						
							| 
									
										
										
										
											2021-10-15 05:20:02 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let layoutId: string = undefined | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const path = window.location.pathname.split("/").slice(-1)[0] | 
					
						
							| 
									
										
										
										
											2021-12-21 18:35:31 +01:00
										 |  |  |         if (path !== "theme.html" && path !== "") { | 
					
						
							| 
									
										
										
										
											2021-10-15 05:20:02 +02:00
										 |  |  |             layoutId = path | 
					
						
							|  |  |  |             if (path.endsWith(".html")) { | 
					
						
							|  |  |  |                 layoutId = path.substr(0, path.length - 5) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             console.log("Using layout", layoutId) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         layoutId = QueryParameters.GetQueryParameter( | 
					
						
							|  |  |  |             "layout", | 
					
						
							|  |  |  |             layoutId, | 
					
						
							|  |  |  |             "The layout to load into MapComplete" | 
					
						
							|  |  |  |         ).data | 
					
						
							| 
									
										
										
										
											2024-08-01 19:34:13 +02:00
										 |  |  |         const id = layoutId?.toLowerCase() | 
					
						
							| 
									
										
										
										
											2025-01-11 01:18:56 +01:00
										 |  |  |         const themes: MinimalThemeInformation[] = theme_overview.themes | 
					
						
							|  |  |  |         if (themes.length == 0) { | 
					
						
							| 
									
										
										
										
											2024-08-12 23:49:46 +02:00
										 |  |  |             throw "Build failed or running, no layouts are known at all" | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-01-18 00:30:06 +01:00
										 |  |  |         const themeInfo = themes.find((th) => th.id === id) | 
					
						
							| 
									
										
										
										
											2025-01-11 01:18:56 +01:00
										 |  |  |         if (themeInfo === undefined) { | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |             const alternatives = Utils.sortedByLevenshteinDistance( | 
					
						
							|  |  |  |                 id, | 
					
						
							| 
									
										
										
										
											2025-01-18 00:30:06 +01:00
										 |  |  |                 themes.map((th) => th.id), | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                 (i) => i | 
					
						
							|  |  |  |             ).slice(0, 3) | 
					
						
							|  |  |  |             const msg = `No builtin map theme with name ${layoutId} exists. Perhaps you meant one of ${alternatives.join( | 
					
						
							|  |  |  |                 ", " | 
					
						
							|  |  |  |             )}`
 | 
					
						
							| 
									
										
										
										
											2024-08-01 19:34:13 +02:00
										 |  |  |             throw msg | 
					
						
							| 
									
										
										
										
											2023-04-15 02:28:24 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-01-11 01:18:56 +01:00
										 |  |  |         // Actually fetch the theme
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-18 00:30:06 +01:00
										 |  |  |         const config = await Utils.downloadJsonCached<ThemeConfigJson>( | 
					
						
							|  |  |  |             "./assets/generated/themes/" + id + ".json", | 
					
						
							|  |  |  |             1000 * 60 * 60 * 60 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2025-01-22 01:24:21 +01:00
										 |  |  |         const withDefault = new PrepareTheme(this.createConversionContext()).convertStrict(config) | 
					
						
							|  |  |  |         return new ThemeConfig(withDefault, true) | 
					
						
							| 
									
										
										
										
											2021-10-15 05:20:02 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-06 23:56:50 +02:00
										 |  |  |     private static getSharedTagRenderings(): Map<string, QuestionableTagRenderingConfigJson> { | 
					
						
							|  |  |  |         const dict = new Map<string, QuestionableTagRenderingConfigJson>() | 
					
						
							| 
									
										
										
										
											2023-06-07 17:33:07 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-15 18:04:30 +02:00
										 |  |  |         for (const tagRendering of questions.tagRenderings) { | 
					
						
							| 
									
										
										
										
											2024-10-19 14:44:55 +02:00
										 |  |  |             dict.set(tagRendering.id, <QuestionableTagRenderingConfigJson>tagRendering) | 
					
						
							| 
									
										
										
										
											2023-06-07 17:33:07 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return dict | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-01 19:34:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-19 01:56:19 +02:00
										 |  |  |     private static getSharedTagRenderingOrder(): string[] { | 
					
						
							| 
									
										
										
										
											2024-06-20 04:21:29 +02:00
										 |  |  |         return questions.tagRenderings.map((tr) => tr.id) | 
					
						
							| 
									
										
										
										
											2024-06-19 01:56:19 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-06-26 11:11:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |     private static prepCustomTheme(json: any, sourceUrl?: string, forceId?: string): ThemeConfig { | 
					
						
							| 
									
										
										
										
											2022-02-08 00:56:47 +01:00
										 |  |  |         if (json.layers === undefined && json.tagRenderings !== undefined) { | 
					
						
							| 
									
										
										
										
											2023-10-06 23:56:50 +02:00
										 |  |  |             // We got fed a layer instead of a theme
 | 
					
						
							| 
									
										
										
										
											2023-10-11 04:16:52 +02:00
										 |  |  |             const layerConfig = <LayerConfigJson>json | 
					
						
							| 
									
										
										
										
											2024-06-20 14:04:01 +02:00
										 |  |  |             let icon = Utils.NoNull( | 
					
						
							| 
									
										
										
										
											2024-06-20 04:21:29 +02:00
										 |  |  |                 layerConfig.pointRendering | 
					
						
							|  |  |  |                     .flatMap((pr) => pr.marker) | 
					
						
							|  |  |  |                     .map((iconSpec) => { | 
					
						
							| 
									
										
										
										
											2024-06-24 13:11:35 +02:00
										 |  |  |                         if (!iconSpec) { | 
					
						
							| 
									
										
										
										
											2024-06-20 14:04:01 +02:00
										 |  |  |                             return undefined | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2024-06-20 04:21:29 +02:00
										 |  |  |                         const icon = new TagRenderingConfig(<TagRenderingConfigJson>iconSpec.icon) | 
					
						
							|  |  |  |                             .render.txt | 
					
						
							|  |  |  |                         if ( | 
					
						
							|  |  |  |                             iconSpec.color === undefined || | 
					
						
							|  |  |  |                             icon.startsWith("http:") || | 
					
						
							|  |  |  |                             icon.startsWith("https:") | 
					
						
							|  |  |  |                         ) { | 
					
						
							|  |  |  |                             return icon | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         const color = new TagRenderingConfig(<TagRenderingConfigJson>iconSpec.color) | 
					
						
							|  |  |  |                             .render.txt | 
					
						
							|  |  |  |                         return icon + ":" + color | 
					
						
							|  |  |  |                     }) | 
					
						
							|  |  |  |             ).join(";") | 
					
						
							| 
									
										
										
										
											2024-06-19 01:56:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-24 13:11:35 +02:00
										 |  |  |             if (!icon) { | 
					
						
							| 
									
										
										
										
											2024-06-20 14:04:01 +02:00
										 |  |  |                 icon = "./assets/svg/bug.svg" | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 00:56:47 +01:00
										 |  |  |             json = { | 
					
						
							|  |  |  |                 id: json.id, | 
					
						
							|  |  |  |                 description: json.description, | 
					
						
							|  |  |  |                 icon, | 
					
						
							|  |  |  |                 title: json.name, | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |                 layers: [json], | 
					
						
							| 
									
										
										
										
											2022-02-08 00:56:47 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-11 04:16:52 +02:00
										 |  |  |         json = new FixLegacyTheme().convertStrict(json) | 
					
						
							| 
									
										
										
										
											2022-04-18 02:39:30 +02:00
										 |  |  |         const raw = json | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |         json = new FixImages(DetermineTheme._knownImages).convertStrict(json) | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |         json.enableNoteImports = json.enableNoteImports ?? false | 
					
						
							| 
									
										
										
										
											2025-01-22 01:24:21 +01:00
										 |  |  |         const convertState = this.createConversionContext() | 
					
						
							| 
									
										
										
										
											2023-10-11 04:16:52 +02:00
										 |  |  |         json = new PrepareTheme(convertState).convertStrict(json) | 
					
						
							| 
									
										
										
										
											2022-01-26 21:40:38 +01:00
										 |  |  |         console.log("The layoutconfig is ", json) | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-21 16:47:54 +02:00
										 |  |  |         json.id = forceId ?? json.id | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-11 01:49:07 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-10-11 04:16:52 +02:00
										 |  |  |             new PrevalidateTheme().convertStrict(json) | 
					
						
							| 
									
										
										
										
											2022-09-11 01:49:07 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-10-11 04:16:52 +02:00
										 |  |  |             new ValidateThemeAndLayers( | 
					
						
							| 
									
										
										
										
											2024-06-19 01:56:19 +02:00
										 |  |  |                 new DoesImageExist(new Set<string>(), () => true), | 
					
						
							| 
									
										
										
										
											2022-09-11 01:49:07 +02:00
										 |  |  |                 "", | 
					
						
							| 
									
										
										
										
											2023-02-03 03:57:30 +01:00
										 |  |  |                 false | 
					
						
							| 
									
										
										
										
											2023-10-11 04:16:52 +02:00
										 |  |  |             ).convertStrict(json) | 
					
						
							| 
									
										
										
										
											2022-09-11 01:49:07 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |         return new ThemeConfig(json, false, { | 
					
						
							| 
									
										
										
										
											2022-04-18 02:39:30 +02:00
										 |  |  |             definitionRaw: JSON.stringify(raw, null, "  "), | 
					
						
							| 
									
										
										
										
											2024-08-14 13:53:56 +02:00
										 |  |  |             definedAtUrl: sourceUrl, | 
					
						
							| 
									
										
										
										
											2022-04-18 02:39:30 +02:00
										 |  |  |         }) | 
					
						
							| 
									
										
										
										
											2022-01-26 21:40:38 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |     private static async LoadRemoteTheme(link: string): Promise<ThemeConfig | null> { | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |         console.log("Downloading map theme from ", link) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         new FixedUiElement(`Downloading the theme from the <a href="${link}">link</a>...`).AttachTo( | 
					
						
							| 
									
										
										
										
											2023-06-07 17:33:07 +02:00
										 |  |  |             "maindiv" | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |         let parsed = <ThemeConfigJson>await Utils.downloadJson(link) | 
					
						
							| 
									
										
										
										
											2023-06-26 11:11:22 +02:00
										 |  |  |         let forcedId = parsed.id | 
					
						
							|  |  |  |         const url = new URL(link) | 
					
						
							|  |  |  |         if (!(url.hostname === "localhost" || url.hostname === "127.0.0.1")) { | 
					
						
							|  |  |  |             forcedId = link | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-06-26 11:11:22 +02:00
										 |  |  |         console.log("Loaded remote link:", link) | 
					
						
							| 
									
										
										
										
											2024-04-23 15:35:18 +02:00
										 |  |  |         parsed = await this.expandRemoteLayers(parsed) | 
					
						
							| 
									
										
										
										
											2024-10-17 04:06:03 +02:00
										 |  |  |         return DetermineTheme.prepCustomTheme(parsed, link, forcedId) | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-15 05:20:02 +02:00
										 |  |  | } |