forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			266 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			266 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import LayoutConfig from "../Models/ThemeConfig/LayoutConfig"
 | |
| import { QueryParameters } from "./Web/QueryParameters"
 | |
| import { AllKnownLayouts } from "../Customizations/AllKnownLayouts"
 | |
| import { FixedUiElement } from "../UI/Base/FixedUiElement"
 | |
| import { Utils } from "../Utils"
 | |
| import Combine from "../UI/Base/Combine"
 | |
| import { SubtleButton } from "../UI/Base/SubtleButton"
 | |
| import BaseUIElement from "../UI/BaseUIElement"
 | |
| import { UIEventSource } from "./UIEventSource"
 | |
| import { LocalStorageSource } from "./Web/LocalStorageSource"
 | |
| import LZString from "lz-string"
 | |
| import { FixLegacyTheme } from "../Models/ThemeConfig/Conversion/LegacyJsonConvert"
 | |
| import { LayerConfigJson } from "../Models/ThemeConfig/Json/LayerConfigJson"
 | |
| import known_layers from "../assets/generated/known_layers.json"
 | |
| import { PrepareTheme } from "../Models/ThemeConfig/Conversion/PrepareTheme"
 | |
| import licenses from "../assets/generated/license_info.json"
 | |
| import TagRenderingConfig from "../Models/ThemeConfig/TagRenderingConfig"
 | |
| import { FixImages } from "../Models/ThemeConfig/Conversion/FixImages"
 | |
| import Svg from "../Svg"
 | |
| import {
 | |
|     DoesImageExist,
 | |
|     PrevalidateTheme,
 | |
|     ValidateTagRenderings,
 | |
|     ValidateThemeAndLayers,
 | |
| } from "../Models/ThemeConfig/Conversion/Validation"
 | |
| import { DesugaringContext } from "../Models/ThemeConfig/Conversion/Conversion"
 | |
| import { RewriteSpecial } from "../Models/ThemeConfig/Conversion/PrepareLayer"
 | |
| import { TagRenderingConfigJson } from "../Models/ThemeConfig/Json/TagRenderingConfigJson"
 | |
| import questions from "../assets/tagRenderings/questions.json"
 | |
| import Hash from "./Web/Hash"
 | |
| 
 | |
