您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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://gf.qytechs.cn/scripts/24851-wazewrap/code/WazeWrap.js // @require https://gf.qytechs.cn/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•"); if (mKadastrsFeaturesData[e.featureId].found_venues.length > 1) { sVenues = "\r\n•" + 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);
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址