mapcomplete/scripts/generateDocs.ts

453 lines
16 KiB
TypeScript
Raw Normal View History

2023-07-17 22:04:35 +02:00
import Combine from "../src/UI/Base/Combine"
import BaseUIElement from "../src/UI/BaseUIElement"
import { existsSync, mkdirSync, readFileSync, writeFile, writeFileSync } from "fs"
2023-07-17 22:04:35 +02:00
import { AllKnownLayouts } from "../src/Customizations/AllKnownLayouts"
import TableOfContents from "../src/UI/Base/TableOfContents"
import SimpleMetaTaggers from "../src/Logic/SimpleMetaTagger"
import SpecialVisualizations from "../src/UI/SpecialVisualizations"
import { ExtraFunctions } from "../src/Logic/ExtraFunctions"
import Title from "../src/UI/Base/Title"
import QueryParameterDocumentation from "../src/UI/QueryParameterDocumentation"
2022-09-08 21:40:48 +02:00
import ScriptUtils from "./ScriptUtils"
2023-07-17 22:04:35 +02:00
import List from "../src/UI/Base/List"
import Translations from "../src/UI/i18n/Translations"
2023-07-20 13:28:38 +02:00
import themeOverview from "../src/assets/generated/theme_overview.json"
2023-07-17 22:04:35 +02:00
import LayoutConfig from "../src/Models/ThemeConfig/LayoutConfig"
2023-07-20 13:28:38 +02:00
import bookcases from "../src/assets/generated/themes/bookcases.json"
import fakedom from "fake-dom"
2023-07-28 00:29:21 +02:00
2023-07-17 22:04:35 +02:00
import Hotkeys from "../src/UI/Base/Hotkeys"
import { QueryParameters } from "../src/Logic/Web/QueryParameters"
import Link from "../src/UI/Base/Link"
import Constants from "../src/Models/Constants"
import LayerConfig from "../src/Models/ThemeConfig/LayerConfig"
import DependencyCalculator from "../src/Models/ThemeConfig/DependencyCalculator"
import { AllSharedLayers } from "../src/Customizations/AllSharedLayers"
import ThemeViewState from "../src/Models/ThemeViewState"
import Validators from "../src/UI/InputElement/Validators"
2023-07-20 13:28:38 +02:00
import questions from "../src/assets/generated/layers/questions.json"
import { LayerConfigJson } from "../src/Models/ThemeConfig/Json/LayerConfigJson"
2023-07-28 00:29:21 +02:00
import { Utils } from "../src/Utils"
import { TagUtils } from "../src/Logic/Tags/TagUtils"
2022-09-08 21:40:48 +02:00
function WriteFile(
filename,
2023-06-22 15:07:14 +02:00
html: string | BaseUIElement,
2022-09-08 21:40:48 +02:00
autogenSource: string[],
options?: {
noTableOfContents: boolean
}
): void {
2023-06-28 22:43:06 +02:00
if (!html) {
return
}
2022-02-15 00:11:11 +01:00
for (const source of autogenSource) {
2022-09-08 21:40:48 +02:00
if (source.indexOf("*") > 0) {
2022-02-15 00:11:11 +01:00
continue
}
2022-09-08 21:40:48 +02:00
if (!existsSync(source)) {
throw (
"While creating a documentation file and checking that the generation sources are properly linked: source file " +
source +
" was not found. Typo?"
)
2022-02-15 00:11:11 +01:00
}
}
2022-09-08 21:40:48 +02:00
if (html instanceof Combine && !options?.noTableOfContents) {
const toc = new TableOfContents(html)
const els = html.getElements()
html = new Combine([els.shift(), toc, ...els]).SetClass("flex flex-col")
2021-11-30 22:50:48 +01:00
}
2022-09-08 21:40:48 +02:00
let md = new Combine([
Translations.W(html),
"\n\nThis document is autogenerated from " +
autogenSource
.map(
(file) =>
`[${file}](https://github.com/pietervdvn/MapComplete/blob/develop/${file})`
)
.join(", "),
]).AsMarkdown()
2022-09-08 21:40:48 +02:00
md.replace(/\n\n\n+/g, "\n\n")
2022-01-26 21:40:38 +01:00
if (!md.endsWith("\n")) {
md += "\n"
}
const warnAutomated =
"[//]: # (WARNING: this file is automatically generated. Please find the sources at the bottom and edit those sources)"
writeFileSync(filename, warnAutomated + md)
}
function GenerateDocumentationForTheme(theme: LayoutConfig): BaseUIElement {
return new Combine([
new Title(
new Combine([
theme.title,
"(",
new Link(theme.id, "https://mapcomplete.org/" + theme.id),
")",
]),
2
),
theme.description,
"This theme contains the following layers:",
new List(
theme.layers
.filter((l) => !l.id.startsWith("note_import_"))
.map((l) => new Link(l.id, "../Layers/" + l.id + ".md"))
),
"Available languages:",
new List(theme.language.filter((ln) => ln !== "_context")),
]).SetClass("flex flex-col")
}
/**
* Generates the documentation for the layers overview page
* @constructor
*/
function GenLayerOverviewText(): BaseUIElement {
for (const id of Constants.priviliged_layers) {
if (!AllSharedLayers.sharedLayers.has(id)) {
2023-06-22 15:07:14 +02:00
console.error("Priviliged layer definition not found: " + id)
2023-06-28 22:43:06 +02:00
return undefined
}
}
const allLayers: LayerConfig[] = Array.from(AllSharedLayers.sharedLayers.values()).filter(
2023-07-28 00:29:21 +02:00
(layer) => layer["source"] === null
)
const builtinLayerIds: Set<string> = new Set<string>()
allLayers.forEach((l) => builtinLayerIds.add(l.id))
const themesPerLayer = new Map<string, string[]>()
for (const layout of Array.from(AllKnownLayouts.allKnownLayouts.values())) {
for (const layer of layout.layers) {
if (!builtinLayerIds.has(layer.id)) {
continue
}
if (!themesPerLayer.has(layer.id)) {
themesPerLayer.set(layer.id, [])
}
themesPerLayer.get(layer.id).push(layout.id)
}
}
// Determine the cross-dependencies
const layerIsNeededBy: Map<string, string[]> = new Map<string, string[]>()
for (const layer of allLayers) {
for (const dep of DependencyCalculator.getLayerDependencies(layer)) {
const dependency = dep.neededLayer
if (!layerIsNeededBy.has(dependency)) {
layerIsNeededBy.set(dependency, [])
}
layerIsNeededBy.get(dependency).push(layer.id)
}
}
return new Combine([
new Title("Special and other useful layers", 1),
"MapComplete has a few data layers available in the theme which have special properties through builtin-hooks. Furthermore, there are some normal layers (which are built from normal Theme-config files) but are so general that they get a mention here.",
new Title("Priviliged layers", 1),
new List(Constants.priviliged_layers.map((id) => "[" + id + "](#" + id + ")")),
2023-06-22 15:07:14 +02:00
...Utils.NoNull(
Constants.priviliged_layers.map((id) => AllSharedLayers.sharedLayers.get(id))
).map((l) =>
l.GenerateDocumentation(
themesPerLayer.get(l.id),
layerIsNeededBy,
DependencyCalculator.getLayerDependencies(l),
Constants.added_by_default.indexOf(<any>l.id) >= 0,
Constants.no_include.indexOf(<any>l.id) < 0
)
),
new Title("Normal layers", 1),
"The following layers are included in MapComplete:",
new List(
Array.from(AllSharedLayers.sharedLayers.keys()).map(
(id) => new Link(id, "./Layers/" + id + ".md")
)
),
])
}
/**
* Generates documentation for the layers.
* Inline layers are included (if the theme is public)
* @param callback
* @constructor
*/
function GenOverviewsForSingleLayer(
callback: (layer: LayerConfig, element: BaseUIElement, inlineSource: string) => void
): void {
const allLayers: LayerConfig[] = Array.from(AllSharedLayers.sharedLayers.values()).filter(
2023-07-28 00:29:21 +02:00
(layer) => layer["source"] !== null
)
const builtinLayerIds: Set<string> = new Set<string>()
allLayers.forEach((l) => builtinLayerIds.add(l.id))
const inlineLayers = new Map<string, string>()
for (const layout of Array.from(AllKnownLayouts.allKnownLayouts.values())) {
if (layout.hideFromOverview) {
continue
}
for (const layer of layout.layers) {
2023-03-25 02:48:24 +01:00
if (layer.source === null) {
continue
}
if (builtinLayerIds.has(layer.id)) {
continue
}
if (layer.source.geojsonSource !== undefined) {
// Not an OSM-source
continue
}
allLayers.push(layer)
builtinLayerIds.add(layer.id)
inlineLayers.set(layer.id, layout.id)
}
}
const themesPerLayer = new Map<string, string[]>()
for (const layout of Array.from(AllKnownLayouts.allKnownLayouts.values())) {
if (layout.hideFromOverview) {
continue
}
for (const layer of layout.layers) {
if (!builtinLayerIds.has(layer.id)) {
// This is an inline layer
continue
}
if (!themesPerLayer.has(layer.id)) {
themesPerLayer.set(layer.id, [])
}
themesPerLayer.get(layer.id).push(layout.id)
}
}
// Determine the cross-dependencies
const layerIsNeededBy: Map<string, string[]> = new Map<string, string[]>()
for (const layer of allLayers) {
for (const dep of DependencyCalculator.getLayerDependencies(layer)) {
const dependency = dep.neededLayer
if (!layerIsNeededBy.has(dependency)) {
layerIsNeededBy.set(dependency, [])
}
layerIsNeededBy.get(dependency).push(layer.id)
}
}
allLayers.forEach((layer) => {
const element = layer.GenerateDocumentation(
themesPerLayer.get(layer.id),
layerIsNeededBy,
DependencyCalculator.getLayerDependencies(layer)
)
callback(layer, element, inlineLayers.get(layer.id))
})
}
2022-10-27 22:16:28 +02:00
/**
* The wikitable is updated as some tools show an overview of apps based on the wiki.
*/
2022-11-02 13:47:34 +01:00
function generateWikipage() {
function generateWikiEntry(layout: {
hideFromOverview: boolean
id: string
shortDescription: any
}) {
2022-10-27 22:03:41 +02:00
if (layout.hideFromOverview) {
2022-11-02 13:47:34 +01:00
return ""
2022-10-27 22:03:41 +02:00
}
const languagesInDescr = Array.from(Object.keys(layout.shortDescription)).filter(
(k) => k !== "_context"
)
2022-11-02 13:47:34 +01:00
const languages = languagesInDescr.map((ln) => `{{#language:${ln}|en}}`).join(", ")
let auth = "Yes"
2022-10-27 22:03:41 +02:00
return `{{service_item
|name= [https://mapcomplete.org/${layout.id} ${layout.id}]
2022-10-27 22:03:41 +02:00
|region= Worldwide
|lang= ${languages}
|descr= A MapComplete theme: ${Translations.T(layout.shortDescription)
.textFor("en")
.replace("<a href='", "[[")
2022-11-02 13:47:34 +01:00
.replace(/'>.*<\/a>/, "]]")}
|material= {{yes|[https://mapcomplete.org/ ${auth}]}}
2022-10-27 22:03:41 +02:00
|image= MapComplete_Screenshot.png
|genre= POI, editor, ${layout.id}
}}`
}
2022-11-02 13:47:34 +01:00
let wikiPage =
'{|class="wikitable sortable"\n' +
2022-10-27 22:03:41 +02:00
"! Name, link !! Genre !! Covered region !! Language !! Description !! Free materials !! Image\n" +
2022-11-02 13:47:34 +01:00
"|-"
2022-10-27 22:03:41 +02:00
for (const layout of themeOverview) {
2022-10-27 22:03:41 +02:00
if (layout.hideFromOverview) {
2022-11-02 13:47:34 +01:00
continue
2022-10-27 22:03:41 +02:00
}
2022-11-02 13:47:34 +01:00
wikiPage += "\n" + generateWikiEntry(layout)
2022-10-27 22:03:41 +02:00
}
wikiPage += "\n|}"
writeFile("Docs/wikiIndex.txt", wikiPage, (err) => {
if (err !== null) {
2022-11-02 13:47:34 +01:00
console.log("Could not save wikiindex", err)
2022-10-27 22:03:41 +02:00
}
2022-11-02 13:47:34 +01:00
})
2022-10-27 22:03:41 +02:00
}
function studioDocs() {
const lines = readFileSync("./Docs/Studio/Introduction.md", "utf8").split("\n")
const sections: string[][] = []
let currentSection: string[] = []
for (let line of lines) {
if (line.trim().startsWith("# ")) {
sections.push(currentSection)
currentSection = []
}
line = line.replace('src="../../public/', 'src="./')
line = line.replace('src="../../', 'src="./')
currentSection.push(line)
}
writeFileSync(
"./src/assets/studio_introduction.json",
JSON.stringify({
sections: sections.map((s) => s.join("\n")).filter((s) => s.length > 0),
})
)
}
console.log("Starting documentation generation...")
ScriptUtils.fixUtils()
studioDocs()
2022-10-27 22:16:28 +02:00
generateWikipage()
GenOverviewsForSingleLayer((layer, element, inlineSource) => {
ScriptUtils.erasableLog("Exporting layer documentation for", layer.id)
2022-09-08 21:40:48 +02:00
if (!existsSync("./Docs/Layers")) {
2022-04-06 16:12:01 +02:00
mkdirSync("./Docs/Layers")
}
let source: string = `assets/layers/${layer.id}/${layer.id}.json`
2022-09-08 21:40:48 +02:00
if (inlineSource !== undefined) {
2022-04-06 16:12:01 +02:00
source = `assets/themes/${inlineSource}/${inlineSource}.json`
}
2022-09-08 21:40:48 +02:00
WriteFile("./Docs/Layers/" + layer.id + ".md", element, [source], { noTableOfContents: true })
})
2022-12-16 13:40:25 +01:00
Array.from(AllKnownLayouts.allKnownLayouts.values()).map((theme) => {
if (!existsSync("./Docs/Themes")) {
mkdirSync("./Docs/Themes")
}
const docs = GenerateDocumentationForTheme(theme)
2022-12-16 13:40:25 +01:00
WriteFile(
"./Docs/Themes/" + theme.id + ".md",
docs,
[`assets/themes/${theme.id}/${theme.id}.json`],
{ noTableOfContents: true }
)
})
2022-09-08 21:40:48 +02:00
WriteFile("./Docs/SpecialRenderings.md", SpecialVisualizations.HelpMessage(), [
2023-07-17 22:04:35 +02:00
"src/UI/SpecialVisualizations.ts",
2022-09-08 21:40:48 +02:00
])
WriteFile(
"./Docs/CalculatedTags.md",
new Combine([
new Title("Metatags", 1),
SimpleMetaTaggers.HelpText(),
ExtraFunctions.HelpText(),
]).SetClass("flex-col"),
2023-07-17 22:04:35 +02:00
["src/Logic/SimpleMetaTagger.ts", "src/Logic/ExtraFunctions.ts"]
2022-09-08 21:40:48 +02:00
)
WriteFile("./Docs/SpecialInputElements.md", Validators.HelpText(), [
2023-07-17 22:04:35 +02:00
"src/UI/InputElement/Validators.ts",
])
WriteFile("./Docs/BuiltinLayers.md", GenLayerOverviewText(), [
"src/Customizations/AllKnownLayouts.ts",
2022-09-08 21:40:48 +02:00
])
2023-07-20 13:28:38 +02:00
const qLayer = new LayerConfig(<LayerConfigJson>questions, "questions.json", true)
WriteFile("./Docs/BuiltinQuestions.md", qLayer.GenerateDocumentation([], new Map(), []), [
2023-07-17 22:04:35 +02:00
"assets/layers/questions/questions.json",
2022-09-08 21:40:48 +02:00
])
2023-07-28 00:29:21 +02:00
WriteFile("./Docs/Tags_format.md", TagUtils.generateDocs(), ["src/Logic/Tags/TagUtils.ts"])
{
2022-02-04 01:05:35 +01:00
// Generate the builtinIndex which shows interlayer dependencies
2022-09-08 21:40:48 +02:00
var layers = ScriptUtils.getLayerFiles().map((f) => f.parsed)
var builtinsPerLayer = new Map<string, string[]>()
var layersUsingBuiltin = new Map<string /* Builtin */, string[]>()
for (const layer of layers) {
2022-09-08 21:40:48 +02:00
if (layer.tagRenderings === undefined) {
continue
}
2022-09-08 21:40:48 +02:00
const usedBuiltins: string[] = []
for (const tagRendering of layer.tagRenderings) {
2022-09-08 21:40:48 +02:00
if (typeof tagRendering === "string") {
usedBuiltins.push(tagRendering)
continue
}
2022-09-08 21:40:48 +02:00
if (tagRendering["builtin"] !== undefined) {
const builtins = tagRendering["builtin"]
2022-09-08 21:40:48 +02:00
if (typeof builtins === "string") {
usedBuiltins.push(builtins)
2022-09-08 21:40:48 +02:00
} else {
usedBuiltins.push(...builtins)
}
}
}
for (const usedBuiltin of usedBuiltins) {
2023-08-10 15:36:32 +02:00
const usingLayers = layersUsingBuiltin.get(usedBuiltin)
if (usingLayers === undefined) {
layersUsingBuiltin.set(usedBuiltin, [layer.id])
2022-09-08 21:40:48 +02:00
} else {
2023-08-10 15:36:32 +02:00
usingLayers.push(layer.id)
}
}
2022-09-08 21:40:48 +02:00
builtinsPerLayer.set(layer.id, usedBuiltins)
}
2022-09-08 21:40:48 +02:00
const docs = new Combine([
2022-09-08 21:40:48 +02:00
new Title("Index of builtin TagRendering", 1),
new Title("Existing builtin tagrenderings", 2),
2022-09-08 21:40:48 +02:00
...Array.from(layersUsingBuiltin.entries()).map(([builtin, usedByLayers]) =>
new Combine([new Title(builtin), new List(usedByLayers)]).SetClass("flex flex-col")
),
]).SetClass("flex flex-col")
WriteFile("./Docs/BuiltinIndex.md", docs, ["assets/layers/*.json"])
}
2022-09-08 21:40:48 +02:00
WriteFile("./Docs/URL_Parameters.md", QueryParameterDocumentation.GenerateQueryParameterDocs(), [
2023-07-17 22:04:35 +02:00
"src/Logic/Web/QueryParameters.ts",
"src/UI/QueryParameterDocumentation.ts",
2022-09-08 21:40:48 +02:00
])
2023-08-10 15:36:32 +02:00
if (fakedom === undefined) {
throw "FakeDom not initialized"
}
QueryParameters.GetQueryParameter(
"mode",
"map",
"The mode the application starts in, e.g. 'map', 'dashboard' or 'statistics'"
)
{
new ThemeViewState(new LayoutConfig(<any>bookcases))
WriteFile("./Docs/Hotkeys.md", Hotkeys.generateDocumentation(), [])
}
2021-03-22 01:06:24 +01:00
console.log("Generated docs")