WME UT Kadastrs

WME UrSuS Tools: LV Kadastrs

// ==UserScript==
// @name        WME UT Kadastrs
// @namespace   http://ursus.id.lv
// @version     1.1.0
// @description WME UrSuS Tools: LV Kadastrs
// @author      UrSuS
// @match       https://*.waze.com/*editor*
// @license     MIT/BSD/X11
// @icon        
// @exclude     https://www.waze.com/user/editor*
// @require     https://cdn.jsdelivr.net/npm/@turf/turf@7/turf.min.js
// @require     https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require     https://greasyfork.org/scripts/383120-proj4-wazedev/code/proj4-Wazedev.js
// @connect     lvmgeoserver.lvm.lv
// @connect     docs.google.com
// @grant       GM_info
// @grant       GM_xmlhttpRequest
// ==/UserScript==

/* global turf */
/* globals proj4 */

(function (turf, proj4) {
    'use strict';

    function _interopNamespaceDefault(e) {
        var n = Object.create(null);
        if (e) {
            Object.keys(e).forEach(function (k) {
                if (k !== 'default') {
                    var d = Object.getOwnPropertyDescriptor(e, k);
                    Object.defineProperty(n, k, d.get ? d : {
                        enumerable: true,
                        get: function () { return e[k]; }
                    });
                }
            });
        }
        n.default = e;
        return Object.freeze(n);
    }

    var turf__namespace = /*#__PURE__*/_interopNamespaceDefault(turf);

    const sScriptName = GM_info.script.name;
    GM_info.script.version;
    let wmeSDK;
    let aWazeVenues;
    let aGlobalFetchedAddresses = [];
    let aWMEValidAddressVenues = [];
    let aWMEPolygonHighlightBlueFeatures = [];
    let aMissingFetchedAddresses = [];
    let iPopupTimeout;
    let $KadastrsMenuPopupDiv;
    let aVasarnicas = [];
    let mKadastrsFeaturesData = {};
    function fetchSheetData() {
        const SHEET_ID = "1W230EX3e0ECeF44Ls0GtvI_Iwn9tceidJYT7T4vjJJw";
        const SHEET_NAME = "Sheet1";
        const url = `https://docs.google.com/spreadsheets/d/${SHEET_ID}/gviz/tq?tqx=out:csv&sheet=${SHEET_NAME}`;
        GM_xmlhttpRequest({
            method: "GET",
            url: url,
            onload: response => {
                if (response.status === 200) {
                    const csvData = response.responseText;
                    const aRows = csvData.trim().split("\n");
                    const aParsedResult = aRows.slice(1).map(row => {
                        const aCols = row.split(",");
                        return {
                            name: aCols[0].trim().replace(/^"(.*)"$/, "$1"),
                            parent: aCols[1].trim().replace(/^"(.*)"$/, "$1")
                        };
                    });
                    aVasarnicas = aParsedResult;
                }
                else {
                    console.error("Failed to fetch data:", response);
                }
            }
        });
    }
    function createElem(type, attrs = {}, eventListener = {}) {
        const oElement = document.createElement(type);
        for (const [sKey, vValue] of Object.entries(attrs)) {
            if (sKey in oElement) {
                oElement[sKey] = vValue;
            }
            else if (sKey === "classList" && Array.isArray(vValue)) {
                oElement.classList.add(...vValue);
            }
            else {
                oElement.setAttribute(sKey, vValue);
            }
        }
        for (const [sEventType, fnHandler] of Object.entries(eventListener)) {
            if (fnHandler) {
                oElement.addEventListener(sEventType, fnHandler);
            }
        }
        return oElement;
    }
    function getOLMapExtent() {
        let extent = W.map.getExtent();
        if (Array.isArray(extent)) {
            extent = new OpenLayers.Bounds(extent);
        }
        return extent;
    }
    function addKFetchButton() {
        const divOverlayMain = document.getElementById("overlay-buttons-region");
        if (!divOverlayMain) {
            return;
        }
        const mStyle = {
            position: "absolute",
            top: "0px",
            right: "60px",
            width: "44px",
            zIndex: "1"
        };
        const mainDiv = createElem("div");
        Object.assign(mainDiv.style, mStyle);
        const btnDiv = createElem("div", {
            classList: ["overlay-buttons-container top"]
        });
        const owz = createElem("wz-button", {
            color: "clear-icon",
            classList: ["overlay-button"],
            disabled: "false"
        }, { click: fetchKadastrsData });
        const h6 = createElem("h6", {
            classList: ["w-icon"],
            textContent: "K"
        });
        h6.style["font-family"] = "Waze Boing Medium";
        h6.style["line-height"] = "24px";
        owz.appendChild(h6);
        btnDiv.appendChild(owz);
        mainDiv.appendChild(btnDiv);
        divOverlayMain.appendChild(mainDiv);
    }
    function initScript() {
        if (!unsafeWindow.getWmeSdk) {
            throw new Error("SDK not available");
        }
        wmeSDK = unsafeWindow.getWmeSdk({
            scriptId: "wmeUTKadastrs",
            scriptName: "UrSuS Tools: Kadastrs"
        });
        if (typeof proj4 === "undefined") {
            const oScript = document.createElement("script");
            oScript.type = "text/javascript";
            oScript.src = "https://cdnjs.cloudflare.com/ajax/libs/proj4js/2.4.4/proj4.js";
            document.getElementsByTagName("head")[0].appendChild(oScript);
        }
        require("Waze/Actions/AddHouseNumber");
        addKFetchButton();
        fetchSheetData();
        wmeSDK.Map.addLayer({
            layerName: "wme-ut-kadastrs-hover",
            styleContext: {
                getLabel: ({ feature }) => feature?.properties.label ?? "",
                getXOffset: ({ feature }) => feature?.properties.xOffset ?? 0,
                getYOffset: ({ feature }) => feature?.properties.yOffset ?? 0
            },
            styleRules: [
                {
                    style: {
                        fillColor: "#00695C",
                        fillOpacity: 0.5,
                        strokeWidth: 9,
                        strokeDashstyle: "longdashdot",
                        strokeColor: "red",
                        label: "",
                        labelYOffset: 45,
                        fontColor: "#FF0",
                        fontWeight: "bold",
                        labelOutlineColor: "#000",
                        labelOutlineWidth: 4,
                        fontSize: "18"
                    }
                }
            ],
            zIndexing: true
        });
        wmeSDK.Map.addLayer({
            layerName: "wme-ut-kadastrs",
            styleContext: {
                getLabel: ({ feature }) => feature?.properties.label ?? "",
                getXOffset: ({ feature }) => feature?.properties.xOffset ?? 0,
                getYOffset: ({ feature }) => feature?.properties.yOffset ?? 0
            },
            styleRules: [
                {
                    style: {
                        label: "K",
                        pointRadius: 12,
                        fillColor: "#00695C",
                        fillOpacity: 0.8,
                        fontColor: "black",
                        labelOutlineColor: "white",
                        labelOutlineWidth: 3,
                        strokeColor: "white"
                    }
                }
            ],
            zIndexing: true
        });
        wmeSDK.Map.addLayer({
            layerName: "wme-ut-kadastrs-lines",
            styleContext: {
                getLabel: ({ feature }) => feature?.properties.label ?? "",
                getXOffset: ({ feature }) => feature?.properties.xOffset ?? 0,
                getYOffset: ({ feature }) => feature?.properties.yOffset ?? 0
            },
            styleRules: [
                {
                    style: {
                        fillColor: "#00695C",
                        fillOpacity: 0.5,
                        strokeWidth: 3,
                        strokeDashstyle: "longdashdot",
                        strokeColor: "blue",
                        label: "",
                        labelYOffset: 45,
                        fontColor: "#FF0",
                        fontWeight: "bold",
                        labelOutlineColor: "#000",
                        labelOutlineWidth: 4,
                        fontSize: "18"
                    }
                }
            ],
            zIndexing: true
        });
        wmeSDK.Map.addLayer({
            layerName: "wme-ut-kadastrs-highlights",
            styleContext: {
                getLabel: ({ feature }) => feature?.properties.label ?? "",
                getXOffset: ({ feature }) => feature?.properties.xOffset ?? 0,
                getYOffset: ({ feature }) => feature?.properties.yOffset ?? 0
            },
            styleRules: [
                {
                    style: {
                        fillColor: "#00695C",
                        fillOpacity: 0.5,
                        strokeWidth: 9,
                        strokeDashstyle: "solid",
                        label: "",
                        labelYOffset: 45,
                        fontColor: "#FF0",
                        fontWeight: "bold",
                        labelOutlineColor: "#000",
                        labelOutlineWidth: 4,
                        fontSize: "18"
                    }
                },
                {
                    predicate: applyRedHightlightStyle,
                    style: {
                        strokeColor: "red",
                        graphicZIndex: 9999
                    }
                },
                {
                    predicate: applyGreenHightlightStyle,
                    style: {
                        strokeColor: "green",
                        graphicZIndex: 9999
                    }
                }
            ],
            zIndexing: true
        });
        wmeSDK.Map.setLayerVisibility({
            layerName: "wme-ut-kadastrs-hover",
            visibility: true
        });
        wmeSDK.Map.setLayerVisibility({
            layerName: "wme-ut-kadastrs-highlights",
            visibility: true
        });
        wmeSDK.Events.trackLayerEvents({ layerName: "wme-ut-kadastrs" });
        wmeSDK.Events.on({
            eventName: "wme-layer-feature-mouse-enter",
            eventHandler: onFeatureHoverEvent
        });
        wmeSDK.Events.on({
            eventName: "wme-layer-feature-mouse-leave",
            eventHandler: onFeatureUnHoverEvent
        });
    }
    void unsafeWindow.SDK_INITIALIZED.then(initScript);
    async function fetchKadastrsData() {
        wazedevtoastr.options = {
            closeButton: false,
            debug: false,
            newestOnTop: false,
            progressBar: false,
            rtl: false,
            positionClass: "toast-bottom-center",
            preventDuplicates: false,
            onclick: null,
            showDuration: 300,
            hideDuration: 1000,
            timeOut: 5000,
            extendedTimeOut: 1000,
            showEasing: "swing",
            hideEasing: "linear",
            showMethod: "fadeIn",
            hideMethod: "fadeOut"
        };
        wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: "wme-ut-kadastrs-highlights" });
        if (wmeSDK.Map.getZoomLevel() < 17) {
            WazeWrap.Alerts.warning(sScriptName, `You are on ${wmeSDK.Map.getZoomLevel()} Zoom Level, HouseNumbers won't be fetched!`);
        }
        WazeWrap.Alerts.info(sScriptName, "KadastrsAddress Fetching started");
        let sURL = "https://lvmgeoserver.lvm.lv/geoserver/publicwfs/wfs?layer=publicwfs&SERVICE=WFS&REQUEST=GetFeature";
        sURL +=
            "&VERSION=2.0.0&TYPENAMES=publicwfs:arisbuilding&STARTINDEX=0&COUNT=500&SRSNAME=urn:ogc:def:crs:EPSG::3059&BBOX=";
        sURL += getBBox3059();
        sURL += ",urn:ogc:def:crs:EPSG::3059&outputFormat=application/json";
        const sVenueResponseJSON = await makeGetRequest(sURL);
        processFetchedDataNewMode(sVenueResponseJSON);
    }
    async function processFetchedDataNewMode(responseText) {
        const oParser = new OpenLayers.Format.GeoJSON();
        aGlobalFetchedAddresses = oParser.read(responseText);
        WazeWrap.Alerts.info(sScriptName, `${aGlobalFetchedAddresses.length} Kadastrs Address Fetched`);
        let sWazeVenueURL = `https://${window.location.hostname}/row-Descartes/app/Features?bbox=${getBBoxv2()}`;
        sWazeVenueURL += "&language=en-GB&v=2&venueLevel=4&venueFilter=1%2C1%2C0%2C0";
        const sVenueResponseJSON = await makeGetRequest(sWazeVenueURL);
        const oParsedObject = JSON.parse(sVenueResponseJSON);
        aWazeVenues = oParsedObject.venues.objects;
        aWMEValidAddressVenues = [];
        WazeWrap.Alerts.info(sScriptName, `${aWazeVenues.length} WME Address Fetched`);
        let aSDKWazeHouseNumbers = [];
        if (wmeSDK.Map.getZoomLevel() >= 17) {
            const aSegments = wmeSDK.DataModel.Segments.getAll().filter((oObject) => oObject.hasHouseNumbers);
            const mBBox = getOLMapExtent();
            const oMapPoly = turf__namespace.polygon([
                [
                    [mBBox.left, mBBox.bottom],
                    [mBBox.right, mBBox.bottom],
                    [mBBox.right, mBBox.top],
                    [mBBox.left, mBBox.top],
                    [mBBox.left, mBBox.bottom]
                ]
            ]);
            const aSegmentsOnScreen = aSegments
                .filter(oSegment => turf__namespace.booleanWithin(oSegment.geometry, oMapPoly) ||
                turf__namespace.booleanIntersects(oSegment.geometry, oMapPoly))
                .map(oSegment => Number(oSegment.id));
            const oResults = await wmeSDK.DataModel.HouseNumbers.fetchHouseNumbers({
                segmentIds: aSegmentsOnScreen
            });
            aSDKWazeHouseNumbers = oResults;
        }
        aMissingFetchedAddresses = [...aGlobalFetchedAddresses];
        aWazeVenues.map((oVenue) => {
            let { name: sStreetName, city: sCity } = findStreetName(oVenue.streetID);
            let oAddressRegex;
            if (!oVenue.houseNumber && sStreetName === "") {
                if (oVenue.name) {
                    const sConverted = `"${oVenue.name.replace(", ", '", ')}${sCity !== "" ? '"' : ""}`;
                    oAddressRegex = getRegex([sConverted, sCity]);
                }
            }
            else if (!oVenue.houseNumber) {
                if (sStreetName.slice(-4) === "pag.") {
                    const aAddress = sStreetName.split(", ");
                    sStreetName = `"${aAddress[0]}", ${aAddress[1]}`;
                }
                else {
                    sStreetName = `"${sStreetName}"`;
                }
                oAddressRegex = getRegex([sStreetName, sCity]);
            }
            else {
                const sHouseNumber = oVenue.houseNumber.toUpperCase().replace("-", " k-");
                const bIsVasarnica = aVasarnicas.some(mRecord => mRecord.name === sStreetName && mRecord.parent === sCity);
                if (bIsVasarnica) {
                    oAddressRegex = getRegex([`"${sStreetName} ${sHouseNumber}"`, sCity]);
                }
                else {
                    oAddressRegex = getRegex([`${sStreetName} ${sHouseNumber}`, sCity]);
                }
            }
            if (oAddressRegex) {
                excludeKadastrsAddressConsistentWithWME(oAddressRegex, oVenue);
            }
        });
        WazeWrap.Alerts.info(sScriptName, `${aMissingFetchedAddresses.length} Kadastrs Addresses Not Present`);
        const aConvertedAddresses = aGlobalFetchedAddresses
            .map(mRowKadastrsAddress => {
            const mKadastrsAddress = convertKadastrsAddressStringToParts(mRowKadastrsAddress.attributes.std, "");
            if (!mKadastrsAddress) {
                return undefined;
            }
            return {
                cityName: mKadastrsAddress.cityName,
                streetName: mKadastrsAddress.streetName,
                houseNumber: mKadastrsAddress.houseNumber,
                name: mKadastrsAddress.name,
                cityId: "",
                streetID: ""
            };
        })
            .filter(oItem => !!oItem);
        const iFailedAddressIndex = populateIds(aConvertedAddresses);
        if (iFailedAddressIndex) {
            const mMissingAddress = aGlobalFetchedAddresses[iFailedAddressIndex];
            const aLonLat = proj4("EPSG:3059", "EPSG:4326", [mMissingAddress.geometry.x, mMissingAddress.geometry.y]);
            wmeSDK.Map.setMapCenter({
                lonLat: { lon: aLonLat[0], lat: aLonLat[1] },
                zoomLevel: 19
            });
        }
        aConvertedAddresses.forEach(mConvertedAddress => {
            const sHN = mConvertedAddress.houseNumber.includes(" k-")
                ? mConvertedAddress.houseNumber.replace(" k-", "-")
                : mConvertedAddress.houseNumber;
            aWazeVenues = aWazeVenues.filter(mVenue => mVenue.streetID !== mConvertedAddress.streetID || mVenue.houseNumber?.toUpperCase() !== sHN);
        });
        WazeWrap.Alerts.info(sScriptName, `${aWazeVenues.length} WME Venues with incorrect or missing Addresses`);
        aSDKWazeHouseNumbers.map(oSDKWazeHouseNumber => {
            const mSegStreet = W.model.segments.getObjectById(oSDKWazeHouseNumber.segmentId).getAddress().attributes
                ?.street?.attributes;
            if (!mSegStreet) {
                return;
            }
            const { name: sStreetName, city: sCity } = findStreetName(mSegStreet.id);
            const oAddressRegex = getRegex([sStreetName + " " + oSDKWazeHouseNumber.number.toUpperCase(), sCity]);
            excludeKadastrsAddressConsistentWithWME(oAddressRegex, oSDKWazeHouseNumber);
        });
        addKadastrsLayerWithFeatures(aGlobalFetchedAddresses);
        aWazeVenues.forEach(oVenue => {
            highlightVenue3(oVenue, "error");
        });
        aWMEValidAddressVenues.forEach(oVenue => {
            highlightVenue3(oVenue, "valid");
        });
    }
    function addKadastrsLayerWithFeatures(aKadastrsFeatures) {
        const aFeaturesForLayerNewSDK = [];
        aKadastrsFeatures.forEach(mFeature => {
            const aCoords = proj4("EPSG:3059", "EPSG:900913", [mFeature.geometry.x, mFeature.geometry.y]);
            mFeature.geometry.x = aCoords[0];
            mFeature.geometry.y = aCoords[1];
            let convertedGeometry = W.userscripts.toGeoJSONGeometry(mFeature.geometry).coordinates;
            const mPointFeature = {
                type: "Feature",
                id: mFeature.id,
                geometry: {
                    type: "Point",
                    coordinates: [convertedGeometry[0], convertedGeometry[1]]
                }
            };
            aFeaturesForLayerNewSDK.push(mPointFeature);
            mKadastrsFeaturesData[mPointFeature.id] = {
                kadastrs_data: mFeature.attributes.std,
                geometry: mPointFeature.geometry,
                found_venues: mFeature.data.Venues,
                found_residential: mFeature.data.Residential,
                found_hn: mFeature.data.HN
            };
        });
        wmeSDK.Map.addFeaturesToLayer({ features: aFeaturesForLayerNewSDK, layerName: "wme-ut-kadastrs" });
    }
    function checkTooltip() {
        window.clearTimeout(iPopupTimeout);
    }
    function onFeatureHoverEvent(e) {
        window.clearTimeout(iPopupTimeout);
        const mKadastrsAddress = convertKadastrsAddressStringToParts(mKadastrsFeaturesData[e.featureId].kadastrs_data, "waze");
        if (!mKadastrsAddress) {
            return;
        }
        const placeGeom = mKadastrsFeaturesData[e.featureId].geometry;
        if (!$KadastrsMenuPopupDiv) {
            $KadastrsMenuPopupDiv = createElem("div", {
                id: "kadastrsMenuDiv",
                style: "z-index:9998; visibility:visible; position:absolute; margin: 0px; top: 0px; left: 0px;",
                "data-tippy-root": false
            }, { mouseenter: checkTooltip, mouseleave: hideTooltipAfterDelay });
            Object.assign($KadastrsMenuPopupDiv.style, {
                zIndex: 9998,
                visibility: "visible",
                position: "absolute",
                margin: "0px",
                top: "0px",
                left: "0px"
            });
            W.map.getEl()[0].appendChild($KadastrsMenuPopupDiv);
        }
        const divElemRoot = createElem("div", {
            id: "kadastrsMenuDiv-tooltip",
            classList: ["tippy-box"],
            "data-state": "hidden",
            tabindex: "-1",
            "data-theme": "light-border",
            "data-animation": "fade",
            role: "tooltip",
            "data-placement": "top",
            style: "max-width: 350px; transition-duration:300ms;"
        });
        const divTippyContent = createElem("div", {
            id: "kadastrsMenuDiv-content",
            classList: ["tippy-content"],
            "data-state": "hidden",
            style: "transition-duration: 300ms;"
        });
        const oAddressTextDiv = createElem("div", {
            classList: ["coordinates-wrapper"]
        });
        const oVenuesTextDiv = createElem("div", {
            classList: ["coordinates-wrapper"],
            style: "white-space: pre-wrap;"
        });
        let sVenues = "❌";
        if (mKadastrsFeaturesData[e.featureId].found_venues) {
            sVenues = mKadastrsFeaturesData[e.featureId].found_venues.map((oVenue) => oVenue.name).join("\r\n&bull;");
            if (mKadastrsFeaturesData[e.featureId].found_venues.length > 1) {
                sVenues = "\r\n&bull;" + sVenues + "\r\n";
            }
        }
        oVenuesTextDiv.innerHTML = `Venues: ${sVenues} \n Residential: ${(mKadastrsFeaturesData[e.featureId].found_residential ?? []).length === 1
        ? "✅"
        : mKadastrsFeaturesData[e.featureId].found_residential ?? "❌"} HN: ${(mKadastrsFeaturesData[e.featureId].found_hn ?? []).length === 1
        ? "✅"
        : mKadastrsFeaturesData[e.featureId].found_hn ?? "❌"}`;
        divTippyContent.appendChild(oVenuesTextDiv);
        const oInputForm = createElem("div", {
            classList: ["wz-text-input-inner-container"]
        });
        const oInputInput = createElem("wz-text-input", {
            value: mKadastrsFeaturesData[e.featureId].kadastrs_data
        });
        oInputInput.appendChild(oInputForm);
        oAddressTextDiv.appendChild(oInputInput);
        divTippyContent.appendChild(oAddressTextDiv);
        const mSDKSelection = wmeSDK.Editing.getSelection();
        if (mSDKSelection?.ids.length === 1) {
            const oApplyAddressForm = createElem("div", {
                classList: ["external-providers-control", "form-group"]
            });
            divTippyContent.appendChild(oApplyAddressForm);
            const oApplyAddressFormLabel = createElem("wz-label", {
                "html-for": ""
            });
            oApplyAddressFormLabel.innerText = "Apply Address to selected Venue:";
            oApplyAddressForm.appendChild(oApplyAddressFormLabel);
            const fnFullyApplyToSelectedClick = () => applyAddress(e, "full");
            const oWZButtonFullyApplyToSelectedPlace = createElem("wz-button", {
                color: "secondary",
                size: "sm",
                classList: ["overlay-button"],
                disabled: "false",
                textContent: "Full"
            }, { click: fnFullyApplyToSelectedClick });
            const oWZIconFullyApplyToSelectedPlace = createElem("i", {
                classList: ["w-icon w-icon-location-check-fill"],
                style: "font-size:18px;"
            });
            oWZButtonFullyApplyToSelectedPlace.prepend(oWZIconFullyApplyToSelectedPlace);
            oApplyAddressForm.appendChild(oWZButtonFullyApplyToSelectedPlace);
            const fnApplyKeepingNameToSelectedClick = () => applyAddress(e, "");
            const oWZButtonApplyKeepingNameToSelectedPlace = createElem("wz-button", {
                color: "secondary",
                size: "sm",
                classList: ["overlay-button"],
                disabled: "false",
                textContent: "Keep Name"
            }, { click: fnApplyKeepingNameToSelectedClick });
            const oWZIconApplyKeepingNameToSelectedPlace = createElem("i", {
                classList: ["w-icon w-icon-location-fill"],
                style: "font-size: 18px;"
            });
            oWZButtonApplyKeepingNameToSelectedPlace.prepend(oWZIconApplyKeepingNameToSelectedPlace);
            oApplyAddressForm.appendChild(oWZButtonApplyKeepingNameToSelectedPlace);
            const fnPasteViensetaAddressClick = () => applyAddress(e, "vienseta");
            const oWZButtonApplyAsViensetaToSelectedPlace = createElem("wz-button", {
                color: "secondary",
                size: "sm",
                classList: ["overlay-button"],
                disabled: "false",
                textContent: "Keep Name adding Vienseta"
            }, { click: fnPasteViensetaAddressClick });
            const oWZIconApplyAsViensetaToSelectedPlace = createElem("i", {
                classList: ["w-icon w-icon-home"],
                style: "font-size: 18px;"
            });
            oWZButtonApplyAsViensetaToSelectedPlace.prepend(oWZIconApplyAsViensetaToSelectedPlace);
            oApplyAddressForm.appendChild(oWZButtonApplyAsViensetaToSelectedPlace);
        }
        const oCreateVenueForm = createElem("div", {
            classList: ["external-providers-control", "form-group"]
        });
        divTippyContent.appendChild(oCreateVenueForm);
        const oCreateVenueFormLabel = createElem("wz-label", {
            "html-for": ""
        });
        oCreateVenueFormLabel.innerText = "Create Venue:";
        oCreateVenueForm.appendChild(oCreateVenueFormLabel);
        if (!mKadastrsFeaturesData[e.featureId].found_residential) {
            const fnCreateResidentialClick = () => createResidential(e);
            const oWZIconCreateResidential = createElem("i", {
                classList: ["w-icon w-icon-navigation-now-fill"],
                style: "font-size:18px;"
            });
            const oWZButtonCreateResidential = createElem("wz-button", {
                color: "secondary",
                size: "sm",
                classList: ["overlay-button"],
                disabled: "false",
                textContent: "Residential"
            }, { click: fnCreateResidentialClick });
            oWZButtonCreateResidential.prepend(oWZIconCreateResidential);
            oCreateVenueForm.appendChild(oWZButtonCreateResidential);
        }
        if (!mKadastrsFeaturesData[e.featureId].found_hn) {
            const fnCreateHNClick = () => createHN(e);
            const oWZButtonCreateHN = createElem("wz-button", {
                color: "secondary",
                size: "sm",
                classList: ["overlay-button"],
                disabled: "false",
                textContent: "HN"
            }, { click: fnCreateHNClick });
            const oWZIconCreateHN = createElem("i", {
                classList: ["w-icon w-icon-home"],
                style: "font-size:18px;"
            });
            oWZButtonCreateHN.prepend(oWZIconCreateHN);
            oCreateVenueForm.appendChild(oWZButtonCreateHN);
        }
        const fnForceCreateClick = () => createVenue(e);
        const oWZButtonCreatePlace = createElem("wz-button", {
            color: "secondary",
            size: "sm",
            classList: ["overlay-button"],
            disabled: "false",
            textContent: `${mKadastrsFeaturesData[e.featureId].found_venues ? "Force Create Place" : "Place"}`
        }, { click: fnForceCreateClick });
        const oWZIconCreatePlace = createElem("i", {
            classList: ["w-icon w-icon-polygon"],
            style: "font-size:18px;"
        });
        oWZButtonCreatePlace.prepend(oWZIconCreatePlace);
        oCreateVenueForm.appendChild(oWZButtonCreatePlace);
        divElemRoot.appendChild(divTippyContent);
        $KadastrsMenuPopupDiv.replaceChildren(divElemRoot);
        const mPopupPixelPosition = wmeSDK.Map.getMapPixelFromLonLat({
            lonLat: { lon: placeGeom.coordinates[0], lat: placeGeom.coordinates[1] }
        });
        const dataPlacement = "right";
        $KadastrsMenuPopupDiv.style.transform = `translate(${Math.round(mPopupPixelPosition.x + 24)}px, ${Math.round(mPopupPixelPosition.y - 24)}px)`;
        $KadastrsMenuPopupDiv.querySelector("#kadastrsMenuDiv-tooltip")?.setAttribute("data-placement", dataPlacement);
        $KadastrsMenuPopupDiv.querySelector("#kadastrsMenuDiv-tooltip")?.setAttribute("data-state", "visible");
        $KadastrsMenuPopupDiv.querySelector("#kadastrsMenuDiv-content")?.setAttribute("data-state", "visible");
        let aFoundVenues = [];
        if (mKadastrsFeaturesData[e.featureId].found_venues) {
            aFoundVenues = [...aFoundVenues, ...mKadastrsFeaturesData[e.featureId].found_venues];
        }
        if (mKadastrsFeaturesData[e.featureId].found_residential) {
            aFoundVenues = [...aFoundVenues, ...mKadastrsFeaturesData[e.featureId].found_residential];
        }
        if (mKadastrsFeaturesData[e.featureId].found_hn) {
            aFoundVenues = [...aFoundVenues, ...mKadastrsFeaturesData[e.featureId].found_hn];
        }
        const placePoint = placeGeom;
        aFoundVenues.forEach(sFoundVenueKey => {
            highlightVenue2(sFoundVenueKey, "blue");
        });
        aFoundVenues.forEach(sFoundVenueKey => {
            drawConnectionLines(sFoundVenueKey, placePoint);
        });
        setTimeout(() => W.map.getLayerByUniqueName("wme-ut-kadastrs-hover").redraw(), 0);
    }
    function createGeometry(e) {
        const vertex = 0.0006;
        const placeGeom = mKadastrsFeaturesData[e.featureId].geometry;
        const [lon, lat] = placeGeom.coordinates;
        const oRectangle = turf__namespace.polygon([
            [
                [lon - vertex, lat - vertex / 2],
                [lon - vertex, lat + vertex / 2],
                [lon + vertex, lat + vertex / 2],
                [lon + vertex, lat - vertex / 2],
                [lon - vertex, lat - vertex / 2]
            ]
        ]);
        const oRotatedRectangle = turf__namespace.transformRotate(oRectangle, 10, {
            pivot: turf__namespace.centroid(oRectangle).geometry.coordinates
        });
        return oRotatedRectangle;
    }
    function applyAddress(e, sMode) {
        const mAddressBuffer = convertKadastrsAddressStringToParts(mKadastrsFeaturesData[e.featureId].kadastrs_data, "waze");
        if (!mAddressBuffer) {
            return;
        }
        const selection = wmeSDK.Editing.getSelection();
        if (!selection || selection.objectType !== "venue") {
            return;
        }
        if (selection.ids.length > 0) {
            if (sMode === "vienseta") {
                let oStreet = findStreetId(mAddressBuffer.cityName, mAddressBuffer.name);
                if (!oStreet) {
                    oStreet = findStreetId(`${mAddressBuffer.cityName}, ${mAddressBuffer.pagastsName}`, mAddressBuffer.name);
                }
                if (!oStreet) {
                    const oNewStreet = wmeSDK.DataModel.Streets.addStreet({
                        cityId: mAddressBuffer.cityID,
                        streetName: mAddressBuffer.name
                    });
                    updateVenue(selection.ids[0].toString(), { streetID: oNewStreet.id });
                }
                else {
                    updateVenue(selection.ids[0].toString(), { streetID: oStreet.attributes.id });
                }
            }
            else {
                let sStreetID = mAddressBuffer.streetID;
                if (!sStreetID) {
                    let oStreet = findStreetId(mAddressBuffer.cityName, mAddressBuffer.streetName ?? "");
                    if (!oStreet) {
                        oStreet = findStreetId(`${mAddressBuffer.cityName}, ${mAddressBuffer.pagastsName}`, mAddressBuffer.name);
                    }
                    if (oStreet) {
                        sStreetID = oStreet.attributes.id;
                    }
                }
                if (!sStreetID) {
                    const oNewStreet = wmeSDK.DataModel.Streets.addStreet({
                        cityId: mAddressBuffer.cityID,
                        streetName: mAddressBuffer.name
                    });
                    sStreetID = oNewStreet.id;
                }
                if (!sStreetID) {
                    console.log("Error: no Street!");
                }
                updateVenue(selection.ids[0].toString(), {
                    houseNumber: mAddressBuffer.houseNumber,
                    streetID: sStreetID,
                    name: sMode === "full" ? mAddressBuffer.name : undefined
                });
            }
        }
    }
    function updateVenue(sVenueId, mAddressData) {
        if (mAddressData.houseNumber?.includes(" k-")) {
            mAddressData.houseNumber = mAddressData.houseNumber.replace(" k-", "-");
        }
        if (mAddressData.streetID) {
            wmeSDK.DataModel.Venues.updateAddress({
                venueId: sVenueId,
                houseNumber: mAddressData.houseNumber,
                streetId: mAddressData.streetID
            });
        }
        if (mAddressData.name) {
            try {
                wmeSDK.DataModel.Venues.updateVenue({
                    venueId: sVenueId,
                    name: mAddressData.name,
                    lockRank: 2
                });
            }
            catch (oError) {
            }
        }
    }
    function createResidential(e) {
        const WMEAddressParams = convertKadastrsAddressStringToParts(mKadastrsFeaturesData[e.featureId].kadastrs_data, "waze");
        if (!WMEAddressParams) {
            return;
        }
        const oGeometry = createGeometry(e);
        const addressPoi = turf__namespace.centroid(oGeometry.geometry);
        const mAddressPoiPixelPosition = wmeSDK.Map.getMapPixelFromLonLat({
            lonLat: { lon: addressPoi.geometry.coordinates[0], lat: addressPoi.geometry.coordinates[1] }
        });
        mAddressPoiPixelPosition.x += 7;
        mAddressPoiPixelPosition.y -= 7;
        const mMovedcoordinates = wmeSDK.Map.getLonLatFromMapPixel(mAddressPoiPixelPosition);
        const poi = turf__namespace.point([mMovedcoordinates.lon, mMovedcoordinates.lat]);
        const sVenueId = wmeSDK.DataModel.Venues.addVenue({
            category: "RESIDENTIAL",
            geometry: poi.geometry
        }).toString();
        updateVenue(sVenueId, WMEAddressParams);
        wmeSDK.Editing.setSelection({
            selection: {
                ids: [sVenueId],
                objectType: "venue"
            }
        });
    }
    function createHN(e) {
        const WMEAddressParams = convertKadastrsAddressStringToParts(mKadastrsFeaturesData[e.featureId].kadastrs_data, "waze");
        if (!WMEAddressParams) {
            return;
        }
        const oGeometry = createGeometry(e);
        const addressPoi = turf__namespace.centroid(oGeometry.geometry);
        const mAddressPoiPixelPosition = wmeSDK.Map.getMapPixelFromLonLat({
            lonLat: { lon: addressPoi.geometry.coordinates[0], lat: addressPoi.geometry.coordinates[1] }
        });
        mAddressPoiPixelPosition.x += 7;
        mAddressPoiPixelPosition.y += 7;
        const mMovedcoordinates = wmeSDK.Map.getLonLatFromMapPixel(mAddressPoiPixelPosition);
        const poi = turf__namespace.point([mMovedcoordinates.lon, mMovedcoordinates.lat]);
        wmeSDK.DataModel.HouseNumbers.addHouseNumber({
            number: WMEAddressParams.houseNumber,
            point: poi.geometry
        });
    }
    function createVenue(e) {
        const WMEAddressParams = convertKadastrsAddressStringToParts(mKadastrsFeaturesData[e.featureId].kadastrs_data, "waze");
        if (!WMEAddressParams) {
            alert(`Street  not found on map. Please check if it actually exist ${mKadastrsFeaturesData[e.featureId].kadastrs_data}`);
            return;
        }
        const oGeometry = createGeometry(e);
        const sVenueId = wmeSDK.DataModel.Venues.addVenue({
            category: "OTHER",
            geometry: oGeometry.geometry
        }).toString();
        updateVenue(sVenueId, WMEAddressParams);
        wmeSDK.Editing.setSelection({
            selection: {
                ids: [sVenueId],
                objectType: "venue"
            }
        });
    }
    function drawConnectionLines(foundVenueKey, placePt) {
        var placeGeomTarget = turf__namespace.centroid(foundVenueKey.geometry);
        wmeSDK.Map.addFeatureToLayer({
            layerName: "wme-ut-kadastrs-lines",
            feature: {
                id: "line",
                type: "Feature",
                geometry: {
                    type: "LineString",
                    coordinates: [placePt.coordinates, placeGeomTarget.geometry.coordinates]
                }
            }
        });
    }
    function hideTooltipAfterDelay() {
        iPopupTimeout = window.setTimeout(hideCustomPopup, 300);
    }
    function onFeatureUnHoverEvent(e) {
        hideTooltipAfterDelay();
        wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: "wme-ut-kadastrs-lines" });
        wmeSDK.Map.removeAllFeaturesFromLayer({ layerName: "wme-ut-kadastrs-hover" });
        aWMEPolygonHighlightBlueFeatures.forEach(feature => {
            feature.style = null;
        });
        aWMEPolygonHighlightBlueFeatures = [];
    }
    function highlightVenue3(foundVenue, sColor) {
        if (foundVenue.geometry.type === "Polygon") {
            const sdkVenue = wmeSDK.DataModel.Venues.getById({ venueId: foundVenue.id });
            const oVenueClone = turf__namespace.polygon(sdkVenue?.geometry.coordinates, { styleName: sColor ? sColor : "error", layerName: "wme-ut-kadastrs-highlights" }, { id: `polygon_${foundVenue.id}` });
            wmeSDK.Map.addFeatureToLayer({
                layerName: "wme-ut-kadastrs-highlights",
                feature: oVenueClone,
                style: {
                    strokeWidth: 3,
                    strokeDashstyle: "12 8",
                    strokeColor: "green"
                }
            });
        }
    }
    function highlightVenue2(foundVenue, sColor) {
        if (foundVenue.geometry.type === "Polygon") {
            wmeSDK.DataModel.Venues.getById({ venueId: foundVenue.id });
            const oVenue = W.model.venues.getByIds([foundVenue.id]).at(0);
            if (oVenue) {
                const poly = W.userscripts.getFeatureElementByDataModel(oVenue);
                poly?.setAttribute("stroke", sColor );
                poly?.setAttribute("stroke-dasharray", "12 8");
                poly?.setAttribute("stroke-width", "3");
                poly?.setAttribute("fill", "#00695C");
            }
        }
    }
    function hideCustomPopup() {
        if ($KadastrsMenuPopupDiv) {
            $KadastrsMenuPopupDiv.querySelector("#kadastrsMenuDiv-content")?.setAttribute("data-state", "hidden");
            $KadastrsMenuPopupDiv.querySelector("#kadastrsMenuDiv-tooltip")?.setAttribute("data-state", "hidden");
            $KadastrsMenuPopupDiv.replaceChildren();
        }
    }
    function findStreetId(sCityName, sStreetName, bVienseta) {
        return Object.values(W.model.streets.objects).find((street) => {
            if (!W.model.cities.objects[street.attributes.cityID]) {
                console.log("Error: no City");
            }
            if (street.attributes.name === sStreetName &&
                W.model.cities.objects[street.attributes.cityID].attributes.name.includes(sCityName)) {
                return true;
            }
        });
    }
    function populateIds(addresses) {
        const streetCache = {};
        for (const [index, address] of addresses.entries()) {
            if (!streetCache[address.streetName]) {
                const oStreet = findStreetId(address.cityName, address.streetName);
                if (!oStreet && address.streetName !== "") {
                    const sErrorMsg = "Error: Can't find street: " + address.streetName + " in City: " + address.cityName;
                    console.log(sErrorMsg);
                    alert(sErrorMsg);
                    return index;
                }
                if (oStreet) {
                    streetCache[address.streetName] = oStreet.attributes.id;
                }
            }
            address.streetID = streetCache[address.streetName];
        }
        return undefined;
    }
    function convertKadastrsAddressStringToParts(sAddress, sMode) {
        const aAddress = sAddress.split(", ");
        const bNoCity = aAddress[1].slice(-4) === "pag.";
        const sCityName = bNoCity ? "" : aAddress[1];
        const sPagastsName = bNoCity ? aAddress[1] : aAddress[2];
        bNoCity ? aAddress[2] : aAddress[3];
        bNoCity ? aAddress[3] : aAddress[4];
        const sStreetNameOrHN = aAddress[0];
        let sName = "";
        let sHN = "";
        let bVienseta = false;
        let bVasarnica = false;
        if (bNoCity) {
            sName = `${sStreetNameOrHN.slice(1, -1)}, ${sPagastsName}`;
            const sStreetName = "";
            return {
                cityName: sCityName,
                streetName: sStreetName,
                houseNumber: sHN,
                name: sName
            };
        }
        else {
            if (sStreetNameOrHN.startsWith('"') && sStreetNameOrHN.endsWith('"')) {
                sName = sStreetNameOrHN.slice(1, -1);
                const aVasarnica = sName.split(" ");
                const sLast = aVasarnica.pop();
                const sVasarnicaName = aVasarnica.join(" ");
                const bIsVasarnica = aVasarnicas.some(mRecord => mRecord.name === sVasarnicaName && mRecord.parent === aAddress[1]);
                if (bIsVasarnica) {
                    bVasarnica = true;
                    sHN = aVasarnica.length > 0 ? sLast ?? "" : "";
                    sName = sHN;
                }
                else {
                    bVienseta = true;
                }
            }
            else {
                const aStreetNameAndHN = sStreetNameOrHN.split(" ");
                sName = "";
                sHN = aStreetNameAndHN.pop() ?? "";
                if (sHN.includes("k-")) {
                    sHN = `${aStreetNameAndHN.pop()}${sHN}`;
                    sName = sHN.replace("k", " k");
                }
                else {
                    sName = sHN;
                }
            }
            const iLastIndex = sStreetNameOrHN.lastIndexOf(" " + sName);
            const sStreetName = sHN === ""
                ? ""
                : iLastIndex === -1
                    ? sStreetNameOrHN
                    : bVasarnica
                        ? sStreetNameOrHN.substring(1, iLastIndex)
                        : sStreetNameOrHN.substring(0, iLastIndex);
            if (sMode !== "waze") {
                return {
                    cityName: sCityName,
                    streetName: bVienseta ? "" : sStreetName,
                    houseNumber: sHN,
                    name: sName
                };
            }
            else {
                const oStreet = Object.values(W.model.streets.objects).find((street) => {
                    if (street.attributes.name === sStreetName &&
                        W.model.cities.objects[street.attributes.cityID].attributes.name.includes(sCityName)) {
                        return true;
                    }
                });
                let oCity;
                if (!oStreet) {
                    oCity = Object.values(W.model.cities.objects).find((oCity) => {
                        return oCity.attributes.name === sCityName;
                    });
                }
                return {
                    cityID: oStreet ? oStreet.attributes.cityID : oCity ? oCity.attributes.id : undefined,
                    streetID: oStreet ? oStreet.attributes.id : undefined,
                    houseNumber: sHN,
                    name: sName,
                    cityName: sCityName,
                    pagastsName: sPagastsName
                };
            }
        }
    }
    function makeGetRequest(sURL) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "GET",
                url: sURL,
                onload: response => resolve(response.responseText),
                onerror: error => reject(error)
            });
        });
    }
    function excludeKadastrsAddressConsistentWithWME(oAddressRegex, oVenueOrHouseNumber) {
        const indexToRemove = aMissingFetchedAddresses.findIndex(mFetchedAddresses => oAddressRegex.test(mFetchedAddresses.attributes.std));
        if (indexToRemove > -1) {
            aMissingFetchedAddresses.splice(indexToRemove, 1);
        }
        const indexToUpdate = aGlobalFetchedAddresses.findIndex(mFetchedAddresses => oAddressRegex.test(mFetchedAddresses.attributes.std));
        if (indexToUpdate > -1) {
            if (oVenueOrHouseNumber.geometry.type === "Point") {
                if (oVenueOrHouseNumber.categories?.[0] === "RESIDENCE_HOME") {
                    if (!aGlobalFetchedAddresses[indexToUpdate].data.Residential) {
                        aGlobalFetchedAddresses[indexToUpdate].data.Residential = [];
                    }
                    aGlobalFetchedAddresses[indexToUpdate].data.Residential.push(oVenueOrHouseNumber);
                }
                else if (oVenueOrHouseNumber.categories) {
                    if (!aGlobalFetchedAddresses[indexToUpdate].data.Venues) {
                        aGlobalFetchedAddresses[indexToUpdate].data.Venues = [];
                    }
                    aGlobalFetchedAddresses[indexToUpdate].data.Venues.push(oVenueOrHouseNumber);
                }
                else {
                    if (!aGlobalFetchedAddresses[indexToUpdate].data.HN) {
                        aGlobalFetchedAddresses[indexToUpdate].data.HN = [];
                    }
                    aGlobalFetchedAddresses[indexToUpdate].data.HN.push(oVenueOrHouseNumber);
                }
            }
            else {
                aWMEValidAddressVenues.push(oVenueOrHouseNumber);
                if (!aGlobalFetchedAddresses[indexToUpdate].data.Venues) {
                    aGlobalFetchedAddresses[indexToUpdate].data.Venues = [];
                }
                aGlobalFetchedAddresses[indexToUpdate].data.Venues.push(oVenueOrHouseNumber);
            }
        }
    }
    function getRegex(aStrings) {
        const regexPattern = aStrings.map(v => v).join("\\s*,\\s*");
        return new RegExp(regexPattern);
    }
    function findStreetName(iStreetId) {
        if (!W.model.streets.objects[iStreetId]) {
            return {
                name: "",
                city: ""
            };
        }
        const sName = W.model.streets.objects[iStreetId].attributes.name;
        const iCity = W.model.streets.objects[iStreetId].attributes.cityID;
        const sCity = W.model.cities.objects[iCity].attributes.countryID === 123 ? W.model.cities.objects[iCity].attributes.name : "";
        return {
            name: sName,
            city: sCity
        };
    }
    function getBBoxv2() {
        const mapExtent = W.map.getExtent();
        return mapExtent.toString();
    }
    function getBBox3059() {
        const extent = W.map.getExtent();
        if (Array.isArray(extent)) {
            proj4.defs("EPSG:3059", "+proj=tmerc +lat_0=0 +lon_0=24 +k=0.9996 +x_0=500000 +y_0=-6000000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs +type=crs");
            const aConvertedPoints = [
                ...proj4("EPSG:4326", "EPSG:3059", extent.slice(0, 2)).reverse(),
                ...proj4("EPSG:4326", "EPSG:3059", extent.slice(2, 4)).reverse()
            ];
            const string = aConvertedPoints.join();
            return string;
        }
    }
    function applyRedHightlightStyle(properties, zoomLevel) {
        return properties.styleName === "error" && properties.layerName === "wme-ut-kadastrs-highlights";
    }
    function applyGreenHightlightStyle(properties, zoomLevel) {
        return properties.styleName === "valid" && properties.layerName === "wme-ut-kadastrs-highlights";
    }

})(turf, proj4);