Shortcuts for multiple WME functions
// ==UserScript==
// @name WME Shortcuts+
// @description Shortcuts for multiple WME functions
// @icon 
// @author N190392
// @match https://*.waze.com/*editor*
// @version 0.5
// @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @namespace https://greasyfork.org/en/scripts/517140-wme-shortcuts
// @grant GM_xmlhttpRequest
// @license MIT
// ==/UserScript==
/* global W */
/* global WazeWrap */
(function() {
'use strict';
const SCRIPT_NAME = 'WME Shortcuts+';
const SCRIPT_VERSION = '0.5';
const UPDATE_MESSAGE = '';
const FORUM_URL = '';
const DOWNLOAD_URL = '';
const EXTERNAL_SETTINGS_NAME = 'WME_SHORTCUTS_PLUS_SETTINGS';
// *** 1 ADD SHORTCUTS HERE *** //
const EXTERNAL_SETTINGS = {
pressPieEditGeomShortcut: '',
clickOtherPointShortcut: '',
toggleRankShortcut: '',
clickOtherPolygonShortcut: '',
highlightSmallVenuesShortcut: '',
wmeutkadastrsShortcut: ''
};
// *** 2 FUNCTIONS *** //
// Function to create Other point
function clickOtherPoint() {
notifyAction("Creating 'Other' point");
const menuItems = document.querySelectorAll('wz-menu-item');
let pointButtonFound = false;
menuItems.forEach(item => {
const label = item.querySelector('.itemLabel--kXZjU');
if (label && ["Cits", "Other"].includes(label.textContent.trim())) {
const pointButton = item.querySelector('wz-button.point');
if (pointButton) {
pointButton.click();
console.log("WME Shortcuts+: Other (point) activated successfully.");
pointButtonFound = true;
}
}
});
if (!pointButtonFound) {
console.log("WME Shortcuts+: 'Other' button not found.");
}
}
// Function to create Other polygon
function clickOtherPolygon() {
notifyAction("Creating 'Other' polygon");
const menuItems = document.querySelectorAll('wz-menu-item');
let PolygonButtonFound = false;
menuItems.forEach(item => {
const label = item.querySelector('.itemLabel--kXZjU');
if (label && ["Cits", "Other"].includes(label.textContent.trim())) {
const PolygonButton = item.querySelector('wz-button.polygon');
if (PolygonButton) {
PolygonButton.click();
console.log("WME Shortcuts+: Other (polygon) activated successfully.");
PolygonButtonFound = true;
}
}
});
if (!PolygonButtonFound) {
console.log("WME Shortcuts+: 'Other' button not found.");
}
}
// Function to toggle rank to 2 and transoform point to polygon
function toggleRank() {
notifyAction("Rank 2 & Polygon");
const checkableChip = document.querySelector('#lockRank-1');
if (checkableChip) {
checkableChip.click();
console.log("WME Shortcuts+: Local rank changed to 2.");
}
const poligonsChip = Array.from(document.querySelectorAll('wz-checkable-chip'))
.find(chip => ['Poligons', 'Area'].some(text => chip.textContent.includes(text)));
if (poligonsChip) {
poligonsChip.click();
console.log("WME Shortcuts+: Point transformed to polygon");
}
}
// Function to open PIE Geometry settings
function pressPieEditGeomButton() {
notifyAction("PIE Geometry settings");
const button = document.querySelector('#pieEditGeom');
if (button) {
button.click();
console.log("WME Shortcuts+: Opened Geometry menu.");
}
}
// Function to highlight a place with specified colors
function highlightSmallVenues() {
notifyAction("Highlight venues < 500 m²");
//console.log("Starting small venues scan...");
function highlightAPlace(venue, fg, bg) {
var poly = W.userscripts.getFeatureElementByDataModel(venue);
if (poly) {
if (venue.isPoint()) {
poly.setAttribute("fill", fg);
} else { // area (polygon)
poly.setAttribute("stroke", fg);
poly.setAttribute("fill", bg);
poly.setAttribute("stroke-width", 4);
}
}
}
// Iterate over all venues
for (var mark in W.model.venues.objects) {
var venue = W.model.venues.getObjectById(mark);
if (!venue) continue;
if (venue.attributes) {
try {
// Try to get geometry directly from the venue
var metersArea = venue.getOLGeometry().getGeodesicArea(W.map.getProjectionObject());
// If area is less than 500m², highlight in red
if (metersArea < 500) {
highlightAPlace(venue, "#f00", "#f99"); // Red stroke, light red fill
const message = venue.attributes.name +": " + metersArea.toFixed(2) + " m²";
showToast(message);
}
} catch(e) {
// Skip venues that cause errors (like points)
continue;
}
}
}
}
// Function WME UT Kadastrs button
function wmeutkadastrs() {
notifyAction("WME UT Kadastrs");
//console.log("WME UT Kadastrs");
const wmeutbutton = document.querySelector('.overlay-buttons-container.top wz-button.overlay-button h6.w-icon');
if (wmeutbutton) {
wmeutbutton.click();
console.log("WME UT Kadastrs");
}
}
// *** 3 SHORTCUT MENU *** //
// WazeWrap Shortcut setup functions
function addShortcuts() {
new WazeWrap.Interface.Shortcut(
'clickOtherPoint',
'Cits: Izveidot punktu',
'shortcuts_plus',
'WME Shortcuts+',
EXTERNAL_SETTINGS.clickOtherPointShortcut,
clickOtherPoint,
null
).add();
new WazeWrap.Interface.Shortcut(
'clickOtherPolygon',
'Cits: Izveidot poligonu',
'shortcuts_plus',
'WME Shortcuts+',
EXTERNAL_SETTINGS.clickOtherPolygonShortcut,
clickOtherPolygon,
null
).add();
new WazeWrap.Interface.Shortcut(
'toggleRank',
'Slēgt ar līmeni (2) & pārveidot punktu par poligonu',
'shortcuts_plus',
'WME Shortcuts+',
EXTERNAL_SETTINGS.toggleRankShortcut,
toggleRank,
null
).add();
new WazeWrap.Interface.Shortcut(
'PressEditGeom',
'Atvērt ģeometrijas izvēlni',
'shortcuts_plus',
'WME Shortcuts+',
EXTERNAL_SETTINGS.pressPieEditGeomShortcut,
pressPieEditGeomButton,
null
).add();
new WazeWrap.Interface.Shortcut(
'highlightSmallVenues',
'Iezīmēt vietas, kas ir mazākas par 500m²',
'shortcuts_plus',
'WME Shortcuts+',
EXTERNAL_SETTINGS.highlightSmallVenuesShortcut,
highlightSmallVenues,
null
).add();
new WazeWrap.Interface.Shortcut(
'wmeutkadastrs',
'WME UT Kadastrs',
'shortcuts_plus',
'WME Shortcuts+',
EXTERNAL_SETTINGS.wmeutkadastrsShortcut,
wmeutkadastrs,
null
).add();
}
// *** 4 MAKE SHORTCUTS CHANGABLE *** //
function sandboxLoadSettings() {
const loadedSettings = JSON.parse(localStorage.getItem(EXTERNAL_SETTINGS_NAME)) || {};
EXTERNAL_SETTINGS.clickOtherPointShortcut = loadedSettings.clickOtherPointShortcut || 'C+1';
EXTERNAL_SETTINGS.clickOtherPolygonShortcut = loadedSettings.clickOtherPolygonShortcut || 'C+2';
EXTERNAL_SETTINGS.toggleRankShortcut = loadedSettings.toggleRankShortcut || 'A+2';
EXTERNAL_SETTINGS.pressPieEditGeomShortcut = loadedSettings.pressPieEditGeomShortcut || 'A+3';
EXTERNAL_SETTINGS.highlightSmallVenuesShortcut = loadedSettings.highlightSmallVenuesShortcut || 'A+1';
EXTERNAL_SETTINGS.wmeutkadastrsShortcut = loadedSettings.wmeutkadastrsShortcut || 'A+4';
addShortcuts();
$(window).on('beforeunload', () => sandboxSaveSettings());
}
function getShortcutKeys(shortcutAction) {
let keys = '';
const { shortcut } = shortcutAction;
if (shortcut) {
if (shortcut.altKey) keys += 'A';
if (shortcut.shiftKey) keys += 'S';
if (shortcut.ctrlKey) keys += 'C';
if (keys.length) keys += '+';
if (shortcut.keyCode) keys += shortcut.keyCode;
}
return keys;
}
function sandboxSaveSettings() {
EXTERNAL_SETTINGS.clickOtherPointShortcut = getShortcutKeys(W.accelerators.Actions.clickOtherPoint);
EXTERNAL_SETTINGS.toggleRankShortcut = getShortcutKeys(W.accelerators.Actions.toggleRank);
EXTERNAL_SETTINGS.pressPieEditGeomShortcut = getShortcutKeys(W.accelerators.Actions.PressEditGeom);
EXTERNAL_SETTINGS.clickOtherPolygonShortcut = getShortcutKeys(W.accelerators.Actions.clickOtherPolygon);
EXTERNAL_SETTINGS.highlightSmallVenuesShortcut = getShortcutKeys(W.accelerators.Actions.highlightSmallVenues); // Add this line
localStorage.setItem(EXTERNAL_SETTINGS_NAME, JSON.stringify(EXTERNAL_SETTINGS));
}
function loadScriptUpdateMonitor() {
let updateMonitor;
try {
updateMonitor = new WazeWrap.Alerts.ScriptUpdateMonitor(
SCRIPT_NAME,
SCRIPT_VERSION,
DOWNLOAD_URL,
GM_xmlhttpRequest
);
updateMonitor.start();
} catch (ex) {
console.error(`${SCRIPT_NAME}:`, ex);
}
}
function bootstrap() {
if (WazeWrap?.Ready) {
WazeWrap.Interface.ShowScriptUpdate(
SCRIPT_NAME,
SCRIPT_VERSION,
UPDATE_MESSAGE,
FORUM_URL
);
loadScriptUpdateMonitor();
sandboxLoadSettings();
} else {
setTimeout(bootstrap, 250);
}
}
let toastMessages = [];
let toastTimeout = null;
let toastElement = null; // Keep global reference to the toast
function showToast(message, options = {}) {
const {
duration = 4000,
position = 'bottom-left',
containerId = 'editor-container'
} = options;
const container = document.getElementById(containerId);
if (!container) return;
container.style.position = 'relative';
const positionStyles = {
'bottom-left': { left: '10px', bottom: '30px' },
'bottom-right': { right: '10px', bottom: '30px' },
'top-left': { left: '10px', top: '30px' },
'top-right': { right: '10px', top: '30px' }
};
if (!toastElement || !document.body.contains(toastElement)) {
toastElement = document.createElement('div');
toastElement.className = 'wme-toast';
Object.assign(toastElement.style, {
position: 'absolute',
background: '#333',
color: '#fff',
padding: '8px 12px',
borderRadius: '20px',
boxShadow: '0 0 10px rgba(0,0,0,0.3)',
zIndex: '9999',
fontFamily: 'Arial, sans-serif',
maxWidth: '300px',
...positionStyles[position] || positionStyles['bottom-left']
});
const list = document.createElement('ul');
list.id = 'wme-toast-list';
list.style.paddingLeft = '20px';
list.style.margin = '0';
list.style.whiteSpace = 'pre-line';
toastElement.appendChild(list);
container.appendChild(toastElement);
}
// Prevent duplicate messages
if (toastMessages.includes(message)) return;
toastMessages.push(message);
const list = toastElement.querySelector('#wme-toast-list');
const item = document.createElement('li');
item.textContent = message;
list.appendChild(item);
if (toastTimeout) clearTimeout(toastTimeout);
toastTimeout = setTimeout(() => {
toastElement.remove();
toastElement = null;
toastMessages = [];
toastTimeout = null;
}, duration);
}
let actionPopupTimeout = null;
let actionPopupElement = null;
function notifyAction(text, duration = 3000) {
if (actionPopupElement) {
actionPopupElement.remove();
clearTimeout(actionPopupTimeout);
}
actionPopupElement = document.createElement('div');
actionPopupElement.className = 'wme-action-popup';
Object.assign(actionPopupElement.style, {
position: 'fixed',
top: '40px',
right: '0px',
background: '#007acc',
color: '#fff',
padding: '5px 12px',
borderRadius: '4px',
boxShadow: '0 2px 6px rgba(0,0,0,0.3)',
fontFamily: 'Arial, sans-serif',
fontSize: '14px',
zIndex: '99999',
maxWidth: '300px',
whiteSpace: 'nowrap'
});
actionPopupElement.textContent = `✓ ${text}`;
document.body.appendChild(actionPopupElement);
actionPopupTimeout = setTimeout(() => {
actionPopupElement.remove();
actionPopupElement = null;
}, duration);
}
bootstrap();
})();