MapComplete/src/Models/ThemeConfig/DependencyCalculator.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

140 lines
5.2 KiB
TypeScript
Raw Normal View History

import TagRenderingConfig from "./TagRenderingConfig"
import { ExtraFuncParams, ExtraFunctions } from "../../Logic/ExtraFunctions"
import LayerConfig from "./LayerConfig"
2022-11-02 14:44:06 +01:00
import { SpecialVisualization } from "../../UI/SpecialVisualization"
import SpecialVisualizations from "../../UI/SpecialVisualizations"
export default class DependencyCalculator {
public static GetTagRenderingDependencies(tr: TagRenderingConfig): string[] {
2022-01-26 21:40:38 +01:00
if (tr === undefined) {
throw "Got undefined tag rendering in getTagRenderingDependencies"
}
const deps: string[] = []
// All translated snippets
const parts: string[] = [].concat(...tr.EnumerateTranslations().map((tr) => tr.AllValues()))
for (const part of parts) {
const specialVizs: { func: SpecialVisualization; args: string[] }[] =
SpecialVisualizations.constructSpecification(part)
.filter((p) => typeof p !== "string")
.map((p) => <{ func: SpecialVisualization; args: string[] }>p)
.filter((o) => o?.func?.getLayerDependencies !== undefined)
for (const specialViz of specialVizs) {
deps.push(...specialViz.func.getLayerDependencies(specialViz.args))
}
}
return deps
}
/**
* Returns a set of all other layer-ids that this layer needs to function.
* E.g. if this layers does snap to another layer in the preset, this other layer id will be mentioned
*/
public static getLayerDependencies(
layer: LayerConfig
2024-10-19 14:44:55 +02:00
): {
neededLayer: string
reason: string
context?: string
neededBy: string
checkHasSnapName: boolean
}[] {
const deps: {
neededLayer: string
reason: string
context?: string
neededBy: string
checkHasSnapName: boolean
}[] = []
for (let i = 0; layer.presets !== undefined && i < layer.presets.length; i++) {
const preset = layer.presets[i]
2023-08-24 12:22:45 +02:00
const snapTo = preset.preciseInput?.snapToLayers
if (snapTo && !Array.isArray(snapTo)) {
2023-09-17 13:11:06 +02:00
throw new Error(
`snapToLayers is not an array; it is ${snapTo}(used in preset ${i} for: ${layer.id})`
)
2023-08-24 12:22:45 +02:00
}
preset.preciseInput?.snapToLayers?.forEach((id) => {
deps.push({
neededLayer: id,
2024-08-16 10:13:18 +02:00
reason: `preset \`${preset.title.textFor("en")}\` snaps to this layer`,
context: `${layer.id}.presets[${i}]`,
neededBy: layer.id,
2024-10-19 14:44:55 +02:00
checkHasSnapName: true,
})
})
}
for (const tr of layer.AllTagRenderings()) {
for (const dep of DependencyCalculator.GetTagRenderingDependencies(tr)) {
deps.push({
neededLayer: dep,
reason: "a tagrendering needs this layer",
context: tr.id,
neededBy: layer.id,
2024-10-19 14:44:55 +02:00
checkHasSnapName: false,
})
}
}
if (layer.calculatedTags?.length > 0) {
const obj = {
type: "Feature",
geometry: {
type: "Point",
coordinates: [0, 0],
},
properties: {
id: "node/1",
},
}
let currentKey = undefined
let currentLine = undefined
const params: ExtraFuncParams = {
2021-12-07 17:46:57 +01:00
getFeatureById: (_) => undefined,
getFeaturesWithin: (layerId, _) => {
2022-01-26 21:40:38 +01:00
if (layerId === "*") {
// This is a wildcard
return []
}
2022-01-26 21:40:38 +01:00
// The important line: steal the dependencies!
deps.push({
2022-04-23 15:20:54 +02:00
neededLayer: layerId,
reason: "a calculated tag loads features from this layer",
context:
"calculatedTag[" +
currentLine +
"] which calculates the value for " +
currentKey,
neededBy: layer.id,
2024-10-19 14:44:55 +02:00
checkHasSnapName: false,
})
return []
},
}
2023-05-16 03:27:49 +02:00
const helpers = ExtraFunctions.constructHelpers(params)
// ... Run the calculated tag code, which will trigger the getFeaturesWithin above...
for (let i = 0; i < layer.calculatedTags.length; i++) {
const [key, code] = layer.calculatedTags[i]
currentLine = i // Leak the state...
currentKey = key
try {
const func = new Function(
"feat",
"{" + ExtraFunctions.types.join(",") + "}",
"return " + code + ";"
)
2023-05-16 03:27:49 +02:00
const result = func(obj, helpers)
obj.properties[key] = JSON.stringify(result)
} catch (e) {}
}
}
return deps
}
}