WME UT Kadastrs

WME UrSuS Tools: LV Kadastrs

// ==UserScript==
// @name        WME UT Kadastrs
// @namespace   http://ursus.id.lv
// @version     1.0.1
// @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://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==

/* globals proj4 */

(function (proj4) {
    'use strict';

    const sScriptName = GM_info.script.name;
    GM_info.script.version;
    let wmeSDK;
    let aGlobalFetchedAddresses = [];
    let aWMEValidAddressVenues = [];
    let aWMEPolygonHighlightFeatures = [];
    let aAddressConnectionLinesFeatures = [];
    let aWMEPolygonHighlightBlueFeatures = [];
    let aMissingFetchedAddresses = [];
    let aHoverFeatures = [];
    let iPopupTimeout;
    let $KadastrsMenuPopupDiv;
    let oAddHouseNumber;
    let aVasarnicas = [];
    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);
            extent.transform("EPSG:4326", "EPSG:3857");
        }
        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);
        }
        oAddHouseNumber = require("Waze/Actions/AddHouseNumber");
        addKFetchButton();
        fetchSheetData();
    }
    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"
        };
        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);
        let aWazeVenues = oParsedObject.venues.objects;
        aWMEValidAddressVenues = [];
        WazeWrap.Alerts.info(sScriptName, `${aWazeVenues.length} WME Address Fetched`);
        let aWazeHouseNumbers = [];
        if (wmeSDK.Map.getZoomLevel() >= 17) {
            const aSegments = W.model.segments
                .getObjectArray()
                .filter((oObject) => oObject.getAttribute("hasHNs"));
            const oMapExtGeometry = getOLMapExtent().toGeometry();
            const aSegmentsOnScreen = aSegments
                .map((oSegment) => {
                if (oMapExtGeometry.intersects(oSegment.getAttribute("geometry"))) {
                    return oSegment.getID();
                }
            })
                .filter((oItem) => !!oItem);
            const oResults = await W.controller.descartesClient.getHouseNumbers(aSegmentsOnScreen);
            aWazeHouseNumbers = oResults.segmentHouseNumbers.objects;
        }
        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`);
        aWazeHouseNumbers.map(oWazeHouseNumber => {
            const mSegStreet = W.model.segments.getObjectById(oWazeHouseNumber.attributes.segID).getAddress().attributes
                ?.street?.attributes;
            const { name: sStreetName, city: sCity } = findStreetName(mSegStreet.id);
            const oAddressRegex = getRegex([sStreetName + " " + oWazeHouseNumber.attributes.number.toUpperCase(), sCity]);
            excludeKadastrsAddressConsistentWithWME(oAddressRegex, oWazeHouseNumber);
        });
        addKadastrsLayerWithFeatures(aGlobalFetchedAddresses);
        clearHighlightedAdressFeatures();
        let bIsVenueLayerHidden = false;
        aWazeVenues.forEach(oVenue => {
            const OLVenue = W.map.venueLayer.getOlFeatureByFeatureId(oVenue.id);
            if (!OLVenue) {
                bIsVenueLayerHidden = true;
                return;
            }
            highlightVenue2(OLVenue);
        });
        aWMEValidAddressVenues.forEach(oVenue => {
            const OLVenue = W.map.venueLayer.getOlFeatureByFeatureId(oVenue.id);
            if (!OLVenue) {
                bIsVenueLayerHidden = true;
                return;
            }
            highlightVenue2(OLVenue, true);
        });
        if (bIsVenueLayerHidden) {
            WazeWrap.Alerts.warning(sScriptName, "Please activate Venues or Residential Layer");
        }
        setTimeout(() => W.map.getLayerByUniqueName("venues").redraw(), 0);
    }
    function addKadastrsLayerWithFeatures(aKadastrsFeatures) {
        const aFeaturesForLayer = [];
        const pointLayer = new OpenLayers.Layer.Vector("LVWMEKadastrsPointLayer", {
            rendererOptions: {
                zIndexing: true
            }
        });
        const oExistingLayer = W.map.getLayersByName("LVWMEKadastrsPointLayer")[0];
        const layerContainer = W.selectionManager.getWebMapSelectionMediator().getRootContainerLayer();
        if (oExistingLayer) {
            oExistingLayer.destroyFeatures();
            W.map.removeLayer(oExistingLayer);
            const indexToRemove = W.map.layers.indexOf(oExistingLayer);
            if (indexToRemove !== -1) {
                layerContainer.layers.splice(indexToRemove, 1);
            }
        }
        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];
            const point = new OpenLayers.Geometry.Point(aCoords[0], aCoords[1]);
            const featureStyle = {
                label: "K",
                pointRadius: 15,
                fillColor: "#00695C",
                fillOpacity: 0.8,
                strokeColor: "#cc6633",
                strokeWidth: 2,
                strokeOpacity: 0.8,
                fontColor: "black",
                labelOutlineColor: "white",
                labelOutlineWidth: 3,
                repositoryObject: {
                    isDeleted: () => false,
                    isNew: () => false,
                    getType: () => null,
                    getID: () => -1
                }
            };
            const feature = new OpenLayers.Feature.Vector(point, null, featureStyle);
            feature.data = mFeature;
            aFeaturesForLayer.push(feature);
        });
        pointLayer.addFeatures(aFeaturesForLayer);
        const selectControl = new OpenLayers.Control.SelectFeature(pointLayer, {
            hover: true,
            onSelect: onFeatureHoverEvent,
            onUnselect: onFeatureUnHoverEvent
        });
        W.map.addControl(selectControl);
        selectControl.activate();
        W.map.addLayer(pointLayer);
        layerContainer.layers.push(pointLayer);
        layerContainer.collectRoots();
    }
    function checkTooltip() {
        window.clearTimeout(iPopupTimeout);
    }
    function onFeatureHoverEvent(e) {
        window.clearTimeout(iPopupTimeout);
        const mKadastrsAddress = convertKadastrsAddressStringToParts(e.data.attributes.std, "waze");
        if (!mKadastrsAddress) {
            return;
        }
        const placeGeom = e.geometry.getCentroid();
        const placePt = new OpenLayers.Geometry.Point(placeGeom.x, placeGeom.y);
        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 (e.data.data.Venues) {
            sVenues = e.data.data.Venues.map((oVenue) => oVenue.name).join("\r\n&bull;");
            if (e.data.data.Venues.length > 1) {
                sVenues = "\r\n&bull;" + sVenues + "\r\n";
            }
        }
        oVenuesTextDiv.innerHTML = `Venues: ${sVenues} \n Residential: ${(e.data.data.Residential ?? []).length === 1 ? "✅" : e.data.data.Residential ?? "❌"} HN: ${(e.data.data.HN ?? []).length === 1 ? "✅" : e.data.data.HN ?? "❌"}`;
        divTippyContent.appendChild(oVenuesTextDiv);
        const oInputForm = createElem("div", {
            classList: ["wz-text-input-inner-container"]
        });
        const oInputInput = createElem("wz-text-input", {
            value: e.data.attributes.std
        });
        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 (!e.data.data.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 (!e.data.data.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: `${e.data.data.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 aLonLat = proj4("EPSG:900913", "EPSG:4326", [placeGeom.x, placeGeom.y]);
        const mPopupPixelPosition = W.map.getPixelFromLonLat({
            lon: aLonLat[0],
            lat: aLonLat[1]
        });
        mPopupPixelPosition.origX = mPopupPixelPosition.x;
        $KadastrsMenuPopupDiv.clientWidth / 2;
        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 (e.data.data.Venues) {
            aFoundVenues = [...aFoundVenues, ...e.data.data.Venues];
        }
        if (e.data.data.Residential) {
            aFoundVenues = [...aFoundVenues, ...e.data.data.Residential];
        }
        if (e.data.data.HN) {
            aFoundVenues = [...aFoundVenues, ...e.data.data.HN];
        }
        aHoverFeatures = [];
        aFoundVenues.forEach(sFoundVenueKey => {
            highlightVenue(sFoundVenueKey);
        });
        aFoundVenues.forEach(sFoundVenueKey => {
            drawConnectionLines(sFoundVenueKey, placePt);
        });
        aHoverFeatures = [...aAddressConnectionLinesFeatures];
        W.map.getLayerByUniqueName("venues").addFeatures(aHoverFeatures);
        setTimeout(() => W.map.getLayerByUniqueName("venues").redraw(), 0);
    }
    function createGeometry(e) {
        const vertex = 29.4;
        const poly = new OpenLayers.Geometry.LinearRing([
            new OpenLayers.Geometry.Point(e.data.geometry.x - vertex, e.data.geometry.y - vertex / 2),
            new OpenLayers.Geometry.Point(e.data.geometry.x - vertex, e.data.geometry.y + vertex / 2),
            new OpenLayers.Geometry.Point(e.data.geometry.x + vertex, e.data.geometry.y + vertex / 2),
            new OpenLayers.Geometry.Point(e.data.geometry.x + vertex, e.data.geometry.y - vertex / 2),
            new OpenLayers.Geometry.Point(e.data.geometry.x - vertex, e.data.geometry.y - vertex / 2)
        ]);
        poly.rotate(10, poly.getCentroid());
        return new OpenLayers.Geometry.Polygon([poly]);
    }
    function applyAddress(e, sMode) {
        const mAddressBuffer = convertKadastrsAddressStringToParts(e.data.attributes.std, "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 {
                updateVenue(selection.ids[0].toString(), {
                    houseNumber: mAddressBuffer.houseNumber,
                    streetID: mAddressBuffer.streetID,
                    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(e.data.attributes.std, "waze");
        if (!WMEAddressParams) {
            return;
        }
        const oGeometry = createGeometry(e);
        const movedPoi = oGeometry.getCentroid();
        movedPoi.x += 7;
        movedPoi.y -= 7;
        const sVenueId = wmeSDK.DataModel.Venues.addVenue({
            category: "RESIDENTIAL",
            geometry: W.userscripts.toGeoJSONGeometry(movedPoi)
        }).toString();
        updateVenue(sVenueId, WMEAddressParams);
        wmeSDK.Editing.setSelection({
            selection: {
                ids: [sVenueId],
                objectType: "venue"
            }
        });
    }
    function createHN(e) {
        const WMEAddressParams = convertKadastrsAddressStringToParts(e.data.attributes.std, "waze");
        if (!WMEAddressParams) {
            return;
        }
        const oGeometry = createGeometry(e);
        const aStreetSegments = W.model.segments.getByAttributes({
            primaryStreetID: WMEAddressParams.streetID
        });
        if (aStreetSegments.length > 0) {
            const closestSegment = aStreetSegments.reduce((min, segment) => oGeometry.distanceTo(segment.getOLGeometry(), { details: true }).distance <
                oGeometry.distanceTo(min.getOLGeometry(), { details: true }).distance
                ? segment
                : min);
            const oHNPoint = oGeometry.getCentroid();
            oHNPoint.x += 7;
            oHNPoint.y += 7;
            const fractionPoint = findClosestPoint(oHNPoint, closestSegment.getOLGeometry());
            if (!fractionPoint) {
                return;
            }
            const oFractionPoint = new OpenLayers.Geometry.Point(fractionPoint.x, fractionPoint.y);
            const epsg900913 = new OpenLayers.Projection("EPSG:900913");
            const epsg4326 = new OpenLayers.Projection("EPSG:4326");
            oHNPoint.transform(epsg900913, epsg4326);
            oFractionPoint.transform(epsg900913, epsg4326);
            const oHouseNumber = new Vs({
                forced: false,
                fractionPoint: {
                    coordinates: [oFractionPoint.x, oFractionPoint.y],
                    type: "Point"
                },
                geoJSONGeometry: {
                    coordinates: [oHNPoint.x, oHNPoint.y],
                    type: "Point"
                },
                number: WMEAddressParams.houseNumber,
                segID: closestSegment.getID(),
                id: W.model.segmentHouseNumbers.generateUniqueID()
            });
            W.model.actionManager.add(new oAddHouseNumber(oHouseNumber));
        }
    }
    function createVenue(e) {
        const WMEAddressParams = convertKadastrsAddressStringToParts(e.data.attributes.std, "waze");
        if (!WMEAddressParams) {
            alert(`Street  not found on map. Please check if it actually exist ${e.data.attributes.std}`);
            return;
        }
        const oGeometry = createGeometry(e);
        const sVenueId = wmeSDK.DataModel.Venues.addVenue({
            category: "OTHER",
            geometry: W.userscripts.toGeoJSONGeometry(oGeometry)
        }).toString();
        updateVenue(sVenueId, WMEAddressParams);
        wmeSDK.Editing.setSelection({
            selection: {
                ids: [sVenueId],
                objectType: "venue"
            }
        });
    }
    function drawConnectionLines(foundVenueKey, placePt) {
        const label = "";
        let placeGeomTarget;
        if (foundVenueKey.type === "houseNumber") {
            placeGeomTarget = foundVenueKey.getOLGeometry().getCentroid();
        }
        else {
            placeGeomTarget = W.model.venues.objects[foundVenueKey.id].getOLGeometry().getCentroid();
        }
        const poiPt = new OpenLayers.Geometry.Point(placeGeomTarget.x, placeGeomTarget.y);
        const lsLine = new OpenLayers.Geometry.LineString([placePt, poiPt]);
        const addressConnectionLineFeature = new OpenLayers.Feature.Vector(lsLine, {}, {
            strokeWidth: 3,
            strokeDashstyle: "12 8",
            strokeColor: "#FF0",
            label,
            labelYOffset: 45,
            fontColor: "#FF0",
            fontWeight: "bold",
            labelOutlineColor: "#000",
            labelOutlineWidth: 4,
            fontSize: "18"
        });
        aAddressConnectionLinesFeatures.push(addressConnectionLineFeature);
    }
    function highlightVenue(foundVenue, placePt) {
        if (foundVenue.type === "houseNumber") ;
        else {
            const placeGeomTarget = W.model.venues.objects[foundVenue.id].getOLGeometry().getCentroid();
            new OpenLayers.Geometry.Point(placeGeomTarget.x, placeGeomTarget.y);
            const OLVenue = W.map.venueLayer.getOlFeatureByFeatureId(foundVenue.id);
            if (OLVenue.geometry.getArea() !== 0) {
                OLVenue.geometry.getGeodesicArea(W.map.getProjectionObject());
            }
            if (OLVenue.geometry.getArea() !== 0) {
                const mStyle = {
                    fillColor: "#00695C",
                    fillOpacity: 0.5,
                    strokeWidth: 3,
                    strokeDashstyle: "12 8",
                    strokeColor: "blue",
                    label: "",
                    labelYOffset: 45,
                    fontColor: "#FF0",
                    fontWeight: "bold",
                    labelOutlineColor: "#000",
                    labelOutlineWidth: 4,
                    fontSize: "18"
                };
                OLVenue.style = mStyle;
                aWMEPolygonHighlightBlueFeatures.push(OLVenue);
            }
        }
    }
    function hideTooltipAfterDelay() {
        iPopupTimeout = window.setTimeout(hideCustomPopup, 300);
    }
    function onFeatureUnHoverEvent(e) {
        hideTooltipAfterDelay();
        aAddressConnectionLinesFeatures.forEach(feature => {
            feature.destroy();
            feature = null;
        });
        aAddressConnectionLinesFeatures = [];
        aWMEPolygonHighlightBlueFeatures.forEach(feature => {
            feature.style = null;
        });
        aWMEPolygonHighlightBlueFeatures = [];
        setTimeout(() => W.map.getLayerByUniqueName("venues").redraw(), 0);
    }
    function highlightVenue2(OLVenue, bIsValid) {
        if (OLVenue.geometry.getArea() !== 0) {
            OLVenue.geometry.getGeodesicArea(W.map.getProjectionObject());
        }
        if (OLVenue.geometry.getArea() !== 0) {
            const mStyle = {
                fillColor: "#00695C",
                fillOpacity: 0.5,
                strokeWidth: 3,
                strokeDashstyle: "12 8",
                strokeColor: `${bIsValid ? "green" : "red"}`,
                label: "",
                labelYOffset: 45,
                fontColor: "#FF0",
                fontWeight: "bold",
                labelOutlineColor: "#000",
                labelOutlineWidth: 4,
                fontSize: "18"
            };
            OLVenue.style = mStyle;
            aWMEPolygonHighlightFeatures.push(OLVenue);
        }
    }
    function clearHighlightedAdressFeatures() {
        if (aWMEPolygonHighlightFeatures.length > 0) {
            let layer = aWMEPolygonHighlightFeatures[0].layer;
            aWMEPolygonHighlightFeatures.forEach(feature => {
                feature.style = null;
                layer = feature.layer;
            });
            layer?.redraw();
            aWMEPolygonHighlightFeatures = [];
        }
    }
    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", "");
                }
                else {
                    sName = sHN;
                }
            }
            const iLastIndex = sStreetNameOrHN.lastIndexOf(" " + sHN);
            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.type === "houseNumber") {
                if (!aGlobalFetchedAddresses[indexToUpdate].data.HN) {
                    aGlobalFetchedAddresses[indexToUpdate].data.HN = [];
                }
                aGlobalFetchedAddresses[indexToUpdate].data.HN.push(oVenueOrHouseNumber);
            }
            else {
                aWMEValidAddressVenues.push(oVenueOrHouseNumber);
                if (oVenueOrHouseNumber.categories[0] === "RESIDENCE_HOME") {
                    if (!aGlobalFetchedAddresses[indexToUpdate].data.Residential) {
                        aGlobalFetchedAddresses[indexToUpdate].data.Residential = [];
                    }
                    aGlobalFetchedAddresses[indexToUpdate].data.Residential.push(oVenueOrHouseNumber);
                }
                else {
                    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;
        }
    }
    class Vs {
        type = "houseNumber";
        attributes;
        _prevID;
        version;
        state;
        constructor(e) {
            this.attributes = e;
        }
        static defaults() {
            return {
                forced: false,
                fractionPoint: null,
                id: "",
                side: null,
                valid: false
            };
        }
        changeID(e) {
            (this._prevID = this.getID()), this.setID(e);
        }
        merge(e) {
            Object.assign(this.attributes, e);
        }
        setID(e) {
            this.attributes.id = e;
        }
        setVersion(e) {
            this.version = e;
        }
        setState(e) {
            this.state = e;
        }
        getAttributes() {
            return this.attributes;
        }
        getAttribute(e) {
            return this.attributes[e];
        }
        getGeometry() {
            return this.getAttribute("geoJSONGeometry");
        }
        getID() {
            return this.attributes.id;
        }
        getUniqueIdentifier() {
            return {
                objectId: this.getID(),
                objectType: this.getType()
            };
        }
        getUniqueID() {
            return `${this.getType()}: ${this.getID()}`;
        }
        getType() {
            return this.type;
        }
        getSide() {
            return this.attributes.side;
        }
        getSegmentId() {
            return this.attributes.segID;
        }
        setSegmentId(e) {
            this.attributes.segID = e;
        }
        getNumber() {
            return this.attributes.number;
        }
        setNumber(e) {
            this.attributes.number = e;
        }
        isForced() {
            return this.attributes.forced;
        }
        isValid() {
            return this.attributes.valid;
        }
        getFractionPoint() {
            return this.attributes.fractionPoint;
        }
        setFractionPoint(e) {
            this.attributes.fractionPoint = e;
        }
        isDeleted() {
            return Boolean(this.state && this.state === "DELETE");
        }
    }
    function findClosestPoint(point, lineString) {
        let closestPoint = null;
        let closestDistance = Infinity;
        for (let i = 0; i < lineString.components.length - 1; i++) {
            const segmentStart = lineString.components[i];
            const segmentEnd = lineString.components[i + 1];
            const segmentVector = {
                x: segmentEnd.x - segmentStart.x,
                y: segmentEnd.y - segmentStart.y
            };
            const pointVector = {
                x: point.x - segmentStart.x,
                y: point.y - segmentStart.y
            };
            const dotProduct = pointVector.x * segmentVector.x + pointVector.y * segmentVector.y;
            const t = dotProduct / (segmentVector.x * segmentVector.x + segmentVector.y * segmentVector.y);
            const closestPointOnSegment = {};
            if (t < 0) {
                closestPointOnSegment.x = segmentStart.x;
                closestPointOnSegment.y = segmentStart.y;
            }
            else if (t > 1) {
                closestPointOnSegment.x = segmentEnd.x;
                closestPointOnSegment.y = segmentEnd.y;
            }
            else {
                closestPointOnSegment.x = segmentStart.x + t * segmentVector.x;
                closestPointOnSegment.y = segmentStart.y + t * segmentVector.y;
            }
            const distance = calculateDistance(closestPointOnSegment, point);
            if (distance < closestDistance) {
                closestDistance = distance;
                closestPoint = closestPointOnSegment;
            }
        }
        return closestPoint;
    }
    function calculateDistance(point1, point2) {
        const dx = point1.x - point2.x;
        const dy = point1.y - point2.y;
        return Math.sqrt(dx * dx + dy * dy);
    }

})(proj4);