MapComplete/Models/ThemeConfig/WithContextLoader.ts

88 lines
3.1 KiB
TypeScript
Raw Normal View History

2022-09-08 21:40:48 +02:00
import TagRenderingConfig from "./TagRenderingConfig"
import SharedTagRenderings from "../../Customizations/SharedTagRenderings"
import { TagRenderingConfigJson } from "./Json/TagRenderingConfigJson"
export default class WithContextLoader {
2022-09-08 21:40:48 +02:00
protected readonly _context: string
private readonly _json: any
2021-10-22 18:53:07 +02:00
constructor(json: any, context: string) {
2022-09-08 21:40:48 +02:00
this._json = json
this._context = context
}
/** Given a key, gets the corresponding property from the json (or the default if not found
*
* The found value is interpreted as a tagrendering and fetched/parsed
* */
2023-06-01 02:52:21 +02:00
public tr(key: string, deflt?: string, translationContext?: string) {
2022-09-08 21:40:48 +02:00
const v = this._json[key]
if (v === undefined || v === null) {
if (deflt === undefined) {
2022-09-08 21:40:48 +02:00
return undefined
}
return new TagRenderingConfig(
deflt,
`${translationContext ?? this._context}.${key}.default value`
)
}
if (typeof v === "string") {
2022-09-08 21:40:48 +02:00
const shared = SharedTagRenderings.SharedTagRendering.get(v)
if (shared) {
2022-09-08 21:40:48 +02:00
return shared
}
}
if (Object.keys(v).length === 1 && typeof v["render"] === "string") {
throw `At ${
translationContext ?? "<unknown>"
}: use the content directly instead of {${key}: ${JSON.stringify(v)}}`
}
return new TagRenderingConfig(v, `${translationContext ?? this._context}.${key}`)
}
/**
* Converts a list of tagRenderingCOnfigJSON in to TagRenderingConfig
* A string is interpreted as a name to call
*/
2021-10-22 18:53:07 +02:00
public ParseTagRenderings(
2021-12-21 18:35:31 +01:00
tagRenderings: TagRenderingConfigJson[],
options?: {
/**
* Throw an error if 'question' is defined
*/
2022-09-08 21:40:48 +02:00
readOnlyMode?: boolean
requiresId?: boolean
2022-09-08 21:40:48 +02:00
prepConfig?: (config: TagRenderingConfigJson) => TagRenderingConfigJson
}
2021-11-07 16:34:51 +01:00
): TagRenderingConfig[] {
if (tagRenderings === undefined) {
2022-09-08 21:40:48 +02:00
return []
}
2021-10-22 18:53:07 +02:00
const context = this._context
options = options ?? {}
if (options.prepConfig === undefined) {
2022-09-08 21:40:48 +02:00
options.prepConfig = (c) => c
2021-10-22 18:53:07 +02:00
}
2021-11-14 18:01:48 +01:00
const renderings: TagRenderingConfig[] = []
2021-12-21 18:35:31 +01:00
for (let i = 0; i < tagRenderings.length; i++) {
2022-09-08 21:40:48 +02:00
const preparedConfig = tagRenderings[i]
const tr = new TagRenderingConfig(preparedConfig, `${context}.tagrendering[${i}]`)
2021-12-21 18:35:31 +01:00
if (options.readOnlyMode && tr.question !== undefined) {
2022-09-08 21:40:48 +02:00
throw (
"A question is defined for " +
`${context}.tagrendering[${i}], but this is not allowed at this position - probably because this rendering is an icon, badge or label`
)
}
2021-12-21 18:35:31 +01:00
if (options.requiresId && tr.id === "") {
throw `${context}.tagrendering[${i}] has an invalid ID - make sure it is defined and not empty`
}
2021-12-21 18:35:31 +01:00
renderings.push(tr)
}
2022-09-08 21:40:48 +02:00
return renderings
}
2022-09-08 21:40:48 +02:00
}