import {UIElement} from "./UI/UIElement"; // We HAVE to mark this while importing UIElement.runningFromConsole = true; import {AllKnownLayouts} from "./Customizations/AllKnownLayouts"; import {Layout} from "./Customizations/Layout"; import {readFileSync, writeFile, writeFileSync} from "fs"; import Locale from "./UI/i18n/Locale"; import svg2img from 'promise-svg2img'; import Translation from "./UI/i18n/Translation"; import Translations from "./UI/i18n/Translations"; console.log("Building the layouts") 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) { if (translation.translations["*"] !== undefined) { continue; } const txt = translation.translations[ln]; const isMissing = txt === undefined || txt === "" || txt.toLowerCase().indexOf("todo") >= 0; if (isMissing) { console.log(` ${layout.id}: No translation for`, ln, "in", translation.translations, "got:", txt) missing[ln]++ } else { present[ln]++; } } } let message = `Translation completenes for theme ${layout.id}` let isComplete = true; for (const ln of layout.supportedLanguages) { const amiss = missing[ln]; const ok = present[ln]; const total = amiss + ok; message += `\n${ln}: ${ok}/${total}` if (ok !== total) { isComplete = false; } } if (isComplete) { console.log(`${layout.id} is fully translated!`) } else { console.log(message) } } function generateWikiEntry(layout: Layout){ if(layout.hideFromOverview){ return ""; } const languages = layout.supportedLanguages.map(ln => `{{#language:${ln}|en}}`).join(", ") let auth = "Yes"; if(layout.maintainer !== "" && layout.maintainer !== "MapComplete"){ auth=`Yes, by ${layout.maintainer};` } return `{{service_item |name= [https://pietervdvn.github.io/MapComplete/${layout.id}.html ${layout.id}] |region= Worldwide |lang= ${languages} |descr= A MapComplete theme: ${Translations.W(layout.description).InnerRender()} |material= {{yes|[https://github.com/pietervdvn/MapComplete ${auth}]}} |image= MapComplete_Screenshot.png |genre= POI, editor, ${layout.id} }}` } 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) { // Errors are normal here if this file exists } try { console.log("Creating icon ", name, newname) // We already read to file, in order to crash here if the file is not found readFileSync(iconPath); 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) }); } catch (e) { console.error("Could not read icon", iconPath, "due to", e) } return newname; } function createManifest(layout: Layout, relativePath: string) { const name = layout.id; const icons = []; let icon = layout.icon; if (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(icon, size); icons.push({ src: name, sizes: size + "x" + size, type: "image/png" }) } icons.push({ src: icon, sizes: "513x513", type: "image/svg" }) } else { throw "Icon is not an svg for " + layout.id } 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.id.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 ?? "Easily add and edit geodata with OpenStreetMap")?.InnerRender(); const ogImage = layout.socialImage; let customCss = ""; if (layout.customCss !== undefined && layout.customCss !== "") { customCss = ` ` let output = template .replace(`./manifest.manifest`, `./${enc(layout.id)}.webmanifest`) .replace("", og) .replace(/.+?<\/title>/, `<title>${ogTitle}`) .replace("Loading MapComplete, hang on...", `Loading MapComplete theme ${ogTitle}...`) .replace("", customCss) .replace(``, ``); try { output = output .replace(/.*/s, ``) .replace(/.*/s, ``); } catch (e) { console.warn("Error while applying logo: ", e) } return output; } const blacklist = ["", "test", ".", "..", "manifest", "index", "land", "preferences", "account", "openstreetmap"] const all = AllKnownLayouts.allSets; let wikiPage = "{|class=\"wikitable sortable\"\n" + "! Name, link !! Genre !! Covered region !! Language !! Description !! Free materials !! Image\n" + "|-"; 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")); const manifestLocation = encodeURIComponent(layout.id.toLowerCase()) + ".webmanifest"; writeFile(manifestLocation, manif, err); const landing = createLandingPage(layout); writeFile(enc(layout.id) + ".html", landing, err) wikiPage += "\n"+generateWikiEntry(layout); } wikiPage += "|}" writeFile("./assets/generated/wikiIndex", wikiPage, (err) => { if (err !== null) { console.log("Could not save wikiindex", err); } }); console.log("Counting all translations") Translations.CountTranslations(); console.log("All done!");