// ==UserScript==
// @name WME UT LV Layers
// @namespace http://ursus.id.lv
// @version 1.0.0
// @description WME UrSuS Tools: LV Layers
// @author UrSuS
// @match https://*.waze.com/*editor*
// @license MIT
// @icon 
// @exclude https://www.waze.com/user/editor*
// @require https://cdnjs.cloudflare.com/ajax/libs/proj4js/2.4.4/proj4.js
// @connect balticmaps.eu
// @grant GM_info
// @grant GM_xmlhttpRequest
// ==/UserScript==
/* globals proj4 */
(function (proj4) {
"use strict";
GM_info.script.name;
GM_info.script.version;
let wmeSDK;
let _settings = {};
let webLocationTilesContainerId;
let currentOverlay = "";
let currentOpacity = 100;
let sBalticMapsKey = "";
const _SETTINGS_STORAGE_NAME = "WME_LV_TOOLS";
const defaultSettings = {
LeftOffset: "400px",
TopOffset: "100px",
HideToolbar: false,
};
function cbLayerChange(oEvent) {
if (oEvent.target instanceof HTMLInputElement) {
const sSetting = oEvent.target.getAttribute("boundSetting");
if (sSetting) {
_settings[sSetting] = oEvent.target.checked;
}
const sLayerId = oEvent.target.getAttribute("layerId");
if (sLayerId) {
CheckUnCheckCustomWMSLayer(oEvent.target.checked, sLayerId);
}
saveSettingsToStorage();
}
}
const aLayersElementsConfig = [
{
id: "lvm-kadastrs",
url: "https://lvmgeoserver.lvm.lv/geoserver/ows?FORMAT=image/png&TRANSPARENT=TRUE&VERSION=1.3.0&SERVICE=WMS&REQUEST=GetMap&LAYERS=publicwfs:kkparcel&STYLES=&CRS=EPSG:4326&WIDTH=256&HEIGHT=256&BBOX={bbox}&STYLES=&request=GetMap",
elements: [
{
type: "input",
attributes: {
type: "checkbox",
id: "cbShowKadastrsLayer",
title: "Kadastrs",
checked: _settings.showKadastrsLayer,
boundSetting: "showKadastrsLayer",
layerId: "lvm-kadastrs",
},
events: {
change: cbLayerChange,
},
triggerChangeEvent: false,
},
{
type: "label",
attributes: {
for: "cbShowKadastrsLayer",
textContent: "Kadastrs",
},
},
],
},
{
id: "lvm-kadastrs-houses",
url: "https://lvmgeoserver.lvm.lv/geoserver/ows?FORMAT=image/png&TRANSPARENT=TRUE&VERSION=1.3.0&SERVICE=WMS&REQUEST=GetMap&LAYERS=publicwfs:kkbuilding&STYLES=&CRS=EPSG:4326&WIDTH=256&HEIGHT=256&BBOX={bbox}",
elements: [
{
type: "input",
attributes: {
type: "checkbox",
id: "cbShowKadastrsHousesLayer",
title: "Kadastrs Houses",
checked: _settings.showKadastrsHousesLayer,
boundSetting: "showKadastrsHousesLayer",
layerId: "lvm-kadastrs-houses",
},
events: {
change: cbLayerChange,
},
triggerChangeEvent: false,
},
{
type: "label",
attributes: {
for: "cbShowKadastrsHousesLayer",
textContent: "Kadastrs Houses",
},
},
],
},
{
id: "new-bing",
url: "https://ecn.t3.tiles.virtualearth.net/tiles/a{quadDigits}.jpeg?g={latestBingImageVersion}&n=z",
elements: [
{
type: "input",
attributes: {
type: "checkbox",
id: "cbShowNewBingLayer",
title: "New Bing",
checked: _settings.showNewBingLayer,
boundSetting: "showNewBingLayer",
layerId: "new-bing",
},
events: {
change: cbLayerChange,
},
triggerChangeEvent: false,
},
{
type: "label",
attributes: {
for: "cbShowNewBingLayer",
textContent: "New Bing",
},
},
],
},
{
id: "lvm-roads",
url: "https://lvmgeoproxy01.lvm.lv/A05F8B5E0EB84CAEB9499496474E3093/LVMLV/LITEvLVMbaseData/MapServer/export?F=image&FORMAT=PNG32&TRANSPARENT=true&LAYERS=show%3A19%2C20&BBOX={bbox3857}&BBOXSR=3857&DPI=350",
elements: [
{
type: "input",
attributes: {
type: "checkbox",
id: "cbShowLVMRoadsLayer",
title: "LVM Roads",
checked: _settings.showLVMRoadsLayer,
boundSetting: "showLVMRoadsLayer",
layerId: "lvm-roads",
},
events: {
change: cbLayerChange,
},
triggerChangeEvent: false,
},
{
type: "label",
attributes: {
for: "cbShowLVMRoadsLayer",
textContent: "LVM Roads",
},
},
],
},
{
id: "balticmaps",
url: "https://wms3.kartes.lv/PIER/wgs/15/{zoom}/{x}/{y}.png",
elements: [
{
type: "input",
attributes: {
type: "checkbox",
id: "cbShowBalticMapsLayer",
title: "Baltic Maps",
checked: _settings.showBalticMapsLayer,
boundSetting: "showBalticMapsLayer",
layerId: "balticmaps",
},
events: {
change: cbLayerChange,
},
triggerChangeEvent: false,
},
{
type: "label",
attributes: {
for: "cbShowBalticMapsLayer",
textContent: "Baltic Maps",
},
},
],
},
{
id: "balticmaps-orto8",
url: "https://wmsbm4.kartes.lv/{balticMapsKey}/wgs/orto_full/?SERVICE=WMS&VERSION=1.0.0&REQUEST=GetMap&FORMAT=image%2Fpng&TRANSPARENT=true&LAYERS=0&DPI=96&MAP_RESOLUTION=96&FORMAT_OPTIONS=dpi%3A96&WIDTH=256&HEIGHT=256&SRS=EPSG%3A3857&STYLES=&BBOX={bbox3857}",
elements: [
{
type: "input",
attributes: {
type: "checkbox",
id: "cbShowBalticMapsOrto8Layer",
title: "Baltic Maps Ortofoto 8th cycle (2022-2024)",
checked: _settings.showBalticMapsOrto8Layer,
boundSetting: "showBalticMapsLayer",
layerId: "balticmaps-orto8",
},
events: {
change: cbLayerChange,
},
triggerChangeEvent: false,
},
{
type: "label",
attributes: {
for: "cbShowBalticMapsOrto8Layer",
textContent: "Baltic Maps Ortofoto 8th cycle",
},
},
],
},
];
unsafeWindow.SDK_INITIALIZED.then(initScript);
function makeGetRequest(sURL) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "GET",
url: sURL,
onload: (response) => resolve(response.responseText),
onerror: (error) => reject(error),
});
});
}
function initScript() {
if (!unsafeWindow.getWmeSdk) {
throw new Error("SDK not available");
}
wmeSDK = unsafeWindow.getWmeSdk({
scriptId: "wmeUTLayers",
scriptName: "UrSuS Tools: Layers",
});
const sStorageItem = localStorage.getItem(_SETTINGS_STORAGE_NAME);
if (sStorageItem) {
const loadedSettings = JSON.parse(sStorageItem);
_settings = { ...defaultSettings, ...loadedSettings };
}
webLocationTilesContainerId =
"#" + W.map.getLayerByUniqueName("satellite_imagery").id;
void fetchBalticMapKey();
void initScriptTab();
}
async function saveSettingsToStorage() {
if (localStorage) {
_settings.lastSaved = Date.now();
localStorage.setItem(_SETTINGS_STORAGE_NAME, JSON.stringify(_settings));
}
}
async function fetchBalticMapKey() {
const sURL = "https://balticmaps.eu/backend/public/v1/key";
const sKeyResponseJSON = await makeGetRequest(sURL);
const mBalticMapsKey = JSON.parse(sKeyResponseJSON);
sBalticMapsKey = mBalticMapsKey.key;
}
async function initScriptTab() {
const { tabLabel, tabPane } = await wmeSDK.Sidebar.registerScriptTab();
tabLabel.innerText = "WME UT Layers";
tabLabel.title = "WME UT Layers";
const description = document.createElement("p");
description.style.fontWeight = "bold";
description.textContent = "WME UT Layers";
tabPane.appendChild(description);
let buttonContainer = createElem("div", {
class: "controls-container",
});
generateDomElements(aLayersElementsConfig, buttonContainer);
tabPane.appendChild(buttonContainer);
}
function generateDomElements(aLayersConfig, oDOMContainer) {
aLayersConfig.forEach((mLayerConfig) => {
mLayerConfig.elements.forEach((item) => {
const $HTMLElement = createElem(
item.type,
item.attributes,
item.events || []
);
oDOMContainer.appendChild($HTMLElement);
if (item.triggerChangeEvent) {
$HTMLElement.dispatchEvent(new Event("change"));
}
});
console.log(
`Elements for '${mLayerConfig.id}' created. URL: ${mLayerConfig.url}`
);
const br = createElem("br");
oDOMContainer.append(br);
});
}
function createElem(type, attrs = {}, eventListener = {}) {
const element = document.createElement(type);
for (const [key, value] of Object.entries(attrs)) {
if (key in element) {
element[key] = value;
} else {
element.setAttribute(key, value);
}
}
for (const [eventType, handler] of Object.entries(eventListener)) {
if (handler) {
element.addEventListener(eventType, handler);
}
}
return element;
}
function CheckUnCheckCustomWMSLayer(checked, sLayerName) {
checked ? setWMSLayer(sLayerName) : unsetWMSLayer();
}
function setWMSLayer(sLayerName) {
setDefaultImages();
currentOverlay = sLayerName;
setImages();
wmeSDK.Events.on({
eventName: "wme-map-move-end",
eventHandler: setImages,
});
}
function unsetWMSLayer() {
setDefaultImages();
wmeSDK.Events.off({
eventName: "wme-map-move-end",
eventHandler: setImages,
});
}
function setDefaultImages() {
var innerTilesContainer = $(webLocationTilesContainerId);
$(".overlayUrSuS", innerTilesContainer).remove();
}
function setImages() {
setImagesWaze();
}
function setImagesWaze() {
var innerTilesContainer = $(webLocationTilesContainerId);
innerTilesContainer.children("img.overlayUrSuS").each(function () {
var default_url = $(this).attr("data-default_url");
var original = innerTilesContainer.find(
'img.olTileImage[src="' + default_url + '"]'
);
if (original.length == 0) {
$(this).remove();
}
});
innerTilesContainer.children("img.olTileImage").each(function () {
var content = $(this);
var coords = getCoordinatesWaze(content);
if (undefined != coords) {
var duplicate = innerTilesContainer.find(
'img.overlayUrSuS[data-coords="' + coords + '"]'
);
if (duplicate.length == 0) {
duplicate = $(
'<img src="" draggable="false" style="width: 256px; height: 256px; position: absolute; border: 0px; padding: 0px; margin: 0px; -webkit-user-select: none;">'
);
duplicate.css("opacity", currentOpacity / 100);
duplicate.addClass("overlayUrSuS");
}
if (!duplicate.is(content.next())) {
const sUrl = content.attr("src");
if (sUrl) {
duplicate.attr("data-default_url", sUrl);
}
duplicate.attr("data-coords", coords);
duplicate.css("top", content.css("top"));
duplicate.css("left", content.css("left"));
duplicate.insertAfter(content);
replaceImage(duplicate);
}
}
});
}
function replaceImage(element) {
var quadLetters = element.data("coords");
if (!quadLetters) return;
element.attr("src", getUrl(quadLetters));
}
function getUrl(quadLetters) {
const sLatestBingImageVersion = "14801";
const mLayerConfig = aLayersElementsConfig.find(
(config) => config.id === currentOverlay
);
if (!mLayerConfig) {
console.error(
`Overlay configuration with id '${currentOverlay}' not found.`
);
return null;
}
const mQuadDigits = QuadLettersToQuadDigits(quadLetters);
const mTiles = QuadDigitsToTileXYZ(mQuadDigits);
const mBBox = TileBounds(mTiles.x, mTiles.y, mTiles.z);
QuadLettersToQuadDigits(quadLetters);
let sUrl = mLayerConfig.url;
const sBBox = `${mBBox.p2.y},${mBBox.p1.x},${mBBox.p1.y},${mBBox.p2.x}`;
const sBBox3857 = getBBox3857(mBBox);
if (sUrl.includes("{bbox}")) {
sUrl = sUrl.replace("{bbox}", sBBox);
} else if (sUrl.includes("{bbox3857}")) {
sUrl = sUrl.replace("{bbox3857}", sBBox3857);
} else if (sUrl.includes("{quadDigits}")) {
sUrl = sUrl.replace("{quadDigits}", mQuadDigits);
} else {
sUrl = sUrl.replace("{x}", mTiles.x.toString());
sUrl = sUrl.replace("{y}", mTiles.y.toString());
sUrl = sUrl.replace("{zoom}", mTiles.z.toString());
}
if (sUrl.includes("{latestBingImageVersion}")) {
sUrl = sUrl.replace("{latestBingImageVersion}", sLatestBingImageVersion);
}
if (sUrl.includes("{balticMapsKey}")) {
sUrl = sUrl.replace("{balticMapsKey}", sBalticMapsKey);
}
return sUrl;
}
function TileBounds(tx, ty, zoom) {
var p = Math.pow(2, zoom);
return {
p1: {
x: (tx * 360) / p - 180,
y:
90 -
(360 * Math.atan(Math.exp(-(0.5 - ty / p) * 2 * Math.PI))) / Math.PI,
},
p2: {
x: ((tx + 1) * 360) / p - 180,
y:
90 -
(360 * Math.atan(Math.exp(-(0.5 - (ty + 1) / p) * 2 * Math.PI))) /
Math.PI,
},
};
}
function getBBox3857(mBBox) {
const aConvertedCoordinates = [
...proj4("EPSG:4326", "EPSG:900913", [mBBox.p1.x, mBBox.p2.y]),
...proj4("EPSG:4326", "EPSG:900913", [mBBox.p2.x, mBBox.p1.y]),
];
const string = aConvertedCoordinates.join();
return string;
}
function QuadDigitsToTileXYZ(quadKey) {
let x = 0;
let y = 0;
let z = quadKey.length;
for (var i = z; i > 0; i--) {
var digit = quadKey[z - i];
var mask = 1 << (i - 1);
if (digit == "0") {
continue;
} else if (digit == "1") {
x |= mask;
} else if (digit == "2") {
y |= mask;
} else if (digit == "3") {
x |= mask;
y |= mask;
}
}
return {
x: x,
y: y,
z: z,
};
}
function QuadLettersToQuadDigits(quadKey) {
var quadDigits = "";
for (var i = 1; i < quadKey.length; i++) {
switch (quadKey[i]) {
case "q":
quadDigits += "0";
break;
case "r":
quadDigits += "1";
break;
case "t":
quadDigits += "2";
break;
case "s":
quadDigits += "3";
break;
}
}
return quadDigits;
}
function getCoordinatesWaze(img) {
var satelliteTileUrl = img.attr("src");
var coords;
if (!satelliteTileUrl) {
return;
}
var pattern = new RegExp("\\/\\/www\\.googleapis\\.com\\/tile\\/v1", "g");
if (pattern.test(satelliteTileUrl)) {
const RegExp =
/\/\/www\.googleapis\.com\/tile\/v1\/tiles\/(\d+)\/(\d+)\/(\d+)/g;
const match = RegExp.exec(satelliteTileUrl);
if (match) {
coords = QuadDigitsToQuadLetters(
TileXYZToQuadDigits(match[2], match[3], parseFloat(match[1]))
);
}
}
return coords;
}
function QuadDigitsToQuadLetters(quadKey) {
var quadLetters = "t";
for (var i = 0; i < quadKey.length; i++) {
switch (quadKey[i]) {
case "0":
quadLetters += "q";
break;
case "1":
quadLetters += "r";
break;
case "2":
quadLetters += "t";
break;
case "3":
quadLetters += "s";
break;
}
}
return quadLetters;
}
function TileXYZToQuadDigits(tileX, tileY, zoom) {
var quadKey = "";
for (var i = zoom; i > 0; i--) {
var digit = 0;
var mask = 1 << (i - 1);
if ((tileX & mask) != 0) {
digit++;
}
if ((tileY & mask) != 0) {
digit++;
digit++;
}
quadKey = quadKey.concat(digit.toString());
}
return quadKey;
}
})(proj4);