// ==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•");
if (e.data.data.Venues.length > 1) {
sVenues = "\r\n•" + 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);