// ==UserScript==
// @name WME Speed Display
// @namespace https://gf.qytechs.cn/scripts/526702-wme-speed-display/
// @version 1.1.4
// @description Displays road speed directly in the center of the segment (taking curves into account) in Waze Map Editor
// @author Luan Tavares
// @icon 
// @license GPLv3
// @include https://www.waze.com/editor*
// @include https://www.waze.com/*/editor*
// @include https://beta.waze.com/*
// @exclude https://www.waze.com/user/editor*
// @grant none
// ==/UserScript==
/*
* @todo:
* Add a speed filter, where the user says from which speed he wants to see
* Add an opacity option for icons
*/
/* global W */
/* global I18n */
/* global OpenLayers */
class WmeSpeedDisplay {
constructor() {
this.version = '1.1.4';
this.layer = null;
this.debugMode = false;
this.segmentsCategories = {
highways: [
{
id: 3,
name: 'freeway'
},
{
id: 4,
name: 'ramp'
},
{
id: 6,
name: 'major_highway'
},
{
id: 7,
name: 'minor_highway'
}
],
non_drivable: [
{
id: 18,
name: 'railroad'
},
{
id: 19,
name: 'runway_taxiway'
},
{
id: 5,
name: 'walking_trail'
},
{
id: 10,
name: 'pedestrian_boardwalk'
},
{
id: 16,
name: 'stairway'
}
],
other_drivable: [
{
id: 8,
name: 'off_road_not_maintained'
},
{
id: 20,
name: 'parking_lot_road'
},
{
id: 17,
name: 'private_road'
},
{
id: 15,
name: 'ferry'
}
],
streets: [
{
id: 2,
name: 'primary_street'
},
{
id: 1,
name: 'street'
},
{
id: 22,
name: 'narrow_street'
}
]
};
this.spdTranslations = I18n.translations[I18n.locale].wmeSpd;
this.debounce = {
updateMapDisplay: this.debounce(this.updateMapDisplay.bind(this), 1000),
saveSettings: this.debounce(this.saveSettings.bind(this))
};
this.settings;
this.prepareAndInitialize();
}
/**
* Checks if WME is ready to initializes the plugin.
*/
prepareAndInitialize() {
this.defineInitialSettings();
this.defineTranslations();
if (W?.userscripts?.state?.isReady) {
this.logDebug(this.spdTranslations.log.wmeReadyStartScript);
this.initializePlugin();
} else {
this.logDebug(this.spdTranslations.log.waitingWme);
document.addEventListener('wme-ready', this.initializePlugin.bind(this), {once: true});
}
}
/**
* Initializes the plugin.
*/
async initializePlugin() {
this.logDebug(this.spdTranslations.log.wmeReadyStartingPlugin, true);
this.checkVersionAndShowChangelog();
this.layer = new OpenLayers.Layer.Vector('Speed Display Layer', {
displayInLayerSwitcher: false
});
W.map.addLayer(this.layer);
this.addLayerCheckbox();
this.addSettingsTab();
this.listen();
this.logDebug(this.spdTranslations.log.shortSleep);
await this.sleep(1000);
this.updateMapDisplay();
}
/**
* Method that listens to this script events.
*/
listen() {
W.map.events.register('moveend', null, this.onTriggeringMapEvent.bind(this));
W.map.events.register('zoomend', null, this.onTriggeringMapEvent.bind(this));
W.model.segments.on('objectschanged', this.onTriggeringMapEvent.bind(this));
}
/**
* When the map triggers an event, updates the map display.
*/
onTriggeringMapEvent() {
if (this.settings.spdEnabled)
this.debounce.updateMapDisplay();
}
/**
* Saves and updates the map display when WME SPD checkbox change.
* @param {Event} event
*/
onSpeedDisplayCheckboxChange(event) {
this.debounce.saveSettings();
this.debounce.updateMapDisplay();
this.toggleLayerVisibility(event.target.checked);
}
/**
* Saves the settings and updates the map display.
*/
onSettingsChange() {
this.debounce.saveSettings();
this.debounce.updateMapDisplay();
}
/**
* Updates the max zoom value display and triggers map update if the zoom level reaches the max.
* @param {Event} event
*/
onSettingZoomChanged(event) {
let zoomLevel = W.map.getZoom();
document.getElementById('spd-max-zoom-value').innerText = event.target.value;
this.debounce.saveSettings();
if (zoomLevel >= this.settings.general.spdMaxZoom)
this.debounce.updateMapDisplay();
}
/**
* Attaches event listeners to the settings elements to track user interactions and changes.
*/
attachSettingsListeners() {
['spd-show-no-speed', 'spd-show-roundabout'].forEach(id => {
document.getElementById(id).addEventListener('change', this.onSettingsChange.bind(this));
});
document.getElementById('spd-max-zoom').addEventListener('input', this.onSettingZoomChanged.bind(this));
document.querySelectorAll('[id^="spd-show-speed-in-"]').forEach(input => {
input.addEventListener('change', this.onSettingsChange.bind(this));
});
if (this.debugMode) {
document.getElementById('btn-spd-update-settings-tab').addEventListener('click', this.updateSettingsTab.bind(this));
document.getElementById('btn-spd-show-modal').addEventListener('click', this.showSpdDialog.bind(this));
}
}
/**
* Detaches event listeners from the settings elements to stop tracking user interactions.
*/
detachSettingsListeners() {
['spd-show-no-speed', 'spd-show-roundabout'].forEach(id => {
document.getElementById(id).removeEventListener('change', this.onSettingsChange.bind(this));
});
document.getElementById('spd-max-zoom').removeEventListener('input', this.onSettingZoomChanged.bind(this));
document.querySelectorAll('[id^="spd-show-speed-in-"]').forEach(input => {
input.removeEventListener('change', this.onSettingsChange.bind(this));
});
if (this.debugMode) {
document.getElementById('btn-spd-update-settings-tab').removeEventListener('click', this.updateSettingsTab.bind(this));
document.getElementById('btn-spd-show-modal').removeEventListener('click', this.showSpdDialog.bind(this));
}
}
/**
* Attaches listeners to close the dialog.
*/
attachDialogListeners() {
document.addEventListener('keydown', this.simulateBtnHideDialogClick.bind(this));
document.querySelector('#spd-dialog-container .dark-overlay').addEventListener('click', this.simulateBtnHideDialogClick.bind(this), {once: true});
document.getElementById('btn-spd-hide-dialog').addEventListener('click', this.hideSpdDialog.bind(this), {once: true});
}
/**
* Updates the map display by removing old features and adding new speed icons based on the settings.
*/
updateMapDisplay() {
this.logDebug(this.spdTranslations.log.updateMapDisplay);
let zoomLevel = W.map.getZoom();
this.layer.removeAllFeatures();
if (zoomLevel < this.settings.general.spdMaxZoom) {
this.logDebug(this.spdTranslations.log.maxZoomExceeded);
return;
}
let segments = W.model.segments.objects || {};
let segmentKeys = Object.keys(segments);
if (segmentKeys.length === 0) {
this.logDebug(this.spdTranslations.log.noneSegment);
return;
}
this.logDebug(`${this.spdTranslations.log.numSegmentsLoaded} ${segmentKeys.length}`);
segmentKeys.forEach(segmentId => {
let segment = segments[segmentId];
if (!segment || !segment.attributes) {
this.logDebug(`${this.spdTranslations.log.attributesNotFound} ${segmentId}.`);
return;
}
let attributes = segment.attributes;
let roadSettingId = this.convertStringCaseStyle('snake', 'camel', `spd_show_speed_in_${this.getRoadSettingNameById(attributes.roadType)}`);
let speedFwd = attributes.fwdMaxSpeed || 'N/A';
let speedRev = attributes.revMaxSpeed || 'N/A';
let isFwd = attributes.fwdDirection;
let isRev = attributes.revDirection;
let hideNoSpeed = !this.settings.general.spdShowNoSpeed && ((speedFwd == 'N/A' && speedRev == 'N/A') || (isFwd && !isRev && speedFwd == 'N/A') || (!isFwd && isRev && speedRev == 'N/A'));
let ignoreOnRoundabout = attributes.junctionID && !this.settings.general.spdShowRoundabout;
if (hideNoSpeed || !this.settings.roads[roadSettingId] || ignoreOnRoundabout) {
this.logDebug(this.spdTranslations.log.disallowedSegment);
return;
}
let geometry = segment.getOLGeometry();
if (!geometry || geometry.components.length < 2) {
this.logDebug(`${this.spdTranslations.log.invalidGeometry} ${segmentId}.`);
return;
}
let midpoint = this.calculateMidpoint(geometry);
let feature = new OpenLayers.Feature.Vector(midpoint, {
segmentId,
speedFwd,
speedRev
});
let graphicWidth;
let graphicXOffset;
if (isFwd && isRev && speedFwd != speedRev) {
graphicWidth = 70;
graphicXOffset = -30;
} else if (isFwd && speedFwd != 'N/A' || isRev && speedRev != 'N/A') {
graphicWidth = 30;
graphicXOffset = -15;
}
feature.style = {
graphic: true,
externalGraphic: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speedFwd, speedRev, isFwd, isRev)),
graphicHeight: 30,
graphicWidth: graphicWidth,
graphicYOffset: -15,
graphicXOffset: graphicXOffset
};
this.layer.addFeatures([feature]);
});
this.logDebug(this.spdTranslations.log.updateMapDisplayFinished);
}
/**
* Toggle visibility of the display of the speed layer.
* @param {Boolean} enabled
*/
toggleLayerVisibility(enabled) {
if (this.layer) {
this.layer.setVisibility(enabled);
this.logDebug(`${this.spdTranslations.log.speedLayer} ${enabled ? this.spdTranslations.enabled : this.spdTranslations.disabled}.`);
}
}
/**
* Saves the current settings to localStorage.
*/
saveSettings() {
this.settings.spdEnabled = document.getElementById('layer-switcher-item_speed_display').checked;
Object.keys(this.settings.general).forEach(setting => {
let element = document.getElementById(this.convertStringCaseStyle('camel', 'kebab', setting));
this.settings.general[setting] = setting != 'spdMaxZoom' ? element.checked : element.value;
});
document.querySelectorAll('[id^="spd-show-speed-in-"]').forEach(input => {
let settingId = this.convertStringCaseStyle('kebab', 'camel', input.id);
this.settings.roads[settingId] = input.checked;
});
localStorage.setItem('wmeSpeedDisplaySettings', JSON.stringify(this.settings));
}
/**
* Loads the settings from localStorage or uses the provided settings to initialize the configuration.
* @param {Object} [savedSettings] - The saved settings object to load; if not provided, it will load from localStorage.
*/
loadSettings(savedSettings) {
savedSettings = savedSettings ?? JSON.parse(localStorage.getItem('wmeSpeedDisplaySettings'));
if (savedSettings)
this.settings = savedSettings;
}
/**
* Adds a settings tab to the user interface with options to configure the speed display settings.
*/
addSettingsTab() {
let scriptTabContainer = document.querySelector('#user-info .nav-tabs');
if (!scriptTabContainer) {
this.logDebug(this.spdTranslations.log.scriptsTabNotFound);
setTimeout(() => this.addSettingsTab(), 1000);
return;
}
if (document.getElementById('wme-spd-tab'))
return;
let tabButton = document.createElement('li');
tabButton.innerHTML = `<a href="#spd-settings" data-toggle="tab">${this.spdTranslations.name}</a>`;
scriptTabContainer.appendChild(tabButton);
let tabContentContainer = document.querySelector('.tab-content');
let userScriptsApiDocsLinkContainer = tabContentContainer.querySelector('.userscripts-api-docs-link-container');
let tabContent = document.createElement('div');
tabContent.id = 'spd-settings';
tabContent.classList.add('tab-pane');
let tabContentHtml = `
<div style="padding: 0 10px;">
<h4>${this.spdTranslations.settingTabName}</h4>
<hr>
<wz-label html-for="">${this.spdTranslations.title.general}</wz-label>
<wz-checkbox checked="${this.settings.general.spdShowNoSpeed}" indeterminate="false" disabled="false" id="spd-show-no-speed" value="true">${this.spdTranslations.label.showNoSpeed}<input type="checkbox" value="true" style="display: none; visibility: hidden;"></wz-checkbox>
<wz-checkbox checked="${this.settings.general.spdShowRoundabout}" indeterminate="false" disabled="false" id="spd-show-roundabout" value="true">${this.spdTranslations.label.showOnRoundabout}<input type="checkbox" value="true" style="display: none; visibility: hidden;"></wz-checkbox>
<br>
<wz-label html-for="" style="margin-top:10px">${this.spdTranslations.label.maxZoom} <span id="spd-max-zoom-value">${this.settings.general.spdMaxZoom}</span></wz-label>
<input type="range" id="spd-max-zoom" min="12" max="22" step="1" value="${this.settings.general.spdMaxZoom}">
<div style="display: flex; justify-content: space-between; width: 100%; color: #777; font-size: 11px; line-height: 15px;"><span>${this.spdTranslations.zoomOut}</span><span>${this.spdTranslations.zoomIn}</span></div>
<br>
<wz-label html-for="" style="margin:0">${this.spdTranslations.title.showOnRoadType}</wz-label>`;
Object.entries(this.segmentsCategories).forEach(segmentCategory => {
tabContentHtml += `<wz-menu-title style="padding:0;">${I18n.translations[I18n.locale].segment.categories[segmentCategory[0]]}</wz-menu-title>`;
Object.values(segmentCategory[1]).forEach(roadType => {
let id = this.convertStringCaseStyle('snake', 'kebab', `spd_show_speed_in_${roadType.name}`);
let settingId = this.convertStringCaseStyle('kebab', 'camel', id);
let checked = this.settings.roads[settingId];
tabContentHtml += `<wz-checkbox checked="${checked}" indeterminate="false" disabled="false" id="${id}" value="true">${I18n.translations[I18n.locale].segment.road_types[roadType.id]}<input type="checkbox" value="true" style="display: none; visibility: hidden;"></wz-checkbox>`;
});
});
if (this.debugMode) {
tabContentHtml += `<wz-button color="primary" id="btn-spd-update-settings-tab" style="margin-top: 10px; width: 100%;">${this.spdTranslations.btn.updateSettingsTab}</wz-button>`;
tabContentHtml += `<wz-button color="secondary" id="btn-spd-show-modal" style="margin-top: 10px; width: 100%;">Exibir modal</wz-button>`;
}
tabContentHtml += `</div>`;
tabContent.innerHTML = tabContentHtml;
tabContentContainer.insertBefore(tabContent, userScriptsApiDocsLinkContainer);
this.attachSettingsListeners();
this.logDebug(this.spdTranslations.log.settingsTabAdded);
}
/**
* Updates the settings tab by removing the old one and adding a new one with updated settings.
*/
updateSettingsTab() {
this.logDebug(this.spdTranslations.log.updatingSettingsTab);
let linkTab = document.querySelector('#user-info .nav-tabs li a[href="#spd-settings"]');
let tabContent = document.getElementById('spd-settings');
if (!linkTab || !tabContent)
return;
this.detachSettingsListeners();
this.defineTranslations();
tabContent.remove();
linkTab.closest('li').remove();
this.addSettingsTab();
this.updateSettingsElementsStates();
}
/**
* Updates the state of the settings elements (checkboxes, inputs) based on the current settings.
*/
updateSettingsElementsStates() {
Object.entries(this.settings.general).forEach(([setting, value]) => {
let id = this.convertStringCaseStyle('camel', 'kebab', setting);
let element = document.getElementById(id);
if (!element)
return;
if (setting != 'spdMaxZoom') {
element.checked = value;
} else {
element.value = value;
document.getElementById('spd-max-zoom-value').innerText = value;
}
});
document.querySelectorAll('[id^="spd-show-speed-in-"]').forEach(input => {
let settingId = this.convertStringCaseStyle('kebab', 'camel', input.id);
if (input.type == 'checkbox')
input.checked = this.settings.roads[settingId];
});
}
/**
* Calculates the real midpoint of a segment, considering curves.
* @param {OpenLayers.Geometry.LineString} geometry Segment geometry.
*
* @returns {OpenLayers.Geometry.Point} Real midpoint.
*/
calculateMidpoint(geometry) {
let length = geometry.getLength();
let cumulativeLength = 0;
for (let i = 0; i < geometry.components.length - 1; i++) {
let start = geometry.components[i];
let end = geometry.components[i + 1];
let segmentLength = start.distanceTo(end);
if (cumulativeLength + segmentLength >= length / 2) {
let ratio = (length / 2 - cumulativeLength) / segmentLength;
return new OpenLayers.Geometry.Point(
start.x + ratio * (end.x - start.x),
start.y + ratio * (end.y - start.y)
);
}
cumulativeLength += segmentLength;
}
return geometry.getCentroid();
}
/**
* Generates an SVG icon with the speeds.
* @param {String} speedFwd Forward speed.
* @param {String} speedRev Reverse speed.
* @param {String} isFwd Direction of the road forward.
* @param {String} isRev Reverse lane direction.
*
* @returns {String} SVG in string format.
*/
getSpeedIcon(speedFwd, speedRev, isFwd, isRev) {
if (isFwd && isRev && speedFwd != speedRev) {
return `
<svg xmlns="http://www.w3.org/2000/svg" width="120" height="50" viewBox="0 0 120 50">
<circle cx="25" cy="25" r="21" fill="white" stroke="red" stroke-width="5"/>
<text x="25" y="31" font-size="20" font-family="Arial" font-weight="bold" fill="black" text-anchor="middle">${speedFwd}</text>
<circle cx="80" cy="25" r="21" fill="white" stroke="red" stroke-width="5"/>
<text x="80" y="31" font-size="20" font-family="Arial" font-weight="bold" fill="black" text-anchor="middle">${speedRev}</text>
</svg>`;
} else {
return `
<svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 50 50">
<circle cx="25" cy="25" r="21" fill="white" stroke="red" stroke-width="5"/>
<text x="25" y="31" font-size="20" font-family="Arial" font-weight="bold" fill="black" text-anchor="middle">${isFwd ? speedFwd : speedRev}</text>
</svg>`;
}
}
/**
* Makes a "dramatic" pause in the code.
* @param {Number} ms Pause time in milliseconds
*/
async sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
/**
* Returns a function with delay.
* @param {Object} func Function that will be called with delay
* @param {(Number|null)} timeout Delay time in milliseconds
*
* @returns Function
*/
debounce(func, timeout = 300) {
let timer;
return (...args) => {
clearTimeout(timer);
timer = setTimeout(() => {
func.apply(this, args);
}, timeout);
}
}
/**
* Converts a string from one case style to another (camel, snake, kebab).
* @param {string} fromType - The original string case ('camel', 'snake', or 'kebab').
* @param {string} toType - The target string case ('camel', 'snake', or 'kebab').
* @param {string} string - The string to be converted.
*
* @returns {string} The converted string.
*/
convertStringCaseStyle(fromType, toType, string) {
if (fromType == toType)
return string;
let conversionMap = {
camel: {
kebab: str => str.match(/[A-Z]?[a-z]+|[0-9]+/g).join('-').toLowerCase(),
snake: str => str.match(/[A-Z]?[a-z]+|[0-9]+/g).join('_').toLowerCase()
},
snake: {
kebab: str => str.replace(/_/g, '-'),
camel: str => str.replace(/_([a-z])/g, (_, letra) => letra.toUpperCase())
},
kebab: {
camel: str => str.replace(/-([a-z])/g, (_, letra) => letra.toUpperCase()),
snake: str => str.replace(/-/g, '_')
}
};
return conversionMap[fromType]?.[toType]?.(string) || string;
}
/**
* Retrieves the road setting name by its ID from the segments categories.
* @param {string} id - The ID of the road setting.
*
* @returns {string|null} The name of the road setting if found, otherwise null.
*/
getRoadSettingNameById(id) {
for (let category of Object.values(this.segmentsCategories)) {
let segment = category.find(item => item.id === id);
if (segment)
return segment.name;
}
// Returns null if ID is not found
return null;
}
/**
* Logs a debug message to the console if debug mode is enabled or forceMessage is true.
* @param {string} message - The message to log.
* @param {boolean} [forceMessage] - Whether to log the message regardless of the debug mode setting.
*/
logDebug(message, forceMessage = false) {
if (this.debugMode || forceMessage)
console.log(`[WME Speed Display]: ${message}`);
}
/**
* Adds a checkbox to the layer switcher to toggle the speed display layer visibility.
*/
addLayerCheckbox() {
let houseNumbersCheckbox = document.getElementById('layer-switcher-item_house_numbers');
if (!houseNumbersCheckbox) {
this.logDebug(this.spdTranslations.log.houseNumberCheckboxNotFound);
setTimeout(() => this.addLayerCheckbox(), 1000);
return;
}
let layerItem = document.createElement('li');
layerItem.innerHTML = `
<div class="layer-selector">
<wz-checkbox id="layer-switcher-item_speed_display" checked="${this.settings.spdEnabled}">
<div class="layer-selector-container" title="WME ${this.spdTranslations.name}">${this.spdTranslations.label.displaySpeeds}</div>
</wz-checkbox>
</div>`;
houseNumbersCheckbox.closest('li').insertAdjacentElement('afterend', layerItem);
let switchElement = document.getElementById('layer-switcher-item_speed_display');
if (!switchElement)
return;
switchElement.addEventListener('change', this.onSpeedDisplayCheckboxChange.bind(this));
this.toggleLayerVisibility(this.settings.spdEnabled);
}
/**
* Initializes settings from localStorage or sets default values.
*/
defineInitialSettings() {
let localSettings = localStorage.getItem('wmeSpeedDisplaySettings');
if (localSettings != null) {
this.loadSettings(JSON.parse(localSettings));
this.updateSettingsElementsStates();
} else {
this.settings = {
spdEnabled: true,
general: {
spdShowNoSpeed: true,
spdShowRoundabout: true,
spdMaxZoom: 18
},
roads: {}
};
Object.values(this.segmentsCategories).forEach(category => {
Object.values(category).forEach(roadType => {
let settingId = this.convertStringCaseStyle('snake', 'camel', `spd_show_speed_in_${roadType.name}`);
this.settings.roads[settingId] = true;
});
});
}
}
/**
* Checks if the plugin version changed and displays the changelog if necessary.
*/
checkVersionAndShowChangelog() {
this.logDebug(this.spdTranslations.log.checkingVersion);
let pluginChangelog = JSON.parse(localStorage.getItem('wmeSpeedDisplayVersion'));
if (pluginChangelog == null || pluginChangelog.version != this.version) {
let spdDialog = document.getElementById('spd-dialog-container');
spdDialog?.remove();
this.showSpdDialog(`${this.spdTranslations.changelog} ${this.spdTranslations.name} v${this.version}`, this.spdTranslations.changelogMessage);
localStorage.setItem('wmeSpeedDisplayVersion', JSON.stringify(this.version));
}
}
/**
* Displays a modal dialog with customizable title, body content, and button text.
* @param {string} title - The title of the modal (default: 'Without title').
* @param {string} body - The HTML content or string inside the modal (default: '<span>Without body message</span>').
* @param {string} [buttonText] - The text for the close button (default: 'Fechar').
*/
showSpdDialog(title = 'Without title', body = '<span>Without body message</span>', buttonText = 'Fechar') {
let wzDialogContainer = document.getElementById('wz-dialog-container');
let divDialog = document.createElement('div');
divDialog.id = 'spd-dialog-container';
Object.assign(divDialog.style, {
display: 'flex',
alignItems: 'center',
justifyContent: 'center',
position: 'absolute',
width: '100vw',
height: '100vh',
top: 0,
left: 0
});
divDialog.innerHTML = `
<div class="dark-overlay" style="background-color: var(--background_modal, rgba(32, 33, 36, 0.6)); width: 100vw; height: 100vh; position: absolute; top: 0px; left: 0px; z-index: 9999;"></div>
<div style="display: flex; flex-direction: column; background-color: var(--background_default, #ffffff); border-radius: var(--wz-dialog-border-radius, 10px); box-shadow: rgba(0, 0, 0, 0.26) 0px 4px 8px 3px; width: var(--wz-dialog-width, 720px); height: var(--wz-dialog-height); padding: var(--wz-dialog-padding, var(--space-l, 24px)); position: fixed; z-index: 99999;">
<h3 style="margin-bottom: 10px;">${title}</h3>
<div style="max-height: 70vh; overflow: auto;">
${body}
</div>
<div style="display: flex; justify-content: end; margin-top: 10px;">
<wz-button color="secondary" id="btn-spd-hide-dialog">${buttonText}</wz-button>
</div>
</div>`;
wzDialogContainer.appendChild(divDialog);
this.attachDialogListeners();
}
/**
* Closes the dialog and removes associated event listeners.
*/
hideSpdDialog() {
let spdDialog = document.getElementById('spd-dialog-container');
document.removeEventListener('keydown', this.simulateBtnHideDialogClick.bind(this));
spdDialog?.remove();
}
/**
* Simulates a click on the close dialog button.
* @param {Event} event
*/
simulateBtnHideDialogClick(event) {
if (event.key && (event.key != 'Escape' || event.key != 'Enter'))
return;
document.getElementById('btn-spd-hide-dialog')?.click();
}
/**
* Sets translations for the current locale.
*/
defineTranslations() {
switch (I18n.locale) {
case 'cs':
// Czech
this.spdTranslations = {
name: 'Zobrazení rychlosti',
settingTabName: 'Nastavení zobrazení rychlosti',
enabled: 'povolené',
disabled: 'zakázané',
zoomOut: 'Oddálit',
zoomIn: 'Přiblížit',
changelog: 'Seznam změn',
changelogMessage: '<h6>Novinky:</h6><ul style="margin-bottom: 10px"><li>Přidán modální okno changelogu.</li></ul><h6>Opravy:</h6><ul><li>Opraven problém s ikonami bez rychlosti, které se zobrazovaly, i když byla možnost zobrazit bez rychlosti deaktivována;</li><li>Další drobné opravy a vylepšení.</li></ul>',
title: {
general: 'Obecné',
showOnRoadType: 'Zobrazit na segmentech typu:'
},
label: {
displaySpeeds: 'Zobrazit rychlosti',
showNoSpeed: 'Zobrazit na segmentu bez rychlosti',
showOnRoundabout: 'Zobrazit na kruhovém objezdu',
maxZoom: 'Renderovat až do úrovně zoomu:'
},
btn: {
updateSettingsTab: 'Aktualizovat menu',
},
log: {
wmeReadyStartScript: 'WME načteno a připraveno. Spouštím skript...',
waitingWme: 'Čekám, až bude WME připraveno...',
wmeReadyStartingPlugin: 'WME je připraveno. Nastavuji vlastní vrstvu...',
shortSleep: 'Čekám asi 1 sekundu, než se vše načte...',
updateMapDisplay: 'Aktualizuji zobrazení na mapě...',
maxZoomExceeded: 'Byl překročen maximální zoom. Rychlost nebude zobrazena.',
noneSegment: 'Žádné segmenty nebyly nalezeny.',
numSegmentsLoaded: 'Celkový počet načtených segmentů:',
attributesNotFound: 'Nelze najít atributy nebo chybí v segmentu',
disallowedSegment: 'Tento typ segmentu by neměl být načten.',
invalidGeometry: 'Neplatná geometrie v segmentu:',
updateMapDisplayFinished: 'Aktualizace dokončena.',
houseNumberCheckboxNotFound: 'Checkbox "Čísla domů" nebyl nalezen. Pokouším se znovu...',
scriptsTabNotFound: 'Záložka skriptů nebyla nalezena. Pokouším se znovu...',
settingsTabAdded: 'Přidána karta nastavení.',
speedLayer: 'Vrstva rychlosti',
opacityUpdated: 'Průhlednost ikon byla nastavena na',
updatingSettingsTab: 'Aktualizuji záložku nastavení...',
languageDefined: 'Jazyky definovány.',
checkingVersion: 'Kontroluji verzi pluginu.',
}
};
break;
case 'es-419':
// Latin-american spanish
this.spdTranslations = {
name: 'Mostrar Velocidad',
settingTabName: 'Configuraciones de Mostrar Velocidad',
enabled: 'habilitada',
disabled: 'deshabilitada',
zoomOut: 'Alejar',
zoomIn: 'Acercar',
changelog: 'Registro de cambios',
changelogMessage: '<h6>Novedades:</h6><ul style="margin-bottom: 10px"><li>Se agregó un modal de registro de cambios.</li></ul><h6>Correcciones:</h6><ul><li>Se corrigió un problema con los íconos sin velocidad que aparecían incluso con la opción de mostrar sin velocidad desactivada;</li><li>Otras pequeñas correcciones y mejoras.</li></ul>',
title: {
general: 'General',
showOnRoadType: 'Mostrar en los segmentos del tipo:'
},
label: {
displaySpeeds: 'Mostrar velocidades',
showNoSpeed: 'Mostrar en segmento sin velocidad',
showOnRoundabout: 'Mostrar en rotonda',
maxZoom: 'Renderizar hasta el nivel de zoom:'
},
btn: {
updateSettingsTab: 'Actualizar menú',
},
log: {
wmeReadyStartScript: 'WME cargado y listo. Iniciando el script...',
waitingWme: 'Esperando que WME esté listo...',
wmeReadyStartingPlugin: 'WME está listo. Configurando la capa personalizada...',
shortSleep: 'Esperando aproximadamente 1 segundo para que todo se cargue...',
updateMapDisplay: 'Actualizando la visualización en el mapa...',
maxZoomExceeded: 'Se ha excedido el zoom máximo. No se mostrará velocidad.',
noneSegment: 'No se encontraron segmentos.',
numSegmentsLoaded: 'Número total de segmentos cargados:',
attributesNotFound: 'No se pudieron encontrar los atributos o están ausentes en el segmento',
disallowedSegment: 'Este tipo de segmento no debe cargarse.',
invalidGeometry: 'Geometría inválida en el segmento:',
updateMapDisplayFinished: 'Actualización completada.',
houseNumberCheckboxNotFound: 'No se encontró el checkbox "Números de Casas". Intentando nuevamente...',
scriptsTabNotFound: 'No se encontró la pestaña de scripts. Intentando nuevamente...',
settingsTabAdded: 'Pestaña de configuraciones añadida.',
speedLayer: 'Capa de velocidad',
opacityUpdated: 'Opacidad de los íconos ajustada a',
updatingSettingsTab: 'Actualizando la pestaña de configuraciones...',
languageDefined: 'Idiomas definidos.',
checkingVersion: 'Comprobando la versión del plugin.',
}
};
break;
case 'fi':
// Finnish
this.spdTranslations = {
name: 'Nopeuden näyttö',
settingTabName: 'Nopeuden näyttöasetukset',
enabled: 'käytössä',
disabled: 'poistettu käytöstä',
zoomOut: 'Loitontaa',
zoomIn: 'Lähentää',
changelog: 'Muutosloki',
changelogMessage: '<h6>Uutta:</h6><ul style="margin-bottom: 10px"><li>Lisätty changelog-modali.</li></ul><h6>Korjaukset:</h6><ul><li>Korjattu ongelma, jossa nopeudettomat kuvakkeet ilmestyivät, vaikka asetus oli pois päältä;</li><li>Muita pieniä korjauksia ja parannuksia.</li></ul>',
title: {
general: 'Yleinen',
showOnRoadType: 'Näytä tyypin segmenttien kohdalla:'
},
label: {
displaySpeeds: 'Näytä nopeudet',
showNoSpeed: 'Näytä segmentissä, jossa ei ole nopeutta',
showOnRoundabout: 'Näytä liikenneympyrässä',
maxZoom: 'Renderöi enintään zoom-tasolle:'
},
btn: {
updateSettingsTab: 'Päivitä valikko',
},
log: {
wmeReadyStartScript: 'WME ladattu ja valmis. Käynnistetään skripti...',
waitingWme: 'Odotetaan, että WME on valmis...',
wmeReadyStartingPlugin: 'WME on valmis. Asetetaan mukautettua kerrosta...',
shortSleep: 'Odota noin 1 sekunti ennen kuin kaikki latautuu...',
updateMapDisplay: 'Päivitetään näyttö kartalla...',
maxZoomExceeded: 'Maksimi zoomaus ylitetty. Nopeutta ei näytetä.',
noneSegment: 'Ei löytynyt segmenttejä.',
numSegmentsLoaded: 'Yhteensä ladattuja segmenttejä:',
attributesNotFound: 'Attribuutteja ei löytynyt tai ne puuttuvat segmentistä',
disallowedSegment: 'Tämän tyyppistä segmenttiä ei tule ladata.',
invalidGeometry: 'Virheellinen geometria segmentissä:',
updateMapDisplayFinished: 'Päivitys valmis.',
houseNumberCheckboxNotFound: '"Talojen numerot" -valintaruutua ei löytynyt. Yritetään uudelleen...',
scriptsTabNotFound: 'Skripti-välilehteä ei löytynyt. Yritetään uudelleen...',
settingsTabAdded: 'Asetusvälilehti lisätty.',
speedLayer: 'Nopeuskerros',
opacityUpdated: 'Ikonien opasiteetti säädetty arvoon',
updatingSettingsTab: 'Päivitetään asetusten välilehteä...',
languageDefined: 'Kielet määritetty.',
checkingVersion: 'Tarkistetaan liitännäisen versiota.',
}
};
break;
case 'fr':
// French
this.spdTranslations = {
name: 'Affichage de la vitesse',
settingTabName: 'Paramètres d\'affichage de la vitesse',
enabled: 'activée',
disabled: 'désactivée',
zoomOut: 'Dézoomer',
zoomIn: 'Zoomer',
changelog: 'Journal des modifications',
changelogMessage: '<h6>Nouveautés :</h6><ul style="margin-bottom: 10px"><li>Ajout d’une fenêtre modale pour le changelog.</li></ul><h6>Corrections :</h6><ul><li>Correction d’un problème où les icônes sans vitesse apparaissaient même lorsque l’option d’affichage sans vitesse était désactivée ;</li><li>Autres petites corrections et améliorations.</li></ul>',
title: {
general: 'Général',
showOnRoadType: 'Afficher sur les segments de type :'
},
label: {
displaySpeeds: 'Afficher les vitesses',
showNoSpeed: 'Afficher sur le segment sans vitesse',
showOnRoundabout: 'Afficher sur le rond-point',
maxZoom: 'Rendre jusqu\'au niveau de zoom :'
},
btn: {
updateSettingsTab: 'Mettre à jour le menu',
},
log: {
wmeReadyStartScript: 'WME chargé et prêt. Démarrage du script...',
waitingWme: 'En attente que WME soit prêt...',
wmeReadyStartingPlugin: 'WME est prêt. Configuration de la couche personnalisée...',
shortSleep: 'Attente d\'environ 1 seconde pour que tout se charge...',
updateMapDisplay: 'Mise à jour de l\'affichage sur la carte...',
maxZoomExceeded: 'Le zoom maximum a été dépassé. Aucune vitesse ne sera affichée.',
noneSegment: 'Aucun segment trouvé.',
numSegmentsLoaded: 'Nombre total de segments chargés :',
attributesNotFound: 'Impossible de trouver les attributs ou ils sont manquants dans le segment',
disallowedSegment: 'Ce type de segment ne doit pas être chargé.',
invalidGeometry: 'Géométrie invalide dans le segment :',
updateMapDisplayFinished: 'Mise à jour terminée.',
houseNumberCheckboxNotFound: 'La case à cocher "Numéros des maisons" n\'a pas été trouvée. Tentative de nouveau...',
scriptsTabNotFound: 'L\'onglet des scripts n\'a pas été trouvé. Tentative de nouveau...',
settingsTabAdded: 'Onglet des paramètres ajouté.',
speedLayer: 'Couche de vitesse',
opacityUpdated: 'Opacité des icônes ajustée à',
updatingSettingsTab: 'Mise à jour de l\'onglet des paramètres...',
languageDefined: 'Langues définies.',
checkingVersion: 'Vérification de la version du plugin.',
}
};
break;
case 'pl':
// Polish
this.spdTranslations = {
name: 'Wyświetlanie prędkości',
settingTabName: 'Ustawienia wyświetlania prędkości',
enabled: 'włączona',
disabled: 'wyłączona',
zoomOut: 'Oddal',
zoomIn: 'Przybliż',
changelog: 'Dziennik zmian',
changelogMessage: '<h6>Nowości:</h6><ul style="margin-bottom: 10px"><li>Dodano modalny changelog.</li></ul><h6>Poprawki:</h6><ul><li>Naprawiono problem z ikonami bez prędkości, które pojawiały się mimo wyłączonej opcji ich wyświetlania;</li><li>Inne drobne poprawki i ulepszenia.</li></ul>',
title: {
general: 'Ogólne',
showOnRoadType: 'Wyświetl na segmentach typu:'
},
label: {
displaySpeeds: 'Wyświetlaj prędkości',
showNoSpeed: 'Pokaż na segmencie bez prędkości',
showOnRoundabout: 'Pokaż na rondzie',
maxZoom: 'Renderuj do poziomu zoomu:'
},
btn: {
updateSettingsTab: 'Zaktualizuj menu',
},
log: {
wmeReadyStartScript: 'WME załadowane i gotowe. Uruchamiam skrypt...',
waitingWme: 'Czekam, aż WME będzie gotowe...',
wmeReadyStartingPlugin: 'WME jest gotowe. Konfigurowanie warstwy niestandardowej...',
shortSleep: 'Czekam około 1 sekundy, aż wszystko się załaduje...',
updateMapDisplay: 'Aktualizuję wyświetlanie na mapie...',
maxZoomExceeded: 'Przekroczono maksymalny poziom zoomu. Prędkość nie będzie wyświetlana.',
noneSegment: 'Nie znaleziono segmentów.',
numSegmentsLoaded: 'Łączna liczba załadowanych segmentów:',
attributesNotFound: 'Nie można znaleźć atrybutów lub są one brakujące w segmencie',
disallowedSegment: 'Tego typu segment nie powinien być ładowany.',
invalidGeometry: 'Nieprawidłowa geometria w segmencie:',
updateMapDisplayFinished: 'Aktualizacja zakończona.',
houseNumberCheckboxNotFound: 'Nie znaleziono pola wyboru „Numery domów”. Próbuję ponownie...',
scriptsTabNotFound: 'Nie znaleziono zakładki skryptów. Próbuję ponownie...',
settingsTabAdded: 'Dodano kartę ustawień.',
speedLayer: 'Warstwa prędkości',
opacityUpdated: 'Przezroczystość ikon została ustawiona na',
updatingSettingsTab: 'Aktualizowanie zakładki ustawień...',
languageDefined: 'Języki zdefiniowane.',
checkingVersion: 'Sprawdzanie wersji wtyczki.',
}
};
break;
case 'pt-BR':
// Brazilian portuguese
this.spdTranslations = {
name: 'Exibição de Velocidade',
settingTabName: 'Configurações de Exibição de Velocidade',
enabled: 'atiavada',
disabled: 'desativada',
zoomOut: 'Mais afastado',
zoomIn: 'Mais aproximado',
changelog: 'Registro de alterações',
changelogMessage: '<h6>Novidades:</h6><ul style="margin-bottom: 10px"><li>Adicionado modal de changelog.</li></ul><h6>Correções:</h6><ul><li>Corrigido problema com ícones sem velocidade que apareciam, mesmo com a opção de exibir sem velocidade desativado;</li><li>Outras pequenas correções e melhorias.</li></ul>',
title: {
general: 'Geral',
showOnRoadType: 'Exibir nos seguimentos do tipo:'
},
label: {
displaySpeeds: 'Exibir velocidades',
showNoSpeed: 'Exibir em seguimento sem velocidade',
showOnRoundabout: 'Exibir em rotatória',
maxZoom: 'Renderizar até o zoom:'
},
btn: {
updateSettingsTab: 'Atualizar menu',
},
log: {
wmeReadyStartScript: 'WME carregado e pronto. Iniciando script...',
waitingWme: 'Aguardando o WME estar pronto...',
wmeReadyStartingPlugin: 'WME está pronto. Configurando camada personalizada...',
shortSleep: 'Aguardando uns 1 segundo para tudo carregar...',
updateMapDisplay: 'Atualizando display no mapa...',
maxZoomExceeded: 'Zoom máximo excedido. Nenhuma velocidade será exibida.',
noneSegment: 'Nenhum segmento encontrado.',
numSegmentsLoaded: 'Total de segmentos carregados:',
attributesNotFound: 'Não foi possível encontrar os atributos ou estão ausentes no segmento',
disallowedSegment: 'Este tipo de seguimento não deve ser carregado.',
invalidGeometry: 'Geometria inválida no seguimento:',
updateMapDisplayFinished: 'Atualização concluída.',
houseNumberCheckboxNotFound: 'O checkbox "Números das Casas" não foi encontrado. Tentando novamente...',
scriptsTabNotFound: 'Painel de scripts não encontrado. Tentando novamente...',
settingsTabAdded: 'Aba de configurações adicionada.',
speedLayer: 'Camada de velocidade',
opacityUpdated: 'Opacidade dos ícones ajustada para',
updatingSettingsTab: 'Atualizando aba de configurações...',
languageDefined: 'Idiomas definidos.',
checkingVersion: 'Verificando a versão do plugin.'
}
};
break;
case 'ru':
// Russian
this.spdTranslations = {
name: 'Отображение скорости',
settingTabName: 'Настройки отображения скорости',
enabled: 'включено',
disabled: 'выключено',
zoomOut: 'Отдалить',
zoomIn: 'Приблизить',
changelog: 'Журнал изменений',
changelogMessage: '<h6>Новое:</h6><ul style="margin-bottom: 10px"><li>Добавлено модальное окно с журналом изменений.</li></ul><h6>Исправления:</h6><ul><li>Исправлена проблема с отображением значков без скорости, даже если эта опция была отключена;</li><li>Другие небольшие исправления и улучшения.</li></ul>',
title: {
general: 'Общие',
showOnRoadType: 'Отображать на сегментах типа:'
},
label: {
displaySpeeds: 'Отображать скорости',
showNoSpeed: 'Показывать на сегменте без скорости',
showOnRoundabout: 'Показывать на круговом движении',
maxZoom: 'Отображать до уровня зума:'
},
btn: {
updateSettingsTab: 'Обновить меню',
},
log: {
wmeReadyStartScript: 'WME загружен и готов. Запуск скрипта...',
waitingWme: 'Ожидание готовности WME...',
wmeReadyStartingPlugin: 'WME готово. Настройка пользовательского слоя...',
shortSleep: 'Ожидание около 1 секунды, пока все загрузится...',
updateMapDisplay: 'Обновление отображения на карте...',
maxZoomExceeded: 'Превышен максимальный уровень зума. Скорость отображаться не будет.',
noneSegment: 'Сегменты не найдены.',
numSegmentsLoaded: 'Общее количество загруженных сегментов:',
attributesNotFound: 'Не удалось найти атрибуты или они отсутствуют в сегменте',
disallowedSegment: 'Этот тип сегмента не должен загружаться.',
invalidGeometry: 'Неверная геометрия в сегменте:',
updateMapDisplayFinished: 'Обновление завершено.',
houseNumberCheckboxNotFound: 'Не найден флажок "Номера домов". Попробую снова...',
scriptsTabNotFound: 'Вкладка скриптов не найдена. Попробую снова...',
settingsTabAdded: 'Вкладка настроек добавлена.',
speedLayer: 'Слой скорости',
opacityUpdated: 'Прозрачность значков обновлена до',
updatingSettingsTab: 'Обновление вкладки настроек...',
languageDefined: 'Языки определены.',
checkingVersion: 'Проверка версии плагина.',
}
};
break;
case 'sv':
// Swedish
this.spdTranslations = {
name: 'Hastighetsvisning',
settingTabName: 'Inställningar för hastighetsvisning',
enabled: 'aktiverad',
disabled: 'inaktiverad',
zoomOut: 'Zooma ut',
zoomIn: 'Zooma in',
changelog: 'Ändringslogg',
changelogMessage: '<h6>Nyheter:</h6><ul style="margin-bottom: 10px"><li>Lagt till en modal för ändringsloggen.</li></ul><h6>Felkorrigeringar:</h6><ul><li>Fixat ett problem där ikoner utan hastighet visades även när alternativet att dölja dem var aktiverat;</li><li>Andra små korrigeringar och förbättringar.</li></ul>',
title: {
general: 'Allmänt',
showOnRoadType: 'Visa på segment av typen:'
},
label: {
displaySpeeds: 'Visa hastigheter',
showNoSpeed: 'Visa på segment utan hastighet',
showOnRoundabout: 'Visa på rondell',
maxZoom: 'Rendera upp till zoom-nivå:'
},
btn: {
updateSettingsTab: 'Uppdatera meny',
},
log: {
wmeReadyStartScript: 'WME laddad och redo. Startar skriptet...',
waitingWme: 'Väntar på att WME ska bli redo...',
wmeReadyStartingPlugin: 'WME är redo. Konfigurerar anpassad lager...',
shortSleep: 'Väntar cirka 1 sekund innan allt laddas...',
updateMapDisplay: 'Uppdaterar visningen på kartan...',
maxZoomExceeded: 'Maximal zoomnivå överskriden. Hastighet visas inte.',
noneSegment: 'Inga segment hittades.',
numSegmentsLoaded: 'Totalt antal laddade segment:',
attributesNotFound: 'Kunde inte hitta attribut eller de saknas i segmentet',
disallowedSegment: 'Den här typen av segment ska inte laddas.',
invalidGeometry: 'Ogiltig geometri i segmentet:',
updateMapDisplayFinished: 'Uppdatering slutförd.',
houseNumberCheckboxNotFound: 'Checkboxen "Husnummer" hittades inte. Försöker igen...',
scriptsTabNotFound: 'Fliken för skript hittades inte. Försöker igen...',
settingsTabAdded: 'Inställningsflik tillagd.',
speedLayer: 'Hastighetslager',
opacityUpdated: 'Opaciteten för ikoner har uppdaterats till',
updatingSettingsTab: 'Uppdaterar inställningsfliken...',
languageDefined: 'Språk definierade.',
checkingVersion: 'Kontrollerar plugin-version.',
}
};
break;
case 'uk':
// Ukrainian
this.spdTranslations = {
name: 'Відображення швидкості',
settingTabName: 'Налаштування відображення швидкості',
enabled: 'увімкнено',
disabled: 'вимкнено',
zoomOut: 'Віддалити',
zoomIn: 'Приблизити',
changelog: 'Журнал змін',
changelogMessage: '<h6>Новини:</h6><ul style="margin-bottom: 10px"><li>Додано модальне вікно для журналу змін.</li></ul><h6>Виправлення:</h6><ul><li>Виправлено проблему, коли значки без швидкості з’являлися, навіть якщо опція їх відображення була вимкнена;</li><li>Інші дрібні виправлення та покращення.</li></ul>',
title: {
general: 'Загальні',
showOnRoadType: 'Відображати на сегментах типу:'
},
label: {
displaySpeeds: 'Відображати швидкості',
showNoSpeed: 'Показувати на сегменті без швидкості',
showOnRoundabout: 'Показувати на круговому русі',
maxZoom: 'Рендерити до рівня масштабування:'
},
btn: {
updateSettingsTab: 'Оновити меню',
},
log: {
wmeReadyStartScript: 'WME завантажено та готове. Запуск скрипта...',
waitingWme: 'Очікуємо, поки WME стане готовим...',
wmeReadyStartingPlugin: 'WME готове. Налаштування кастомного шару...',
shortSleep: 'Чекаємо близько 1 секунди, поки все завантажиться...',
updateMapDisplay: 'Оновлення відображення на карті...',
maxZoomExceeded: 'Максимальний рівень масштабування перевищено. Швидкість не буде відображена.',
noneSegment: 'Сегменти не знайдено.',
numSegmentsLoaded: 'Загальна кількість завантажених сегментів:',
attributesNotFound: 'Не вдалося знайти атрибути або вони відсутні в сегменті',
disallowedSegment: 'Цей тип сегмента не повинен завантажуватися.',
invalidGeometry: 'Невірна геометрія в сегменті:',
updateMapDisplayFinished: 'Оновлення завершено.',
houseNumberCheckboxNotFound: 'Не знайдено чекбокс "Номери будинків". Спробую ще раз...',
scriptsTabNotFound: 'Не знайдена вкладка скриптів. Спробую ще раз...',
settingsTabAdded: 'Вкладку налаштувань додано.',
speedLayer: 'Шар швидкості',
opacityUpdated: 'Прозорість значків оновлено до',
updatingSettingsTab: 'Оновлення вкладки налаштувань...',
languageDefined: 'Мови визначено.',
checkingVersion: 'Перевірка версії плагіна.',
}
};
break;
default:
// Default language (english)
this.spdTranslations = {
name: 'Speed Display',
settingTabName: 'Speed Display Settings',
enabled: 'enabled',
disabled: 'disabled',
zoomOut: 'Zoom out',
zoomIn: 'Zoom in',
changelog: 'Changelog',
changelogMessage: '<h6>What’s New:</h6><ul style="margin-bottom: 10px"><li>Added a changelog modal.</li></ul><h6>Fixes:</h6><ul><li>Fixed an issue where speedless icons appeared even when the option to show them was disabled;</li><li>Other minor fixes and improvements.</li></ul>',
title: {
general: 'General',
showOnRoadType: 'Display on segments of type:'
},
label: {
displaySpeeds: 'Display speeds',
showNoSpeed: 'Show on segment with no speed',
showOnRoundabout: 'Show on roundabout',
maxZoom: 'Render up to zoom level:'
},
btn: {
updateSettingsTab: 'Update menu',
},
log: {
wmeReadyStartScript: 'WME loaded and ready. Starting script...',
waitingWme: 'Waiting for WME to be ready...',
wmeReadyStartingPlugin: 'WME is ready. Setting up custom layer...',
shortSleep: 'Waiting about 1 second for everything to load...',
updateMapDisplay: 'Updating display on the map...',
maxZoomExceeded: 'Maximum zoom exceeded. No speed will be displayed.',
noneSegment: 'No segments found.',
numSegmentsLoaded: 'Total segments loaded:',
attributesNotFound: 'Could not find attributes or they are missing in the segment',
disallowedSegment: 'This type of segment should not be loaded.',
invalidGeometry: 'Invalid geometry in segment:',
updateMapDisplayFinished: 'Update completed.',
houseNumberCheckboxNotFound: 'The "House Numbers" checkbox was not found. Trying again...',
settingsTabAdded: 'Settings tab added.',
settingsTabAdded: 'Aba de configurações adicionada.',
speedLayer: 'Speed layer',
opacityUpdated: 'Icon opacity adjusted to',
updatingSettingsTab: 'Updating settings tab...',
languageDefined: 'Languages defined.',
checkingVersion: 'Checking plugin version.',
}
};
break;
};
this.logDebug(this.spdTranslations.log.languageDefined);
}
}
new WmeSpeedDisplay();