| export default class DetermineLayout {
 | |
|     private static readonly _knownImages = new Set(Array.from(licenses).map((l) => l.path))
 | |
|     private static readonly loadCustomThemeParam = QueryParameters.GetQueryParameter(
 | |
|         "userlayout",
 | |
|         "false",
 | |
|         "If not 'false', a custom (non-official) theme is loaded. This custom layout can be done in multiple ways: \n\n- The hash of the URL contains a base64-encoded .json-file containing the theme definition\n- The hash of the URL contains a lz-compressed .json-file, as generated by the custom theme generator\n- The parameter itself is an URL, in which case that URL will be downloaded. It should point to a .json of a theme"
 | |
|     )
 | |
|     public static getCustomDefinition(): string {
 | |
|         const layoutFromBase64 = decodeURIComponent(DetermineLayout.loadCustomThemeParam.data)
 | |
| 
 | |
|         if (layoutFromBase64.startsWith("http")) {
 | |
|             return layoutFromBase64
 | |
|         }
 | |
| 
 | |
|         if (layoutFromBase64 !== "false") {
 | |
|             // We have to load something from the hash (or from disk)
 | |
|             const hash = Hash.hash.data
 | |
|             try {
 | |
|                 JSON.parse(atob(hash))
 | |
|                 return atob(hash)
 | |
|             } catch (e) {
 | |
|                 // We try to decode with lz-string
 | |
|                 JSON.parse(Utils.UnMinify(LZString.decompressFromBase64(hash)))
 | |
|                 return Utils.UnMinify(LZString.decompressFromBase64(hash))
 | |
|             }
 | |
|         }
 | |
|         return undefined
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Gets the correct layout for this website
 | |
|      */
 | |
|     public static async GetLayout(): Promise<LayoutConfig | undefined> {
 | |
|         const layoutFromBase64 = decodeURIComponent(DetermineLayout.loadCustomThemeParam.data)
 | |
| 
 | |
|         if (layoutFromBase64.startsWith("http")) {
 | |
|             return await DetermineLayout.LoadRemoteTheme(layoutFromBase64)
 | |
|         }
 | |
| 
 | |
|         if (layoutFromBase64 !== "false") {
 | |
|             // We have to load something from the hash (or from disk)
 | |
|             return DetermineLayout.LoadLayoutFromHash(DetermineLayout.loadCustomThemeParam)
 | |
|         }
 | |
| 
 | |
|         let layoutId: string = undefined
 | |
| 
 | |
|         const path = window.location.pathname.split("/").slice(-1)[0]
 | |
|         if (path !== "theme.html" && path !== "") {
 | |
|             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
 | |
|         const layout = AllKnownLayouts.allKnownLayouts.get(layoutId?.toLowerCase())
 | |
|         if (layout === undefined) {
 | |
|             throw "No builtin map theme with name " + layoutId + " exists"
 | |
|         }
 | |
|         return layout
 | |
|     }
 | |
| 
 | |
|     public static LoadLayoutFromHash(userLayoutParam: UIEventSource<string>): LayoutConfig | null {
 | |
|         let hash = location.hash.substr(1)
 | |
|         let json: any
 | |
| 
 | |
|         // layoutFromBase64 contains the name of the theme. This is partly to do tracking with goat counter
 | |
|         const dedicatedHashFromLocalStorage = LocalStorageSource.Get(
 | |
|             "user-layout-" + userLayoutParam.data?.replace(" ", "_")
 | |
|         )
 | |
|         if (dedicatedHashFromLocalStorage.data?.length < 10) {
 | |
|             dedicatedHashFromLocalStorage.setData(undefined)
 | |
|         }
 | |
| 
 | |
|         const hashFromLocalStorage = LocalStorageSource.Get("last-loaded-user-layout")
 | |
|         if (hash.length < 10) {
 | |
|             hash = dedicatedHashFromLocalStorage.data ?? hashFromLocalStorage.data
 | |
|         } else {
 | |
|             console.log("Saving hash to local storage")
 | |
|             hashFromLocalStorage.setData(hash)
 | |
|             dedicatedHashFromLocalStorage.setData(hash)
 | |
|         }
 | |
| 
 | |
|         try {
 | |
|             json = JSON.parse(atob(hash))
 | |
|         } catch (e) {
 | |
|             // We try to decode with lz-string
 | |
|             json = JSON.parse(Utils.UnMinify(LZString.decompressFromBase64(hash)))
 | |
|         }
 | |
| 
 | |
|         const layoutToUse = DetermineLayout.prepCustomTheme(json)
 | |
|         userLayoutParam.setData(layoutToUse.id)
 | |
|         return layoutToUse
 | |
|     }
 | |
| 
 | |
|     public static ShowErrorOnCustomTheme(
 | |
|         intro: string = "Error: could not parse the custom layout:",
 | |
|         error: BaseUIElement,
 | |
|         json?: any
 | |
|     ) {
 | |
|         new Combine([
 | |
|             intro,
 | |
|             error.SetClass("alert"),
 | |
|             new SubtleButton(Svg.back_svg(), "Go back to the theme overview", {
 | |
|                 url: window.location.protocol + "//" + window.location.host + "/index.html",
 | |
|                 newTab: false,
 | |
|             }),
 | |
|             json !== undefined
 | |
|                 ? new SubtleButton(Svg.download_svg(), "Download the JSON file").onClick(() => {
 | |
|                       Utils.offerContentsAsDownloadableFile(
 | |
|                           JSON.stringify(json, null, "  "),
 | |
|                           "theme_definition.json"
 | |
|                       )
 | |
|                   })
 | |
|                 : undefined,
 | |
|         ])
 | |
|             .SetClass("flex flex-col clickable")
 | |
|             .AttachTo("maindiv")
 | |
|     }
 | |
| 
 | |
|     private static getSharedTagRenderings(): Map<string, TagRenderingConfigJson> {
 | |
|         const dict = new Map<string, TagRenderingConfigJson>()
 | |
| 
 | |
|         const prep = new RewriteSpecial()
 | |
|         const validator = new ValidateTagRenderings()
 | |
|         for (const key in questions) {
 | |
|             if (key === "id") {
 | |
|                 continue
 | |
|             }
 | |
|             questions[key].id = key
 | |
|             questions[key]["source"] = "shared-questions"
 | |
|             const config = prep.convertStrict(
 | |
|                 <TagRenderingConfigJson>questions[key],
 | |
|                 "questions.json:" + key
 | |
|             )
 | |
|             delete config["#"]
 | |
|             validator.convertStrict(
 | |
|                 config,
 | |
|                 "generate-layer-overview:tagRenderings/questions.json:" + key
 | |
|             )
 | |
|             dict.set(key, config)
 | |
|         }
 | |
| 
 | |
|         dict.forEach((value, key) => {
 | |
|             if (key === "id") {
 | |
|                 return
 | |
|             }
 | |
|             value.id = value.id ?? key
 | |
|         })
 | |
| 
 | |
|         return dict
 | |
|     }
 | |
| 
 | |
|     private static prepCustomTheme(json: any, sourceUrl?: string, forceId?: string): LayoutConfig {
 | |
|         if (json.layers === undefined && json.tagRenderings !== undefined) {
 | |
|             const iconTr = json.mapRendering.map((mr) => mr.icon).find((icon) => icon !== undefined)
 | |
|             const icon = new TagRenderingConfig(iconTr).render.txt
 | |
|             json = {
 | |
|                 id: json.id,
 | |
|                 description: json.description,
 | |
|                 descriptionTail: {
 | |
|                     en: "<div class='alert'>Layer only mode.</div> The loaded custom theme actually isn't a custom theme, but only contains a layer.",
 | |
|                 },
 | |
|                 icon,
 | |
|                 title: json.name,
 | |
|                 layers: [json],
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         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: DetermineLayout.getSharedTagRenderings(),
 | |
|             sharedLayers: knownLayersDict,
 | |
|             publicLayers: new Set<string>(),
 | |
|         }
 | |
|         json = new FixLegacyTheme().convertStrict(json, "While loading a dynamic theme")
 | |
|         const raw = json
 | |
| 
 | |
|         json = new FixImages(DetermineLayout._knownImages).convertStrict(
 | |
|             json,
 | |
|             "While fixing the images"
 | |
|         )
 | |
|         json.enableNoteImports = json.enableNoteImports ?? false
 | |
|         json = new PrepareTheme(convertState).convertStrict(json, "While preparing a dynamic theme")
 | |
|         console.log("The layoutconfig is ", json)
 | |
| 
 | |
|         json.id = forceId ?? json.id
 | |
| 
 | |
|         {
 | |
|             let { errors } = new PrevalidateTheme().convert(json, "validation")
 | |
|             if (errors.length > 0) {
 | |
|                 throw "Detected errors: " + errors.join("\n")
 | |
|             }
 | |
|         }
 | |
|         {
 | |
|             let { errors } = new ValidateThemeAndLayers(
 | |
|                 new DoesImageExist(new Set<string>(), (_) => true),
 | |
|                 "",
 | |
|                 false
 | |
|             ).convert(json, "validation")
 | |
|             if (errors.length > 0) {
 | |
|                 throw "Detected errors: " + errors.join("\n")
 | |
|             }
 | |
|         }
 | |
|         return new LayoutConfig(json, false, {
 | |
|             definitionRaw: JSON.stringify(raw, null, "  "),
 | |
|             definedAtUrl: sourceUrl,
 | |
|         })
 | |
|     }
 | |
| 
 | |
|     private static async LoadRemoteTheme(link: string): Promise<LayoutConfig | null> {
 | |
|         console.log("Downloading map theme from ", link)
 | |
| 
 | |
|         new FixedUiElement(`Downloading the theme from the <a href="${link}">link</a>...`).AttachTo(
 | |
|             "maindiv"
 | |
|         )
 | |
| 
 | |
|         let parsed = await Utils.downloadJson(link)
 | |
|         let forcedId = parsed.id
 | |
|         const url = new URL(link)
 | |
|         if (!(url.hostname === "localhost" || url.hostname === "127.0.0.1")) {
 | |
|             forcedId = link
 | |
|         }
 | |
|         console.log("Loaded remote link:", link)
 | |
|         return DetermineLayout.prepCustomTheme(parsed, link, forcedId)
 | |
|     }
 | |
| }
 |