| 
									
										
										
										
											2020-07-31 17:11:44 +02:00
										 |  |  | import {UIElement} from "./UI/UIElement"; | 
					
						
							| 
									
										
										
										
											2020-08-17 17:23:15 +02:00
										 |  |  | // We HAVE to mark this while importing
 | 
					
						
							| 
									
										
										
										
											2020-07-31 17:11:44 +02:00
										 |  |  | UIElement.runningFromConsole = true; | 
					
						
							| 
									
										
										
										
											2020-10-23 02:15:58 +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"; | 
					
						
							| 
									
										
										
										
											2020-07-31 18:27:40 +02:00
										 |  |  | import Locale from "./UI/i18n/Locale"; | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | import svg2img from 'promise-svg2img'; | 
					
						
							|  |  |  | import Translation from "./UI/i18n/Translation"; | 
					
						
							|  |  |  | import Translations from "./UI/i18n/Translations"; | 
					
						
							| 
									
										
										
										
											2020-10-17 03:42:30 +02:00
										 |  |  | import {TagRendering} from "./UI/Popup/TagRendering"; | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | TagRendering.injectFunction(); | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  |     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) { | 
					
						
							| 
									
										
										
										
											2020-09-25 23:59:19 +02:00
										 |  |  |             if (translation.translations["*"] !== undefined) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |             const txt = translation.translations[ln]; | 
					
						
							|  |  |  |             const isMissing = txt === undefined || txt === "" || txt.toLowerCase().indexOf("todo") >= 0; | 
					
						
							|  |  |  |             if (isMissing) { | 
					
						
							| 
									
										
										
										
											2020-09-25 23:59:19 +02:00
										 |  |  |                 console.log(`   ${layout.id}: No translation for`, ln, "in", translation.translations, "got:", txt) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |                 missing[ln]++ | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 present[ln]++; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 23:59:19 +02:00
										 |  |  |     let message = `Translation completenes for theme ${layout.id}` | 
					
						
							|  |  |  |     let isComplete = true; | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     for (const ln of layout.supportedLanguages) { | 
					
						
							|  |  |  |         const amiss = missing[ln]; | 
					
						
							|  |  |  |         const ok = present[ln]; | 
					
						
							|  |  |  |         const total = amiss + ok; | 
					
						
							| 
									
										
										
										
											2020-09-25 23:59:19 +02:00
										 |  |  |         message += `\n${ln}: ${ok}/${total}` | 
					
						
							|  |  |  |         if (ok !== total) { | 
					
						
							|  |  |  |             isComplete = false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (isComplete) { | 
					
						
							|  |  |  |         console.log(`${layout.id} is fully translated!`) | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         console.log(message) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-17 19:06:32 +02:00
										 |  |  | function generateWikiEntry(layout: Layout){ | 
					
						
							|  |  |  |     if(layout.hideFromOverview){ | 
					
						
							|  |  |  |         return ""; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     let image = "MapComplete_Screenshot.png"; | 
					
						
							|  |  |  |     if(layout.socialImage){ | 
					
						
							|  |  |  |     //    image = layout.socialImage;
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     if(!image.startsWith("http")){ | 
					
						
							|  |  |  |    //     image = "https://pietervdvn.github.io/MapComplete/"+image
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |    return `{{Software
 | 
					
						
							|  |  |  | |name           = ${layout.id} | 
					
						
							|  |  |  | |author         = ${layout.maintainer ?? "MapComplete builtin"} | 
					
						
							|  |  |  | |web            = https://pietervdvn.github.io/MapComplete/${layout.id}.html
 | 
					
						
							|  |  |  | |repo           = https://github.com/pietervdvn/MapComplete
 | 
					
						
							|  |  |  | |platform       = web | 
					
						
							|  |  |  | |code           = Typescript;HTML;CSS | 
					
						
							|  |  |  | |languages      = ${layout.supportedLanguages.join(";")} | 
					
						
							|  |  |  | |genre          = display;editor | 
					
						
							|  |  |  | |screenshot     = ${image} | 
					
						
							|  |  |  | |description    = A MapComplete theme: ${Translations.W(layout.description)?.InnerRender() ?? ""} | 
					
						
							|  |  |  | |map        = yes | 
					
						
							|  |  |  | |findLocation            = yes | 
					
						
							|  |  |  | |findNearbyPOI           = yes | 
					
						
							|  |  |  | |addPOI          = yes | 
					
						
							|  |  |  | |editPOI         = yes | 
					
						
							|  |  |  | |editTags        = yes | 
					
						
							|  |  |  | | | 
					
						
							|  |  |  | }}`
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 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) { | 
					
						
							| 
									
										
										
										
											2020-09-20 20:28:35 +02:00
										 |  |  |         // Errors are normal here if this file exists
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  |         console.log("Creating icon ", name, newname) | 
					
						
							|  |  |  |         // We already read to file, in order to crash here if the file is not found
 | 
					
						
							|  |  |  |         readFileSync(iconPath);  | 
					
						
							| 
									
										
										
										
											2020-09-25 23:37:59 +02:00
										 |  |  |         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 while writing" + iconPath, error) | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2020-08-22 03:15:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 23:37:59 +02:00
										 |  |  |     } catch (e) { | 
					
						
							|  |  |  |         console.error("Could not read icon", iconPath, "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
										 |  |  | 
 | 
					
						
							|  |  |  | function createManifest(layout: Layout, relativePath: string) { | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |     const name = layout.id; | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const icons = []; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-22 03:15:42 +02:00
										 |  |  |     let icon = layout.icon; | 
					
						
							|  |  |  |     if (icon.endsWith(".svg")) { | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         // 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) { | 
					
						
							| 
									
										
										
										
											2020-08-22 03:15:42 +02:00
										 |  |  |             const name = createIcon(icon, size); | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |             icons.push({ | 
					
						
							|  |  |  |                 src: name, | 
					
						
							|  |  |  |                 sizes: size + "x" + size, | 
					
						
							|  |  |  |                 type: "image/png" | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         icons.push({ | 
					
						
							| 
									
										
										
										
											2020-08-22 03:15:42 +02:00
										 |  |  |             src: icon, | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |             sizes: "513x513", | 
					
						
							|  |  |  |             type: "image/svg" | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  |     } | 
					
						
							|  |  |  |     const ogTitle = Translations.W(layout.title).InnerRender(); | 
					
						
							| 
									
										
										
										
											2020-08-22 03:15:42 +02:00
										 |  |  |     const ogDescr = Translations.W(layout.description ?? "").InnerRender(); | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const manif = { | 
					
						
							|  |  |  |         name: name, | 
					
						
							|  |  |  |         short_name: ogTitle, | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |         start_url: `${relativePath}/${layout.id.toLowerCase()}.html`, | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         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]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |     const ogTitle = Translations.W(layout.title)?.InnerRender(); | 
					
						
							|  |  |  |     const ogDescr = Translations.W(layout.description ?? "Easily add and edit geodata with OpenStreetMap")?.InnerRender(); | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     const ogImage = layout.socialImage; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-23 02:15:58 +02:00
										 |  |  |     let customCss = ""; | 
					
						
							|  |  |  |     if (layout.customCss !== undefined && layout.customCss !== "") { | 
					
						
							|  |  |  |         customCss = `<link rel='stylesheet" href=${layout.customCss}"/>` | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     const og = `
 | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |     <meta property="og:image" content="${ogImage ?? './assets/add.svg'}"> | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     <meta property="og:title" content="${ogTitle}"> | 
					
						
							|  |  |  |     <meta property="og:description" content="${ogDescr}">`
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |     let output = template | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |         .replace(`./manifest.manifest`, `./${enc(layout.id)}.webmanifest`) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         .replace("<!-- $$$OG-META -->", og) | 
					
						
							| 
									
										
										
										
											2020-10-23 02:15:58 +02:00
										 |  |  |         .replace(/<title>.+?<\/title>/, `<title>${ogTitle}</title>`) | 
					
						
							|  |  |  |         .replace("Loading MapComplete, hang on...", `Loading MapComplete theme <i>${ogTitle}</i>...`) | 
					
						
							|  |  |  |         .replace("<!-- $$$CUSTOM-CSS -->", customCss) | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |         .replace(`<link rel="icon" href="assets/add.svg" sizes="any" type="image/svg+xml">`, | 
					
						
							| 
									
										
										
										
											2020-08-27 11:11:20 +02:00
										 |  |  |             `<link rel="icon" href="${layout.icon}" sizes="any" type="image/svg+xml">`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |         output = output | 
					
						
							|  |  |  |             .replace(/<!-- DECORATION 0 START -->.*<!-- DECORATION 0 END -->/s, `<img src='${layout.icon}' width="100%" height="100%">`) | 
					
						
							|  |  |  |             .replace(/<!-- DECORATION 1 START -->.*<!-- DECORATION 1 END -->/s, `<img src='${layout.icon}' width="100%" height="100%">`); | 
					
						
							|  |  |  |     } catch (e) { | 
					
						
							|  |  |  |         console.warn("Error while applying logo: ", e) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return output; | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const blacklist = ["", "test", ".", "..", "manifest", "index", "land", "preferences", "account", "openstreetmap"] | 
					
						
							|  |  |  | const all = AllKnownLayouts.allSets; | 
					
						
							| 
									
										
										
										
											2020-09-17 19:06:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | let wikiPage = ""; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | 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) | 
					
						
							|  |  |  |     const manif = JSON.stringify(createManifest(layout, "/MapComplete")); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |     const manifestLocation = encodeURIComponent(layout.id.toLowerCase()) + ".webmanifest"; | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  |     writeFile(manifestLocation, manif, err); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const landing = createLandingPage(layout); | 
					
						
							| 
									
										
										
										
											2020-09-03 00:00:37 +02:00
										 |  |  |     writeFile(enc(layout.id) + ".html", landing, err) | 
					
						
							| 
									
										
										
										
											2020-09-17 19:06:32 +02:00
										 |  |  |      | 
					
						
							|  |  |  |     wikiPage += "\n\n"+generateWikiEntry(layout); | 
					
						
							| 
									
										
										
										
											2020-07-26 02:01:34 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-09-26 01:11:17 +02:00
										 |  |  | writeFile("./assets/generated/wikiIndex", wikiPage, (err) => { | 
					
						
							| 
									
										
										
										
											2020-09-25 23:59:19 +02:00
										 |  |  |     if (err !== null) { | 
					
						
							|  |  |  |         console.log("Could not save wikiindex", err); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
										
										
											2020-08-22 03:15:42 +02:00
										 |  |  | console.log("Counting all translations") | 
					
						
							| 
									
										
										
										
											2020-07-31 17:11:44 +02:00
										 |  |  | Translations.CountTranslations(); | 
					
						
							| 
									
										
										
										
											2020-09-17 19:06:32 +02:00
										 |  |  | console.log("All done!"); |