| 
									
										
										
										
											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"; | 
					
						
							| 
									
										
										
										
											2021-02-28 00:30:58 +01:00
										 |  |  | import Constants from "../Models/Constants"; | 
					
						
							| 
									
										
										
										
											2021-04-10 15:01:28 +02:00
										 |  |  | import * as all_known_layouts from "../assets/generated/known_layers_and_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-09-09 00:05:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-03-17 14:17:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-10 16:26:25 +01:00
										 |  |  |     const newname = `assets/generated/images/${name.replace(/\//g,"_")}${size}.png`; | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (alreadyWritten.indexOf(newname) >= 0) { | 
					
						
							|  |  |  |         return newname; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     alreadyWritten.push(newname); | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     if (existsSync(newname)) { | 
					
						
							|  |  |  |         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) | 
					
						
							|  |  |  |         await resized.toFile(newname) | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const path = `./assets/generated/images/social_image_${layout.id}_${template}.svg` | 
					
						
							|  |  |  |     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[] | 
					
						
							|  |  |  | }> { | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |     const name = layout.id; | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 = "./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
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |             path = "./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
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-18 01:24:31 +02:00
										 |  |  |     const ogTitle = Translations.WT(layout.title).txt; | 
					
						
							|  |  |  |     const ogDescr = Translations.WT(layout.description ?? "").txt; | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     const manifest = { | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         name: name, | 
					
						
							|  |  |  |         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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | async function createLandingPage(layout: LayoutConfig, manifest, whiteIcons, alreadyWritten) { | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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] | 
					
						
							|  |  |  |     const ogTitle = Translations.WT(layout.title).textFor(targetLanguage).replace(/"/g, '\\"'); | 
					
						
							|  |  |  |     const ogDescr = Translations.WT(layout.shortDescription ?? "Easily add and edit geodata with OpenStreetMap").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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     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
 | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |         icon = `./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}); | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |     let output = template | 
					
						
							| 
									
										
										
										
											2022-02-06 03:45:32 +01:00
										 |  |  |         .replace("Loading MapComplete, hang on...", loadingText.textFor(targetLanguage)) | 
					
						
							|  |  |  |         .replace("Powered by OpenStreetMap", Translations.t.general.poweredByOsm.textFor(targetLanguage)) | 
					
						
							| 
									
										
										
										
											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, layout.shortDescription.textFor(targetLanguage)) | 
					
						
							| 
									
										
										
										
											2021-12-21 18:35:31 +01:00
										 |  |  |         .replace("<script src=\"./index.ts\"></script>", `<script src='./index_${layout.id}.ts'></script>`); | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |         output = output | 
					
						
							| 
									
										
										
										
											2020-11-06 03:17:27 +01:00
										 |  |  |             .replace(/<!-- DECORATION 0 START -->.*<!-- DECORATION 0 END -->/s, `<img src='${icon}' width="100%" height="100%">`) | 
					
						
							|  |  |  |             .replace(/<!-- DECORATION 1 START -->.*<!-- DECORATION 1 END -->/s, `<img src='${icon}' width="100%" height="100%">`); | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |     } catch (e) { | 
					
						
							|  |  |  |         console.warn("Error while applying logo: ", e) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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" | 
					
						
							| 
									
										
										
										
											2021-12-21 18:35:31 +01:00
										 |  |  |     writeFileSync(filename, `import * as themeConfig from "./assets/generated/themes/${theme.id}.json"\n`) | 
					
						
							|  |  |  |     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") | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     createDir("./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-02-06 01:57:33 +01:00
										 |  |  |         const layout = new LayoutConfig(layoutConfigJson, true, "generating layouts") | 
					
						
							|  |  |  |         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"; | 
					
						
							|  |  |  |         writeFile(manifestLocation, manif, err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Create a landing page for the given theme
 | 
					
						
							|  |  |  |         createLandingPage(layout, manifest, whiteIcons, alreadyWritten).then(landing => { | 
					
						
							|  |  |  |             writeFile(enc(layout.id) + ".html", landing, err) | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         createIndexFor(layout) | 
					
						
							| 
									
										
										
										
											2022-02-06 01:57:33 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const {manifest, whiteIcons} = 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: [], | 
					
						
							|  |  |  |         maintainer: "Pieter Vander Vennet", | 
					
						
							|  |  |  |         socialImage: "assets/SocialImage.png", | 
					
						
							|  |  |  |         startLat: 0, | 
					
						
							|  |  |  |         startLon: 0, | 
					
						
							|  |  |  |         startZoom: 0, | 
					
						
							|  |  |  |         title: {en: "MapComplete"}, | 
					
						
							|  |  |  |         version: Constants.vNumber, | 
					
						
							|  |  |  |         description: {en: "A thematic map viewer and editor based on OpenStreetMap"} | 
					
						
							| 
									
										
										
										
											2022-03-08 04:09:03 +01:00
										 |  |  |     }), alreadyWritten); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const manif = JSON.stringify(manifest, undefined, 2); | 
					
						
							|  |  |  |     writeFileSync("index.manifest", 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!") | 
					
						
							|  |  |  | }) |