| 
									
										
										
										
											2020-07-31 17:11:44 +02:00
										 |  |  | import {UIElement} from "./UI/UIElement"; | 
					
						
							|  |  |  | UIElement.runningFromConsole = true; | 
					
						
							| 
									
										
										
										
											2020-07-31 17:38:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | import {AllKnownLayouts} from "./Customizations/AllKnownLayouts"; | 
					
						
							|  |  |  | import {Layout} from "./Customizations/Layout"; | 
					
						
							|  |  |  | import {readFileSync, writeFile, writeFileSync} from "fs"; | 
					
						
							|  |  |  | import svg2img from 'promise-svg2img'; | 
					
						
							|  |  |  | import Translation from "./UI/i18n/Translation"; | 
					
						
							|  |  |  | import Locale from "./UI/i18n/Locale"; | 
					
						
							|  |  |  | import Translations from "./UI/i18n/Translations"; | 
					
						
							| 
									
										
										
										
											2020-07-31 17:38:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | import {All} from "./Customizations/Layouts/All"; | 
					
						
							|  |  |  | import {StreetWidth} from "./Customizations/Layouts/StreetWidth"; | 
					
						
							|  |  |  | import {LayerDefinition}  from"./Customizations/LayerDefinition"; | 
					
						
							|  |  |  | import {ClimbingTrees} from "./Customizations/Layouts/ClimbingTrees"; | 
					
						
							|  |  |  | import {Smoothness} from "./Customizations/Layouts/Smoothness"; | 
					
						
							|  |  |  | import {Natuurpunt} from "./Customizations/Layouts/Natuurpunt"; | 
					
						
							|  |  |  | import {Groen} from "./Customizations/Layouts/Groen"; | 
					
						
							|  |  |  | import Cyclofix from "./Customizations/Layouts/Cyclofix"; | 
					
						
							|  |  |  | import {CustomLayout} from "./Logic/CustomLayers"; | 
					
						
							|  |  |  | import {GRB} from "./Customizations/Layouts/GRB"; | 
					
						
							|  |  |  | import {Artworks} from "./Customizations/Layouts/Artworks"; | 
					
						
							|  |  |  | import {WalkByBrussels} from "./Customizations/Layouts/WalkByBrussels"; | 
					
						
							|  |  |  | import {MetaMap} from "./Customizations/Layouts/MetaMap"; | 
					
						
							|  |  |  | import {Bookcases} from "./Customizations/Layouts/Bookcases"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 11:30:04 +02:00
										 |  |  | console.log("Building the layouts") | 
					
						
							| 
									
										
										
										
											2020-07-25 18:00:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | function enc(str: string): string { | 
					
						
							|  |  |  |     return encodeURIComponent(str.toLowerCase()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function validate(layout: Layout) { | 
					
						
							| 
									
										
										
										
											2020-07-31 17:38:03 +02:00
										 |  |  |     console.log("Validationg ", layout.name) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     const translations: Translation[] = []; | 
					
						
							|  |  |  |     const queue: any[] = [layout] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (queue.length > 0) { | 
					
						
							|  |  |  |         const item = queue.pop(); | 
					
						
							|  |  |  |         for (const key in item) { | 
					
						
							|  |  |  |             const v = item[key]; | 
					
						
							|  |  |  |             if (v === undefined) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (v instanceof Translation || v?.translations !== undefined) { | 
					
						
							|  |  |  |                 translations.push(v); | 
					
						
							|  |  |  |             } else if ( | 
					
						
							|  |  |  |                 ["string", "function", "boolean", "number"].indexOf(typeof (v)) < 0) { | 
					
						
							|  |  |  |                 queue.push(v) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const missing = {} | 
					
						
							|  |  |  |     const present = {} | 
					
						
							|  |  |  |     for (const ln of layout.supportedLanguages) { | 
					
						
							|  |  |  |         missing[ln] = 0; | 
					
						
							|  |  |  |         present[ln] = 0; | 
					
						
							|  |  |  |         for (const translation of translations) { | 
					
						
							|  |  |  |             const txt = translation.translations[ln]; | 
					
						
							|  |  |  |             const isMissing = txt === undefined || txt === "" || txt.toLowerCase().indexOf("todo") >= 0; | 
					
						
							|  |  |  |             if (isMissing) { | 
					
						
							| 
									
										
										
										
											2020-07-30 11:30:04 +02:00
										 |  |  |                 console.log(`Missing or suspicious ${ln}-translation for '`, translation.txt, ":", txt) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |                 missing[ln]++ | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 present[ln]++; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     console.log("Translation completenes for", layout.name); | 
					
						
							|  |  |  |     for (const ln of layout.supportedLanguages) { | 
					
						
							|  |  |  |         const amiss = missing[ln]; | 
					
						
							|  |  |  |         const ok = present[ln]; | 
					
						
							|  |  |  |         const total = amiss + ok; | 
					
						
							|  |  |  |         console.log(`${ln}: ${ok}/${total}`) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const alreadyWritten = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function createIcon(iconPath: string, size: number) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let name = iconPath.split(".").slice(0, -1).join("."); | 
					
						
							|  |  |  |     if(name.startsWith("./")){ | 
					
						
							|  |  |  |         name = name.substr(2) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const newname = `${name}${size}.png` | 
					
						
							|  |  |  |         .replace(/\//g,"_") | 
					
						
							|  |  |  |         .replace("assets_","assets/generated/"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (alreadyWritten.indexOf(newname) >= 0) { | 
					
						
							|  |  |  |         return newname; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     alreadyWritten.push(newname); | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |         readFileSync(newname); | 
					
						
							|  |  |  |         return newname; // File already exists - nothing to do
 | 
					
						
							|  |  |  |     } catch (e) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     svg2img(iconPath, | 
					
						
							|  |  |  |         // @ts-ignore
 | 
					
						
							|  |  |  |         {width: size, height: size, preserveAspectRatio: true}) | 
					
						
							|  |  |  |         .then((buffer) => { | 
					
						
							|  |  |  |             console.log("Writing icon", newname) | 
					
						
							|  |  |  |             writeFileSync(newname, buffer); | 
					
						
							|  |  |  |         }).catch((error) => { | 
					
						
							|  |  |  |             console.log("ERROR", error) | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     return newname; | 
					
						
							| 
									
										
										
										
											2020-07-25 18:00:08 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | function createManifest(layout: Layout, relativePath: string) { | 
					
						
							| 
									
										
										
										
											2020-07-31 17:11:44 +02:00
										 |  |  |     const name = layout.name;  | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const icons = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (layout.icon.endsWith(".svg")) { | 
					
						
							|  |  |  |         // This is an svg. Lets create the needed pngs!
 | 
					
						
							|  |  |  |         const sizes = [72, 96, 120, 128, 144, 152, 180, 192, 384, 512]; | 
					
						
							|  |  |  |         for (const size of sizes) { | 
					
						
							|  |  |  |             const name = createIcon(layout.icon, size); | 
					
						
							|  |  |  |             icons.push({ | 
					
						
							|  |  |  |                 src: name, | 
					
						
							|  |  |  |                 sizes: size + "x" + size, | 
					
						
							|  |  |  |                 type: "image/png" | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         icons.push({ | 
					
						
							|  |  |  |             src: layout.icon, | 
					
						
							|  |  |  |             sizes: "513x513", | 
					
						
							|  |  |  |             type: "image/svg" | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         throw "Icon is not an svg for " + layout.name | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const ogTitle = Translations.W(layout.title).InnerRender(); | 
					
						
							|  |  |  |     const ogDescr = Translations.W(layout.description).InnerRender(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const manif = { | 
					
						
							|  |  |  |         name: name, | 
					
						
							|  |  |  |         short_name: ogTitle, | 
					
						
							|  |  |  |         start_url: `${relativePath}/${layout.name.toLowerCase()}.html`, | 
					
						
							|  |  |  |         display: "standalone", | 
					
						
							|  |  |  |         background_color: "#fff", | 
					
						
							|  |  |  |         description: ogDescr, | 
					
						
							|  |  |  |         orientation: "portrait-primary, landscape-primary", | 
					
						
							|  |  |  |         icons: icons | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return manif; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const template = readFileSync("index.html", "utf8"); | 
					
						
							|  |  |  | function createLandingPage(layout: Layout) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Locale.language.setData(layout.supportedLanguages[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const ogTitle = Translations.W(layout.title).InnerRender(); | 
					
						
							|  |  |  |     const ogDescr = Translations.W(layout.description).InnerRender(); | 
					
						
							|  |  |  |     const ogImage = layout.socialImage; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const og = `
 | 
					
						
							|  |  |  |      <meta property="og:image" content="${ogImage}"> | 
					
						
							|  |  |  |     <meta property="og:title" content="${ogTitle}"> | 
					
						
							|  |  |  |     <meta property="og:description" content="${ogDescr}">`
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return template | 
					
						
							|  |  |  |         .replace(`./manifest.manifest`, `./${enc(layout.name)}.webmanifest`) | 
					
						
							|  |  |  |         .replace("<!-- $$$OG-META -->", og) | 
					
						
							|  |  |  |         .replace(`<link rel="icon" href="assets/add.svg" sizes="any" type="image/svg+xml">`, | 
					
						
							|  |  |  |             `<link rel="icon" href="${layout.icon}" sizes="any" type="image/svg+xml">`) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const blacklist = ["", "test", ".", "..", "manifest", "index", "land", "preferences", "account", "openstreetmap"] | 
					
						
							|  |  |  | const all = AllKnownLayouts.allSets; | 
					
						
							|  |  |  | for (const layoutName in all) { | 
					
						
							|  |  |  |     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) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     const layout = all[layoutName]; | 
					
						
							|  |  |  |     validate(layout) | 
					
						
							| 
									
										
										
										
											2020-07-31 17:38:03 +02:00
										 |  |  |     console.log("Generating manifest") | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     const manif = JSON.stringify(createManifest(layout, "/MapComplete")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const manifestLocation = encodeURIComponent(layout.name.toLowerCase()) + ".webmanifest"; | 
					
						
							|  |  |  |     writeFile(manifestLocation, manif, err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const landing = createLandingPage(layout); | 
					
						
							| 
									
										
										
										
											2020-07-31 17:38:03 +02:00
										 |  |  |     console.log("Generating html-file for ",layout.name) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     writeFile(enc(layout.name) + ".html", landing, err) | 
					
						
							| 
									
										
										
										
											2020-07-31 17:38:03 +02:00
										 |  |  |     console.log("done") | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-31 17:38:03 +02:00
										 |  |  | console.log("COunting all translations") | 
					
						
							| 
									
										
										
										
											2020-07-31 17:11:44 +02:00
										 |  |  | Translations.CountTranslations(); | 
					
						
							| 
									
										
										
										
											2020-07-31 17:38:03 +02:00
										 |  |  | console.log("All done!") |