forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			241 lines
		
	
	
		
			No EOL
		
	
	
		
			7.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			241 lines
		
	
	
		
			No EOL
		
	
	
		
			7.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
import Translations from "../../UI/i18n/Translations";
 | 
						|
import TagRenderingConfig from "./TagRenderingConfig";
 | 
						|
import {Tag, TagsFilter} from "../../Logic/Tags";
 | 
						|
import {LayerConfigJson} from "./LayerConfigJson";
 | 
						|
import {FromJSON} from "./FromJSON";
 | 
						|
import SharedTagRenderings from "../SharedTagRenderings";
 | 
						|
import {TagRenderingConfigJson} from "./TagRenderingConfigJson";
 | 
						|
import {Translation} from "../../UI/i18n/Translation";
 | 
						|
import {Img} from "../../UI/Img";
 | 
						|
import Svg from "../../Svg";
 | 
						|
import {SubstitutedTranslation} from "../../UI/SpecialVisualizations";
 | 
						|
import {Utils} from "../../Utils";
 | 
						|
import Combine from "../../UI/Base/Combine";
 | 
						|
import {VariableUiElement} from "../../UI/Base/VariableUIElement";
 | 
						|
 | 
						|
export default class LayerConfig {
 | 
						|
 | 
						|
 | 
						|
    id: string;
 | 
						|
 | 
						|
    name: Translation
 | 
						|
 | 
						|
    description: Translation;
 | 
						|
    overpassTags: TagsFilter;
 | 
						|
    doNotDownload: boolean;
 | 
						|
 | 
						|
    passAllFeatures: boolean;
 | 
						|
 | 
						|
    minzoom: number;
 | 
						|
 | 
						|
    title?: TagRenderingConfig;
 | 
						|
 | 
						|
    titleIcons: TagRenderingConfig[];
 | 
						|
 | 
						|
    icon: TagRenderingConfig;
 | 
						|
    iconSize: TagRenderingConfig;
 | 
						|
    rotation: TagRenderingConfig;
 | 
						|
    color: TagRenderingConfig;
 | 
						|
    width: TagRenderingConfig;
 | 
						|
    dashArray: TagRenderingConfig;
 | 
						|
 | 
						|
 | 
						|
    wayHandling: number;
 | 
						|
 | 
						|
    static WAYHANDLING_DEFAULT = 0;
 | 
						|
    static WAYHANDLING_CENTER_ONLY = 1;
 | 
						|
    static WAYHANDLING_CENTER_AND_WAY = 2;
 | 
						|
 | 
						|
    hideUnderlayingFeaturesMinPercentage?: number;
 | 
						|
 | 
						|
    presets: {
 | 
						|
        title: Translation,
 | 
						|
        tags: Tag[],
 | 
						|
        description?: Translation,
 | 
						|
    }[];
 | 
						|
 | 
						|
    tagRenderings: TagRenderingConfig [];
 | 
						|
 | 
						|
    constructor(json: LayerConfigJson, roamingRenderings: TagRenderingConfig[],
 | 
						|
                context?: string) {
 | 
						|
        context = context + "." + json.id;
 | 
						|
 | 
						|
        this.id = json.id;
 | 
						|
        this.name = Translations.T(json.name);
 | 
						|
        this.description = Translations.T(json.name);
 | 
						|
        this.overpassTags = FromJSON.Tag(json.overpassTags, context + ".overpasstags");
 | 
						|
        this.doNotDownload = json.doNotDownload ?? false,
 | 
						|
            this.passAllFeatures = json.passAllFeatures ?? false;
 | 
						|
        this.minzoom = json.minzoom;
 | 
						|
        this.wayHandling = json.wayHandling ?? 0;
 | 
						|
        this.hideUnderlayingFeaturesMinPercentage = json.hideUnderlayingFeaturesMinPercentage ?? 0;
 | 
						|
        this.presets = (json.presets ?? []).map(pr =>
 | 
						|
            ({
 | 
						|
                title: Translations.T(pr.title),
 | 
						|
                tags: pr.tags.map(t => FromJSON.SimpleTag(t)),
 | 
						|
                description: Translations.T(pr.description)
 | 
						|
            }))
 | 
						|
 | 
						|
 | 
						|
        /**
 | 
						|
         * Converts a list of tagRenderingCOnfigJSON in to TagRenderingConfig
 | 
						|
         * A string is interpreted as a name to call
 | 
						|
         * @param tagRenderings
 | 
						|
         */
 | 
						|
        function trs(tagRenderings?: (string | TagRenderingConfigJson)[]) {
 | 
						|
            if (tagRenderings === undefined) {
 | 
						|
                return [];
 | 
						|
            }
 | 
						|
            return tagRenderings.map(
 | 
						|
                (renderingJson, i) => {
 | 
						|
                    if (typeof renderingJson === "string") {
 | 
						|
                        const shared = SharedTagRenderings.SharedTagRendering[renderingJson];
 | 
						|
                        if (shared !== undefined) {
 | 
						|
                            return shared;
 | 
						|
                        }
 | 
						|
                        throw `Predefined tagRendering ${renderingJson} not found in ${context}`;
 | 
						|
                    }
 | 
						|
                    return new TagRenderingConfig(renderingJson, `${context}.tagrendering[${i}]`);
 | 
						|
                });
 | 
						|
        }
 | 
						|
 | 
						|
        this.tagRenderings = trs(json.tagRenderings).concat(roamingRenderings);
 | 
						|
        this.titleIcons = trs(json.titleIcons ?? ["wikipedialink","osmlink"]);
 | 
						|
        
 | 
						|
 | 
						|
        function tr(key, deflt) {
 | 
						|
            const v = json[key];
 | 
						|
            if (v === undefined || v === null) {
 | 
						|
                if (deflt === undefined) {
 | 
						|
                    return undefined;
 | 
						|
                }
 | 
						|
                return new TagRenderingConfig(deflt);
 | 
						|
            }
 | 
						|
            if (typeof v === "string") {
 | 
						|
                const shared = SharedTagRenderings.SharedTagRendering[v];
 | 
						|
                if (shared) {
 | 
						|
                    console.log("Got shared TR:", v, "-->", shared)
 | 
						|
                    return shared;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            return new TagRenderingConfig(v, context + "." + key);
 | 
						|
        }
 | 
						|
 | 
						|
 | 
						|
        this.title = tr("title", undefined);
 | 
						|
        this.icon = tr("icon", Img.AsData(Svg.bug));
 | 
						|
        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 = tr("iconSize", "40,40,center");
 | 
						|
        this.color = tr("color", "#0000ff");
 | 
						|
        this.width = tr("width", "7");
 | 
						|
        this.rotation = tr("rotation", "0");
 | 
						|
        this.dashArray = tr("dashArray", "");
 | 
						|
 | 
						|
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    public GenerateLeafletStyle(tags: any, clickable: boolean):
 | 
						|
        {
 | 
						|
            color: string;
 | 
						|
            icon: {
 | 
						|
                iconUrl: string,
 | 
						|
                popupAnchor: [number, number];
 | 
						|
                iconAnchor: [number, number];
 | 
						|
                iconSize: [number, number];
 | 
						|
                html: string;
 | 
						|
                rotation: string;
 | 
						|
                className?: string;
 | 
						|
            };
 | 
						|
            weight: number; dashArray: number[]
 | 
						|
        } {
 | 
						|
 | 
						|
        function num(str, deflt = 40) {
 | 
						|
            const n = Number(str);
 | 
						|
            if (isNaN(n)) {
 | 
						|
                return deflt;
 | 
						|
            }
 | 
						|
            return n;
 | 
						|
        }
 | 
						|
 | 
						|
        function rendernum(tr: TagRenderingConfig, deflt: number) {
 | 
						|
            const str = Number(render(tr, "" + deflt));
 | 
						|
            const n = Number(str);
 | 
						|
            if (isNaN(n)) {
 | 
						|
                return deflt;
 | 
						|
            }
 | 
						|
            return n;
 | 
						|
        }
 | 
						|
 | 
						|
        function render(tr: TagRenderingConfig, deflt?: string) {
 | 
						|
            const str = (tr?.GetRenderValue(tags)?.txt ?? deflt);
 | 
						|
            return SubstitutedTranslation.SubstituteKeys(str, tags);
 | 
						|
        }
 | 
						|
 | 
						|
        const iconUrl = render(this.icon);
 | 
						|
        const iconSize = render(this.iconSize, "40,40,center").split(",");
 | 
						|
        const dashArray = render(this.dashArray).split(" ").map(Number);
 | 
						|
        let color = render(this.color, "#00f");
 | 
						|
 | 
						|
        if (color.startsWith("--")) {
 | 
						|
            color = getComputedStyle(document.body).getPropertyValue("--catch-detail-color")
 | 
						|
        }
 | 
						|
 | 
						|
        const weight = rendernum(this.width, 5);
 | 
						|
        const rotation = render(this.rotation, "0deg");
 | 
						|
 | 
						|
 | 
						|
        const iconW = num(iconSize[0]);
 | 
						|
        const iconH = num(iconSize[1]);
 | 
						|
        const mode = iconSize[2] ?? "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;
 | 
						|
        }
 | 
						|
 | 
						|
        
 | 
						|
        let html = `<img src="${iconUrl}" style="width:100%;height:100%;rotate:${rotation};display:block;" />`;
 | 
						|
        
 | 
						|
        if (iconUrl.startsWith(Utils.assets_path)) {
 | 
						|
            const key = iconUrl.substr(Utils.assets_path.length);
 | 
						|
            html = new Combine([
 | 
						|
                (Svg.All[key] as string).replace(/stop-color:#000000/g, 'stop-color:' + color)
 | 
						|
            ]).SetStyle(`width:100%;height:100%;rotate:${rotation};display:block;`).Render();
 | 
						|
        }
 | 
						|
        return {
 | 
						|
            icon:
 | 
						|
                {
 | 
						|
                    html: html,
 | 
						|
                    iconSize: [iconW, iconH],
 | 
						|
                    iconAnchor: [anchorW, anchorH],
 | 
						|
                    popupAnchor: [0, 3 - anchorH],
 | 
						|
                    rotation: rotation,
 | 
						|
                    iconUrl: iconUrl,
 | 
						|
                    className: clickable ? "leaflet-div-icon" : "leaflet-div-icon unclickable"
 | 
						|
                },
 | 
						|
            color: color,
 | 
						|
            weight: weight,
 | 
						|
            dashArray: dashArray
 | 
						|
        };
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
} |