import PointRenderingConfigJson from "./Json/PointRenderingConfigJson"; import TagRenderingConfig from "./TagRenderingConfig"; import {TagsFilter} from "../../Logic/Tags/TagsFilter"; import SharedTagRenderings from "../../Customizations/SharedTagRenderings"; import {TagUtils} from "../../Logic/Tags/TagUtils"; import {Utils} from "../../Utils"; import Svg from "../../Svg"; import WithContextLoader from "./WithContextLoader"; import {UIEventSource} from "../../Logic/UIEventSource"; import BaseUIElement from "../../UI/BaseUIElement"; import {FixedUiElement} from "../../UI/Base/FixedUiElement"; import Img from "../../UI/Base/Img"; import Combine from "../../UI/Base/Combine"; import {VariableUiElement} from "../../UI/Base/VariableUIElement"; export default class PointRenderingConfig extends WithContextLoader { private static readonly allowed_location_codes = new Set(["point", "centroid","start","end"]) public readonly location: Set<"point" | "centroid" | "start" | "end"> public readonly icon: TagRenderingConfig; public readonly iconBadges: { if: TagsFilter; then: TagRenderingConfig }[]; public readonly iconSize: TagRenderingConfig; public readonly label: TagRenderingConfig; public readonly rotation: TagRenderingConfig; constructor(json: PointRenderingConfigJson, context: string) { super(json, context) if(typeof json.location === "string"){ json.location = [json.location] } this.location = new Set(json.location) this.location.forEach(l => { const allowed = PointRenderingConfig.allowed_location_codes if(!allowed.has(l)){ throw `A point rendering has an invalid location: '${l}' is not one of ${Array.from(allowed).join(", ")} (at ${context}.location)` } }) if(this.location.size == 0){ throw "A pointRendering should have at least one 'location' to defined where it should be rendered. (At "+context+".location)" } this.icon = this.tr("icon", ""); this.iconBadges = (json.iconBadges ?? []).map((overlay, i) => { let tr : TagRenderingConfig; if (typeof overlay.then === "string" && SharedTagRenderings.SharedIcons.get(overlay.then) !== undefined) { tr = SharedTagRenderings.SharedIcons.get(overlay.then); }else{ tr = new TagRenderingConfig( overlay.then, `iconBadges.${i}` ); } return { if: TagUtils.Tag(overlay.if), then: tr }; }); const iconPath = this.icon.GetRenderValue({id: "node/-1"}).txt; if (iconPath.startsWith(Utils.assets_path)) { const iconKey = iconPath.substr(Utils.assets_path.length); if (Svg.All[iconKey] === undefined) { throw "Builtin SVG asset not found: " + iconPath; } } this.iconSize = this.tr("iconSize", "40,40,center"); this.label = this.tr("label", undefined); this.rotation = this.tr("rotation", "0"); } public ExtractImages(): Set { const parts: Set[] = []; parts.push(this.icon?.ExtractImages(true)); parts.push( ...this.iconBadges?.map((overlay) => overlay.then.ExtractImages(true)) ); const allIcons = new Set(); for (const part of parts) { part?.forEach(allIcons.add, allIcons); } return allIcons; } /** * Given a single HTML spec (either a single image path OR "image_path_to_known_svg:fill-colour", returns a fixedUIElement containing that * The element will fill 100% and be positioned absolutely with top:0 and left: 0 */ private static FromHtmlSpec(htmlSpec: string, style: string, isBadge = false): BaseUIElement { if (htmlSpec === undefined) { return undefined; } const match = htmlSpec.match(/([a-zA-Z0-9_]*):([^;]*)/); if (match !== null && Svg.All[match[1] + ".svg"] !== undefined) { const svg = (Svg.All[match[1] + ".svg"] as string) const targetColor = match[2] const img = new Img(svg.replace(/#000000/g, targetColor), true) .SetStyle(style) if(isBadge){ img.SetClass("badge") } return img } else { return new FixedUiElement(``); } } private static FromHtmlMulti(multiSpec: string, rotation: string , isBadge: boolean, defaultElement: BaseUIElement = undefined){ if(multiSpec === undefined){ return defaultElement } const style = `width:100%;height:100%;transform: rotate( ${rotation} );display:block;position: absolute; top: 0; left: 0`; const htmlDefs = multiSpec.trim()?.split(";") ?? [] const elements = Utils.NoEmpty(htmlDefs).map(def => PointRenderingConfig.FromHtmlSpec(def, style, isBadge)) if (elements.length === 0) { return defaultElement } else { return new Combine(elements).SetClass("relative block w-full h-full") } } public GetSimpleIcon(tags: UIEventSource): BaseUIElement { const self = this; if (this.icon === undefined) { return undefined; } return new VariableUiElement(tags.map(tags => { const rotation = self.rotation?.GetRenderValue(tags)?.txt ?? "0deg" const htmlDefs = Utils.SubstituteKeys(self.icon.GetRenderValue(tags)?.txt, tags) let defaultPin : BaseUIElement = undefined if(self.label === undefined){ defaultPin = Svg.teardrop_with_hole_green_svg() } return PointRenderingConfig.FromHtmlMulti(htmlDefs, rotation,false, defaultPin) })).SetClass("w-full h-full block") } private GetBadges(tags: UIEventSource): BaseUIElement { if (this.iconBadges.length === 0) { return undefined } return new VariableUiElement( tags.map(tags => { const badgeElements = this.iconBadges.map(badge => { if (!badge.if.matchesProperties(tags)) { // Doesn't match... return undefined } const htmlDefs = Utils.SubstituteKeys(badge.then.GetRenderValue(tags)?.txt, tags) const badgeElement= PointRenderingConfig.FromHtmlMulti(htmlDefs, "0", true)?.SetClass("block relative") if(badgeElement === undefined){ return undefined; } return new Combine([badgeElement]).SetStyle("width: 1.5rem").SetClass("block") }) return new Combine(badgeElements).SetClass("inline-flex h-full") })).SetClass("absolute bottom-0 right-1/3 h-1/2 w-0") } private GetLabel(tags: UIEventSource): BaseUIElement { if (this.label === undefined) { return undefined; } const self = this; return new VariableUiElement(tags.map(tags => { const label = self.label ?.GetRenderValue(tags) ?.Subs(tags) ?.SetClass("block text-center") return new Combine([label]).SetClass("flex flex-col items-center mt-1") })) } public GenerateLeafletStyle( tags: UIEventSource, clickable: boolean ): { html: BaseUIElement; iconSize: [number, number]; iconAnchor: [number, number]; popupAnchor: [number, number]; iconUrl: string; className: string; } { function num(str, deflt = 40) { const n = Number(str); if (isNaN(n)) { return deflt; } return n; } function render(tr: TagRenderingConfig, deflt?: string) { if (tags === undefined) { return deflt } const str = tr?.GetRenderValue(tags.data)?.txt ?? deflt; return Utils.SubstituteKeys(str, tags.data).replace(/{.*}/g, ""); } const iconSize = render(this.iconSize, "40,40,center").split(","); const iconW = num(iconSize[0]); let iconH = num(iconSize[1]); const mode = iconSize[2]?.trim()?.toLowerCase() ?? "center"; let anchorW = iconW / 2; let anchorH = iconH / 2; if (mode === "left") { anchorW = 0; } if (mode === "right") { anchorW = iconW; } if (mode === "top") { anchorH = 0; } if (mode === "bottom") { anchorH = iconH; } const iconAndBadges = new Combine([this.GetSimpleIcon(tags), this.GetBadges(tags)]) .SetStyle(`width: ${iconW}px; height: ${iconH}px`) .SetClass("block relative") return { html: new Combine([iconAndBadges, this.GetLabel(tags)]).SetStyle("flex flex-col"), iconSize: [iconW, iconH], iconAnchor: [anchorW, anchorH], popupAnchor: [0, 3 - anchorH], iconUrl: undefined, className: clickable ? "leaflet-div-icon" : "leaflet-div-icon unclickable", }; } }