2021-10-15 05:20:02 +02:00
import LayoutConfig from "../Models/ThemeConfig/LayoutConfig"
2023-07-28 01:12:42 +02:00
import { QueryParameters } from "./Web/QueryParameters"
import { AllKnownLayouts } from "../Customizations/AllKnownLayouts"
import { FixedUiElement } from "../UI/Base/FixedUiElement"
import { Utils } from "../Utils"
2021-10-15 05:20:02 +02:00
import Combine from "../UI/Base/Combine"
2023-07-28 01:12:42 +02:00
import { SubtleButton } from "../UI/Base/SubtleButton"
2021-10-15 05:20:02 +02:00
import BaseUIElement from "../UI/BaseUIElement"
2023-07-28 01:12:42 +02:00
import { UIEventSource } from "./UIEventSource"
import { LocalStorageSource } from "./Web/LocalStorageSource"
2021-10-15 05:20:02 +02:00
import LZString from "lz-string"
2023-07-28 01:12:42 +02:00
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"
2022-02-14 02:26:03 +01:00
import Svg from "../Svg"
2023-07-15 18:04:30 +02:00
import questions from "../assets/generated/layers/questions.json"
2023-07-28 01:12:42 +02:00
import {
DoesImageExist ,
PrevalidateTheme ,
ValidateThemeAndLayers ,
} from "../Models/ThemeConfig/Conversion/Validation"
import { DesugaringContext } from "../Models/ThemeConfig/Conversion/Conversion"
2023-10-06 23:56:50 +02:00
import {
MinimalTagRenderingConfigJson ,
TagRenderingConfigJson
} from "../Models/ThemeConfig/Json/TagRenderingConfigJson" ;
2023-07-08 02:53:45 +02:00
import Hash from "./Web/Hash"
2023-10-06 23:56:50 +02:00
import { QuestionableTagRenderingConfigJson } from "../Models/ThemeConfig/Json/QuestionableTagRenderingConfigJson" ;
2022-02-08 00:56:47 +01:00
2021-10-15 05:20:02 +02:00
export default class DetermineLayout {
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" ,
"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
}
2021-10-15 05:20:02 +02:00
/ * *
* Gets the correct layout for this website
* /
2023-03-02 05:20:53 +01:00
public static async GetLayout ( ) : Promise < LayoutConfig | undefined > {
2023-07-08 02:53:45 +02:00
const layoutFromBase64 = decodeURIComponent ( DetermineLayout . loadCustomThemeParam . data )
2021-10-15 05:20:02 +02:00
if ( layoutFromBase64 . startsWith ( "http" ) ) {
2021-12-21 19:56:04 +01:00
return await DetermineLayout . LoadRemoteTheme ( layoutFromBase64 )
2021-10-15 05:20:02 +02:00
}
if ( layoutFromBase64 !== "false" ) {
// We have to load something from the hash (or from disk)
2023-07-08 02:53:45 +02:00
return DetermineLayout . LoadLayoutFromHash ( DetermineLayout . loadCustomThemeParam )
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
2023-04-15 02:28:24 +02:00
const layout = AllKnownLayouts . allKnownLayouts . get ( layoutId ? . toLowerCase ( ) )
if ( layout === undefined ) {
2023-06-07 17:33:07 +02:00
throw "No builtin map theme with name " + layoutId + " exists"
2023-04-15 02:28:24 +02:00
}
return layout
2021-10-15 05:20:02 +02:00
}
public static LoadLayoutFromHash ( userLayoutParam : UIEventSource < string > ) : LayoutConfig | null {
let hash = location . hash . substr ( 1 )
2022-02-14 02:26:03 +01:00
let json : any
2023-06-26 11:11:22 +02:00
// 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 )
}
2021-10-15 05:20:02 +02:00
2023-06-26 11:11:22 +02:00
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 )
}
2021-10-15 05:20:02 +02:00
2023-06-26 11:11:22 +02:00
try {
json = JSON . parse ( atob ( hash ) )
2021-10-15 05:20:02 +02:00
} catch ( e ) {
2023-06-26 11:11:22 +02:00
// We try to decode with lz-string
json = JSON . parse ( Utils . UnMinify ( LZString . decompressFromBase64 ( hash ) ) )
2021-10-15 05:20:02 +02:00
}
2023-06-26 11:11:22 +02:00
const layoutToUse = DetermineLayout . prepCustomTheme ( json )
userLayoutParam . setData ( layoutToUse . id )
return layoutToUse
2021-10-15 05:20:02 +02:00
}
public static ShowErrorOnCustomTheme (
intro : string = "Error: could not parse the custom layout:" ,
2022-02-14 02:26:03 +01:00
error : BaseUIElement ,
json? : any
) {
2021-10-15 05:20:02 +02:00
new Combine ( [
intro ,
error . SetClass ( "alert" ) ,
2022-02-14 02:26:03 +01:00
new SubtleButton ( Svg . back_svg ( ) , "Go back to the theme overview" , {
2022-07-01 00:16:05 +02:00
url : window.location.protocol + "//" + window . location . host + "/index.html" ,
newTab : false ,
} ) ,
2022-02-14 02:26:03 +01:00
json !== undefined
? new SubtleButton ( Svg . download_svg ( ) , "Download the JSON file" ) . onClick ( ( ) = > {
2022-10-27 01:50:01 +02:00
Utils . offerContentsAsDownloadableFile (
JSON . stringify ( json , null , " " ) ,
"theme_definition.json"
)
} )
2022-02-14 02:26:03 +01:00
: undefined ,
2021-10-15 05:20:02 +02:00
] )
. SetClass ( "flex flex-col clickable" )
2023-06-07 17:33:07 +02:00
. AttachTo ( "maindiv" )
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 ) {
dict . set ( tagRendering . id , tagRendering )
2023-06-07 17:33:07 +02:00
}
return dict
}
2023-06-26 11:11:22 +02:00
2022-06-21 16:47:54 +02:00
private static prepCustomTheme ( json : any , sourceUrl? : string , forceId? : string ) : LayoutConfig {
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
const layerConfig = < LayerConfigJson > json
const iconTr : string | TagRenderingConfigJson = layerConfig . pointRendering . map ( ( mr ) = > mr . marker . find ( icon = > icon . icon !== undefined ) . icon ) . find ( ( i ) = > i !== undefined )
2022-02-08 00:56:47 +01:00
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 ] ,
}
}
2022-09-08 21:40:48 +02:00
2022-01-26 21:40:38 +01:00
const knownLayersDict = new Map < string , LayerConfigJson > ( )
2022-01-29 02:45:59 +01:00
for ( const key in known_layers . layers ) {
const layer = known_layers . layers [ key ]
knownLayersDict . set ( layer . id , < LayerConfigJson > layer )
2022-01-26 21:40:38 +01:00
}
2023-06-07 17:33:07 +02:00
const convertState : DesugaringContext = {
tagRenderings : DetermineLayout.getSharedTagRenderings ( ) ,
2022-06-13 03:13:42 +02:00
sharedLayers : knownLayersDict ,
publicLayers : new Set < string > ( ) ,
2022-01-26 21:40:38 +01:00
}
2022-02-04 13:17:50 +01:00
json = new FixLegacyTheme ( ) . convertStrict ( json , "While loading a dynamic theme" )
2022-04-18 02:39:30 +02:00
const raw = json
2022-02-04 15:48:26 +01:00
json = new FixImages ( DetermineLayout . _knownImages ) . convertStrict (
json ,
"While fixing the images"
)
2022-05-06 12:41:24 +02:00
json . enableNoteImports = json . enableNoteImports ? ? false
2023-06-07 17:33:07 +02:00
json = new PrepareTheme ( convertState ) . convertStrict ( json , "While preparing a dynamic theme" )
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
{
2022-10-27 01:50:01 +02:00
let { errors } = new PrevalidateTheme ( ) . convert ( json , "validation" )
2022-09-11 01:49:07 +02:00
if ( errors . length > 0 ) {
throw "Detected errors: " + errors . join ( "\n" )
}
}
{
2022-10-27 01:50:01 +02:00
let { errors } = new ValidateThemeAndLayers (
new DoesImageExist ( new Set < string > ( ) , ( _ ) = > true ) ,
2022-09-11 01:49:07 +02:00
"" ,
2023-02-03 03:57:30 +01:00
false
2022-09-11 01:49:07 +02:00
) . convert ( json , "validation" )
if ( errors . length > 0 ) {
throw "Detected errors: " + errors . join ( "\n" )
}
}
2022-04-18 02:39:30 +02:00
return new LayoutConfig ( json , false , {
definitionRaw : JSON.stringify ( raw , null , " " ) ,
definedAtUrl : sourceUrl ,
} )
2022-01-26 21:40:38 +01:00
}
2021-11-07 16:34:51 +01:00
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 (
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
2023-06-26 11:11:22 +02:00
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
2021-11-07 16:34:51 +01:00
}
2023-06-26 11:11:22 +02:00
console . log ( "Loaded remote link:" , link )
return DetermineLayout . prepCustomTheme ( parsed , link , forcedId )
2021-11-07 16:34:51 +01:00
}
2021-10-15 05:20:02 +02:00
}