| 
									
										
										
										
											2021-12-21 18:35:31 +01:00
										 |  |  | import { appendFileSync, existsSync, mkdirSync, readFileSync, writeFile, writeFileSync } from "fs" | 
					
						
							| 
									
										
										
										
											2020-11-17 02:22:48 +01:00
										 |  |  | import Locale from "../UI/i18n/Locale" | 
					
						
							|  |  |  | import Translations from "../UI/i18n/Translations" | 
					
						
							|  |  |  | import { Translation } from "../UI/i18n/Translation" | 
					
						
							| 
									
										
										
										
											2023-03-02 14:44:28 +01:00
										 |  |  | import all_known_layouts from "../assets/generated/known_themes.json" | 
					
						
							| 
									
										
										
										
											2021-08-07 23:11:34 +02:00
										 |  |  | import { LayoutConfigJson } from "../Models/ThemeConfig/Json/LayoutConfigJson" | 
					
						
							|  |  |  | import LayoutConfig from "../Models/ThemeConfig/LayoutConfig" | 
					
						
							| 
									
										
										
										
											2022-02-04 00:42:02 +01:00
										 |  |  | import xml2js from "xml2js" | 
					
						
							| 
									
										
										
										
											2022-02-06 03:02:45 +01:00
										 |  |  | import ScriptUtils from "./ScriptUtils" | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | import { Utils } from "../Utils" | 
					
						
							| 
									
										
										
										
											2021-03-17 14:17:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | const sharp = require("sharp") | 
					
						
							| 
									
										
										
										
											2021-12-21 18:35:31 +01:00
										 |  |  | const template = readFileSync("theme.html", "utf8") | 
					
						
							|  |  |  | const codeTemplate = readFileSync("index_theme.ts.template", "utf8") | 
					
						
							| 
									
										
										
										
											2021-01-06 02:21:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | function enc(str: string): string { | 
					
						
							|  |  |  |     return encodeURIComponent(str.toLowerCase()) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  | async function createIcon(iconPath: string, size: number, alreadyWritten: string[]) { | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     let name = iconPath.split(".").slice(0, -1).join(".") // drop svg suffix
 | 
					
						
							| 
									
										
										
										
											2020-11-17 16:29:51 +01:00
										 |  |  |     if (name.startsWith("./")) { | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         name = name.substr(2) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-12 18:32:10 +01:00
										 |  |  |     const newname = `assets/generated/images/${name.replace(/\//g, "_")}${size}.png` | 
					
						
							|  |  |  |     const targetpath = `public/${newname}` | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     if (alreadyWritten.indexOf(newname) >= 0) { | 
					
						
							|  |  |  |         return newname | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     alreadyWritten.push(newname) | 
					
						
							| 
									
										
										
										
											2023-04-13 23:40:28 +02:00
										 |  |  |     if (existsSync(targetpath)) { | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |         return newname | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!existsSync(iconPath)) { | 
					
						
							|  |  |  |         throw "No file at " + iconPath | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 23:37:59 +02:00
										 |  |  |     try { | 
					
						
							| 
									
										
										
										
											2020-09-30 23:34:44 +02:00
										 |  |  |         // We already read to file, in order to crash here if the file is not found
 | 
					
						
							| 
									
										
										
										
											2021-01-18 03:25:15 +01:00
										 |  |  |         let img = await sharp(iconPath) | 
					
						
							|  |  |  |         let resized = await img.resize(size) | 
					
						
							| 
									
										
										
										
											2023-02-12 18:32:10 +01:00
										 |  |  |         await resized.toFile(targetpath) | 
					
						
							| 
									
										
										
										
											2022-03-10 16:26:25 +01:00
										 |  |  |         console.log("Created png version at ", newname) | 
					
						
							| 
									
										
										
										
											2020-09-25 23:37:59 +02:00
										 |  |  |     } catch (e) { | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |         console.error("Could not read icon", iconPath, " to create a PNG due to", e) | 
					
						
							| 
									
										
										
										
											2020-09-20 20:28:35 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-09-25 23:37:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     return newname | 
					
						
							| 
									
										
										
										
											2020-07-25 18:00:08 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | async function createSocialImage(layout: LayoutConfig, template: "" | "Wide"): Promise<string> { | 
					
						
							|  |  |  |     if (!layout.icon.endsWith(".svg")) { | 
					
						
							|  |  |  |         console.warn( | 
					
						
							|  |  |  |             "Not creating a social image for " + | 
					
						
							|  |  |  |                 layout.id + | 
					
						
							|  |  |  |                 " as it is _not_ a .svg: " + | 
					
						
							|  |  |  |                 layout.icon | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         return undefined | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-09 16:19:08 +01:00
										 |  |  |     const path = `./public/assets/generated/images/social_image_${layout.id}_${template}.svg` | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     if (existsSync(path)) { | 
					
						
							| 
									
										
										
										
											2022-03-10 16:26:25 +01:00
										 |  |  |         return path | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     const svg = await ScriptUtils.ReadSvg(layout.icon) | 
					
						
							|  |  |  |     let width: string = svg.$.width | 
					
						
							|  |  |  |     if (width === undefined) { | 
					
						
							|  |  |  |         throw "The logo at " + layout.icon + " does not have a defined width" | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (width?.endsWith("px")) { | 
					
						
							|  |  |  |         width = width.substring(0, width.length - 2) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (width?.endsWith("%")) { | 
					
						
							|  |  |  |         throw "The logo at " + layout.icon + " has a relative width; this is not supported" | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     delete svg["defs"] | 
					
						
							|  |  |  |     delete svg["$"] | 
					
						
							|  |  |  |     let templateSvg = await ScriptUtils.ReadSvg("./assets/SocialImageTemplate" + template + ".svg") | 
					
						
							|  |  |  |     templateSvg = Utils.WalkJson( | 
					
						
							|  |  |  |         templateSvg, | 
					
						
							|  |  |  |         (leaf) => { | 
					
						
							|  |  |  |             const { cx, cy, r } = leaf["circle"][0].$ | 
					
						
							|  |  |  |             return { | 
					
						
							|  |  |  |                 $: { | 
					
						
							|  |  |  |                     id: "icon", | 
					
						
							|  |  |  |                     transform: `translate(${cx - r},${cy - r}) scale(${(r * 2) / Number(width)}) `, | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 g: [svg], | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         (mightBeTokenToReplace) => { | 
					
						
							|  |  |  |             if (mightBeTokenToReplace?.circle === undefined) { | 
					
						
							|  |  |  |                 return false | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return mightBeTokenToReplace.circle[0]?.$?.style?.indexOf("fill:#ff00ff") >= 0 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const builder = new xml2js.Builder() | 
					
						
							|  |  |  |     const xml = builder.buildObject({ svg: templateSvg }) | 
					
						
							|  |  |  |     writeFileSync(path, xml) | 
					
						
							| 
									
										
										
										
											2022-03-10 16:26:25 +01:00
										 |  |  |     console.log("Created social image at ", path) | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     return path | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function createManifest( | 
					
						
							|  |  |  |     layout: LayoutConfig, | 
					
						
							|  |  |  |     alreadyWritten: string[] | 
					
						
							|  |  |  | ): Promise<{ | 
					
						
							|  |  |  |     manifest: any | 
					
						
							|  |  |  |     whiteIcons: string[] | 
					
						
							|  |  |  | }> { | 
					
						
							| 
									
										
										
										
											2021-03-16 20:03:19 +01:00
										 |  |  |     Translation.forcedLanguage = "en" | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     const icons = [] | 
					
						
							| 
									
										
										
										
											2021-09-09 00:05:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     const whiteIcons: string[] = [] | 
					
						
							| 
									
										
										
										
											2020-08-22 03:15:42 +02:00
										 |  |  |     let icon = layout.icon | 
					
						
							| 
									
										
										
										
											2020-11-06 03:17:27 +01:00
										 |  |  |     if (icon.endsWith(".svg") || icon.startsWith("<svg") || icon.startsWith("<?xml")) { | 
					
						
							| 
									
										
										
										
											2022-02-04 00:42:02 +01:00
										 |  |  |         // This is an svg. Lets create the needed pngs and do some checkes!
 | 
					
						
							| 
									
										
										
										
											2021-01-17 21:04:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |         const whiteBackgroundPath = | 
					
						
							| 
									
										
										
										
											2023-02-09 16:19:08 +01:00
										 |  |  |             "./public/assets/generated/images/theme_" + layout.id + "_white_background.svg" | 
					
						
							| 
									
										
										
										
											2022-02-04 00:42:02 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-02-10 23:10:39 +01:00
										 |  |  |             const svg = await ScriptUtils.ReadSvg(icon) | 
					
						
							| 
									
										
										
										
											2022-02-04 00:42:02 +01:00
										 |  |  |             const width: string = svg.$.width | 
					
						
							|  |  |  |             const height: string = svg.$.height | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             const builder = new xml2js.Builder() | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |             const withRect = { rect: { $: { width, height, style: "fill:#ffffff;" } }, ...svg } | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |             const xml = builder.buildObject({ svg: withRect }) | 
					
						
							|  |  |  |             writeFileSync(whiteBackgroundPath, xml) | 
					
						
							| 
									
										
										
										
											2022-02-04 00:42:02 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 03:17:27 +01:00
										 |  |  |         let path = layout.icon | 
					
						
							|  |  |  |         if (layout.icon.startsWith("<")) { | 
					
						
							|  |  |  |             // THis is already the svg
 | 
					
						
							| 
									
										
										
										
											2023-02-09 16:19:08 +01:00
										 |  |  |             path = "./public/assets/generated/images/" + layout.id + "_logo.svg" | 
					
						
							| 
									
										
										
										
											2020-11-06 03:17:27 +01:00
										 |  |  |             writeFileSync(path, layout.icon) | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-17 21:04:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         const sizes = [72, 96, 120, 128, 144, 152, 180, 192, 384, 512] | 
					
						
							|  |  |  |         for (const size of sizes) { | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |             const name = await createIcon(path, size, alreadyWritten) | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |             const whiteIcon = await createIcon(whiteBackgroundPath, size, alreadyWritten) | 
					
						
							|  |  |  |             whiteIcons.push(whiteIcon) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |             icons.push({ | 
					
						
							| 
									
										
										
										
											2022-02-06 03:45:32 +01:00
										 |  |  |                 src: name, | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |                 sizes: size + "x" + size, | 
					
						
							|  |  |  |                 type: "image/png", | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         icons.push({ | 
					
						
							| 
									
										
										
										
											2020-11-06 03:17:27 +01:00
										 |  |  |             src: path, | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |             sizes: "513x513", | 
					
						
							|  |  |  |             type: "image/svg", | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2021-09-09 00:05:51 +02:00
										 |  |  |     } else if (icon.endsWith(".png")) { | 
					
						
							| 
									
										
										
										
											2021-07-06 15:43:21 +02:00
										 |  |  |         icons.push({ | 
					
						
							|  |  |  |             src: icon, | 
					
						
							|  |  |  |             sizes: "513x513", | 
					
						
							| 
									
										
										
										
											2021-07-06 15:53:42 +02:00
										 |  |  |             type: "image/png", | 
					
						
							| 
									
										
										
										
											2021-07-06 15:43:21 +02:00
										 |  |  |         }) | 
					
						
							| 
									
										
										
										
											2021-09-09 00:05:51 +02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2020-11-06 03:17:27 +01:00
										 |  |  |         console.log(icon) | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |         throw "Icon is not an svg for " + layout.id | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-21 12:39:28 +02:00
										 |  |  |     const ogTitle = Translations.T(layout.title).txt | 
					
						
							|  |  |  |     const ogDescr = Translations.T(layout.description ?? "").txt | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     const manifest = { | 
					
						
							| 
									
										
										
										
											2022-07-12 14:22:14 +02:00
										 |  |  |         name: ogTitle, | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         short_name: ogTitle, | 
					
						
							| 
									
										
										
										
											2021-05-10 23:43:30 +02:00
										 |  |  |         start_url: `${layout.id.toLowerCase()}.html`, | 
					
						
							| 
									
										
										
										
											2022-01-27 02:10:28 +01:00
										 |  |  |         lang: "en", | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         display: "standalone", | 
					
						
							|  |  |  |         background_color: "#fff", | 
					
						
							|  |  |  |         description: ogDescr, | 
					
						
							|  |  |  |         orientation: "portrait-primary, landscape-primary", | 
					
						
							| 
									
										
										
										
											2022-01-06 21:05:52 +01:00
										 |  |  |         icons: icons, | 
					
						
							| 
									
										
										
										
											2022-01-26 21:40:38 +01:00
										 |  |  |         categories: ["map", "navigation"], | 
					
						
							| 
									
										
										
										
											2021-03-17 14:17:33 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     return { | 
					
						
							|  |  |  |         manifest, | 
					
						
							|  |  |  |         whiteIcons, | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-13 23:40:28 +02:00
										 |  |  | function asLangSpan(t: Translation, tag = "span"): string { | 
					
						
							|  |  |  |     const values: string[] = [] | 
					
						
							|  |  |  |     for (const lang in t.translations) { | 
					
						
							|  |  |  |         if (lang === "_context") { | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         values.push(`<${tag} lang='${lang}'>${t.translations[lang]}</${tag}>`) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return values.join("\n") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | async function createLandingPage(layout: LayoutConfig, manifest, whiteIcons, alreadyWritten) { | 
					
						
							| 
									
										
										
										
											2020-11-11 16:23:49 +01:00
										 |  |  |     Locale.language.setData(layout.language[0]) | 
					
						
							| 
									
										
										
										
											2022-02-06 03:45:32 +01:00
										 |  |  |     const targetLanguage = layout.language[0] | 
					
						
							| 
									
										
										
										
											2022-04-21 12:39:28 +02:00
										 |  |  |     const ogTitle = Translations.T(layout.title).textFor(targetLanguage).replace(/"/g, '\\"') | 
					
						
							|  |  |  |     const ogDescr = Translations.T( | 
					
						
							|  |  |  |         layout.shortDescription ?? "Easily add and edit geodata with OpenStreetMap" | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-04-21 12:39:28 +02:00
										 |  |  |         .textFor(targetLanguage) | 
					
						
							|  |  |  |         .replace(/"/g, '\\"') | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     let ogImage = layout.socialImage | 
					
						
							|  |  |  |     let twitterImage = ogImage | 
					
						
							|  |  |  |     if (ogImage === LayoutConfig.defaultSocialImage && layout.official) { | 
					
						
							|  |  |  |         ogImage = (await createSocialImage(layout, "")) ?? layout.socialImage | 
					
						
							|  |  |  |         twitterImage = (await createSocialImage(layout, "Wide")) ?? layout.socialImage | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (twitterImage.endsWith(".svg")) { | 
					
						
							|  |  |  |         // svgs are badly supported as social image, we use a generated svg instead
 | 
					
						
							|  |  |  |         twitterImage = await createIcon(twitterImage, 512, alreadyWritten) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     if (ogImage.endsWith(".svg")) { | 
					
						
							|  |  |  |         ogImage = await createIcon(ogImage, 512, alreadyWritten) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-23 02:15:58 +02:00
										 |  |  |     let customCss = "" | 
					
						
							|  |  |  |     if (layout.customCss !== undefined && layout.customCss !== "") { | 
					
						
							| 
									
										
										
										
											2020-11-14 02:54:33 +01:00
										 |  |  |         try { | 
					
						
							|  |  |  |             const cssContent = readFileSync(layout.customCss) | 
					
						
							|  |  |  |             customCss = "<style>" + cssContent + "</style>" | 
					
						
							|  |  |  |         } catch (e) { | 
					
						
							|  |  |  |             customCss = `<link rel='stylesheet' href="${layout.customCss}"/>` | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-10-23 02:15:58 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     const og = `
 | 
					
						
							| 
									
										
										
										
											2022-01-18 20:18:12 +01:00
										 |  |  |     <meta property="og:image" content="${ogImage ?? "assets/SocialImage.png"}"> | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     <meta property="og:title" content="${ogTitle}"> | 
					
						
							| 
									
										
										
										
											2022-01-18 20:18:12 +01:00
										 |  |  |     <meta property="og:description" content="${ogDescr}"> | 
					
						
							|  |  |  |     <meta name="twitter:card" content="summary_large_image"> | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     <meta name="twitter:site" content="@mapcomplete.osm.be"> | 
					
						
							| 
									
										
										
										
											2022-01-18 20:18:12 +01:00
										 |  |  |     <meta name="twitter:creator" content="@pietervdvn"> | 
					
						
							|  |  |  |     <meta name="twitter:title" content="${ogTitle}"> | 
					
						
							|  |  |  |     <meta name="twitter:description" content="${ogDescr}"> | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     <meta name="twitter:image" content="${twitterImage}">`
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 03:17:27 +01:00
										 |  |  |     let icon = layout.icon | 
					
						
							|  |  |  |     if (icon.startsWith("<?xml") || icon.startsWith("<svg")) { | 
					
						
							|  |  |  |         // This already is an svg
 | 
					
						
							| 
									
										
										
										
											2023-02-09 16:19:08 +01:00
										 |  |  |         icon = `./public/assets/generated/images/${layout.id}_icon.svg` | 
					
						
							| 
									
										
										
										
											2020-11-06 03:17:27 +01:00
										 |  |  |         writeFileSync(icon, layout.icon) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-09 00:05:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-17 14:17:33 +01:00
										 |  |  |     const apple_icons = [] | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     for (const icon of whiteIcons) { | 
					
						
							|  |  |  |         if (!existsSync(icon)) { | 
					
						
							| 
									
										
										
										
											2022-02-06 12:51:23 +01:00
										 |  |  |             continue | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |         const size = icon.replace(/[^0-9]/g, "") | 
					
						
							|  |  |  |         apple_icons.push(`<link rel="apple-touch-icon" sizes="${size}x${size}" href="${icon}">`) | 
					
						
							| 
									
										
										
										
											2021-03-17 14:17:33 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-09 00:05:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-16 20:03:19 +01:00
										 |  |  |     let themeSpecific = [ | 
					
						
							|  |  |  |         `<title>${ogTitle}</title>`, | 
					
						
							|  |  |  |         `<link rel="manifest" href="${enc(layout.id)}.webmanifest">`, | 
					
						
							|  |  |  |         og, | 
					
						
							|  |  |  |         customCss, | 
					
						
							|  |  |  |         `<link rel="icon" href="${icon}" sizes="any" type="image/svg+xml">`, | 
					
						
							| 
									
										
										
										
											2021-03-17 14:17:33 +01:00
										 |  |  |         ...apple_icons, | 
					
						
							| 
									
										
										
										
											2021-03-16 20:03:19 +01:00
										 |  |  |     ].join("\n") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 03:45:32 +01:00
										 |  |  |     const loadingText = Translations.t.general.loadingTheme.Subs({ theme: ogTitle }) | 
					
						
							| 
									
										
										
										
											2023-04-13 23:40:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |     let output = template | 
					
						
							| 
									
										
										
										
											2023-04-13 23:40:28 +02:00
										 |  |  |         .replace("Loading MapComplete, hang on...", asLangSpan(loadingText, "h1")) | 
					
						
							| 
									
										
										
										
											2022-02-06 03:45:32 +01:00
										 |  |  |         .replace( | 
					
						
							|  |  |  |             "Powered by OpenStreetMap", | 
					
						
							|  |  |  |             Translations.t.general.poweredByOsm.textFor(targetLanguage) | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-12-21 18:35:31 +01:00
										 |  |  |         .replace(/<!-- THEME-SPECIFIC -->.*<!-- THEME-SPECIFIC-END-->/s, themeSpecific) | 
					
						
							| 
									
										
										
										
											2022-02-06 03:45:32 +01:00
										 |  |  |         .replace( | 
					
						
							|  |  |  |             /<!-- DESCRIPTION START -->.*<!-- DESCRIPTION END -->/s, | 
					
						
							| 
									
										
										
										
											2023-04-13 23:40:28 +02:00
										 |  |  |             asLangSpan(layout.shortDescription) | 
					
						
							| 
									
										
										
										
											2023-05-25 10:46:32 +02:00
										 |  |  |         )  .replace(/<!-- IMAGE-START -->.*<!-- IMAGE-END-->/s, "<img class='p-8 h-32 w-32 self-start' src='"+ icon+"' />") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-21 18:35:31 +01:00
										 |  |  |         .replace( | 
					
						
							| 
									
										
										
										
											2023-01-17 03:51:46 +01:00
										 |  |  |             '<script type="module" src="./index.ts"></script>', | 
					
						
							|  |  |  |             `<script type="module"  src='./index_${layout.id}.ts'></script>` | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2023-02-12 18:32:10 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |     return output | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-26 21:40:38 +01:00
										 |  |  | async function createIndexFor(theme: LayoutConfig) { | 
					
						
							|  |  |  |     const filename = "index_" + theme.id + ".ts" | 
					
						
							| 
									
										
										
										
											2023-04-13 23:40:28 +02:00
										 |  |  |     writeFileSync(filename, `import layout from "./assets/generated/themes/${theme.id}.json"\n`) | 
					
						
							| 
									
										
										
										
											2021-12-21 18:35:31 +01:00
										 |  |  |     appendFileSync(filename, codeTemplate) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | function createDir(path) { | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |     if (!existsSync(path)) { | 
					
						
							|  |  |  |         mkdirSync(path) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-17 16:29:51 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | async function main(): Promise<void> { | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |     const alreadyWritten = [] | 
					
						
							|  |  |  |     createDir("./assets/generated") | 
					
						
							|  |  |  |     createDir("./assets/generated/layers") | 
					
						
							|  |  |  |     createDir("./assets/generated/themes") | 
					
						
							| 
									
										
										
										
											2023-02-09 16:19:08 +01:00
										 |  |  |     createDir("./public/assets/") | 
					
						
							|  |  |  |     createDir("./public/assets/generated") | 
					
						
							|  |  |  |     createDir("./public/assets/generated/images") | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const blacklist = [ | 
					
						
							|  |  |  |         "", | 
					
						
							|  |  |  |         "test", | 
					
						
							|  |  |  |         ".", | 
					
						
							|  |  |  |         "..", | 
					
						
							|  |  |  |         "manifest", | 
					
						
							|  |  |  |         "index", | 
					
						
							|  |  |  |         "land", | 
					
						
							|  |  |  |         "preferences", | 
					
						
							|  |  |  |         "account", | 
					
						
							|  |  |  |         "openstreetmap", | 
					
						
							|  |  |  |         "custom", | 
					
						
							|  |  |  |         "theme", | 
					
						
							|  |  |  |     ] | 
					
						
							|  |  |  |     // @ts-ignore
 | 
					
						
							|  |  |  |     const all: LayoutConfigJson[] = all_known_layouts.themes | 
					
						
							|  |  |  |     const args = process.argv | 
					
						
							|  |  |  |     const theme = args[2] | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     if (theme !== undefined) { | 
					
						
							|  |  |  |         console.warn("Only generating layout " + theme) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |     for (const i in all) { | 
					
						
							|  |  |  |         const layoutConfigJson: LayoutConfigJson = all[i] | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |         if (theme !== undefined && layoutConfigJson.id !== theme) { | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |             continue | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-04-18 02:39:30 +02:00
										 |  |  |         const layout = new LayoutConfig(layoutConfigJson, true) | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |         const layoutName = layout.id | 
					
						
							|  |  |  |         if (blacklist.indexOf(layoutName.toLowerCase()) >= 0) { | 
					
						
							|  |  |  |             console.log(`Skipping a layout with name${layoutName}, it is on the blacklist`) | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const err = (err) => { | 
					
						
							|  |  |  |             if (err !== null) { | 
					
						
							|  |  |  |                 console.log("Could not write manifest for ", layoutName, " because ", err) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |         const { manifest, whiteIcons } = await createManifest(layout, alreadyWritten) | 
					
						
							|  |  |  |         const manif = JSON.stringify(manifest, undefined, 2) | 
					
						
							|  |  |  |         const manifestLocation = encodeURIComponent(layout.id.toLowerCase()) + ".webmanifest" | 
					
						
							| 
									
										
										
										
											2023-02-06 01:47:11 +01:00
										 |  |  |         writeFile("public/" + manifestLocation, manif, err) | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Create a landing page for the given theme
 | 
					
						
							| 
									
										
										
										
											2022-07-12 14:22:14 +02:00
										 |  |  |         const landing = await createLandingPage(layout, manifest, whiteIcons, alreadyWritten) | 
					
						
							|  |  |  |         writeFile(enc(layout.id) + ".html", landing, err) | 
					
						
							|  |  |  |         await createIndexFor(layout) | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-24 01:37:20 +02:00
										 |  |  |     const { manifest } = await createManifest( | 
					
						
							|  |  |  |         new LayoutConfig({ | 
					
						
							| 
									
										
										
										
											2022-02-06 03:45:32 +01:00
										 |  |  |             icon: "./assets/svg/mapcomplete_logo.svg", | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |             id: "index", | 
					
						
							|  |  |  |             layers: [], | 
					
						
							|  |  |  |             socialImage: "assets/SocialImage.png", | 
					
						
							|  |  |  |             startLat: 0, | 
					
						
							|  |  |  |             startLon: 0, | 
					
						
							|  |  |  |             startZoom: 0, | 
					
						
							|  |  |  |             title: { en: "MapComplete" }, | 
					
						
							|  |  |  |             description: { en: "A thematic map viewer and editor based on OpenStreetMap" }, | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         }), | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |         alreadyWritten | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const manif = JSON.stringify(manifest, undefined, 2) | 
					
						
							| 
									
										
										
										
											2023-02-06 01:47:11 +01:00
										 |  |  |     writeFileSync("public/index.webmanifest", manif) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-10-25 17:26:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  | main().then(() => { | 
					
						
							|  |  |  |     console.log("All done!") | 
					
						
							|  |  |  | }) |