import SimpleMetaTaggers, {MetataggingState, SimpleMetaTagger} from "./SimpleMetaTagger" import {ExtraFuncParams, ExtraFunctions, ExtraFuncType} from "./ExtraFunctions" import LayerConfig from "../Models/ThemeConfig/LayerConfig" import {Feature} from "geojson" import FeaturePropertiesStore from "./FeatureSource/Actors/FeaturePropertiesStore" import LayoutConfig from "../Models/ThemeConfig/LayoutConfig" import {GeoIndexedStoreForLayer} from "./FeatureSource/Actors/GeoIndexedStore" import {IndexedFeatureSource} from "./FeatureSource/FeatureSource" import OsmObjectDownloader from "./Osm/OsmObjectDownloader" import {Utils} from "../Utils"; import {GeoJSONFeature} from "maplibre-gl"; /** * Metatagging adds various tags to the elements, e.g. lat, lon, surface area, ... * * All metatags start with an underscore */ export default class MetaTagging { private static errorPrintCount = 0 private static readonly stopErrorOutputAt = 10 private static retaggingFuncCache = new Map void)[]>() constructor(state: { layout: LayoutConfig osmObjectDownloader: OsmObjectDownloader perLayer: ReadonlyMap indexedFeatures: IndexedFeatureSource featureProperties: FeaturePropertiesStore }) { const params: ExtraFuncParams = { getFeatureById: (id) => state.indexedFeatures.featuresById.data.get(id), getFeaturesWithin: (layerId, bbox) => { if(layerId === '*' || layerId === null || layerId === undefined){ const feats: Feature[][] = [] state.perLayer.forEach((layer) => { feats.push(layer.GetFeaturesWithin(bbox)) }) return feats } return [state.perLayer.get(layerId).GetFeaturesWithin(bbox)]; }, } for (const layer of state.layout.layers) { if (layer.source === null) { continue } const featureSource = state.perLayer.get(layer.id) featureSource.features?.stabilized(1000)?.addCallbackAndRunD((features) => { if (!(features?.length > 0)) { // No features to handle return } console.debug( "Recalculating metatags for layer ", layer.id, "due to a change in the upstream features. Contains ", features.length, "items" ) MetaTagging.addMetatags( features, params, layer, state.layout, state.osmObjectDownloader, state.featureProperties ) }) } } /** * This method (re)calculates all metatags and calculated tags on every given feature. * The given features should be part of the given layer * * Returns true if at least one feature has changed properties */ public static addMetatags( features: Feature[], params: ExtraFuncParams, layer: LayerConfig, layout: LayoutConfig, osmObjectDownloader: OsmObjectDownloader, featurePropertiesStores?: FeaturePropertiesStore, options?: { includeDates?: true | boolean includeNonDates?: true | boolean evaluateStrict?: false | boolean } ): boolean { if (features === undefined || features.length === 0) { return } const metatagsToApply: SimpleMetaTagger[] = [] for (const metatag of SimpleMetaTaggers.metatags) { if (metatag.includesDates) { if (options?.includeDates ?? true) { metatagsToApply.push(metatag) } } else { if (options?.includeNonDates ?? true) { metatagsToApply.push(metatag) } } } // The calculated functions - per layer - which add the new keys // Calculated functions are defined by the layer const layerFuncs = this.createRetaggingFunc(layer, ExtraFunctions.constructHelpers(params)) const state: MetataggingState = { layout, osmObjectDownloader } let atLeastOneFeatureChanged = false let strictlyEvaluated = 0 for (let i = 0; i < features.length; i++) { const feature = features[i] const tags = featurePropertiesStores?.getStore(feature.properties.id) let somethingChanged = false let definedTags = new Set(Object.getOwnPropertyNames(feature.properties)) if (layerFuncs !== undefined) { let retaggingChanged = false try { retaggingChanged = layerFuncs(feature) } catch (e) { console.error(e) } somethingChanged = somethingChanged || retaggingChanged } for (const metatag of metatagsToApply) { try { if (!metatag.keys.some((key) => !(key in feature.properties))) { // All keys are already defined, we probably already ran this one // Note that we use 'key in properties', not 'properties[key] === undefined'. The latter will cause evaluation of lazy properties continue } if (metatag.isLazy) { if (!metatag.keys.some((key) => !definedTags.has(key))) { // All keys are defined - lets skip! continue } somethingChanged = true metatag.applyMetaTagsOnFeature(feature, layer, tags, state) if (options?.evaluateStrict) { for (const key of metatag.keys) { const evaluated = feature.properties[key] if(evaluated !== undefined){ strictlyEvaluated++ } } } } else { const newValueAdded = metatag.applyMetaTagsOnFeature( feature, layer, tags, state ) /* Note that the expression: * `somethingChanged = newValueAdded || metatag.applyMetaTagsOnFeature(feature, freshness)` * Is WRONG * * IF something changed is `true` due to an earlier run, it will short-circuit and _not_ evaluate the right hand of the OR, * thus not running an update! */ somethingChanged = newValueAdded || somethingChanged } } catch (e) { console.error( "Could not calculate metatag for ", metatag.keys.join(","), ":", e, e.stack ) } } if (somethingChanged) { try { featurePropertiesStores?.getStore(feature.properties.id)?.ping() } catch (e) { console.error("Could not ping a store for a changed property due to", e) } atLeastOneFeatureChanged = true } } return atLeastOneFeatureChanged } /** * Creates a function that implements that calculates a property and adds this property onto the feature properties * @param specification * @param helperFunctions * @param layerId * @private */ private static createFunctionForFeature( [key, code, isStrict]: [string, string, boolean], helperFunctions: Record Function>, layerId: string = "unkown layer" ): ((feature: GeoJSONFeature) => void) | undefined { if (code === undefined) { return undefined } const calculateAndAssign: ((feat: GeoJSONFeature) => (string | undefined)) = (feat) => { try { let result = new Function("feat", "{"+ExtraFunctions.types.join(", ")+"}", "return " + code + ";")(feat, helperFunctions) if (result === "") { result = undefined } if (result !== undefined && typeof result !== "string") { // Make sure it is a string! result = JSON.stringify(result) } delete feat.properties[key] feat.properties[key] = result return result } catch (e) { if (MetaTagging.errorPrintCount < MetaTagging.stopErrorOutputAt) { console.warn( "Could not calculate a " + (isStrict ? "strict " : "") + " calculated tag for key " + key + " defined by " + code + " (in layer" + layerId + ") due to \n" + e + "\n. Are you the theme creator? Doublecheck your code. Note that the metatags might not be stable on new features", e, e.stack ) MetaTagging.errorPrintCount++ if (MetaTagging.errorPrintCount == MetaTagging.stopErrorOutputAt) { console.error( "Got ", MetaTagging.stopErrorOutputAt, " errors calculating this metatagging - stopping output now" ) } } return undefined } } if(isStrict){ return calculateAndAssign } return (feature: any) => { delete feature.properties[key] Utils.AddLazyProperty(feature.properties, key, () => calculateAndAssign(feature)) return undefined } } /** * Creates the function which adds all the calculated tags to a feature. Called once per layer */ private static createRetaggingFunc( layer: LayerConfig, helpers: Record Function> ): (feature: any) => boolean { const calculatedTags: [string, string, boolean][] = layer.calculatedTags if (calculatedTags === undefined || calculatedTags.length === 0) { return undefined } let functions: ((feature: Feature) => void)[] = MetaTagging.retaggingFuncCache.get(layer.id) if (functions === undefined) { functions = calculatedTags.map(spec => this.createFunctionForFeature(spec, helpers, layer.id)) MetaTagging.retaggingFuncCache.set(layer.id, functions) } return (feature: Feature) => { const tags = feature.properties if (tags === undefined) { return } try { for (const f of functions) { f(feature) } } catch (e) { console.error("Invalid syntax in calculated tags or some other error: ", e) } return true // Something changed } } }