WME Speed Display

Displays road speed directly in the center of the segment (taking curves into account) in Waze Map Editor

// ==UserScript==
// @name              WME Speed Display
// @name:pt-BR        WME Exibição de Velocidade
// @namespace         https://greasyfork.org/scripts/526702-wme-speed-display/
// @version           2.1.3
// @description       Displays road speed directly in the center of the segment (taking curves into account) in Waze Map Editor
// @description:pt-BR Exibe a velocidade da via diretamente no centro do segmento (levando em consideração as curvas) no 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:
 * Check a way to get all road types and groups automatically
 * Script layer switch is not loading based on the saved setting, it is always false in the new SDK
 * When I change the layer switch it does not toggle the layer on and off in the new SDK
 * Change "Hide in segments smaller than" so that it applies transparency instead of ignoring the icon
 * There is a snackbar that makes my wrapper go up and not go down anymore. Identify what it is and fix it
 */

/* global I18n */
/* global getWmeSdk */

// TODO: Remove this when SDK migration is complete
/* global W */

class WmeSpeedDisplay {
    constructor() {
        this.version           = '2.1.3';
        this.debugMode         = false;
        this.needResetSettings = false;
        this.settings          = {};
        this.zIndex            = {
            min: 2065,
            max: 6500,
            warn: 3000,
            warnDisplayed: 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'
                }
            ],
            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.theme = [
            'default',
            'altViennaConv',
            'japan',
            'northAmerica',
            'sacu',
            'uk',
        ];

        this.wmeTranslations = I18n.translations[I18n.locale];
        this.spdTranslations = this.wmeTranslations.wmeSpd = {};

        this.debounce = {
            updateMapDisplay: this.debounce(this.updateMapDisplay.bind(this), 800),
            saveSettings: this.debounce(this.saveSettings.bind(this)),
            onZindexFieldChanged: this.debounce(this.onZindexFieldChanged.bind(this), 1000),
            onSpeedFilterChanged: this.debounce(this.onSpeedFilterChanged.bind(this), 500),
            onSettingsChange: this.debounce(this.onSettingsChange.bind(this), 500)
        };

        
        this.channel = new BroadcastChannel('wmeSpeedDisplay');
        
        this.layerName = 'wme-speed-display-layer';
        
        // TODO: Remove this when SDK migration is complete
        this.layerCheckbox = {
            roadGroup: null,
            speedDisplay: null
        };
        
        this.snackbarWrapper;
        this.displaySpeedsSwitch;
        this.wmeSdk;
        this.layer;
        // TODO: SDK migration
        // this.layerSwitcherElement;

        this.prepareAndInitialize();
    }

    /**
     * Checks if WME is ready to initializes the plugin.
     */
    prepareAndInitialize() {
        this.defineSettings();
        this.defineTranslations();

        this.wmeSdk = getWmeSdk({
            scriptId: 'wme-speed-display',
            scriptName: this.getTranslation('name')
        });

        this.wmeSdk.Events.once({
            eventName: 'wme-ready'
        }).then(this.initializePlugin.bind(this));
     }

    /**
     * Initializes the plugin.
     */
    async initializePlugin() {
        this.logDebug('WME loaded and ready. Starting script...', true);

        // TODO: Remove this when SDK migration is complete
        this.layer = new OpenLayers.Layer.Vector(this.layerName, {
            displayInLayerSwitcher: false,
            rendererOptions: {zIndexing: true}
        });

        // TODO: SDK migration
        // this.layer = this.wmeSdk.Map.addLayer({
        //     layerName: this.layerName,
        //     displayName: this.getTranslation('name'),
        //     shouldDisplayInLayerSwitcher: true,
        //     // styleRules: [
        //     //     {
        //     //         predicate: featureProperties => featureProperties.isHighlighted,
        //     //         style: {
        //     //             strokeColor: 'yellow',
        //     //             strokeWidth: 10,
        //     //         },
        //     //     },
        //     //     {
        //     //         style: { // Default style
        //     //             strokeColor: 'lightgray',
        //     //             strokeWidth: 5,
        //     //         },
        //     //     }
        //     // ]
        // });
        
        // this.wmeSdk.LayerSwitcher.addLayerCheckbox({name: this.getTranslation('name')});

        // this.getLayerSwitcher();
        // TODO: SDK migration - end

        this.setZIndex();
        
        // TODO: Remove this when SDK migration is complete
        W.map.addLayer(this.layer);
        this.addLayerCheckbox();
        // TODO: Remove this when SDK migration is complete - end

        this.createSnackbarWrapper();
        this.observeAllOriginalSnackbars();
        await this.addSettingsTab();
        this.updateSettingsElementsStates();
        this.addShortcut();
        this.addStyle();
        this.checkVersionAndShowChangelog();
        this.listen();

        this.logDebug('Waiting about 1 second for everything to load...');

        await this.sleep(1000);
        
        this.preUpdateMapDisplay();
    }

    /**
     * Method that listens to this script events.
     */
    listen() {
        this.channel.addEventListener('message', this.onChannelMessage.bind(this));
        
        this.wmeSdk.Events.on({
            eventName: 'wme-map-move-end',
            eventHandler: this.preUpdateMapDisplay.bind(this),
        });
        
        this.wmeSdk.Events.on({
            eventName: 'wme-map-zoom-changed',
            eventHandler: this.preUpdateMapDisplay.bind(this),
        });
        
        // TODO: Remove this when SDK migration is complete
        W.model.segments.on('objectschanged', this.preUpdateMapDisplay.bind(this));

        // TODO: SDK migration
        // this.wmeSdk.Events.trackDataModelEvents({dataModelName: 'segments'});
        
        // this.wmeSdk.Events.on({
        //     eventName: 'wme-data-model-objects-changed',
        //     eventHandler: this.preUpdateMapDisplay.bind(this),
        // });
        // TODO: SDK migration - end

        this.wmeSdk.Events.on({
            eventName: 'wme-user-settings-changed',
            eventHandler: this.onUserSettingsChanged.bind(this),
        });

        this.wmeSdk.Events.on({
            eventName: 'wme-map-data-loaded',
            eventHandler: this.preUpdateMapDisplay.bind(this),
        });

        // TODO: SDK migration
        // this.wmeSdk.Events.on({
        //     eventName: 'wme-layer-checkbox-toggled',
        //     eventHandler: this.onLayerSwitcherToggled.bind(this),
        // });
    }

    /**
     * Saves the settings and updates the map display.
     */
    async onSettingsChange() {
        await this.debounce.saveSettings();
        this.preUpdateMapDisplay();
    }

    /**
     * Updates the max zoom value display and triggers map update if the zoom level reaches the max.
     * @param {Event} event
     */
    async onSettingZoomChanged(event) {
        let zoomLevel   = this.wmeSdk.Map.getZoomLevel();
        let newMaxZoom  = parseInt(event.target.value, 10);
        let prevMaxZoom = this.settings.general.spdMaxZoom;

        document.getElementById('spd-max-zoom-value').innerText = newMaxZoom;

        await this.debounce.saveSettings();
    
        let wasVisible      = prevMaxZoom > zoomLevel;
        let shouldBeVisible = newMaxZoom > zoomLevel;
    
        if (wasVisible != shouldBeVisible)
            this.preUpdateMapDisplay();
    }

    /**
     * Updates the filter opacity value and triggers map update when opacity changes.
     * @param {Event} event
     */
    async onSettingOpacityChanged(event) {
        document.getElementById('spd-filter-opacity-value').innerText = parseFloat(event.target.value);

        await this.debounce.saveSettings();
        this.preUpdateMapDisplay();
    }

    /**
     * Handles user settings change.
     */
    onUserSettingsChanged() {
        let speedUnitTexts = document.querySelectorAll('.spd-filter-speed-unit');
        let isImperial     = this.wmeSdk.Settings.getUserSettings().isImperial;

        document.querySelector('.spd-filter-distance-unit').innerHTML = isImperial ? this.wmeTranslations.units_short.feet : this.wmeTranslations.units_short.meters;

        speedUnitTexts.forEach(text => {
            text.innerText = isImperial ? I18n.translate('measurements.speed.mi', {speed: ''}) : I18n.translate('measurements.speed.km', {speed: ''});
        });

        this.preUpdateMapDisplay();
    }

    /**
     * Changes layer's z-index.
     * @param {Event} event 
     */
    increaseOrDecreaseZindex(event) {
        let zIndexInput = document.getElementById('spd-layer-zindex');
        let decrease    = event.target.id == 'spd-btn-zindex-decrease';
    
        let min = this.zIndex.min;
        let max = this.zIndex.max;
    
        let currentZindex = parseInt(zIndexInput.value);
    
        if ((decrease && currentZindex > min) || (!decrease && currentZindex < max)) {
            zIndexInput.value = decrease ? currentZindex - 100 : currentZindex + 100;

            if (!this.zIndex.warnDisplayed)
                this.displayZIndexWarning();

            this.debounce.saveSettings();
            this.setZIndex();
        }
    }

    /**
     * Updates z-index input value within allowed range and saves settings with debounce.
     */
    onZindexFieldChanged() {
        let zIndexInput = document.getElementById('spd-layer-zindex');
        let min         = this.zIndex.min;
        let max         = this.zIndex.max;
        let value       = parseInt(zIndexInput.value, 10) || max;
        value           = Math.min(Math.max(value, min), max);

        zIndexInput.value = value;

        this.settings.general.spdLayerZindex = value;

        if (!this.zIndex.warnDisplayed)
            this.displayZIndexWarning();

        this.debounce.saveSettings();
        this.setZIndex();
    }

    /**
     * Displays a warning alert if the z-index value exceeds the allowed threshold. 
     */
    displayZIndexWarning() {
        if (parseInt(document.getElementById('spd-layer-zindex')?.value, 10) >= this.zIndex.warn && !this.settings.alertDismissed.infoZindex) {
            this.addAlert('info-zindex', this.getTranslation('alertMessage.zIndex'), 'warning');

            this.zIndex.warnDisplayed = true;
        }
    }

    /**
     * Sets the z-index of the layer.
     */
    setZIndex() {
        // TODO: Remove this when SDK migration is complete
        this.layer.setZIndex(this.settings.general.spdLayerZindex);
        
        // TODO: SDK migration
        // this.wmeSdk.Map.setLayerZIndex({
        //     layerName: this.layerName, 
        //     zIndex: this.settings.general.spdLayerZindex
        // });
    }

    /**
     * Resets the z-index of the layer.
     */
    resetZindex() {
        let zIndexInput = document.getElementById('spd-layer-zindex');
        
        this.settings.general.spdLayerZindex = this.zIndex.min;

        zIndexInput.value = this.settings.general.spdLayerZindex;

        this.onZindexFieldChanged();
    }

    /**
     * Save alert dismissed.
     * @param {Event} event
     */
    async onDismissAlert(event) {
        let currentDismissedAlertId = this.convertStringCaseStyle('kebab', 'camel', event.target.parentNode.id);

        this.settings.alertDismissed[currentDismissedAlertId] = true;
        
        await this.debounce.saveSettings();

        let warningGroup       = document.getElementById('spd-alert-group');
        let allAlertsDismissed = warningGroup.querySelectorAll('wz-alert:not([dismissed]), wz-alert[dismissed="false"]');

        if (allAlertsDismissed.length <= 1)
            warningGroup.classList.add('d-none');
    }

    /**
     * Validates the speed filter input's and save settings.
     * @param {Event} event
     */
    onSpeedFilterChanged(event) {
        let minSpeed = parseInt(document.getElementById('spd-min-speed').value) || 0;
        let maxSpeed = parseInt(document.getElementById('spd-max-speed').value) || 0;

        if (maxSpeed > 0 && minSpeed > maxSpeed) {
            this.logDebug('Min speed filter can\'t be bigger than the max speed filter.', false, 'error');

            this.removeAllSnackbars();
            this.addSnackbar('spd-snackbar-error-filtering', `<i class="w-icon w-icon-alert-danger-outline"></i><span><strong>${this.getTranslation('name')}:</strong><br>${this.getTranslation('snackbar.errorFiltering')}</span>`, `<wz-button color="text">${this.wmeTranslations.zoom_in_message.ok}</wz-button>`);

            event.target.value = '';

            return;
        }

        this.onSettingsChange();
    }

    // TODO: SDK migration
    /**
     * Updates the plugin's enabled state in settings when the corresponding checkbox is toggled and changes the plugin's layer visibility.
     * @param {Object} checkboxToggled - The object representing the toggled checkbox.
     * @param {boolean} checkboxToggled.checked - Whether the checkbox is checked or not.
     * @param {String} checkboxToggled.name - The name of the checkbox (used to identify the plugin's settings).
     */
    // async onLayerSwitcherToggled(checkboxToggled) {
    //     if (checkboxToggled.name == this.getTranslation('name')) {
    //         await this.debounce.saveSettings();
    //         this.preUpdateMapDisplay();
    //         this.toggleLayerVisibility();
    //     }
    // }

    /**
     * Select the speed display theme.
     * @param {Event} event 
     */
    async onThemeCardClicked(event) {
        this.selectThemeCard(event.target);
        await this.debounce.saveSettings();
        this.preUpdateMapDisplay();
    }

    /**
     * Simulate a click on the speed display layer switcher.
     */
    simulateLayerSwitchClick() {
        // TODO: Remove this when SDK migration is complete
        this.layerCheckbox.speedDisplay.click();

        // TODO: SDK migration
        // this.layerSwitcherElement.click();
    }

    /**
     * Handles channel messages and sincronizing settings on the others open WME tabs.
     * @param {Event} event 
     */
    onChannelMessage(event) {
        switch (event.data) {
            case 'saved':
                this.defineSettings();
                this.updateSettingsElementsStates();
                this.preUpdateMapDisplay();

                break;
        }

        // TODO: Remove this when SDK migration is complete (just the function param)
        this.toggleLayerVisibility(this.layerCheckbox.roadGroup.checked && this.settings.spdEnabled);
    }

    // TODO: Remove this when SDK migration is complete
    /**
     * Saves and updates the map display when WME SPD checkbox change.
     * @param {Event} event
     */
    onSpeedDisplayCheckboxChange(event) {
        this.debounce.saveSettings();
        this.preUpdateMapDisplay();

        this.toggleLayerVisibility(event.target.checked);
        
        this.displaySpeedsSwitch.checked = event.target.checked;
    }
    
    /**
     * Updates the layer visibility based on the layer switcher road group state.
     */
    onLayerSwithcerGroupRoadChange() {
        let roadGroupActived = this.layerCheckbox.roadGroup.checked;

        this.toggleLayerVisibility(roadGroupActived && this.settings.spdEnabled);

        this.layerCheckbox.speedDisplay.disabled = !roadGroupActived;
        this.displaySpeedsSwitch.disabled        = !roadGroupActived;
    }
    // TODO: Remove this when SDK migration is complete - end

    /**
     * Observes all 'wz-snackbar' elements for changes in children, attributes, or subtree.
     */
    observeAllOriginalSnackbars() {
        let snackbars = document.querySelectorAll('wz-snackbar');
    
        if (snackbars.length === 0) {
            this.logDebug('No "wz-snackbar" found.', false, 'warn');

            return;
        }
    
        snackbars.forEach(snackbar => {
            let observer = new MutationObserver(this.handleMutations.bind(this));
    
            observer.observe(snackbar, {
                childList: true,
                attributes: true,
                subtree: true
            });
    
            this.logDebug(`Observing changes in 'wz-snackbar' with class: ${snackbar.classList.value || 'without class'}`);
        });
    }

    /**
     * Handles mutations by adjusting 'translate' on 'snackbarWrapper' based on shadowRoot content.
     * @param {MutationRecord[]} mutationsList List of observed mutations.
     */
    handleMutations(mutationsList) {
        mutationsList.forEach((mutation) => {
            if (mutation.target.shadowRoot.children.length < 1) {
                this.snackbarWrapper.style.translate = '0px -95px';
                    
            } else {
                this.snackbarWrapper.style.removeProperty('translate');
            }
        });
    }

    /**
     * Attaches event listeners to the settings elements to track user interactions and changes.
     */
    attachSettingsListeners() {
        this.displaySpeedsSwitch.addEventListener('change', this.simulateLayerSwitchClick.bind(this));

        // Change event -> onSettingsChange
        ['spd-show-no-speed', 'spd-show-roundabout', 'spd-show-unpaved', 'spd-hide-segment-smaller-than'].forEach(id => {
            document.getElementById(id).addEventListener('change', this.onSettingsChange.bind(this));
        });

        document.getElementById('spd-max-zoom').addEventListener('input', this.onSettingZoomChanged.bind(this));
        document.getElementById('spd-filter-opacity').addEventListener('input', this.onSettingOpacityChanged.bind(this));

        ['spd-btn-zindex-decrease', 'spd-btn-zindex-increase'].forEach(id => {
            document.getElementById(id).addEventListener('click', this.increaseOrDecreaseZindex.bind(this));
        });

        document.getElementById('spd-btn-reset-z-index').addEventListener('click', this.resetZindex.bind(this));
        document.getElementById('spd-layer-zindex').addEventListener('input', this.debounce.onZindexFieldChanged.bind(this));
        document.getElementById('spd-hide-segment-smaller-than').addEventListener('input', this.onSettingsChange.bind(this));
        document.getElementById('spd-min-speed').addEventListener('input', this.debounce.onSpeedFilterChanged.bind(this));
        document.getElementById('spd-max-speed').addEventListener('input', this.debounce.onSpeedFilterChanged.bind(this));

        document.querySelectorAll('[id^="spd-show-speed-in-"]').forEach(input => {
            input.addEventListener('change', this.onSettingsChange.bind(this));
        });

        document.querySelectorAll('[id^="spd-theme-"]').forEach(card => {
            card.addEventListener('click', this.onThemeCardClicked.bind(this));
        });

        document.getElementById('spd-reset-default-settings').addEventListener('click', this.resetSettings.bind(this));

        if (this.debugMode) {
            document.getElementById('btn-spd-update-settings-tab').addEventListener('click', this.updateSettingsTab.bind(this));
            document.getElementById('btn-spd-test').addEventListener('click', this.addSnackbar.bind(this));
        }

        document.getElementById('btn-spd-open-changelog-dialog').addEventListener('click', this.openChangelogDialog.bind(this));
    }

    /**
     * Detaches event listeners from the settings elements to stop tracking user interactions.
     */
    detachSettingsListeners() {
        this.displaySpeedsSwitch.removeEventListener('change', this.simulateLayerSwitchClick.bind(this));

        // Change event -> onSettingsChange
        ['spd-show-no-speed', 'spd-show-roundabout', 'spd-show-unpaved', 'spd-hide-segment-smaller-than'].forEach(id => {
            document.getElementById(id).removeEventListener('change', this.onSettingsChange.bind(this));
        });

        document.getElementById('spd-max-zoom').removeEventListener('input', this.onSettingZoomChanged.bind(this));
        document.getElementById('spd-filter-opacity').removeEventListener('input', this.onSettingOpacityChanged.bind(this));

        ['spd-btn-zindex-decrease', 'spd-btn-zindex-increase'].forEach(id => {
            document.getElementById(id).removeEventListener('click', this.increaseOrDecreaseZindex.bind(this));
        });

        document.getElementById('spd-btn-reset-z-index').removeEventListener('click', this.resetZindex.bind(this));
        document.getElementById('spd-layer-zindex').removeEventListener('input', this.debounce.onZindexFieldChanged.bind(this));
        document.getElementById('spd-hide-segment-smaller-than').removeEventListener('input', this.onSettingsChange.bind(this));
        document.getElementById('spd-min-speed').removeEventListener('input', this.debounce.onSpeedFilterChanged.bind(this));
        document.getElementById('spd-max-speed').removeEventListener('input', this.debounce.onSpeedFilterChanged.bind(this));

        document.querySelectorAll('[id^="spd-show-speed-in-"]').forEach(input => {
            input.removeEventListener('change', this.onSettingsChange.bind(this));
        });

        document.querySelectorAll('[id^="spd-theme-"]').forEach(card => {
            card.removeEventListener('click', this.onThemeCardClicked.bind(this));
        });

        document.getElementById('spd-reset-default-settings').removeEventListener('click', this.resetSettings.bind(this));

        if (this.debugMode) {
            document.getElementById('btn-spd-update-settings-tab').removeEventListener('click', this.updateSettingsTab.bind(this));
            document.getElementById('btn-spd-test').removeEventListener('click', this.showSpeedDisplayDialog.bind(this));
        }

        document.getElementById('btn-spd-open-changelog-dialog').removeEventListener('click', this.openChangelogDialog.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.hideSpeedDisplayDialog.bind(this), {once: true});
    }

    /**
     * Updates the map display if the road group checkbox is checked and speed display is enabled, showing a snackbar during the process.
     */
    async preUpdateMapDisplay() {
        if (!this.layerCheckbox.roadGroup.checked || !this.settings.spdEnabled) {
            this.logDebug('Can\'t update the map display.', false, 'warn');

            return;
        }

        let snackbarId = 'spd-snackbar-updating-map';
        
        if (W.map.getZoom() >= this.settings.general.spdMaxZoom)
            this.addSnackbar(snackbarId, '<div class="loader"></div>', null, true);

        await this.debounce.updateMapDisplay();
        this.removeSnackbar(document.getElementById(snackbarId));
    }

    /**
     * Updates the map display by removing old features and adding new speed icons based on the settings.
     */
    // TODO: Remove this when SDK migration is complete
    updateMapDisplay() {
        return new Promise((resolve, reject) => {
            try {
                this.logDebug('Updating map display...');
        
                this.layer.removeAllFeatures();
        
                if (W.map.getZoom() < this.settings.general.spdMaxZoom) {
                    this.logDebug('Maximum zoom exceeded. No speeds will be displayed.', false, 'warn');

                    resolve();
        
                    return;
                }
        
                let segments    = W.model.segments.objects || {};
                let segmentKeys = Object.keys(segments);
        
                if (segmentKeys.length == 0) {
                    this.logDebug('No segments found.', false, 'error');

                    reject();
        
                    return;
                }
        
                this.logDebug(`Total segments loaded: ${segmentKeys.length}`);
        
                segmentKeys.forEach(segmentId => {
                    let segment = segments[segmentId];
        
                    if (!segment || !segment.attributes) {
                        this.logDebug(`Could not find attributes or they are missing on the segment ${segmentId}.`, false, 'error');

                        return;
                    }
        
                    let attributes    = segment.attributes;
                    let roadSettingId = this.convertStringCaseStyle('snake', 'camel', `spd_show_speed_in_${this.getRoadSettingNameById(attributes.roadType)}`);
                    
                    let isLeftHandTraffic = this.wmeSdk.DataModel.Countries.getTopCountry().isLeftHandTraffic;
                    let isFwd             = attributes.fwdDirection;
                    let isRev             = attributes.revDirection;
                    let speedFwd          = this.getSpeed(isLeftHandTraffic ? attributes.revMaxSpeed : attributes.fwdMaxSpeed);
                    let speedRev          = this.getSpeed(isLeftHandTraffic ? attributes.fwdMaxSpeed : attributes.revMaxSpeed);
        
                    let hideNoSpeed        = !this.settings.general.spdShowNoSpeed && ((speedFwd == 'N/A' && speedRev == 'N/A') || (isFwd && !isRev && speedFwd == 'N/A') || (!isFwd && isRev && speedRev == 'N/A') || (!isFwd && !isRev));
                    let ignoreOnRoundabout = attributes.junctionID && !this.settings.general.spdShowRoundabout;
                    let ignoreOnUnpaved    = attributes.flags == 16 && !this.settings.general.spdShowUnpaved;
        
                    if (hideNoSpeed || !this.settings.roads[roadSettingId] || ignoreOnRoundabout || ignoreOnUnpaved) {
                        this.logDebug('This type of segment should not be loaded.', false, 'warn');
        
                        return;
                    }
        
                    let geometry = segment.getOLGeometry();
        
                    if (!geometry || geometry.components.length < 2) {
                        this.logDebug(`Invalid geometry on segment: ${segmentId}.`, false, 'error');
        
                        return;
                    }
                    
                    let minLength     = this.settings.filter.spdHideSegmentSmallerThan;
                    let segmentLength = attributes.length;
                    let opacity       = minLength != null && segmentLength < minLength ? this.getSettingOpacity() : null;

                    let midpoint                = this.calculateMidpoint(geometry);
                    let angle                   = this.calculateAngleAtMidpoint(geometry);
                    let {leftPoint, rightPoint} = this.getPerpendicularPoints(midpoint, angle, 5);
        
                    if (isFwd && isRev && speedFwd != speedRev) {
                        let leftFeature  = new OpenLayers.Feature.Vector(leftPoint, {speed: speedRev});
                        let rightFeature = new OpenLayers.Feature.Vector(rightPoint, {speed: speedFwd});

                        leftFeature.style = {
                            graphic: true,
                            externalGraphic: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speedRev)),
                            graphicHeight: 30,
                            graphicWidth: 30,
                            graphicYOffset: -15,
                            graphicXOffset: -15,
                            graphicOpacity: opacity ?? this.calculateOpacityBasedOnSpeed(speedRev)
                        };

                        rightFeature.style = {
                            graphic: true,
                            externalGraphic: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speedFwd)),
                            graphicHeight: 30,
                            graphicWidth: 30,
                            graphicYOffset: -15,
                            graphicXOffset: -15,
                            graphicOpacity: opacity ?? this.calculateOpacityBasedOnSpeed(speedFwd)
                        };
        
                        this.layer.addFeatures([leftFeature, rightFeature]);
        
                    } else {
                        let speed           = isLeftHandTraffic ? (isFwd ? speedRev : speedFwd) : (isFwd ? speedFwd : speedRev);
                        let centerFeature   = new OpenLayers.Feature.Vector(midpoint, {speed});
                        
                        centerFeature.style = {
                            graphic: true,
                            externalGraphic: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speed, isFwd && isRev)),
                            graphicHeight: 30,
                            graphicWidth: 30,
                            graphicYOffset: -15,
                            graphicXOffset: -15,
                            graphicOpacity: opacity ?? this.calculateOpacityBasedOnSpeed(speed)
                        };
        
                        this.layer.addFeatures([centerFeature]);
                    }
                });
        
                this.logDebug('Update finished.');

                resolve();

            } catch (error) {
                this.logDebug(error, false, 'error');
                
                reject(error);
            }
        });
    }
    // TODO: SDK migration
    // updateMapDisplay() {
    //     return new Promise(async (resolve, reject) => {
    //         try {
    //             this.logDebug('Updating map display...');
        
    //             this.wmeSdk.Map.removeAllFeaturesFromLayer({layerName: this.layerName});
        
    //             if (this.wmeSdk.Map.getZoomLevel() < this.settings.general.spdMaxZoom) {
    //                 this.logDebug('Maximum zoom exceeded. No speeds will be displayed.', false, 'warn');

    //                 resolve();
        
    //                 return;
    //             }
        
    //             let segments = await this.wmeSdk.DataModel.Segments.getAll();
        
    //             if (segments.length == 0) {
    //                 this.logDebug('No segments found.', false, 'error');

    //                 reject();
        
    //                 return;
    //             }
        
    //             this.logDebug('Total segments loaded: ' + segments.length);
        
    //             segments.forEach(segment => {
    //                 let roadSettingId = this.convertStringCaseStyle('snake', 'camel', `spd_show_speed_in_${this.getRoadSettingNameById(segment.roadType)}`);
        
    //                 let isLeftHandTraffic = this.wmeSdk.DataModel.Countries.getTopCountry().isLeftHandTraffic;
    //                 let isFwd             = segment.isAtoB;
    //                 let isRev             = segment.isBtoA;
    //                 let isTwoWay          = segment.isTwoWay;
    //                 let speedFwd          = this.getSpeed(isFwd ? (isLeftHandTraffic ? segment.revSpeedLimit : segment.fwdSpeedLimit) : null);
    //                 let speedRev          = this.getSpeed(isRev ? (isLeftHandTraffic ? segment.fwdSpeedLimit : segment.revSpeedLimit) : null);
        
    //                 let hideNoSpeed        = !this.settings.general.spdShowNoSpeed && ((speedFwd == 'N/A' && speedRev == 'N/A') || (isFwd && !isRev && speedFwd == 'N/A') || (!isFwd && isRev && speedRev == 'N/A') || (!isFwd && !isRev && !isTwoWay));
    //                 let ignoreOnRoundabout = segment.junctionId && !this.settings.general.spdShowRoundabout;
    //                 let ignoreOnUnpaved    = segment.flagAttributes.unpaved && !this.settings.general.spdShowUnpaved;
        
    //                 if (hideNoSpeed || !this.settings.roads[roadSettingId] || ignoreOnRoundabout || ignoreOnUnpaved) {
    //                     this.logDebug('This type of segment should not be loaded.', false, 'warn');
        
    //                     return;
    //                 }
        
    //                 let geometry = segment.geometry;
                    
    //                 if (!geometry || geometry.coordinates.length < 2) {
    //                     this.logDebug(`${this.getTranslation('log.invalidGeometry')} ${segment.id}.`, false, 'error');
        
    //                     return;
    //                 }
        
    //                 let minLength     = this.settings.filter.spdHideSegmentSmallerThan;
    //                 let segmentLength = this.getLength(geometry.coordinates);
    //                 let opacity       = minLength != null && segmentLength < minLength ? this.getSettingOpacity() : null;
        
    //                 let midpoint                = this.calculateMidpoint(geometry.coordinates, segmentLength);
    //                 let angle                   = this.calculateAngleAtMidpoint(geometry.coordinates);
    //                 let {leftPoint, rightPoint} = this.getPerpendicularPoints(midpoint, angle, 5);
        
    //                 if (isFwd && isRev && speedFwd != speedRev) {
    //                     this.wmeSdk.Map.addFeaturesToLayer({
    //                         layerName: this.layerName,
    //                         features: [
    //                             {
    //                                 type: 'Feature',
    //                                 id: `speed-${segment.id}-rev`,
    //                                 geometry: {
    //                                     type: 'Point',
    //                                     coordinates: [leftPoint[0], leftPoint[1]]
    //                                 },
    //                                 properties: {
    //                                     speed: speedRev
    //                                 },
    //                                 style: {
    //                                     icon: {
    //                                         type: 'image',
    //                                         url: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speedRev)),
    //                                         size: {
    //                                             width: 30,
    //                                             height: 30
    //                                         },
    //                                         anchor: {
    //                                             x: 15,
    //                                             y: 15
    //                                         },
    //                                         opacity: opacity ?? this.calculateOpacityBasedOnSpeed(speedRev)
    //                                     }
    //                                 }
    //                             },
    //                             {
    //                                 type: 'Feature',
    //                                 id: `speed-${segment.id}-fwd`,
    //                                 geometry: {
    //                                     type: 'Point',
    //                                     coordinates: [rightPoint[0], rightPoint[1]]
    //                                 },
    //                                 properties: {
    //                                     speed: speedFwd
    //                                 },
    //                                 style: {
    //                                     icon: {
    //                                         type: 'image',
    //                                         url: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speedFwd)),
    //                                         size: {
    //                                             width: 30,
    //                                             height: 30
    //                                         },
    //                                         anchor: {
    //                                             x: 15,
    //                                             y: 15
    //                                         },
    //                                         opacity: opacity ?? this.calculateOpacityBasedOnSpeed(speedFwd)
    //                                     }
    //                                 }
    //                             }
    //                         ]
    //                     });
        
    //                     // let leftFeature   = new OpenLayers.Feature.Vector(leftPoint, {speed: speedRev});
    //                     // leftFeature.style = {
    //                     //     graphic: true,
    //                     //     externalGraphic: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speedRev)),
    //                     //     graphicHeight: 30,
    //                     //     graphicWidth: 30,
    //                     //     graphicYOffset: -15,
    //                     //     graphicXOffset: -15,
    //                     //     graphicOpacity: opacity ?? this.calculateOpacityBasedOnSpeed(speedRev)
    //                     // };
        
    //                     // let rightFeature   = new OpenLayers.Feature.Vector(rightPoint, {speed: speedFwd});
    //                     // rightFeature.style = {
    //                     //     graphic: true,
    //                     //     externalGraphic: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speedFwd)),
    //                     //     graphicHeight: 30,
    //                     //     graphicWidth: 30,
    //                     //     graphicYOffset: -15,
    //                     //     graphicXOffset: -15,
    //                     //     graphicOpacity: opacity ?? this.calculateOpacityBasedOnSpeed(speedFwd)
    //                     // };
        
    //                     // this.layer.addFeatures([leftFeature, rightFeature]);
    //                     // this.layer.addFeatures([leftFeature, rightFeature]);
        
    //                 } else {
    //                     let speed;
        
    //                     if (isFwd && speedFwd != null) {
    //                         speed = speedFwd;
        
    //                     } else if (isRev && speedRev != null) {
    //                         speed = speedRev;
    //                     }
        
    //                     speed = isFwd ? speedFwd : speedRev;
        
    //                     this.wmeSdk.Map.addFeatureToLayer({
    //                         layerName: this.layerName,
    //                         feature: {
    //                             type: 'Feature',
    //                             id: `speed-${segment.id}`,
    //                             geometry: {
    //                                 type: 'Point',
    //                                 coordinates: [midpoint[0], midpoint[1]]
    //                             },
    //                             properties: {
    //                                 speed,
    //                             },
    //                             style: {
    //                                 icon: {
    //                                     type: 'image',
    //                                     url: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speed, isTwoWay)),
    //                                     size: {
    //                                         width: 30,
    //                                         height: 30
    //                                     },
    //                                     anchor: {
    //                                         x: 15,
    //                                         y: 15
    //                                     },
    //                                     opacity: opacity ?? this.calculateOpacityBasedOnSpeed(speed)
    //                                 }
    //                             }
    //                         }
    //                     });
        
    //                     // let centerFeature   = new OpenLayers.Feature.Vector(midpoint, {speed});
    //                     // centerFeature.style = {
    //                     //     graphic: true,
    //                     //     externalGraphic: 'data:image/svg+xml;base64,' + btoa(this.getSpeedIcon(speed)),
    //                     //     graphicHeight: 30,
    //                     //     graphicWidth: 30,
    //                     //     graphicYOffset: -15,
    //                     //     graphicXOffset: -15,
    //                     //     graphicOpacity: opacity ?? this.calculateOpacityBasedOnSpeed(speed)
    //                     // };
        
    //                     // this.layer.addFeatures([centerFeature]);
    //                 }
    //             });
        
    //             this.logDebug('Update finished.');

    //             resolve();

    //         } catch (error) {
    //             this.logDebug(error, false, 'error');
                
    //             reject(error);
    //         }
    //     });
    // }

    // TODO: Remove this when SDK migration is complete
    /**
     * Toggle visibility of the display of the speed layer.
     * @param {Boolean} enabled
     */
    toggleLayerVisibility(enabled) {
        if (this.layer) {
            this.layer.setVisibility(enabled);
            
            this.logDebug(`Speed layer ${enabled ? 'enabled' : 'disabled'}.`);
        }
    }
    // TODO: SDK migration
    /**
     * Toggle visibility of the display of the speed layer.
     */
    // toggleLayerVisibility() {
    //     if (this.layer) {
    //         let enabled = this.settings.spdEnabled;

    //         this.wmeSdk.Map.setLayerVisibility({layerName: this.layerName, visibility: this.settings.spdEnabled});
            
    //         this.logDebug(`Speed layer ${enabled ? 'enabled' : 'disabled'}.`);
    //     }
    // }

    /**
     * Saves the current settings to localStorage.
     */
    saveSettings() {
        return new Promise((resolve, reject) => {
            try {
                let selectedTheme = this.getSelectedTheme();
                
                // TODO: Remove this when SDK migration is complete
                this.settings.spdEnabled = this.layerCheckbox.speedDisplay.checked;

                // TODO: SDK migration
                // this.settings.spdEnabled = this.layerSwitcherElement.checked;

                ['general', 'filter', 'roads'].forEach(group => this.updateSettingsGroup(group));

                this.settings.theme = selectedTheme?.id ? this.convertStringCaseStyle('kebab', 'camel', selectedTheme.id.replace('spd-theme-', '')) : 'default';
        
                localStorage.setItem('wmeSpeedDisplaySettings', JSON.stringify(this.settings));

                resolve();
                
                this.channel.postMessage('saved');

            } catch (error) {
                this.logDebug(error, false, 'error');
                
                reject(error);
            }
        });
    }

    /**
     * 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)
            return;

        // To ensure that if it enters a new settings, it doesn't discard them when loading the old settings
        Object.entries(savedSettings).forEach(([setting, value]) => {
            if (typeof value == 'object' && value != null) {
                if (!this.settings[setting])
                    this.settings[setting] = {};
                
                Object.entries(value).forEach(([subSetting, subValue]) => {
                    this.settings[setting][subSetting] = subValue ?? true;
                });
            } else {
                // If the saved theme no longer exists, select the default
                if (setting == 'theme' && !this.theme.includes(value))
                    value = 'default';
                
                this.settings[setting] = value ?? true;
            }
        });
    }

    /**
     * Resets the speed display settings to default.
     */
    async resetSettings() {
        localStorage.removeItem('wmeSpeedDisplaySettings');

        this.zIndex.warnDisplayed = false;

        this.defineSettings();
        this.updateSettingsElementsStates();
        this.resetZindex();
        this.displayZIndexWarning();
        await this.debounce.saveSettings();
        this.preUpdateMapDisplay();
    }

    /**
     * Updates the specified settings group by retrieving values from the corresponding HTML elements.
     * @param {string} group - The settings group to update (e.g., 'general', 'filter', 'roads').
     */
    updateSettingsGroup(group) {
        Object.entries(this.settings[group]).forEach(([setting, _]) => {
            let id      = this.convertStringCaseStyle('camel', 'kebab', setting);
            let element = document.getElementById(id);

            if (!element) {
                this.logDebug(`Setting field not found: ${id}`, false, 'warn');

                return;
            }

            let value;
            
            if (['range', 'number'].includes(element.type)) {
                value = element.value != null ? parseFloat(element.value) : element.value;
                
            } else {
                value = element.checked != undefined ? element.checked : element.value;
            }

            this.settings[group][setting] = value;
        });
    }

    /**
     * Adds a settings tab to the user interface with options to configure the speed display settings.
     */
    addSettingsTab() {
        return new Promise(async (resolve, reject) => {
            try {
                let {tabLabel, tabPane} = await this.wmeSdk.Sidebar.registerScriptTab();
                let speedUnit           = this.wmeSdk.Settings.getUserSettings().isImperial ? I18n.translate('measurements.speed.mi', {speed: ''}) : I18n.translate('measurements.speed.km', {speed: ''});

                tabLabel.innerHTML = this.getTranslation('name');

                let tabPaneHtml = `
                    <div class="segment sidebar-column" style="margin: -15px;">
                        <wz-section-header headline="${this.getTranslation('name')}" size="section-header2" back-button="false">
                            <svg slot="icon" xmlns="http://www.w3.org/2000/svg" width="22px" height="22px" fill-rule="evenodd" clip-rule="evenodd" viewBox="0 0 1.08 1.08" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:xodm="http://www.corel.com/coreldraw/odm/2003">
                                <path fill="var(--leading_icon, #90959c)" d="M0.54 0c0.3,0 0.54,0.24 0.54,0.54 0,0.3 -0.24,0.54 -0.54,0.54 -0.3,0 -0.54,-0.24 -0.54,-0.54 0,-0.3 0.24,-0.54 0.54,-0.54zm-0.18 0.74c0.09,0 0.13,-0.05 0.13,-0.14 0,-0.08 -0.04,-0.13 -0.12,-0.13 -0.01,0 -0.03,0 -0.05,0.01l0.01 -0.06 0.14 0 0 -0.07 -0.2 0 -0.02 0.19 0.07 0.03c0.01,-0.02 0.02,-0.03 0.05,-0.03 0.03,0 0.06,0.02 0.06,0.06 0,0.05 -0.03,0.07 -0.07,0.07 -0.02,0 -0.05,-0.01 -0.08,-0.03l-0.03 0.07c0.03,0.02 0.07,0.03 0.11,0.03zm0.33 -0.33c0.06,0 0.08,0.04 0.08,0.13 0,0.09 -0.03,0.13 -0.08,0.13 -0.05,0 -0.07,-0.04 -0.07,-0.13 0,-0.09 0.02,-0.13 0.07,-0.13zm0 0.33c0.1,0 0.14,-0.07 0.14,-0.2 0,-0.13 -0.04,-0.2 -0.14,-0.2 -0.09,0 -0.14,0.07 -0.14,0.2 0,0.13 0.05,0.2 0.14,0.2zm-0.15 -0.64c0.24,0 0.44,0.2 0.44,0.44 0,0.24 -0.2,0.44 -0.44,0.44 -0.24,0 -0.44,-0.2 -0.44,-0.44 0,-0.24 0.2,-0.44 0.44,-0.44z"/>
                            </svg>
                        </wz-section-header>

                        <div class="segment-edit-section">                  
                            <wz-alerts-group multiline="true" id="spd-alert-group" class="alertsGroup--cx2Lh ${Object.values(this.settings.alertDismissed).every(value => value == true) ? 'd-none' : ''}" style="padding: 0; margin: 15px 15px 15px 15px;"></wz-alerts-group>

                            <wz-tabs>
                                <wz-tab is-active="true" label="${this.getTranslation('title.settings')}" tooltip="${this.getTranslation('title.settings')}">
                                    <div style="display: flex; flex-direction: column; gap: 15px;">
                                        <div>
                                            <wz-label html-for="">${this.getTranslation('label.displaySpeeds')}</wz-label>

                                            <wz-toggle-switch name="spd-display-speeds" checked="${this.settings.spdEnabled}" id="spd-display-speeds" tabindex="0" value="true" disabled="${!this.layerCheckbox.roadGroup.checked}">
                                                <input type="checkbox" name="spd-display-speeds" value="true" style="display: none; visibility: hidden;">
                                            </wz-toggle-switch>
                                        </div>
                                        
                                        <div>
                                            <wz-checkbox checked="${this.settings.general.spdShowNoSpeed}" indeterminate="false" disabled="false" id="spd-show-no-speed" value="true">${this.getTranslation('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.getTranslation('label.showOnRoundabout')}<input type="checkbox" value="true" style="display: none; visibility: hidden;"></wz-checkbox>
                                            <wz-checkbox checked="${this.settings.general.spdShowUnpaved}" indeterminate="false" disabled="false" id="spd-show-unpaved" value="true">${this.getTranslation('label.showOnUnpaved')}<input type="checkbox" value="true" style="display: none; visibility: hidden;"></wz-checkbox>
                                        </div>

                                        <div>
                                            <wz-label html-for="">
                                                ${this.getTranslation('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 class="spd-info-message" style="display: flex; justify-content: space-between; width: 100%;">
                                                <span>${this.getTranslation('zoomOut')}</span><span>${this.getTranslation('zoomIn')}</span>
                                            </div>
                                        </div>

                                        <div>
                                            <wz-label html-for="">
                                                ${this.getTranslation('label.opacity')} <span id="spd-filter-opacity-value">${this.settings.general.spdFilterOpacity}</span>
                                            </wz-label>

                                            <input type="range" id="spd-filter-opacity" min="0" max="1" step="0.1" value="${this.settings.general.spdFilterOpacity}">

                                            <div class="spd-info-message" style="display: flex; justify-content: space-between; width: 100%;">
                                                <span>${this.getTranslation('label.min')} 0</span><span>${this.getTranslation('label.max')} 1</span>
                                            </div>
                                        </div>
                                        
                                        <div class="segment-level-select">
                                            <div class="label-container">
                                                <wz-label html-for="">${this.getTranslation('label.zIndex')}</wz-label>
                                                <wz-button color="text" size="sm" disabled="false" id="spd-btn-reset-z-index">${this.getTranslation('reset')}</wz-button>
                                            </div>

                                            <div class="select-container">
                                                <wz-button color="clear-icon" disabled="false" id="spd-btn-zindex-decrease">
                                                    <i class="w-icon w-icon-minus"></i>
                                                </wz-button>
                                                
                                                <wz-text-input size="sm" type="number" id="spd-layer-zindex" min="${this.zIndex.min}" max="${this.zIndex.max}" value="${this.settings.general.spdLayerZindex}" disabled="false" style="width: 100%;">
                                                    <input style="display: none; visibility: hidden;">
                                                </wz-text-input>
                                                
                                                <wz-button color="clear-icon" disabled="false" id="spd-btn-zindex-increase">
                                                    <i class="w-icon w-icon-plus"></i>
                                                </wz-button>
                                            </div>
                                        </div>
                                    </div>
                                </wz-tab>

                                <wz-tab is-active="false" label="${this.getTranslation('title.roadTypes')}" tooltip="${this.getTranslation('title.roadTypes')}">
                                    <div>
                                        <wz-label html-for="" style="margin:0">${this.getTranslation('title.showOnRoadType')}</wz-label>`;

                Object.entries(this.segmentsCategories).forEach(segmentCategory => {
                    tabPaneHtml += `<wz-menu-title style="padding:0;">${this.wmeTranslations.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];

                        tabPaneHtml += `<wz-checkbox checked="${checked}" indeterminate="false" disabled="false" id="${id}" value="true">${this.wmeTranslations.segment.road_types[roadType.id]}<input type="checkbox" value="true" style="display: none; visibility: hidden;"></wz-checkbox>`;
                    });
                });

                tabPaneHtml += `
                                    </div>
                                </wz-tab>

                                <wz-tab is-active="false" label="${this.getTranslation('title.filter')}" tooltip="${this.getTranslation('title.filter')}">
                                    <div style="display: flex; flex-direction: column; gap: 15px;">
                                        <div>
                                            <wz-label html-for="">${this.getTranslation('label.hideShorterSegments')}</wz-label>

                                            <div class="speedLimit--ZIWbK" style="margin-bottom: 0">
                                                <wz-text-input size="sm" type="number" id="spd-hide-segment-smaller-than"  min="0" max="9999" value="${this.settings.filter.spdHideSegmentSmallerThan}" disabled="false" class="speedLimitInput--aWKs1">
                                                    <input style="display: none; visibility: hidden;">
                                                </wz-text-input>

                                                <span class="spd-filter-distance-unit">${this.wmeSdk.Settings.getUserSettings().isImperial ? this.wmeTranslations.units_short.feet : this.wmeTranslations.units_short.meters}</span>
                                            </div>
                                        </div>
                                        
                                        <div>
                                            <wz-label html-for="">${this.getTranslation('label.speedFilter')}</wz-label>
                                            <div class="speedLimit--ZIWbK">
                                                <div class="directionLabel--U8iep">${this.getTranslation('label.min')}</div>

                                                <wz-text-input size="sm" type="number" id="spd-min-speed" min="0" max="999" value="${this.settings.filter.spdMinSpeed}" disabled="false" class="speedLimitInput--aWKs1">
                                                    <input style="display: none; visibility: hidden;">
                                                </wz-text-input>

                                                <span class="spd-filter-speed-unit">${speedUnit}</span>
                                            </div>

                                            <div class="speedLimit--ZIWbK"  style="margin-bottom: 0">
                                                <div class="directionLabel--U8iep">${this.getTranslation('label.max')}</div>

                                                <wz-text-input size="sm" type="number" id="spd-max-speed" min="0" max="999" value="${this.settings.filter.spdMaxSpeed}" disabled="false" class="speedLimitInput--aWKs1">
                                                    <input style="display: none; visibility: hidden;">
                                                </wz-text-input>

                                                <span class="spd-filter-speed-unit">${speedUnit}</span>
                                            </div>
                                        </div>
                                    </div>
                                </wz-tab>

                                <wz-tab is-active="false" label="${this.getTranslation('title.appearence')}" tooltip="${this.getTranslation('title.appearence')}">
                                    <div style="display: flex; flex-direction: column; gap: 15px;">
                                        <div class="drives-container">
                                            <div class="drive-list">`;
                
                this.theme.forEach(theme => {
                    tabPaneHtml += `
                                                <wz-card id="spd-theme-${this.convertStringCaseStyle('camel', 'kebab', theme)}" selected="${this.settings.theme == theme}" elevation-on-hover="4" class="list-item-card drive-list-item">
                                                    <div class="list-item-card-layout">
                                                        ${this.getSpeedIcon(60, theme == 'uk', theme)}

                                                        <div class="list-item-card-info">
                                                            <div class="list-item-card-title">${this.getTranslation('theme.' + theme)}</div>
                                                        </div>
                                                    </div>
                                                </wz-card>`;
                });
                
                tabPaneHtml += `
                                            </div>
                                        </div>
                                    </div>
                                </wz-tab>
                            </wz-tabs>
                            
                            <div class="form-group">
                                <div style="display: flex; flex-direction: column; gap: 15px; padding: 0 15px;">
                                    <wz-button color="secondary" id="spd-reset-default-settings">${this.getTranslation('btn.resetSettings')}</wz-button>`;

                if (this.debugMode) {
                    tabPaneHtml += `
                                    <wz-button color="primary" id="btn-spd-update-settings-tab">${this.getTranslation('btn.updateSettingsTab')}</wz-button>
                                    <wz-button color="secondary" id="btn-spd-test">Test something</wz-button>`;
                }

                tabPaneHtml += `
                                    <wz-button color="text" size="sm" id="btn-spd-open-changelog-dialog" title="${this.getTranslation('viewChangelog')}"><b>${this.getTranslation('name')}</b> v${this.version}</wz-button>

                                    <a href="https://www.waze.com/discuss/t/script-wme-speed-display/373172" target="_blank" style="width: 100%">
                                        <wz-button color="text" size="sm" title="${this.getTranslation('title.seeOnDiscuss')}" style="width: 100%">${this.getTranslation('btn.seeOnDiscuss')}</wz-button>
                                    </a>

                                    <a href="https://discord.gg/T6cDTGZj" target="_blank" style="width: 100%">
                                        <wz-button color="text" size="sm" title="${this.getTranslation('title.seeOnDiscord')}" style="width: 100%">${this.getTranslation('btn.seeOnDiscord')}</wz-button>
                                    </a>

                                    <a href="https://www.waze.com/pt-BR/user/editor/luan_tavares_127" target="_blank" style="width: 100%">
                                        <wz-button color="text" size="sm" title="${this.getTranslation('title.scriptDevBy')}" style="width: 100%">${this.getTranslation('btn.scriptDevBy')}</wz-button>
                                    </a>
                                </div>
                            </div>
                        </div>
                    </div>`;

                tabPane.innerHTML = tabPaneHtml;

                if (!this.settings.alertDismissed.infoToggleVisibility)
                    this.addAlert('info-toggle-visibility', this.getTranslation('alertMessage.toggleVisibility'));
                
                if (!this.zIndex.warnDisplayed)
                    this.displayZIndexWarning();

                this.displaySpeedsSwitch = document.getElementById('spd-display-speeds');
                
                this.attachSettingsListeners();

                this.logDebug('Settings tab added.');

                resolve();

            } catch (error) {
                this.logDebug(error, false, 'error');
                
                reject(error);
            }
        });
    }

    /**
     * Updates the settings tab by removing the old one and adding a new one with updated settings.
     */
    async updateSettingsTab() {
        this.logDebug('Updating settings tab...');

        this.detachSettingsListeners();
        this.defineTranslations();

        this.wmeSdk.Sidebar.removeScriptTab();

        await this.addSettingsTab();
        this.updateSettingsElementsStates();
        
        if (!this.zIndex.warnDisplayed)
            this.displayZIndexWarning();
    }

    /**
     * Updates the state of the settings elements (checkboxes, inputs) based on the current settings.
     */
    updateSettingsElementsStates() {
        // TODO: Remove this when SDK migration is complete
        this.layerCheckbox.speedDisplay.checked = this.settings.spdEnabled;
        
        // TODO: SDK migration
        // this.layerSwitcherElement.checked = this.settings.spdEnabled;

        if (this.displaySpeedsSwitch) {
            this.displaySpeedsSwitch.checked  = this.settings.spdEnabled;
            this.displaySpeedsSwitch.disabled = !this.layerCheckbox.roadGroup.checked;
        }
        
        Object.entries(this.settings.general).forEach(([setting, value]) => {
            let id      = this.convertStringCaseStyle('camel', 'kebab', setting);
            let element = document.getElementById(id);

            if (!element)
                return;

            switch (element.type) {
                case 'number':
                    element.value = (value == null || value == 0) ? '' : value;

                    break;
                    
                case 'range':
                    element.value                                    = value;
                    document.getElementById(`${id}-value`).innerText = value;

                    break;

                default:
                    element.checked = value;
            }
        });

        document.querySelectorAll('[id^="spd-show-speed-in-"]').forEach(input => {
            let settingId = this.convertStringCaseStyle('kebab', 'camel', input.id);

            input.checked = this.settings.roads[settingId];
        });

        Object.entries(this.settings.filter).forEach(([setting, value]) => {
            let id      = this.convertStringCaseStyle('camel', 'kebab', setting);
            let element = document.getElementById(id);

            if (!element)
                return;

            element.value = (value == null || value == 0) ? '' : value;
        });

        this.selectThemeCard(document.querySelector(`wz-card#spd-theme-${this.convertStringCaseStyle('camel', 'kebab', this.settings.theme)}`));
    }

    /**
     * Adds a global keyboard shortcut listener and add the script shortcut on shortcuts dialog.
     */
    addShortcut() {
        this.wmeSdk.Shortcuts.createShortcut({
            callback: this.shortcutCallback.bind(this),
            description: this.getTranslation('label.toggleLayerShortcut'),
            shortcutId: 'ks-wme-speed-display',
            shortcutKeys: 'AS+s',
        });
    }

    /**
     * Toggles the layer switcher speed display when shortcut is pressed.
     */
    shortcutCallback() {
        // TODO: Remove this when SDK migration is complete
        this.layerCheckbox.speedDisplay.click();
        
        this.displaySpeedsSwitch.checked = this.layerCheckbox.speedDisplay.checked;
        // TODO: Remove this when SDK migration is complete - end
        
        // TODO: SDK migration
        // this.layerSwitcherElement.click();

        // this.displaySpeedsSwitch.checked = this.layerSwitcherElement.checked;
        // TODO: SDK migration - end
    }

    // TODO: Remove this when SDK migration is complete
    /**
     * 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();
    }
    // TODO: SDK migration
    /**
     * Calculates the real midpoint of a segment, considering curves.
     * @param {Array} coordinates - Array of [longitude, latitude] pairs representing the segment geometry.
     *
     * @returns {Array} The midpoint as an array [longitude, latitude].
     */
    // calculateMidpoint(coordinates) {
    //     let totalLength      = this.getLength(coordinates);
    //     let cumulativeLength = 0;
    
    //     for (let i = 0; i < coordinates.length - 1; i++) {
    //         let start         = coordinates[i];
    //         let end           = coordinates[i + 1];
    //         let segmentLength = this.getDistance(start, end);
    
    //         if (cumulativeLength + segmentLength >= totalLength / 2) {
    //             let ratio = (totalLength / 2 - cumulativeLength) / segmentLength;
    
    //             return [
    //                 start[0] + ratio * (end[0] - start[0]),
    //                 start[1] + ratio * (end[1] - start[1])
    //             ];
    //         }
    
    //         cumulativeLength += segmentLength;
    //     }
    
    //     // If something goes wrong, return a simple midpoint as a fallback
    //     let firstPoint = coordinates[0];
    //     let lastPoint  = coordinates[coordinates.length - 1];
    
    //     return [
    //         (firstPoint[0] + lastPoint[0]) / 2,
    //         (firstPoint[1] + lastPoint[1]) / 2
    //     ];
    // }

    /**
     * Generates an SVG icon with the speeds.
     * @param {String} speed The speed.
     * @param {Boolean} [sameSpeedBothWays] It is the segment two ways with same speed? (default: false)
     * @param {String} [theme] Force a theme. (default: this.settings.theme)
     *
     * @returns {String} SVG in string format.
     */
    getSpeedIcon(speed, sameSpeedBothWays = false, theme) {
        let icon;

        theme = theme ?? this.settings.theme;

        switch (theme) {
            case 'altViennaConv':
                icon = `<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="#FBE353" stroke="#FF5251" stroke-width="5"/>
                        <text x="25" y="31" font-size="20" font-family="Arial" font-weight="bold" fill="#1F2125" text-anchor="middle">${speed}</text>
                    </svg>`;
                
                break;
                
            case 'japan':
                icon = `<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="#FF5251" stroke-width="5"/>
                        <text x="25" y="31" font-size="20" font-family="Arial" font-weight="bold" fill="#0074E3" text-anchor="middle">${speed}</text>
                    </svg>`;
                
                break;

            case 'northAmerica':
                icon = `
                    <svg xmlns="http://www.w3.org/2000/svg" width="50" height="50" viewBox="0 0 50 50">
                        <rect x="2.5" y="2.5" width="45" height="45" fill="white" stroke="#1F2125" stroke-width="3" rx="5"/>
                        <text x="25" y="12" font-size="9" font-family="Arial" font-weight="bold" fill="#1F2125" text-anchor="middle">SPEED</text>
                        <text x="25" y="22" font-size="9" font-family="Arial" font-weight="bold" fill="#1F2125" text-anchor="middle">LIMIT</text>
                        <text x="25" y="42" font-size="22" font-family="Arial" font-weight="bold" fill="#1F2125" text-anchor="middle">${speed}</text>
                    </svg>`;

                break;
                
            case 'sacu':
                icon = `<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="#235AA9" stroke="#FF5251" stroke-width="5"/>
                        <text x="25" y="31" font-size="20" font-family="Arial" font-weight="bold" fill="white" text-anchor="middle">${speed}</text>
                    </svg>`;
                
                break;

            case 'uk':
                if (sameSpeedBothWays && [60, 70].includes(speed)) {
                    icon = `<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="#1F2125" stroke-width="2"/>
                            <line x1="10" y1="40" x2="40" y2="10" stroke="#1F2125" stroke-width="9"/>
                        </svg>`;

                } else {
                    icon = `<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="#FF5251" stroke-width="5"/>
                            <text x="25" y="31" font-size="20" font-family="Arial" font-weight="bold" fill="#1F2125" text-anchor="middle">${speed}</text>
                        </svg>`;
                }
                
                break;

            default:
                icon = `<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="#FF5251" stroke-width="5"/>
                        <text x="25" y="31" font-size="20" font-family="Arial" font-weight="bold" fill="#1F2125" text-anchor="middle">${speed}</text>
                    </svg>`;
        }
        
        return icon;
    }

    /**
     * Calculates the opacity of a speed icon based on speed filter settings.
     * @param {number} speed - The speed value to evaluate.
     * 
     * @returns {number} Opacity value (1 for visible, this.settings.general.spdFilterOpcaity for dimmed) depending on whether the speed falls within the defined min and max speed range.
     */
    calculateOpacityBasedOnSpeed(speed) {
        let minSpeed = parseInt(this.settings.filter.spdMinSpeed, 10);
        let maxSpeed = parseInt(this.settings.filter.spdMaxSpeed, 10);
        let validMin = isNaN(minSpeed) || minSpeed <= 0 || speed >= minSpeed;
        let validMax = isNaN(maxSpeed) || maxSpeed <= 0 || speed <= maxSpeed;
        let opacity  = this.getSettingOpacity();
    
        return validMin && validMax ? 1 : opacity;
    }

    /**
     * Get the opacity value defined in the settings.
     * @returns {number} Opacity value.
     */
    getSettingOpacity() {
        return this.settings.general.spdFilterOpacity < 0.1 ? 0.01 : this.settings.general.spdFilterOpacity;
    }

    // TODO: Remove this when SDK migration is complete
    /**
     * Calculates the angle at the midpoint of a geometry based on its components.
     * @param {Object} geometry - The geometry with points.
     * 
     * @returns {number} The angle in radians at the midpoint.
     */
    calculateAngleAtMidpoint(geometry) {
        let totalLength      = 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];

            if (!start || !end) continue;

            let segmentLength = start.distanceTo(end);

            if (cumulativeLength + segmentLength >= totalLength / 2)
                return Math.atan2(end.y - start.y, end.x - start.x);

            cumulativeLength += segmentLength;
        }

        if (geometry.components.length >= 2) {
            let firstPoint = geometry.components[0];
            let lastPoint  = geometry.components[geometry.components.length - 1];

            return Math.atan2(lastPoint.y - firstPoint.y, lastPoint.x - firstPoint.x);
        }

        this.logDebug(this.spdTranslations.log.insufficientGeometryAngleCalc, false, 'error');

        return 0;
    }
    // TODO: SDK migration
    /**
     * Calculates the angle at the midpoint of a geometry based on its components.
     * @param {Array} coordinates - The geometry coordinates.
     * 
     * @returns {Number} The angle in radians at the midpoint.
     */
    // calculateAngleAtMidpoint(coordinates) {
    //     let totalLength      = this.getLength(coordinates);
    //     let cumulativeLength = 0;

    //     for (let i = 0; i < coordinates.length - 1; i++) {
    //         let start = coordinates[i];
    //         let end   = coordinates[i + 1];

    //         if (!start || !end) continue;

    //         let segmentLength = this.getDistance(start, end);

    //         if (cumulativeLength + segmentLength >= totalLength / 2)
    //             return Math.atan2(end.y - start.y, end.x - start.x);

    //         cumulativeLength += segmentLength;
    //     }

    //     if (coordinates.length >= 2) {
    //         let firstPoint = coordinates[0];
    //         let lastPoint  = coordinates[coordinates.length - 1];

    //         return Math.atan2(lastPoint.y - firstPoint.y, lastPoint.x - firstPoint.x);
    //     }

    //     this.logDebug('Insufficient geometry for angle calculation.', false, 'error');

    //     return 0;
    // }

    /**
     * Calculates the distance in meters between two geographic coordinates using the Haversine formula.
     * @param {Array} startPoint - The starting point [longitude, latitude].
     * @param {Array} endPoint - The ending point [longitude, latitude].
     *
     * @returns {Number} The distance in meters between the two points.
     */
    getDistance(startPoint, endPoint) {
        let earthRadius      = 6371000;
        let latitude1        = this.toRadians(startPoint[1]);
        let latitude2        = this.toRadians(endPoint[1]);
        let deltaLatitude    = this.toRadians(endPoint[1] - startPoint[1]);
        let deltaLongitude   = this.toRadians(endPoint[0] - startPoint[0]);
        let haversineFormula = Math.sin(deltaLatitude / 2) * Math.sin(deltaLatitude / 2) + Math.cos(latitude1) * Math.cos(latitude2) * Math.sin(deltaLongitude / 2) * Math.sin(deltaLongitude / 2);
        let centralAngle     = 2 * Math.atan2(Math.sqrt(haversineFormula), Math.sqrt(1 - haversineFormula));
    
        return earthRadius * centralAngle;
    }

    /**
     * Converts an angle from degrees to radians.
     * @param {Number} angle - The angle in degrees.
     *
     * @returns {Number} The angle in radians.
     */
    toRadians(angle) {
        return angle * (Math.PI / 180);
    }

    /**
     * Returns two perpendicular points (left and right) from a midpoint at a given angle and distance, adjusted by map zoom.
     * @param {Object} midpoint - Central point with `x` and `y` coordinates.
     * @param {Number} angle - Angle in radians from the reference direction.
     * @param {Number} baseDistance - Base distance from the midpoint before zoom adjustment.
     * 
     * @returns {Object} {leftPoint, rightPoint} - Perpendicular points to the left and right.
     */
    getPerpendicularPoints(midpoint, angle, baseDistance) {
        let currentZoom      = this.wmeSdk.Map.getZoomLevel();
        let referenceZoom    = 19;
        let scaleFactor      = Math.pow(2, referenceZoom - currentZoom);
        let adjustedDistance = baseDistance * scaleFactor;

        let angleLeft  = angle + Math.PI / 2;
        let angleRight = angle - Math.PI / 2;

        // TODO: Remove this when SDK migration is complete
        let leftPoint = new OpenLayers.Geometry.Point(
            midpoint.x + adjustedDistance * Math.cos(angleLeft),
            midpoint.y + adjustedDistance * Math.sin(angleLeft)
        );

        let rightPoint = new OpenLayers.Geometry.Point(
            midpoint.x + adjustedDistance * Math.cos(angleRight),
            midpoint.y + adjustedDistance * Math.sin(angleRight)
        );
        // TODO: Remove this when SDK migration is complete - end

        // TODO: SDK migration
        // let leftPoint = [
        //     midpoint.x + adjustedDistance * Math.cos(angleLeft),
        //     midpoint.y + adjustedDistance * Math.sin(angleLeft)
        // ];

        // let rightPoint = [
        //     midpoint.x + adjustedDistance * Math.cos(angleRight),
        //     midpoint.y + adjustedDistance * Math.sin(angleRight)
        // ];
        // TODO: SDK migration -end

        return {
            leftPoint,
            rightPoint
        };
    }

    /**
     * Returns speed in mph/kmh or 'N/A'.
     * @param {Number} speed The speed value in kilometers per hour (km/h) to be formatted.
     * 
     * @returns {(Number|String)} The speed.
     */
    getSpeed(speed) {
        if (!speed)
            return 'N/A';

        return this.wmeSdk.Settings.getUserSettings().isImperial ? Math.round(speed * 0.621371) : speed;
    }

    /** 
     * Returns length in feet (rounded) if imperial units are preferred; otherwise, returns meters. 
     * @param {Array} coordinates - The length value in meters to be converted.
     * @param {Boolean} [forceMeters] - If needs to force results in meters.
     *
     * @returns {Number} The converted length in feet or the original length in meters.
     */
    getLength(coordinates, forceMeters = false) {
        let totalDistance = 0;
    
        // Iterate over the coordinates and add the distance between consecutive points
        for (let i = 0; i < coordinates.length - 1; i++) {
            let pointA = {
                lat: coordinates[i][1],
                lng: coordinates[i][0]
            };

            let pointB = {
                lat: coordinates[i + 1][1],
                lng: coordinates[i + 1][0]
            };
    
            totalDistance += this.calculateDistance(pointA, pointB);
        }
    
        return this.wmeSdk.Settings.getUserSettings().isImperial && !forceMeters ? Math.round(totalDistance * 3.28084) : totalDistance;
    }

    /**
     * Calculates the distance between two points using the Haversine formula.
     * @param {Array} pointA The first point, containing latitude (lat) and longitude (lng).
     * @param {Array} pointB The second point, containing latitude (lat) and longitude (lng).
     * @returns 
     */
    calculateDistance(pointA, pointB) {
        let earthRadius = 6371e3;
        let latitudeA   = pointA.lat * Math.PI / 180;
        let latitudeB   = pointB.lat * Math.PI / 180;
    
        // Calculate the differences in coordinates
        let deltaLatitude  = (pointB.lat - pointA.lat) * Math.PI / 180;
        let deltaLongitude = (pointB.lng - pointA.lng) * Math.PI / 180;
    
        // Haversine formula
        let haversineComponent = Math.sin(deltaLatitude / 2) * Math.sin(deltaLatitude / 2) + Math.cos(latitudeA) * Math.cos(latitudeB) * Math.sin(deltaLongitude / 2) * Math.sin(deltaLongitude / 2);
        let angularDistance    = 2 * Math.atan2(Math.sqrt(haversineComponent), Math.sqrt(1 - haversineComponent));
    
        // Calculate the distance in meters
        let distance = earthRadius * angularDistance;

        return distance;
    }

    /**
     * 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} [timeout] Delay time in milliseconds. (default: 300)
     *
     * @returns {Function} Function
     */
    debounce(func, timeout = 300) {
        let promise = null;
        let timer;
    
        return (...args) => {
            clearTimeout(timer);
    
            return new Promise((resolve, reject) => {
                timer = setTimeout(async () => {
                    try {
                        promise    = func.apply(this, args);
                        let result = await promise;

                        resolve(result);
                        
                    } catch (error) {
                        reject(error);
                    }
                }, 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, (_, letter) => letter.toUpperCase())
            },
            kebab: {
                camel: str => str.replace(/-([a-z])/g, (_, letter) => letter.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.
    * @param {String} [type] - The type of message ('warn', or 'error'). Default is 'log'.
     */
    logDebug(message, forceMessage = false, type) {
        if (this.debugMode || forceMessage)
            (console[type] || console.log)(`[WME Speed Display v${this.version}]: ${message}`);
    }

    // TODO: Remove this when SDK migration is complete
    /**
     * Adds a checkbox to the layer switcher to toggle the speed display layer visibility.
     */
    addLayerCheckbox() {
        let houseNumbersCheckbox     = document.getElementById('layer-switcher-item_house_numbers');
        this.layerCheckbox.roadGroup = document.getElementById('layer-switcher-group_road');

        if (!houseNumbersCheckbox) {
            this.logDebug('The "House Numbers" checkbox was not found. Trying again...', false, 'warn');

            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}" disabled="${!this.layerCheckbox.roadGroup.checked}">
                    <div class="layer-selector-container" title="WME ${this.getTranslation('name')}">${this.getTranslation('label.displaySpeeds')}</div>
                </wz-checkbox>
            </div>`;

        houseNumbersCheckbox.closest('li').insertAdjacentElement('afterend', layerItem);

        this.layerCheckbox.speedDisplay = document.getElementById('layer-switcher-item_speed_display');

        if (!this.layerCheckbox.speedDisplay)
            return;

        this.layerCheckbox.speedDisplay.addEventListener('change', this.onSpeedDisplayCheckboxChange.bind(this));
        this.layerCheckbox.roadGroup.addEventListener('change', this.onLayerSwithcerGroupRoadChange.bind(this));

        this.toggleLayerVisibility(this.layerCheckbox.roadGroup.checked && this.settings.spdEnabled);
    }

    /**
     * Initializes settings from localStorage or sets default values.
     */
    defineSettings() {
        let localSettings = localStorage.getItem('wmeSpeedDisplaySettings');
        localSettings     = localSettings ? JSON.parse(localSettings) : null;

        this.settings = {
            spdEnabled: true,
            alertDismissed: {
                infoToggleVisibility: false,
                infoZindex: false
            },
            theme: 'default',
            general: {
                spdShowNoSpeed: true,
                spdShowRoundabout: true,
                spdShowUnpaved: true,
                spdMaxZoom: 18,
                spdLayerZindex: this.zIndex.min,
                spdFilterOpacity: 0.2
            },
            roads: {},
            filter: {
                spdHideSegmentSmallerThan: 0,
                spdMinSpeed: 0,
                spdMaxSpeed: 0
            }
        };

        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;
            });
        });

        if (localSettings)
            this.loadSettings(localSettings);
    }

    /**
     * Checks if the plugin version changed and displays the changelog if necessary.
     */
    checkVersionAndShowChangelog() {
        this.logDebug('Checking plugin version.');
        
        let localSpdVersion = JSON.parse(localStorage.getItem('wmeSpeedDisplayVersion'));

        if (localSpdVersion == null || localSpdVersion != this.version) {
            if (this.needResetSettings && !this.debugMode)
                this.resetSettings();
        
            this.hideSpeedDisplayDialog();
            this.openChangelogDialog();

            localStorage.setItem('wmeSpeedDisplayVersion', JSON.stringify(this.version));
        }
    }
    
    /**
     * Opens changelog dialog.
     */
    openChangelogDialog() {
        this.showSpeedDisplayDialog(`${this.getTranslation('changelog')} ${this.getTranslation('name')} v${this.version}`, this.getTranslation('changelogMessage'), null, `<a href="https://greasyfork.org/scripts/526702-wme-speed-display" target="_blank"><wz-button color="primary">${this.getTranslation('viewScript')}</wz-button></a>`);
    }

    /**
     * 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} [closeButtonText] - The text for the close button (default: 'Close').
     * @param {String} [customHtmlButtons] - Custom HTML button(s).
     */
    showSpeedDisplayDialog(title, body = '<span>Without body message</span>', closeButtonText, customHtmlButtons) {
        title                 = title && typeof title == 'string' ? title : 'Without title';
        closeButtonText       = closeButtonText ?? this.getTranslation('close');
        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
        });

        let divDialogHtml = `
            <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; gap: 10px; margin-top: 10px;">
                    <wz-button color="secondary" id="btn-spd-hide-dialog">${closeButtonText}</wz-button>`;

        if (customHtmlButtons)
            divDialogHtml += `${customHtmlButtons}`;
        
        divDialogHtml += `
                </div>
            </div>`;

        divDialog.innerHTML = divDialogHtml;

        wzDialogContainer.appendChild(divDialog);
        
        this.attachDialogListeners();
    }

    /**
     * Closes the dialog and removes associated event listeners.
     */
    hideSpeedDisplayDialog() {
        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 && !['Escape', 'Enter'].includes(event.key))
            return;

        document.getElementById('btn-spd-hide-dialog')?.click();
    }

    /**
     * Adds an alert to the warning group.
     * 
     * @param {String} id - The unique identifier for the alert.
     * @param {(String|Element)} body - The content of the alert, either as a string or an HTML element.
     * @param {String} [variant] - The alert variant ('info', 'warning', 'danger' and 'success'). (default: 'info')
     * @param {Boolean} [truncable] - Whether the alert content should be truncated. (default: true)
     */
    addAlert(id, body, variant = 'info', truncable = true) {
        let warningGroup = document.getElementById('spd-alert-group');
        let alertHtml    = `
            <wz-alert id="${id}" role="alert" level="page" variant="${variant}" multiline="true" class="sidebar-alert inconsistent-direction-alert">
                <div class="alertContent--O8sZl">`;

        if (truncable) {
            alertHtml +=`
                    <wz-truncate is-expanded="false" limit="50">
                        <wz-body2>${body}</wz-body2>
                        <span slot="ellipsis">${this.wmeTranslations.common.show_more}</span>
                    </wz-truncate>`;
        } else {
            alertHtml +=`<wz-body2>${body}</wz-body2>`;
        }

        alertHtml += `
                </div>

                <wz-button size="sm" color="text" slot="dismiss" class="spd-btn-dismiss-alert">${this.wmeTranslations.zoom_in_message.ok}</wz-button>
            </wz-alert>`;

        warningGroup.classList.remove('d-none');
            
        warningGroup.insertAdjacentHTML('beforeend', alertHtml);

        document.querySelector(`#${id} .spd-btn-dismiss-alert`).addEventListener('click', this.onDismissAlert.bind(this), {once: true});
    }

    /**
     * Adds an snackbar on message container.
     * @param {String} id Snackbar ID
     * @param {(String|Element)} body Snackbar message
     * @param {String} [btnMsg] Snackbar button text (default: no button)
     * @param {Boolean} [loadingSnackbar] Is loading snackbar? (default: false)
     */
    addSnackbar(id, body, btnMsg, loadingSnackbar = false) {
        let snackbarClass = loadingSnackbar ? 'spd-snackbar spd-snackbar-loading' : 'spd-snackbar';
        
        if (id instanceof Event) {
            id     = 'snackbar-id-' + Math.floor(Math.random() * 100000);
            btnMsg = `<wz-button color="text">${this.wmeTranslations.zoom_in_message.ok}</wz-button>`;
        }

        if (document.getElementById(id))
            return;

        let snackbarHtml = `
            <div class="${snackbarClass}" id="${id}">
                <span class="text-wrapper">
                    ${body}
                </span>`;

        if (btnMsg)
            snackbarHtml += btnMsg;

        snackbarHtml += `
            </div>`;

        this.snackbarWrapper.insertAdjacentHTML('beforeend', snackbarHtml);

        if (btnMsg)
            document.querySelector(`.spd-snackbar#${id} wz-button`).addEventListener('click', this.removeSnackbar.bind(this), {once: true});
    }

    /**
     * Remove current snackbar.
     * @param {(Element|Event)} element - The element or event triggering the removal.
     */
    removeSnackbar(element) {
        if (element instanceof Event)
            element = element.currentTarget;

        element?.closest('.spd-snackbar')?.remove();
    }

    /**
     * Remove all snackbars.
     */
    removeAllSnackbars() {
        let allSnackbars = document.querySelectorAll('.spd-snackbar');

        allSnackbars.forEach(snackbar => snackbar.remove());
    }

    /**
     * Create a custom manual snackbar.
     */
    createSnackbarWrapper() {
        let msgContainer     = document.getElementById('map-message-container');
        this.snackbarWrapper = document.querySelector('.spd-snackbar-wrapper');

        if (!this.snackbarWrapper) {
            this.snackbarWrapper = document.createElement('div');
            this.snackbarWrapper.classList.add('spd-snackbar-wrapper');

            msgContainer.appendChild(this.snackbarWrapper);
        }
    }

    /**
     * Adds script custom stylesheet.
     */
    addStyle() {
        let style = document.createElement('style');

        style.innerHTML = `
            /* General */
            .d-none {
                display: none;
            }

            .spd-info-message {
                color: #777;
                font-size: 11px;
                line-height: 15px;
            }
            
            wz-card[id^="spd-theme-"] * {
                pointer-events: none;
            }
                
            /* Snackbar */
            #map-message-container {
                display: flex;
                flex-direction: column-reverse;
                gap: 15px;
            }
            
            .spd-snackbar-wrapper {
                display: flex;
                flex-direction: column-reverse;
                gap: 15px;
                align-items: flex-start;
                justify-content: flex-start;
                position: absolute;
                top: 0;
                bottom: 0;
                left: 0;
                right: 0;
                translate: 0 -15px;
                z-index: 10001;
                pointer-events: none;
            }

            .spd-snackbar {
                display: flex;
                align-items: center;
                gap: 15px;
                position: relative;
                padding: var(--space-m, 16px) var(--space-always-m, 16px);
                bottom: var(--wz-snackbar-bottom, initial);
                left: 50%;
                translate: -50%;
                min-width: var(--wz-snackbar-min-width, 195px);
                background-color: rgba(0, 0, 0, 0.8);
                border: none;
                border-radius: 6px;
                box-shadow: 0 1px 2px 0 rgba(60, 64, 67, 0.3), 0 2px 6px 2px rgba(60, 64, 67, 0.15);
                box-sizing: border-box;
                color: var(--wz-snackbar-color, var(--always_white, #ffffff));
                font-family: "Rubik", sans-serif;
                font-size: 14px;
                font-weight: 400;
                justify-content: space-between;
                pointer-events: all;
                letter-spacing: 0.2px;
                white-space: normal;
            }

            .spd-snackbar.spd-snackbar-loading {
                justify-content: center;
                background-color: transparent;
                box-shadow: none;
                min-width: auto;
            }
            
            .spd-snackbar-wrapper .text-wrapper {
                display: flex;
                align-items: center;
                gap: 15px;
            }

            .spd-snackbar-wrapper i.w-icon {
                font-size: 20px;
            }
                
            .spd-snackbar-wrapper wz-button {
                pointer-events: all;
            }
                
            .loader {
                width: 40px;
                aspect-ratio: 1;
                display: flex;
            }

            .loader::before,
            .loader::after {
                content: "";
                flex: 1;
                animation: l14 1.5s -.25s infinite linear alternate both;
            }

            .loader::after {
                --s: -1, -1;
                animation-delay: .5s;
            }

            @keyframes l14 {
                0%,
                30% {
                        transform: scale(var(--s, 1)) translate(0px) perspective(150px) rotateY(0deg);
                        clip-path: polygon(50% 0, 100% 0, 100% 100%, 50% 100%, 0 75%, 0 25%);
                        background: #FF5251;
                        border: 7px solid #FF5251;
                        border-right: none;
                }

                37.5% {
                        transform: scale(var(--s, 1)) translate(-10px) perspective(150px) rotateX(0deg);
                        clip-path: polygon(50% 0, 100% 0, 100% 100%, 50% 100%, 0 75%, 0 25%);
                        background: #FF5251;
                        border: 7px solid #FF5251;
                        border-right: none;
                }

                50% {
                        transform: scale(var(--s, 1)) translate(-10px) perspective(150px) rotateX(-90deg);
                        clip-path: polygon(50% 0, 100% 0, 100% 100%, 50% 100%, 0 75%, 0 25%);
                        background: #FF5251;
                        border: 7px solid #FF5251;
                        border-right: none;
                        border-radius: 0;
                }

                50.01% {
                        transform: scale(var(--s, 1)) translate(-10px) perspective(150px) rotateX(-90.1deg);
                        clip-path: polygon(0 0, 100% 0, 100% 100%, 50% 100%, 0 100%, 0 25%);
                        background: #fff;
                        border: 7px solid #FF5251;
                        border-right: none;
                        border-radius: 100px 0 0 100px;
                }

                62.5% {
                        transform: scale(var(--s, 1)) translate(-10px) perspective(150px) rotateX(-180deg);
                        clip-path: polygon(0 0, 100% 0, 100% 100%, 50% 100%, 0 100%, 0 25%);
                        background: #fff;
                        border: 7px solid #FF5251;
                        border-right: none;
                        border-radius: 100px 0 0 100px;
                }

                70%,
                100% {
                        transform: scale(var(--s, 1)) translate(0px) perspective(150px) rotateX(-180deg);
                        clip-path: polygon(0 0, 100% 0, 100% 100%, 50% 100%, 0 100%, 0 25%);
                        background: #fff;
                        border: 7px solid #FF5251;
                        border-right: none;
                        border-radius: 100px 0 0 100px;
                }
            }`;

        document.head.appendChild(style);
    }

    /**
     * Get the layer switcher element.
     */
    getLayerSwitcher() {
        let userScriptLayerSwitchers = document.querySelectorAll('.layer-selector-sdk-checkbox');

        if (userScriptLayerSwitchers.length < 1) {
            this.logDebug('Userscripts layer switches not found. Trying again...', false, 'warn');

            setTimeout(() => this.getLayerSwitcher(), 1000);
            
            return;
        }
        
        this.layerSwitcherElement = [...userScriptLayerSwitchers].find(checkbox => {
            return checkbox.querySelector('.layer-selector-container').textContent.trim() == this.getTranslation('name');
        });

        this.layerSwitcherElement.checked = this.settings.spdEnabled;
    }

    /**
     * Select a card theme.
     * @param {Element} card 
     */
    selectThemeCard(card) {
        let selectedTheme = this.getSelectedTheme();

        selectedTheme?.removeAttribute('selected');

        card.setAttribute('selected', 'true');
    }

    /**
     * Get card element of the selected theme.
     * @returns {Element} Returns the element of the selected theme.
     */
    getSelectedTheme() {
        return document.querySelector('wz-card[id^="spd-theme-"][selected=true]');
    }

    /**
     * Gets the translation of the word or displays the key if no translation is found (to make it easier to find missing translations).
     * @param {String} key Key in string format, separating subcategories by periods ".".
     * @param {Object} [replaceStrings={}] An object containing placeholder values for dynamic translations.
     *
     * @returns {String} Translated string.
     */
    getTranslation(key, replaceStrings = {}) {
        return I18n.translate(`wmeSpd.${key}`, replaceStrings);
    }

    /**
     * Sets translations for the current locale.
     */
    defineTranslations() {
        let translations;

        switch (I18n.locale) {
            case 'af':
                // Afrikaans
                translations = {
                    name: 'Spoedweergawes',
                    zoomOut: 'Verder',
                    zoomIn: 'Naby',
                    changelog: 'Veranderingsregister',
                    changelogMessage: `<h6>Wat is nuut:</h6><ul style="margin-bottom: 10px"><li>Die boodskap "werk ikone op met vertraging" is uiteindelik vervang deur 'n eenvoudige ikoon, wat aandui wanneer die skrif steeds spoedikone verwerk (onthou altyd dat daar 'n doelbewuste vertraging in die opdatering is).</li></ul><h6>Herstelwerk en verbeterings:</h6><ul><li>Die naam van die "algemeen"-oortjie is verander na "aanpassings" vir beter begrip van die funksie van elke oortjie;</li><li>Die werking van die segmentgroottefilter is aangepas — dit verwyder nou nie meer die ikoon nie, maar pas eerder dieselfde deursigtigheid toe as die spoedfilter.</li></ul>`,
                    close: 'Sluit',
                    viewScript: 'Besoek Skrip',
                    viewChangelog: 'Besoek veranderingsregister',
                    reset: 'Herstel',
                    snackbar: {
                        errorFiltering: 'Fout met toepassing van filter. Minimum spoed kan nie groter as maksimum wees nie.',
                    },
                    title: {
                        settings: 'Instellings',
                        roadTypes: 'Padsoorte',
                        filter: 'Filtreer',
                        appearence: 'Vorm',
                        showOnRoadType: 'Wys op die volgende padsoorte:',
                        seeOnDiscuss: 'Kyk na die skrip se draad op die Discuss',
                        seeOnDiscord: 'Kyk na die skrip se kanaal op die Waze Script se Discord-bediener',
                        scriptDevBy: 'Bekyk my redakteur gebruikersprofiel',
                    },
                    label: {
                        displaySpeeds: 'Toon spoed',
                        showNoSpeed: 'Toon op segmente sonder spoed',
                        showOnRoundabout: 'Toon op rotondes',
                        showOnUnpaved: 'Toon op nie-geplaveide paaie',
                        maxZoom: 'Render tot die zoom:',
                        toggleLayerShortcut: 'Wissel laag sigbaarheid',
                        zIndex: 'Laagdiepte',
                        hideShorterSegments: 'Versteek segmente kleiner as',
                        speedFilter: 'Filter volgens snelheid',
                        min: 'Min:',
                        max: 'Max:',
                        opacity: 'Filter deursigtigheidsvlak:',
                    },
                    alertMessage: {
                        zIndex: `As jy \'n laagdiepte groter as <b>${this.zIndex.warn}</b> gebruik, sal die spoedlaag ander lae oorskadu en jy sal nie enige element op die skerm kan kies totdat jy die laagdiepte weer laer as <b>${this.zIndex.warn}</b> maak.`,
                        toggleVisibility: `Om die laag sigbaarheid van spoed aan/af te skakel, gaan eenvoudig na <b>Kaartlae > Paaie > Wys spoed</b> of gebruik die kortpad <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Standaard',
                        altViennaConv: 'Alternatiewe Wene Konvensie',
                        japan: 'Japan',
                        northAmerica: 'Noord-Amerika',
                        sacu: 'Suidelike Afrikaanse Douane-unie',
                        uk: 'Verenigde Koninkryk',
                    },
                    btn: {
                        updateSettingsTab: 'Werk menu op',
                        resetSettings: 'Herstel instellings',
                        seeOnDiscuss: 'Sien op die Bespreking',
                        seeOnDiscord: 'Sien op die Discord',
                        scriptDevBy: 'Skrip ontwikkel deur Luan Tavares',
                    }
                };

                break;

            case 'ar':
                // Arabic
                translations = {
                    name: 'عرض السرعة',
                    zoomOut: 'أبعد',
                    zoomIn: 'أقرب',
                    changelog: 'سجل التغييرات',
                    changelogMessage: `<h6>ما الجديد:</h6><ul style="margin-bottom: 10px"><li>تمت إزالة رسالة "تحديث الأيقونات بتأخير" واستبدالها بأيقونة بسيطة تشير إلى أن السكربت لا يزال يعالج أيقونات السرعة (تذكر أن هناك تأخيرًا مقصودًا في التحديث).</li></ul><h6>تصحيحات وتحسينات:</h6><ul><li>تم تغيير اسم علامة التبويب "عام" إلى "تعديلات" لتوضيح وظيفة كل علامة تبويب بشكل أفضل؛</li><li>تم تعديل طريقة تصفية حجم المقطع — بدلاً من إزالة الأيقونة، يتم الآن تطبيق نفس مستوى الشفافية المستخدم في فلتر السرعة.</li></ul>`,
                    close: 'إغلاق',
                    viewScript: 'عرض السكربت',
                    viewChangelog: 'عرض سجل التغييرات',
                    reset: 'إعادة تعيين',
                    snackbar: {
                        errorFiltering: 'خطأ في تطبيق الفلتر. لا يمكن أن تكون السرعة الدنيا أكبر من القصوى.',
                    },
                    title: {
                        settings: 'الإعدادات',
                        roadTypes: 'أنواع الطرق',
                        filter: 'تصفية',
                        appearence: 'المظهر',
                        showOnRoadType: 'عرض على أنواع الطرق التالية:',
                        seeOnDiscuss: 'شاهد موضوع السكربت على Discuss',
                        seeOnDiscord: 'شاهد قناة السكربت على خادم Discord الخاص بـ Waze Script',
                        scriptDevBy: 'عرض ملف تعريف مستخدم المحرر الخاص بي',
                    },
                    label: {
                        displaySpeeds: 'عرض السرعات',
                        showNoSpeed: 'عرض على القطاعات التي لا تحتوي على سرعة',
                        showOnRoundabout: 'عرض على الدوار',
                        showOnUnpaved: 'عرض على الطريق غير المعبدة',
                        maxZoom: 'عرض حتى التكبير:',
                        toggleLayerShortcut: 'تبديل طبقة عرض السرعة',
                        zIndex: 'عمق الطبقة',
                        hideShorterSegments: 'إخفاء في القطاعات الأصغر من',
                        speedFilter: 'تصفية حسب السرعة',
                        min: 'أدنى:',
                        max: 'أقصى:',
                        opacity: 'مستوى تعتيم الفلتر:',
                    },
                    alertMessage: {
                        zIndex: `عند استخدام عمق طبقة أكبر من <b>${this.zIndex.warn}</b>، ستتداخل طبقة السرعة مع الطبقات الأخرى ولن يمكنك تحديد أي عنصر على الشاشة حتى تعيد عمق الطبقة إلى أقل من <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `لتفعيل/إلغاء تفعيل طبقة عرض السرعات، يمكنك الذهاب إلى <b>الطبقات > الطرق > عرض السرعات</b> أو استخدام الاختصار <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'افتراضي',
                        altViennaConv: 'اتفاقية فيينا البديلة',
                        japan: 'اليابان',
                        northAmerica: 'أمريكا الشمالية',
                        sacu: 'اتحاد الجمارك في جنوب أفريقيا',
                        uk: 'المملكة المتحدة',
                    },
                    btn: {
                        updateSettingsTab: 'تحديث القائمة',
                        resetSettings: 'إعادة تعيين الإعدادات',
                        seeOnDiscuss: 'انظر في المناقشة',
                        seeOnDiscord: 'انظر في الديسكورد',
                        scriptDevBy: 'البرنامج النصي تم تطويره بواسطة لوان تافاريس',
                    }
                };
            
                break;
            
            case 'bg':
                // Bulgarian
                translations = {
                    name: 'Показване на скоростта',
                    zoomOut: 'Отдалечи',
                    zoomIn: 'Приближи',
                    changelog: 'Журнал с промени',
                    changelogMessage: `<h6>Какво ново:</h6><ul style="margin-bottom: 10px"><li>Съобщението "актуализиране на иконите със закъснение" беше премахнато и заменено с проста икона, показваща кога скриптът все още обработва иконите за скорост (не забравяйте, че има умишлено закъснение).</li></ul><h6>Поправки и подобрения:</h6><ul><li>Името на раздела "общи" е променено на "настройки" за по-добро разбиране на всяка функция;</li><li>Функцията за филтриране по дължина на сегмента вече не премахва иконата, а прилага същата непрозрачност както във филтъра за скорост.</li></ul>`,
                    close: 'Затвори',
                    viewScript: 'Преглед на скрипта',
                    viewChangelog: 'Преглед на журнала с промени',
                    reset: 'Нулиране',
                    snackbar: {
                        errorFiltering: 'Грешка при прилагане на филтъра. Минималната скорост не може да бъде по-голяма от максималната.',
                    },
                    title: {
                        settings: 'Настройки',
                        roadTypes: 'Типове пътища',
                        filter: 'Филтър',
                        appearence: 'Изглед',
                        showOnRoadType: 'Покажи на следните типове пътища:',
                        seeOnDiscuss: 'Вижте нишката на скрипта в Discuss',
                        seeOnDiscord: 'Вижте канала на скрипта в Discord сървъра на Waze Script',
                        scriptDevBy: 'Вижте потребителския профил на моя редактор',
                    },
                    label: {
                        displaySpeeds: 'Показване на скорости',
                        showNoSpeed: 'Показване на сегменти без скорост',
                        showOnRoundabout: 'Показване на кръгови кръстовища',
                        showOnUnpaved: 'Показване на непавирани пътища',
                        maxZoom: 'Изобразяване до ниво на мащаб:',
                        toggleLayerShortcut: 'Превключване на слоя за показване на скоростта',
                        zIndex: 'Дълбочина на слоя',
                        hideShorterSegments: 'Скриване на сегменти по-малки от',
                        speedFilter: 'Филтриране по скорост',
                        min: 'Мин.:',
                        max: 'Макс.:',
                        opacity: 'Ниво на непрозрачност на филтъра:',
                    },
                    alertMessage: {
                        zIndex: `Ако използвате дълбочина на слоя, по-голяма от <b>${this.zIndex.warn}</b>, слойът за скорост ще се припокрие с други и няма да можете да изберете елемент на екрана, докато не го намалите под <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `За да активирате/деактивирате слоя за показване на скорости, можете да отидете в <b>Картови слоеве > Пътища > Показване на скорости</b> или да използвате клавишната комбинация <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'По подразбиране',
                        altViennaConv: 'Алтернативна Виенска конвенция',
                        japan: 'Япония',
                        northAmerica: 'Северна Америка',
                        sacu: 'Южноафрикански съюз за митници',
                        uk: 'Обединено кралство',
                    },
                    btn: {
                        updateSettingsTab: 'Актуализиране на менюто',
                        resetSettings: 'Нулиране на настройките',
                        seeOnDiscuss: 'Вижте в обсъждането',
                        seeOnDiscord: 'Вижте в Discord',
                        scriptDevBy: 'Скриптът е разработен от Луaн Таварес',
                    }
                };
            
                break;
            
            case 'ca':
                // Catalan
                translations = {
                    name: 'Visualització de la Velocitat',
                    zoomOut: 'Més allunyat',
                    zoomIn: 'Més a prop',
                    changelog: 'Registre de canvis',
                    changelogMessage: `<h6>Novetats:</h6><ul style="margin-bottom: 10px"><li>S'ha eliminat finalment el missatge de "actualització de les icones amb retard" i s'ha substituït per una icona senzilla per indicar que el script encara està processant les icones de velocitat (recorda que hi ha un retard intencionat en l'actualització).</li></ul><h6>Correccions i millores:</h6><ul><li>S'ha canviat el nom de la pestanya "general" a "ajustos" per facilitar la comprensió de la funció de cada pestanya;</li><li>La filtració per mida del segment ja no elimina la icona sinó que aplica la mateixa opacitat que el filtre de velocitat.</li></ul>`,
                    close: 'Tancar',
                    viewScript: 'Veure el script',
                    viewChangelog: 'Veure el registre de canvis',
                    reset: 'Restablir',
                    snackbar: {
                        errorFiltering: 'Error en aplicar el filtre. La velocitat mínima no pot ser superior a la màxima.',
                    },
                    title: {
                        settings: 'Configuració',
                        roadTypes: 'Tipus de carreteres',
                        filter: 'Filtre',
                        appearence: 'Aparença',
                        showOnRoadType: 'Mostrar en els següents tipus de carreteres:',
                        seeOnDiscuss: 'Veure el fil del script a Discuss',
                        seeOnDiscord: 'Veure el canal del script al servidor de Discord de Waze Script',
                        scriptDevBy: 'Veure el meu perfil d\'usuari de l\'editor',
                    },
                    label: {
                        displaySpeeds: 'Mostrar velocitats',
                        showNoSpeed: 'Mostrar en segments sense velocitat',
                        showOnRoundabout: 'Mostrar en rotondes',
                        showOnUnpaved: 'Mostrar en vies no pavimentades',
                        maxZoom: 'Renderitzar fins al zoom:',
                        toggleLayerShortcut: 'Alternar la capa de visualització de la velocitat',
                        zIndex: 'Profunditat de la capa',
                        hideShorterSegments: 'Amagar en segments més petits que',
                        speedFilter: 'Filtrar per velocitat',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Nivell d’opacitat del filtre:',
                    },
                    alertMessage: {
                        zIndex: `Si utilitzeu una profunditat de capa superior a <b>${this.zIndex.warn}</b>, la capa de velocitat es superposarà a altres i no podreu seleccionar cap element a la pantalla fins que la rebaixeu per sota de <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Per activar/desactivar la capa de visualització de velocitats, podeu anar a <b>Capes del mapa > Vies > Mostrar velocitats</b> o utilitzar la drecera <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Per defecte',
                        altViennaConv: 'Convenció alternativa de Viena',
                        japan: 'Japó',
                        northAmerica: 'Amèrica del Nord',
                        sacu: 'Unió duanera de l\'Àfrica Austral',
                        uk: 'Regne Unit',
                    },
                    btn: {
                        updateSettingsTab: 'Actualitzar el menú',
                        resetSettings: 'Restablir la configuració',
                        seeOnDiscuss: 'Veure a la discussió',
                        seeOnDiscord: 'Veure a Discord',
                        scriptDevBy: 'Script desenvolupat per Luan Tavares',
                    }
                };
            
                break;

            case 'cs':
                // Czech
                translations = {
                    name: 'Zobrazení Rychlosti',
                    zoomOut: 'Více vzdálené',
                    zoomIn: 'Více přiblížené',
                    changelog: 'Seznam změn',
                    changelogMessage: `<h6>Novinky:</h6><ul style="margin-bottom: 10px"><li>Zpráva "aktualizace ikon se zpožděním" byla konečně odstraněna a nahrazena jednoduchou ikonou, která jasně ukazuje, že skript stále zpracovává rychlostní ikony (pamatujte, že existuje záměrné zpoždění aktualizace).</li></ul><h6>Opravy a vylepšení:</h6><ul><li>Název karty "obecné" byl změněn na "úpravy" pro lepší porozumění funkci jednotlivých karet skriptu;</li><li>Změněno chování filtrování podle délky segmentu – nyní neodstraňuje ikonu, ale pouze aplikuje stejnou průhlednost jako rychlostní filtr.</li></ul>`,
                    close: 'Zavřít',
                    viewScript: 'Zobrazit skript',
                    viewChangelog: 'Zobrazit seznam změn',
                    reset: 'Obnovit',
                    snackbar: {
                        errorFiltering: 'Chyba při aplikaci filtru. Minimální rychlost nemůže být větší než maximální.',
                    },
                    title: {
                        settings: 'Nastavení',
                        roadTypes: 'Typy silnic',
                        filter: 'Filtr',
                        appearence: 'Vzhled',
                        showOnRoadType: 'Zobrazit na následujících typech silnic:',
                        seeOnDiscuss: 'Podívejte se na vlákno skriptu na Discuss',
                        seeOnDiscord: 'Podívejte se na kanál skriptu na Discord serveru Waze Script',
                        scriptDevBy: 'Zobrazit můj uživatelský profil editoru',
                    },
                    label: {
                        displaySpeeds: 'Zobrazit rychlosti',
                        showNoSpeed: 'Zobrazit na segmentu bez rychlosti',
                        showOnRoundabout: 'Zobrazit na kruhovém objezdu',
                        showOnUnpaved: 'Zobrazit na nezpevněné cestě',
                        maxZoom: 'Renderovat až na zoom:',
                        toggleLayerShortcut: 'Přepnout vrstvu zobrazení rychlosti',
                        zIndex: 'Hloubka vrstvy',
                        hideShorterSegments: 'Skrýt na segmentech menších než',
                        speedFilter: 'Filtrovat podle rychlosti',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Úroveň neprůhlednosti filtru:',
                    },
                    alertMessage: {
                        zIndex: `Pokud použijete hloubku vrstvy vyšší než <b>${this.zIndex.warn}</b>, vrstva rychlosti se překryje s ostatními a nebude možné vybrat žádný prvek na obrazovce, dokud ji opět nezmenšíte pod <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Pro aktivaci/deaktivaci vrstvy zobrazení rychlostí stačí přejít do <b>Mapové vrstvy > Cesty > Zobrazit rychlosti</b> nebo použít klávesovou zkratku <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Výchozí',
                        altViennaConv: 'Alternativní Vídeňská úmluva',
                        japan: 'Japonsko',
                        northAmerica: 'Severní Amerika',
                        sacu: 'Jižní africká celní unie',
                        uk: 'Spojené království',
                    },
                    btn: {
                        updateSettingsTab: 'Aktualizovat nabídku',
                        resetSettings: 'Obnovit nastavení',
                        seeOnDiscuss: 'Zobrazit v diskusi',
                        seeOnDiscord: 'Zobrazit na Discordu',
                        scriptDevBy: 'Skript vyvinutý Luanem Tavaresem',
                    }
                };                

                break;

            case 'da':
                // Danish
                translations = {
                    name: 'Hastighedsvisning',
                    zoomOut: 'Mere afstand',
                    zoomIn: 'Mere tæt på',
                    changelog: 'Ændringslog',
                    changelogMessage: `<h6>Nyheder:</h6><ul style="margin-bottom: 10px"><li>Beskeden "opdatering af ikoner med forsinkelse" er nu fjernet og erstattet med et simpelt ikon, der tydeligt viser, når scriptet stadig behandler hastighedsikoner (husk, der er en tilsigtet forsinkelse ved opdatering).</li></ul><h6>Rettelser og forbedringer:</h6><ul><li>Fanen "generelt" er blevet omdøbt til "justeringer" for at gøre funktionen af hver fane tydeligere;</li><li>Segmentstørrelsesfilteret fjerner ikke længere ikoner, men anvender nu samme gennemsigtighed som hastighedsfilteret.</li></ul>`,
                    close: 'Luk',
                    viewScript: 'Se script',
                    viewChangelog: 'Se ændringslog',
                    reset: 'Nulstil',
                    snackbar: {
                        errorFiltering: 'Fejl ved anvendelse af filter. Minimumshastighed kan ikke være større end maksimum.',
                    },
                    title: {
                        settings: 'Indstillinger',
                        roadTypes: 'Straßentypen',
                        filter: 'Filter',
                        appearence: 'Erscheinungsbild',
                        showOnRoadType: 'An den folgenden Straßentypen anzeigen:',
                        seeOnDiscuss: 'Siehe den Thread des Skripts auf Discuss',
                        seeOnDiscord: 'Siehe den Kanal des Skripts auf dem Waze Script Discord-Server',
                        scriptDevBy: 'Se min redaktørbrugerprofil',
                    },
                    label: {
                        displaySpeeds: 'Vis hastigheder',
                        showNoSpeed: 'Vis på segmenter uden hastighed',
                        showOnRoundabout: 'Vis på rundkørsler',
                        showOnUnpaved: 'Vis på ubelagte veje',
                        maxZoom: 'Render op til zoom:',
                        toggleLayerShortcut: 'Skift synlighed af hastighedslaget',
                        zIndex: 'Lagdybde',
                        hideShorterSegments: 'Skjul på segmenter mindre end',
                        speedFilter: 'Filtrer efter hastighed',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Filterets opacitetsniveau:',
                    },
                    alertMessage: {
                        zIndex: `Når du bruger en lagdybde større end <b>${this.zIndex.warn}</b>, vil hastighedslaget dække de andre, og det vil ikke være muligt at vælge et element på skærmen, før du sænker dybden igen under <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `For at aktivere/deaktivere hastighedslaget, skal du gå til <b>Kortlag > Veje > Vis hastigheder</b> eller bruge genvejstasten <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Standard',
                        altViennaConv: 'Alternativ Wienerkonvention',
                        japan: 'Japan',
                        northAmerica: 'Nordamerika',
                        sacu: 'Southern African Customs Union',
                        uk: 'Storbritannien',
                    },
                    btn: {
                        updateSettingsTab: 'Opdater menu',
                        resetSettings: 'Nulstil indstillinger',
                        seeOnDiscuss: 'Se i diskussionen',
                        seeOnDiscord: 'Se på Discord',
                        scriptDevBy: 'Script udviklet af Luan Tavares',
                    }
                };
            
                break;
            
            case 'de':
                // German
                translations = {
                    name: 'Geschwindigkeitsanzeige',
                    zoomOut: 'Weiter entfernt',
                    zoomIn: 'Mehr heran',
                    changelog: 'Änderungsprotokoll',
                    changelogMessage: `<h6>Neues:</h6><ul style="margin-bottom: 10px"><li>Die Nachricht "Symbole mit Verzögerung aktualisieren" wurde endlich entfernt und durch ein einfaches Symbol ersetzt, das anzeigt, wenn das Skript die Geschwindigkeitssymbole noch verarbeitet (denken Sie daran, dass es eine beabsichtigte Verzögerung gibt).</li></ul><h6>Fehlerbehebungen und Verbesserungen:</h6><ul><li>Der Reiter "Allgemein" wurde in "Anpassungen" umbenannt, um die Funktion jeder Registerkarte besser zu verdeutlichen;</li><li>Die Segmentgrößenfilterung entfernt das Symbol nicht mehr, sondern wendet jetzt dieselbe Transparenz wie der Geschwindigkeitsfilter an.</li></ul>`,
                    close: 'Schließen',
                    viewScript: 'Skript anzeigen',
                    viewChangelog: 'Änderungsprotokoll anzeigen',
                    reset: 'Zurücksetzen',
                    snackbar: {
                        errorFiltering: 'Fehler beim Anwenden des Filters. Die Mindestgeschwindigkeit darf nicht größer als die Höchstgeschwindigkeit sein.',
                    },
                    title: {
                        settings: 'Einstellungen',
                        roadTypes: 'Straßentypen',
                        filter: 'Filter',
                        appearence: 'Erscheinungsbild',
                        showOnRoadType: 'Auf den folgenden Straßentypen anzeigen:',
                        seeOnDiscuss: 'Siehe den Thread des Skripts auf Discuss',
                        seeOnDiscord: 'Siehe den Kanal des Skripts auf dem Waze Script Discord-Server',
                        scriptDevBy: 'Mein Editor-Benutzerprofil anzeigen',
                    },
                    label: {
                        displaySpeeds: 'Geschwindigkeiten anzeigen',
                        showNoSpeed: 'Auf Segmenten ohne Geschwindigkeit anzeigen',
                        showOnRoundabout: 'Auf Kreisverkehr anzeigen',
                        showOnUnpaved: 'Auf unbefestigten Straßen anzeigen',
                        maxZoom: 'Rendern bis Zoom:',
                        toggleLayerShortcut: 'Schicht für Geschwindigkeitsanzeige umschalten',
                        zIndex: 'Schichttiefe',
                        hideShorterSegments: 'Verstecken bei Segmenten kürzer als',
                        speedFilter: 'Nach Geschwindigkeit filtern',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Filterdeckkraftstufe:',
                    },
                    alertMessage: {
                        zIndex: `Wenn Sie eine Schichttiefe von mehr als <b>${this.zIndex.warn}</b> verwenden, wird die Geschwindigkeits-Schicht andere überlagern, und es wird nicht möglich sein, ein Element auf dem Bildschirm auszuwählen, bis Sie die Schichttiefe wieder unter <b>${this.zIndex.warn}</b> verringern.`,
                        toggleVisibility: `Um die Sichtbarkeit der Geschwindigkeits-Schicht zu aktivieren/deaktivieren, gehen Sie zu <b>Kartenebenen > Straßen > Geschwindigkeiten anzeigen</b> oder verwenden Sie die Verknüpfung <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Standard',
                        altViennaConv: 'Alternative Wien-Konvention',
                        japan: 'Japan',
                        northAmerica: 'Nordamerika',
                        sacu: 'Südafrikanische Zollunion',
                        uk: 'Vereinigtes Königreich',
                    },
                    btn: {
                        updateSettingsTab: 'Menü aktualisieren',
                        resetSettings: 'Einstellungen zurücksetzen',
                        seeOnDiscuss: 'Im Diskussionsthread ansehen',
                        seeOnDiscord: 'Auf Discord ansehen',
                        scriptDevBy: 'Skript entwickelt von Luan Tavares',
                    }
                };
            
                break;
            
            case 'el':
                // Greek
                translations = {
                    name: 'Εμφάνιση Ταχύτητας',
                    zoomOut: 'Περισσότερη απόσταση',
                    zoomIn: 'Πιο κοντά',
                    changelog: 'Αρχείο αλλαγών',
                    changelogMessage: `<h6>Τι νέο υπάρχει:</h6><ul style="margin-bottom: 10px"><li>Το μήνυμα "ενημέρωση εικονιδίων με καθυστέρηση" αφαιρέθηκε και αντικαταστάθηκε με ένα απλό εικονίδιο, το οποίο δείχνει πότε το script εξακολουθεί να επεξεργάζεται τα εικονίδια ταχύτητας (θυμηθείτε ότι υπάρχει σκόπιμη καθυστέρηση).</li></ul><h6>Διορθώσεις και βελτιώσεις:</h6><ul><li>Το όνομα της καρτέλας "γενικά" άλλαξε σε "ρυθμίσεις" για καλύτερη κατανόηση της λειτουργίας κάθε καρτέλας του script.</li><li>Η λειτουργία του φιλτραρίσματος μεγεθών τμημάτων άλλαξε — δεν αφαιρεί πλέον το εικονίδιο, αλλά εφαρμόζει την ίδια διαφάνεια όπως το φίλτρο ταχύτητας.</li></ul>`,
                    close: 'Κλείσιμο',
                    viewScript: 'Δείτε το script',
                    viewChangelog: 'Δείτε το αρχείο αλλαγών',
                    reset: 'Επαναφορά',
                    snackbar: {
                        errorFiltering: 'Σφάλμα στην εφαρμογή του φίλτρου. Η ελάχιστη ταχύτητα δεν μπορεί να είναι μεγαλύτερη από τη μέγιστη.',
                    },
                    title: {
                        settings: 'Ρυθμίσεις',
                        roadTypes: 'Τύποι Δρόμων',
                        filter: 'Φίλτρο',
                        appearence: 'Εμφάνιση',
                        showOnRoadType: 'Εμφάνιση στους εξής τύπους δρόμων:',
                        seeOnDiscuss: 'Δείτε το νήμα του script στο Discuss',
                        seeOnDiscord: 'Δείτε το κανάλι του script στον server Discord του Waze Script',
                        scriptDevBy: 'Δείτε το προφίλ χρήστη του επεξεργαστή μου',
                    },
                    label: {
                        displaySpeeds: 'Εμφάνιση ταχυτήτων',
                        showNoSpeed: 'Εμφάνιση σε τμήματα χωρίς ταχύτητα',
                        showOnRoundabout: 'Εμφάνιση σε κυκλικούς κόμβους',
                        showOnUnpaved: 'Εμφάνιση σε α paved δρόμους',
                        maxZoom: 'Απεικόνιση μέχρι το ζουμ:',
                        toggleLayerShortcut: 'Αλλαγή ορατότητας του επιπέδου ταχύτητας',
                        zIndex: 'Βάθος επιπέδου',
                        hideShorterSegments: 'Απόκρυψη σε τμήματα μικρότερα από',
                        speedFilter: 'Φίλτρο κατά ταχύτητα',
                        min: 'Ελάχιστο:',
                        max: 'Μέγιστο:',
                        opacity: 'Επίπεδο αδιαφάνειας φίλτρου:',
                    },
                    alertMessage: {
                        zIndex: `Όταν χρησιμοποιείτε βάθος επιπέδου μεγαλύτερο από <b>${this.zIndex.warn}</b>, το επίπεδο ταχύτητας θα επικαλύψει τα άλλα και δεν θα μπορείτε να επιλέξετε κανένα στοιχείο στην οθόνη μέχρι να το χαμηλώσετε ξανά κάτω από <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Για να ενεργοποιήσετε/απενεργοποιήσετε το επίπεδο ταχύτητας, πηγαίνετε στο <b>Επίπεδα Χάρτη > Οδοί > Εμφάνιση ταχυτήτων</b> ή χρησιμοποιήστε την συντόμευση <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Προεπιλογή',
                        altViennaConv: 'Εναλλακτική Σύμβαση της Βιέννης',
                        japan: 'Ιαπωνία',
                        northAmerica: 'Βόρεια Αμερική',
                        sacu: 'Νότια Αφρικανική Τελωνειακή Ένωση',
                        uk: 'Ηνωμένο Βασίλειο',
                    },
                    btn: {
                        updateSettingsTab: 'Ενημέρωση μενού',
                        resetSettings: 'Επαναφορά ρυθμίσεων',
                        seeOnDiscuss: 'Δείτε στη Συζήτηση',
                        seeOnDiscord: 'Δείτε στο Discord',
                        scriptDevBy: 'Το σενάριο αναπτύχθηκε από τον Luan Tavares',
                    }
                };
            
                break;
            
            case 'en-AU':
                // English (Australia)
                translations = {
                    name: 'Speed Display',
                    zoomOut: 'Zoom Out',
                    zoomIn: 'Zoom In',
                    changelog: 'Changelog',
                    changelogMessage: `<h6>What's New:</h6><ul style="margin-bottom: 10px"><li>Finally removed the "updating icons with delay" message and replaced it with a simple icon, making it clear when the script is still processing speed icons (remember there’s an intentional delay).</li></ul><h6>Fixes and improvements:</h6><ul><li>Renamed the "general" tab to "settings" for better clarity on each tab's function;</li><li>The segment size filter no longer removes the icon but now applies the same opacity as the speed filter.</li></ul>`,
                    close: 'Close',
                    viewScript: 'View Script',
                    viewChangelog: 'View Changelog',
                    reset: 'Reset',
                    snackbar: {
                        errorFiltering: 'Error applying filter. Minimum speed cannot be greater than maximum.',
                    },
                    title: {
                        settings: 'Settings',
                        roadTypes: 'Road Types',
                        filter: 'Filter',
                        appearence: 'Appearance',
                        showOnRoadType: 'Show on the following road types:',
                        seeOnDiscuss: 'See the script\'s thread on Discuss',
                        seeOnDiscord: 'See the script\'s channel on the Waze Script Discord server',
                        scriptDevBy: 'View my editor user profile',
                    },
                    label: {
                        displaySpeeds: 'Display Speeds',
                        showNoSpeed: 'Show on segments without speed',
                        showOnRoundabout: 'Show on roundabouts',
                        showOnUnpaved: 'Show on unpaved roads',
                        maxZoom: 'Render up to zoom level:',
                        toggleLayerShortcut: 'Toggle speed display layer',
                        zIndex: 'Layer Depth',
                        hideShorterSegments: 'Hide segments shorter than',
                        speedFilter: 'Filter by speed',
                        min: 'Min:',
                        max: 'Max:',
                        opacity: 'Filter opacity level:',
                    },
                    alertMessage: {
                        zIndex: `When using a layer depth greater than <b>${this.zIndex.warn}</b>, the speed layer will overlap other layers, and no element can be selected until you lower it back below <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `To toggle the speed display layer, go to <b>Map Layers > Roads > Display Speeds</b> or use the shortcut <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Default',
                        altViennaConv: 'Alternative Vienna Convention',
                        japan: 'Japan',
                        northAmerica: 'North America',
                        sacu: 'Southern African Customs Union',
                        uk: 'United Kingdom',
                    },                    
                    btn: {
                        updateSettingsTab: 'Update Menu',
                        resetSettings: 'Reset Settings',
                        seeOnDiscuss: 'See on the Discuss',
                        seeOnDiscord: 'See on Discord',
                        scriptDevBy: 'Script developed by Luan Tavares',
                    }
                };
            
                break;
            
            case 'en-GB':
                // English (UK)
                translations = {
                    name: 'Speed Display',
                    zoomOut: 'Zoom Out',
                    zoomIn: 'Zoom In',
                    changelog: 'Changelog',
                    changelogMessage: `<h6>What's New:</h6><ul style="margin-bottom: 10px"><li>Finally removed the "updating icons with delay" message and replaced it with a simple icon, making it clear when the script is still processing speed icons (remember there’s an intentional delay).</li></ul><h6>Fixes and improvements:</h6><ul><li>Renamed the "general" tab to "settings" for better clarity on each tab's function;</li><li>The segment size filter no longer removes the icon but now applies the same opacity as the speed filter.</li></ul>`,
                    close: 'Close',
                    viewScript: 'View Script',
                    viewChangelog: 'View Changelog',
                    reset: 'Reset',
                    snackbar: {
                        errorFiltering: 'Error applying filter. Minimum speed cannot be greater than maximum.',
                    },
                    title: {
                        settings: 'Settings',
                        roadTypes: 'Road Types',
                        filter: 'Filter',
                        appearence: 'Appearance',
                        showOnRoadType: 'Show on the following road types:',
                        seeOnDiscuss: 'See the script\'s thread on Discuss',
                        seeOnDiscord: 'See the script\'s channel on the Waze Script Discord server',
                        scriptDevBy: 'View my editor user profile',
                    },
                    label: {
                        displaySpeeds: 'Display Speeds',
                        showNoSpeed: 'Show on segments without speed',
                        showOnRoundabout: 'Show on roundabouts',
                        showOnUnpaved: 'Show on unpaved roads',
                        maxZoom: 'Render up to zoom level:',
                        toggleLayerShortcut: 'Toggle speed display layer',
                        zIndex: 'Layer Depth',
                        hideShorterSegments: 'Hide segments shorter than',
                        speedFilter: 'Filter by speed',
                        min: 'Min:',
                        max: 'Max:',
                        opacity: 'Filter opacity level:',
                    },
                    alertMessage: {
                        zIndex: `When using a layer depth greater than <b>${this.zIndex.warn}</b>, the speed layer will overlap other layers, and no element can be selected until you lower it back below <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `To toggle the speed display layer, go to <b>Map Layers > Roads > Display Speeds</b> or use the shortcut <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Default',
                        altViennaConv: 'Alternative Vienna Convention',
                        japan: 'Japan',
                        northAmerica: 'North America',
                        sacu: 'Southern African Customs Union',
                        uk: 'United Kingdom',
                    },
                    btn: {
                        updateSettingsTab: 'Update Menu',
                        resetSettings: 'Reset Settings',
                        seeOnDiscuss: 'See on the Discuss',
                        seeOnDiscord: 'See on Discord',
                        scriptDevBy: 'Script developed by Luan Tavares',
                    }
                };
            
                break;
            
            case 'en-US':
                // US English
                translations = {
                    name: 'Speed Display',
                    zoomOut: 'Zoom Out',
                    zoomIn: 'Zoom In',
                    changelog: 'Changelog',
                    changelogMessage: `<h6>What's New:</h6><ul style="margin-bottom: 10px"><li>Finally removed the "updating icons with delay" message and replaced it with a simple icon, making it clear when the script is still processing speed icons (remember there’s an intentional delay).</li></ul><h6>Fixes and improvements:</h6><ul><li>Renamed the "general" tab to "settings" for better clarity on each tab's function;</li><li>The segment size filter no longer removes the icon but now applies the same opacity as the speed filter.</li></ul>`,
                    close: 'Close',
                    viewScript: 'View Script',
                    viewChangelog: 'View Changelog',
                    reset: 'Reset',
                    snackbar: {
                        errorFiltering: 'Error applying filter. Minimum speed cannot be greater than maximum.',
                    },
                    title: {
                        settings: 'Settings',
                        roadTypes: 'Road Types',
                        filter: 'Filter',
                        appearence: 'Appearance',
                        showOnRoadType: 'Show on the following road types:',
                        seeOnDiscuss: 'See the script\'s thread on Discuss',
                        seeOnDiscord: 'See the script\'s channel on the Waze Script Discord server',
                        scriptDevBy: 'View my editor user profile',
                    },
                    label: {
                        displaySpeeds: 'Display Speeds',
                        showNoSpeed: 'Show on segments without speed',
                        showOnRoundabout: 'Show on roundabouts',
                        showOnUnpaved: 'Show on unpaved roads',
                        maxZoom: 'Render up to zoom level:',
                        toggleLayerShortcut: 'Toggle speed display layer',
                        zIndex: 'Layer Depth',
                        hideShorterSegments: 'Hide segments shorter than',
                        speedFilter: 'Filter by speed',
                        min: 'Min:',
                        max: 'Max:',
                        opacity: 'Filter opacity level:',
                    },
                    alertMessage: {
                        zIndex: `When using a layer depth greater than <b>${this.zIndex.warn}</b>, the speed layer will overlap other layers, and no element can be selected until you lower it back below <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `To toggle the speed display layer, go to <b>Map Layers > Roads > Display Speeds</b> or use the shortcut <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Default',
                        altViennaConv: 'Alternative Vienna Convention',
                        japan: 'Japan',
                        northAmerica: 'North America',
                        sacu: 'Southern African Customs Union',
                        uk: 'United Kingdom',
                    },
                    btn: {
                        updateSettingsTab: 'Update Menu',
                        resetSettings: 'Reset Settings',
                        seeOnDiscuss: 'See on the Discuss',
                        seeOnDiscord: 'See on Discord',
                        scriptDevBy: 'Script developed by Luan Tavares',
                    }
                };
            
                break;

            case 'es':
                // Spanish
                translations = {
                    name: 'Exhibición de Velocidad',
                    zoomOut: 'Alejar',
                    zoomIn: 'Acercar',
                    changelog: 'Registro de cambios',
                    changelogMessage: `<h6>Novedades:</h6><ul style="margin-bottom: 10px"><li>Por fin se eliminó el mensaje de "actualizando los íconos con retraso" y se reemplazó por un ícono simple, dejando claro cuándo el script aún está procesando los íconos de velocidad (recuerda que hay un retraso intencionado).</li></ul><h6>Correcciones y mejoras:</h6><ul><li>Se cambió el nombre de la pestaña "general" a "ajustes" para clarificar mejor la función de cada pestaña del script;</li><li>Se modificó el funcionamiento del filtro por tamaño de segmento: ya no elimina el ícono, ahora solo aplica la misma opacidad del filtro de velocidad.</li></ul>`,
                    close: 'Cerrar',
                    viewScript: 'Ver script',
                    viewChangelog: 'Ver registro de cambios',
                    reset: 'Restablecer',
                    snackbar: {
                        errorFiltering: 'Error al aplicar el filtro. La velocidad mínima no puede ser mayor que la máxima.',
                    },
                    title: {
                        settings: 'Configuración',
                        roadTypes: 'Tipos de carretera',
                        filter: 'Filtro',
                        appearence: 'Apariencia',
                        showOnRoadType: 'Mostrar en los siguientes tipos de carretera:',
                        seeOnDiscuss: 'Ver el hilo del script en Discuss',
                        seeOnDiscord: 'Ver el canal del script en el servidor de Discord de Waze Script',
                        scriptDevBy: 'Ver mi perfil de usuario del editor',
                    },
                    label: {
                        displaySpeeds: 'Mostrar velocidades',
                        showNoSpeed: 'Mostrar en segmentos sin velocidad',
                        showOnRoundabout: 'Mostrar en rotondas',
                        showOnUnpaved: 'Mostrar en caminos no pavimentados',
                        maxZoom: 'Renderizar hasta el nivel de zoom:',
                        toggleLayerShortcut: 'Alternar capa de exhibición de velocidad',
                        zIndex: 'Profundidad de la capa',
                        hideShorterSegments: 'Ocultar segmentos más pequeños que',
                        speedFilter: 'Filtrar por velocidad',
                        min: 'Mín.:',
                        max: 'Máx.:',
                        opacity: 'Nivel de opacidad del filtro:',
                    },
                    alertMessage: {
                        zIndex: `Al utilizar una profundidad de capa mayor a <b>${this.zIndex.warn}</b>, la capa de velocidad se superpondrá a las demás y no se podrá seleccionar ningún elemento hasta que la bajes de nuevo por debajo de <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Para activar/desactivar la capa de exhibición de velocidades, dirígete a <b>Capas del Mapa > Vías > Mostrar velocidades</b> o usa el acceso directo <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Predeterminado',
                        altViennaConv: 'Convención de Viena Alternativa',
                        japan: 'Japón',
                        northAmerica: 'América del Norte',
                        sacu: 'Unión Aduanera de África Austral',
                        uk: 'Reino Unido',
                    },
                    btn: {
                        updateSettingsTab: 'Actualizar menú',
                        resetSettings: 'Restablecer configuración',
                        seeOnDiscuss: 'Ver en la Discusión',
                        seeOnDiscord: 'Ver en Discord',
                        scriptDevBy: 'Script desarrollado por Luan Tavares',
                    }
                };
            
                break;

            case 'es-419':
                // Latin-american spanish
                translations = {
                    name: 'Exhibición de Velocidad',
                    zoomOut: 'Alejar',
                    zoomIn: 'Acercar',
                    changelog: 'Registro de cambios',
                    changelogMessage: `<h6>Novedades:</h6><ul style="margin-bottom: 10px"><li>Finalmente se eliminó el mensaje de "actualización de íconos con retraso" y fue reemplazado por un ícono simple, lo que deja claro cuándo el script aún está procesando los íconos de velocidad (recuerda que hay un retraso intencional).</li></ul><h6>Correcciones y mejoras:</h6><ul><li>Se renombró la pestaña "general" a "ajustes" para mayor claridad sobre la función de cada pestaña del script;</li><li>El filtro por tamaño de segmento ya no elimina el ícono, ahora solo aplica la misma opacidad que el filtro de velocidad.</li></ul>`,
                    close: 'Cerrar',
                    viewScript: 'Ver script',
                    viewChangelog: 'Ver registro de cambios',
                    reset: 'Restablecer',
                    snackbar: {
                        errorFiltering: 'Error al aplicar el filtro. La velocidad mínima no puede ser mayor que la máxima.',
                    },
                    title: {
                        settings: 'Configuración',
                        roadTypes: 'Tipos de carretera',
                        filter: 'Filtro',
                        appearence: 'Apariencia',
                        showOnRoadType: 'Mostrar en los siguientes tipos de carretera:',
                        seeOnDiscuss: 'Ver el hilo del script en Discuss',
                        seeOnDiscord: 'Ver el canal del script en el servidor de Discord de Waze Script',
                        scriptDevBy: 'Ver mi perfil de usuario del editor',
                    },
                    label: {
                        displaySpeeds: 'Mostrar velocidades',
                        showNoSpeed: 'Mostrar en segmentos sin velocidad',
                        showOnRoundabout: 'Mostrar en rotondas',
                        showOnUnpaved: 'Mostrar en caminos no pavimentados',
                        maxZoom: 'Renderizar hasta el nivel de zoom:',
                        toggleLayerShortcut: 'Alternar capa de exhibición de velocidad',
                        zIndex: 'Profundidad de la capa',
                        hideShorterSegments: 'Ocultar segmentos más pequeños que',
                        speedFilter: 'Filtrar por velocidad',
                        min: 'Mín.:',
                        max: 'Máx.:',
                        opacity: 'Nivel de opacidad del filtro:',
                    },
                    alertMessage: {
                        zIndex: `Al utilizar una profundidad de capa mayor a <b>${this.zIndex.warn}</b>, la capa de velocidad se superpondrá a las demás y no se podrá seleccionar ningún elemento hasta que la bajes de nuevo por debajo de <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Para activar/desactivar la capa de exhibición de velocidades, dirígete a <b>Capas del Mapa > Vías > Mostrar velocidades</b> o usa el acceso directo <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Predeterminado',
                        altViennaConv: 'Convención de Viena Alternativa',
                        japan: 'Japón',
                        northAmerica: 'América del Norte',
                        sacu: 'Unión Aduanera de África Austral',
                        uk: 'Reino Unido',
                    },
                    btn: {
                        updateSettingsTab: 'Actualizar menú',
                        resetSettings: 'Restablecer configuración',
                        seeOnDiscuss: 'Ver en el Debate',
                        seeOnDiscord: 'Ver en Discord',
                        scriptDevBy: 'Script desarrollado por Luan Tavares',
                    }
                };                
                
                break;

            case 'et':
                // Estonian
                translations = {
                    name: 'Kiiruskuva',
                    zoomOut: 'Kaugemal',
                    zoomIn: 'Lähemal',
                    changelog: 'Muudatuste ajalugu',
                    changelogMessage: `<h6>Mis on uut:</h6><ul style="margin-bottom: 10px"><li>"Ikoonide värskendamine viivitusega" teade eemaldati ja asendati lihtsa ikooniga, mis näitab, kui skript töötleb veel kiirusikoonide andmeid (pea meeles, et värskendusel on tahtlik viivitus).</li></ul><h6>Parandused ja täiendused:</h6><ul><li>Vahekaardi "üldine" nimi muudeti "seadistused", et paremini mõista iga vahekaardi funktsiooni skriptis;</li><li>Segmendi pikkuse filter ei eemalda enam ikooni, vaid rakendab nüüd sama läbipaistvuse nagu kiirusfiltris.</li></ul>`,
                    close: 'Sulge',
                    viewScript: 'Vaata skripti',
                    viewChangelog: 'Vaata muudatuste ajalugu',
                    reset: 'Lähtesta',
                    snackbar: {
                        errorFiltering: 'Viga filtri rakendamisel. Minimaalne kiirus ei saa olla suurem kui maksimaalne.',
                    },
                    title: {
                        settings: 'Seaded',
                        roadTypes: 'Teetüübid',
                        filter: 'Filtrid',
                        appearence: 'Välimus',
                        showOnRoadType: 'Kuva järgmistes teetüüpides:',
                        seeOnDiscuss: 'Vaata skripti teemat Discussis',
                        seeOnDiscord: 'Vaata skripti kanalit Waze Scripti Discord serveris',
                        scriptDevBy: 'Vaadake minu redaktori kasutajaprofiili',
                    },
                    label: {
                        displaySpeeds: 'Kuva kiirus',
                        showNoSpeed: 'Kuva ilma kiiruseta segmentidel',
                        showOnRoundabout: 'Kuva ringteel',
                        showOnUnpaved: 'Kuva kõvaketšimata teedel',
                        maxZoom: 'Kuva kuni suumi tasemeni:',
                        toggleLayerShortcut: 'Kiiruskuva kihi vahetamise otsetee',
                        zIndex: 'Kihi sügavus',
                        hideShorterSegments: 'Peida lühemad segmendid',
                        speedFilter: 'Filtreeri kiiruse järgi',
                        min: 'Miinimum:',
                        max: 'Maksimum:',
                        opacity: 'Filtri opakuse tasand:',
                    },
                    alertMessage: {
                        zIndex: `Kui kasutad kihi sügavust üle <b>${this.zIndex.warn}</b>, kattub kiiruskiht teiste kihtidega ja te ei saa valida ühtegi elementi ekraanil enne, kui vähendate kihi sügavust alla <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Kiiruskihtide nähtavuse vahetamiseks mine <b>Kaardi kihid > Teed > Kuvada kiirus</b> või kasuta kiirklahvi <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Vaikimisi',
                        altViennaConv: 'Alternatiivne Viini konventsioon',
                        japan: 'Jaapan',
                        northAmerica: 'Põhja-Ameerika',
                        sacu: 'Lõuna-Aafrika Tolliliit',
                        uk: 'Ühendkuningriik',
                    },
                    btn: {
                        updateSettingsTab: 'Uuenda menüü',
                        resetSettings: 'Lähtesta seaded',
                        seeOnDiscuss: 'Vaata arutelus',
                        seeOnDiscord: 'Vaata Discordis',
                        scriptDevBy: 'Skripti arendas Luan Tavares',
                    }
                };
            
                break;
            
            
            case 'eu':
                // Basque
                translations = {
                    name: 'Abiadura Erakustea',
                    zoomOut: 'Hurbilago',
                    zoomIn: 'Aldez aurretik',
                    changelog: 'Aldaketa erregistroa',
                    changelogMessage: `<h6>Berrikuntzak:</h6><ul style="margin-bottom: 10px"><li>"Ikonoak atzerapenarekin eguneratzen" mezua kendu eta ikono sinple batekin ordezkatu da, script-ak oraindik abiadura-ikonoak prozesatzen ari dela adierazteko (gogoratu eguneratzean atzerapen bat dagoela nahita).</li></ul><h6>Zuzendutakoak eta hobekuntzak:</h6><ul><li>"Orokorra" izeneko fitxa "doikuntzak" izenera aldatu da, fitxa bakoitzaren funtzioa argiago ulertzeko;</li><li>Segmentuaren tamainaren iragazkia ez du ikonoa kentzen, baizik eta abiadura-iragazkiarekin erabiltzen den gardentasun bera aplikatzen du.</li></ul>`,
                    close: 'Itxi',
                    viewScript: 'Ikusi script-a',
                    viewChangelog: 'Ikusi aldaketak',
                    reset: 'Berrezarri',
                    snackbar: {
                        errorFiltering: 'Errorea iragazkia aplikatzean. Gutxieneko abiadura ezin da handiagoa izan gehienekoa baino.',
                    },
                    title: {
                        settings: 'Ezarpenak',
                        roadTypes: 'Errepide motak',
                        filter: 'Filtratu',
                        appearence: 'Itxura',
                        showOnRoadType: 'Erakutsi hurrengo errepide motetan:',
                        seeOnDiscuss: 'Ikusi skriptaren haria Discuss-en',
                        seeOnDiscord: 'Ikusi skriptaren kanala Waze Script-en Discord zerbitzuan',
                        scriptDevBy: 'Ikusi nire editorearen erabiltzaile-profila',
                    },
                    label: {
                        displaySpeeds: 'Erakutsi abiadurak',
                        showNoSpeed: 'Erakutsi abiadurarik gabeko segmentuetan',
                        showOnRoundabout: 'Erakutsi biribilguneetan',
                        showOnUnpaved: 'Erakutsi asfaltatu gabeko bideetan',
                        maxZoom: 'Erakutsi zoom maila honetara arte:',
                        toggleLayerShortcut: 'Abiadura geruza aldatu laburdura',
                        zIndex: 'Geruzaren sakonera',
                        hideShorterSegments: 'Ezkutatu segmentu txikiagoak',
                        speedFilter: 'Iragazi abiaduraren arabera',
                        min: 'Gutxiengoa:',
                        max: 'Maximoa:',
                        opacity: 'Filtroko opazitatea maila:',
                    },
                    alertMessage: {
                        zIndex: `Geruza sakonera <b>${this.zIndex.warn}</b> baino handiagoa bada, abiadura geruzak beste geruzak estaliko ditu eta ezin izango duzu elementurik aukeratu pantailan, <b>${this.zIndex.warn}</b> baino baxuagoa den geruzarekin berriro jaitsi arte.`,
                        toggleVisibility: `Abiadura geruza bistaratzea aktibatu/itzaltzeko, joan <b>Mapa geruzak > Bideak > Abiadura erakutsi</b> eta erabili laburdura <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Definitibo',
                        altViennaConv: 'Alternatibo Vienna Konbentzioa',
                        japan: 'Japonia',
                        northAmerica: 'Ipar Amerika',
                        sacu: 'Hegoaldeko Afrikako Aduana Batasuna',
                        uk: 'Erresuma Batua',
                    },
                    btn: {
                        updateSettingsTab: 'Eguneratu menu',
                        resetSettings: 'Berrezarri ezarpenak',
                        seeOnDiscuss: 'Ikusi eztabaidan',
                        seeOnDiscord: 'Ikusi Discord-en',
                        scriptDevBy: 'Script-a Luan Tavares-ek garatu du',
                    }
                };
            
                break;
            
            case 'fa-IR':
                // Persian (Iran)
                translations = {
                    name: 'نمایش سرعت',
                    zoomOut: 'بیشتر دور',
                    zoomIn: 'بیشتر نزدیک',
                    changelog: 'تغییرات ثبت شده',
                    changelogMessage: `<h6>جدیدترین‌ها:</h6><ul style="margin-bottom: 10px"><li>پیام "در حال به‌روزرسانی آیکون‌ها با تأخیر" بالاخره حذف و با یک آیکون ساده جایگزین شد که نشان می‌دهد اسکریپت هنوز در حال پردازش آیکون‌های سرعت است (به خاطر داشته باشید که تأخیری عمدی وجود دارد).</li></ul><h6>اصلاحات و بهبودها:</h6><ul><li>نام تب "عمومی" به "تنظیمات" تغییر یافت تا عملکرد هر تب واضح‌تر شود؛</li><li>فیلتر اندازه بخش دیگر آیکون را حذف نمی‌کند بلکه از شفافیت مشابه فیلتر سرعت استفاده می‌کند.</li></ul>`,
                    close: 'بستن',
                    viewScript: 'مشاهده اسکریپت',
                    viewChangelog: 'مشاهده تغییرات ثبت شده',
                    reset: 'بازنشانی',
                    snackbar: {
                        errorFiltering: 'خطا در اعمال فیلتر. حداقل سرعت نمی‌تواند بیشتر از حداکثر باشد.',
                    },
                    title: {
                        settings: 'تنظیمات',
                        roadTypes: 'انواع جاده‌ها',
                        filter: 'فیلتر',
                        appearence: 'ظاهر',
                        showOnRoadType: 'نمایش در انواع جاده‌های زیر:',
                        seeOnDiscuss: 'مطالعه موضوع اسکریپت در Discuss',
                        seeOnDiscord: 'مشاهده کانال اسکریپت در سرور Discord اسکریپت‌های Waze',
                        scriptDevBy: 'مشاهده نمایه کاربری ویرایشگر من',
                    },
                    label: {
                        displaySpeeds: 'نمایش سرعت ها',
                        showNoSpeed: 'نمایش در بخش هایی بدون سرعت',
                        showOnRoundabout: 'نمایش در چرخشگاه ها',
                        showOnUnpaved: 'نمایش در جاده های خاکی',
                        maxZoom: 'نمایش تا زوم:',
                        toggleLayerShortcut: 'تغییر لایه نمایش سرعت',
                        zIndex: 'عمق لایه',
                        hideShorterSegments: 'مخفی کردن بخش های کوتاه تر از',
                        speedFilter: 'فیلتر بر اساس سرعت',
                        min: 'حداقل:',
                        max: 'حداکثر:',
                        opacity: 'سطح شفافیت فیلتر:',
                    },
                    alertMessage: {
                        zIndex: `اگر از عمق لایه بیشتر از <b>${this.zIndex.warn}</b> استفاده کنید، لایه سرعت بر روی لایه های دیگر قرار می گیرد و شما نمی توانید هیچ عنصری را در صفحه انتخاب کنید تا زمانی که عمق لایه را به کمتر از <b>${this.zIndex.warn}</b> کاهش دهید.`,
                        toggleVisibility: `برای فعال/غیرفعال کردن لایه نمایش سرعت، به <b>لایه های نقشه > جاده ها > نمایش سرعت</b> بروید یا از میانبر <b>ALT</b> + <b>SHIFT</b> + <b>S</b> استفاده کنید`
                    },
                    theme: {
                        default: 'پیش‌فرض',
                        altViennaConv: 'کنوانسیون وین جایگزین',
                        japan: 'ژاپن',
                        northAmerica: 'آمریکای شمالی',
                        sacu: 'اتحادیه گمرکی جنوب آفریقا',
                        uk: 'پادشاهی متحد',
                    },
                    btn: {
                        updateSettingsTab: 'به روز رسانی منو',
                        resetSettings: 'بازنشانی تنظیمات',
                        seeOnDiscuss: 'مشاهده در بحث',
                        seeOnDiscord: 'مشاهده در دیسکورد',
                        scriptDevBy: 'اسکریپت توسط لوان تاوارس توسعه داده شده است',
                    }
                };
            
                break;

            case 'fi':
                // Finnish
                translations = {
                    name: 'Nopeusnäyttö',
                    zoomOut: 'Laajenna',
                    zoomIn: 'Lähennä',
                    changelog: 'Muutokset',
                    changelogMessage: `<h6>Uutta:</h6><ul style="margin-bottom: 10px"><li>"Päivitän kuvakkeita viiveellä" -viesti on poistettu ja korvattu yksinkertaisella kuvakkeella, joka osoittaa, kun skripti edelleen käsittelee nopeusikoneita (muista, että päivityksessä on tarkoituksellinen viive).</li></ul><h6>Korjaukset ja parannukset:</h6><ul><li>"Yleinen"-välilehden nimi on muutettu muotoon "asetukset", jotta skriptin välilehtien tarkoitus olisi selkeämpi;</li><li>Segmenttikoon suodatin ei enää poista kuvaketta vaan käyttää samaa läpinäkyvyyttä kuin nopeussuodatin.</li></ul>`,
                    close: 'Sulje',
                    viewScript: 'Näytä skripti',
                    viewChangelog: 'Näytä muutokset',
                    reset: 'Palauta',
                    snackbar: {
                        errorFiltering: 'Virhe suodatinta sovellettaessa. Vähimmäisnopeus ei voi olla suurempi kuin enimmäisnopeus.',
                    },
                    title: {
                        settings: 'Asetukset',
                        roadTypes: 'Tiestötyypit',
                        filter: 'Suodatin',
                        appearence: 'Ulkoasu',
                        showOnRoadType: 'Näytä seuraavilla tietyyppien alueilla:',
                        seeOnDiscuss: 'Katso skriptin ketju Discussissa',
                        seeOnDiscord: 'Katso skriptin kanava Waze Scriptin Discord-palvelimella',
                        scriptDevBy: 'Näytä editorini käyttäjäprofiili',
                    },
                    label: {
                        displaySpeeds: 'Näytä nopeudet',
                        showNoSpeed: 'Näytä segmentissä, jossa ei ole nopeutta',
                        showOnRoundabout: 'Näytä liikenneympyrässä',
                        showOnUnpaved: 'Näytä päällystämättömällä tiellä',
                        maxZoom: 'Renderöi zoom-tasoon asti:',
                        toggleLayerShortcut: 'Vaihda nopeuskerroksen näkyvyyttä',
                        zIndex: 'Kerrosjärjestys',
                        hideShorterSegments: 'Piilota lyhyemmät segmentit kuin',
                        speedFilter: 'Suodata nopeuden mukaan',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Suodattimen läpinäkyvyysaste:',
                    },
                    alertMessage: {
                        zIndex: `Kun käytät suurempaa kerrosjärjestystä kuin <b>${this.zIndex.warn}</b>, nopeuskerros peittää muut ja et voi valita mitään elementtiä näytöltä, ennen kuin lasket sen takaisin alle <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Voit aktivoida/poistaa käytöstä nopeuskerroksen menemällä kohtaan <b>Karttatasot > Tiet > Näytä nopeudet</b> tai käyttämällä pikanäppäintä <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Oletus',
                        altViennaConv: 'Vaihtoehtoinen Wienin sopimus',
                        japan: 'Japani',
                        northAmerica: 'Pohjois-Amerikka',
                        sacu: 'Etelä-Afrikan tulliliitto',
                        uk: 'Yhdistynyt kuningaskunta',
                    },
                    btn: {
                        updateSettingsTab: 'Päivitä valikko',
                        resetSettings: 'Palauta asetukset',
                        seeOnDiscuss: 'Katso keskustelussa',
                        seeOnDiscord: 'Katso Discordissa',
                        scriptDevBy: 'Skriptin on kehittänyt Luan Tavares',
                    }
                };

                break;

            case 'fr':
                // French
                translations = {
                    name: 'Affichage de la vitesse',
                    zoomOut: 'Zoom arrière',
                    zoomIn: 'Zoom avant',
                    changelog: 'Journal des modifications',
                    changelogMessage: `<h6>Nouveautés :</h6><ul style="margin-bottom: 10px"><li>Le message "mise à jour des icônes avec délai" a été supprimé et remplacé par une icône simple indiquant que le script traite encore les icônes de vitesse (n'oubliez pas qu'il y a un délai intentionnel).</li></ul><h6>Corrections et améliorations :</h6><ul><li>L'onglet "général" a été renommé en "réglages" pour mieux clarifier la fonction de chaque onglet du script ;</li><li>Le filtre de taille de segment n'enlève plus l'icône mais applique maintenant la même opacité que le filtre de vitesse.</li></ul>`,
                    close: 'Fermer',
                    viewScript: 'Voir le script',
                    viewChangelog: 'Voir le journal des modifications',
                    reset: 'Réinitialiser',
                    snackbar: {
                        errorFiltering: 'Erreur lors de l’application du filtre. La vitesse minimale ne peut pas être supérieure à la maximale.',
                    },
                    title: {
                        settings: 'Paramètres',
                        roadTypes: 'Types de route',
                        filter: 'Filtrer',
                        appearence: 'Apparence',
                        showOnRoadType: 'Afficher sur les types de route suivants :',
                        seeOnDiscuss: 'Voir le fil du script sur Discuss',
                        seeOnDiscord: 'Voir le canal du script sur le serveur Discord de Waze Script',
                        scriptDevBy: 'Voir mon profil utilisateur de l’éditeur',
                    },
                    label: {
                        displaySpeeds: 'Afficher les vitesses',
                        showNoSpeed: 'Afficher sur les segments sans vitesse',
                        showOnRoundabout: 'Afficher sur le rond-point',
                        showOnUnpaved: 'Afficher sur la voie non pavée',
                        maxZoom: 'Rendre jusqu\'au zoom :',
                        toggleLayerShortcut: 'Alterner la visibilité de la couche de vitesse',
                        zIndex: 'Profondeur de la couche',
                        hideShorterSegments: 'Masquer sur les segments plus petits que',
                        speedFilter: 'Filtrer par vitesse',
                        min: 'Min. :',
                        max: 'Max. :',
                        opacity: 'Niveau de transparence du filtre:',
                    },
                    alertMessage: {
                        zIndex: `Lorsque vous utilisez une profondeur de couche supérieure à <b>${this.zIndex.warn}</b>, la couche de vitesse va se superposer aux autres et vous ne pourrez sélectionner aucun élément sur l\'écran jusqu\'à ce que vous la fassiez descendre sous <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Pour activer/désactiver la couche de vitesse, allez dans <b>Couches de la carte > Routes > Afficher les vitesses</b> ou utilisez le raccourci <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Par défaut',
                        altViennaConv: 'Convention de Vienne alternative',
                        japan: 'Japon',
                        northAmerica: 'Amérique du Nord',
                        sacu: 'Union douanière d\'Afrique australe',
                        uk: 'Royaume-Uni',
                    },
                    btn: {
                        updateSettingsTab: 'Mettre à jour le menu',
                        resetSettings: 'Réinitialiser les paramètres',
                        seeOnDiscuss: 'Voir sur la discussion',
                        seeOnDiscord: 'Voir sur Discord',
                        scriptDevBy: 'Script développé par Luan Tavares',
                    }
                };
                
                break;

            case 'gl':
                // Galician
                translations = {
                    name: 'Exhibición de velocidade',
                    zoomOut: 'Máis afastado',
                    zoomIn: 'Máis achegado',
                    changelog: 'Rexistro de cambios',
                    changelogMessage: `<h6>Novedades:</h6><ul style="margin-bottom: 10px"><li>Eliminouse finalmente a mensaxe de "actualización dos iconos con atraso" e substituíuse por un icono sinxelo, indicando claramente cando o script segue a procesar os iconos de velocidade (lembre que hai un atraso intencionado).</li></ul><h6>Correccións e melloras:</h6><ul><li>Cambiouse o nome da lapela "xeral" a "axustes" para unha mellor comprensión da función de cada lapela do script;</li><li>O filtro por tamaño de segmento xa non elimina o icono, agora aplica a mesma opacidade que o filtro de velocidade.</li></ul>`,
                    close: 'Pechar',
                    viewScript: 'Ver script',
                    viewChangelog: 'Ver rexistro de cambios',
                    reset: 'Restablecer',
                    snackbar: {
                        errorFiltering: 'Erro ao aplicar o filtro. A velocidade mínima non pode ser maior que a máxima.',
                    },
                    title: {
                        settings: 'Configuracións',
                        roadTypes: 'Tipos de estrada',
                        filter: 'Filtro',
                        appearence: 'Aparencia',
                        showOnRoadType: 'Amosar nos seguintes tipos de estrada:',
                        seeOnDiscuss: 'Ver o fío do script en Discuss',
                        seeOnDiscord: 'Ver o canal do script no servidor Discord de Waze Script',
                        scriptDevBy: 'Ver o meu perfil de usuario do editor',
                    },
                    label: {
                        displaySpeeds: 'Exhibir velocidades',
                        showNoSpeed: 'Exhibir en segmentos sen velocidade',
                        showOnRoundabout: 'Exhibir en rotondas',
                        showOnUnpaved: 'Exhibir en vías non pavimentadas',
                        maxZoom: 'Renderizar ata o zoom:',
                        toggleLayerShortcut: 'Alternar capa de exhibición de velocidade',
                        zIndex: 'Profundidade da capa',
                        hideShorterSegments: 'Ocultar en segmentos máis pequenos que',
                        speedFilter: 'Filtrar por velocidade',
                        min: 'Mín.:',
                        max: 'Máx.:',
                        opacity: 'Nivel de opacidade do filtro:',
                    },
                    alertMessage: {
                        zIndex: `Ao usar unha profundidade de capa maior que <b>${this.zIndex.warn}</b>, a capa de velocidade cubrirá outras capas e non poderá seleccionar ningún elemento na pantalla ata que a baixe por debaixo de <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Para activar/desactivar a capa de exhibición de velocidades, vai a <b>Capas do mapa > Vías > Exhibir velocidades</b> ou utiliza o atallo <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Por defecto',
                        altViennaConv: 'Convención de Viena alternativa',
                        japan: 'Japón',
                        northAmerica: 'América do Norte',
                        sacu: 'Unión Aduanera de África Austral',
                        uk: 'Reino Unido',
                    },
                    btn: {
                        updateSettingsTab: 'Actualizar menú',
                        resetSettings: 'Restablecer configuracións',
                        seeOnDiscuss: 'Ver na discusión',
                        seeOnDiscord: 'Ver en Discord',
                        scriptDevBy: 'Script desenvolvido por Luan Tavares',
                    }
                };
            
                break;
            
            case 'he':
                // Hebrew
                translations = {
                    name: 'הצגת מהירות',
                    zoomOut: 'מרחוק יותר',
                    zoomIn: 'מקרוב יותר',
                    changelog: 'יומן שינויים',
                    changelogMessage: `<h6>מה חדש:</h6><ul style="margin-bottom: 10px"><li>הוסר סוף סוף ההודעה "מעדכן סמלים עם השהיה" והוחלף באייקון פשוט, המבהיר מתי הסקריפט עדיין מעבד את סמלי המהירות (זכור שיש עיכוב מכוון).</li></ul><h6>תיקונים ושיפורים:</h6><ul><li>שונה שם הלשונית מ"כללי" ל"הגדרות" לשם הבנה טובה יותר של תפקוד כל לשונית;</li><li>הפילטר לפי אורך קטע אינו מסיר עוד את האייקון, אלא מיישם שקיפות כפי שנעשה בפילטר המהירות.</li></ul>`,
                    close: 'סגור',
                    viewScript: 'צפה בסקריפט',
                    viewChangelog: 'צפה ביומן השינויים',
                    reset: 'אפס',
                    snackbar: {
                        errorFiltering: 'שגיאה ביישום המסנן. המהירות המינימלית לא יכולה להיות גבוהה מהמהירות המרבית.',
                    },
                    title: {
                        settings: 'הגדרות',
                        roadTypes: 'סוגי דרכים',
                        filter: 'סנן',
                        appearence: 'מראה',
                        showOnRoadType: 'הצג על סוגי הדרכים הבאים:',
                        seeOnDiscuss: 'ראה את השרשור של הסקריפט ב-Discuss',
                        seeOnDiscord: 'ראה את הערוץ של הסקריפט בשרת Discord של Waze Script',
                        scriptDevBy: 'הצג את פרופיל המשתמש של העורך שלי',
                    },
                    label: {
                        displaySpeeds: 'הצג מהירויות',
                        showNoSpeed: 'הצג בקטעים ללא מהירות',
                        showOnRoundabout: 'הצג בכיכרות',
                        showOnUnpaved: 'הצג בדרכים לא סלוליות',
                        maxZoom: 'הצג עד לזום:',
                        toggleLayerShortcut: 'החלף שכבת הצגת מהירות',
                        zIndex: 'עומק השכבה',
                        hideShorterSegments: 'הסתר בקטעים קצרים מ-',
                        speedFilter: 'סנן לפי מהירות',
                        min: 'מינימום:',
                        max: 'מקסימום:',
                        opacity: 'רמת שקיפות המסנן:',
                    },
                    alertMessage: {
                        zIndex: `כאשר משתמשים בעומק שכבה גבוה יותר מ- <b>${this.zIndex.warn}</b>, שכבת המהירות תכסה שכבות אחרות ולא ניתן יהיה לבחור אף פריט על המסך עד שתרד את עומק השכבה מתחת ל- <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `כדי להפעיל/לכבות את שכבת הצגת המהירויות, פשוט לכו ל- <b>שכבות המפה > דרכים > הצג מהירויות</b> או השתמשו בקיצור הדרך <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'ברירת מחדל',
                        altViennaConv: 'אמנת וינה חלופין',
                        japan: 'יפן',
                        northAmerica: 'צפון אמריקה',
                        sacu: 'איחוד המכס של דרום אפריקה',
                        uk: 'הממלכה המאוחדת',
                    },
                    btn: {
                        updateSettingsTab: 'עדכן תפריט',
                        resetSettings: 'אפס הגדרות',
                        seeOnDiscuss: 'ראה בדיון',
                        seeOnDiscord: 'ראה ב-Discord',
                        scriptDevBy: 'הסקריפט פותח על ידי לואן טאווארס',
                    }
                };
            
                break;
            
            case 'hi':
                // Hindi
                translations = {
                    name: 'स्पीड डिस्प्ले',
                    zoomOut: 'और दूर',
                    zoomIn: 'और पास',
                    changelog: 'चेंजलॉग',
                    changelogMessage: `<h6>नवीनतम:</h6><ul style="margin-bottom: 10px"><li>"विलंब से आइकन अपडेट हो रहे हैं" संदेश को अंततः हटा दिया गया है और इसे एक सरल आइकन से बदल दिया गया है, जो स्पष्ट करता है कि स्क्रिप्ट अभी भी गति आइकन को प्रोसेस कर रही है (ध्यान दें कि अद्यतन में जानबूझकर देरी है)।</li></ul><h6>सुधार और बेहतरियाँ:</h6><ul><li>"सामान्य" टैब का नाम "सेटिंग्स" में बदल दिया गया है ताकि प्रत्येक टैब की भूमिका स्पष्ट हो सके;</li><li>सेगमेंट साइज़ फ़िल्टर अब आइकन को नहीं हटाता, बल्कि गति फ़िल्टर जैसी ही पारदर्शिता लागू करता है।</li></ul>`,
                    close: 'बंद करें',
                    viewScript: 'स्क्रिप्ट देखें',
                    viewChangelog: 'चेंजलॉग देखें',
                    reset: 'रीसेट',
                    snackbar: {
                        errorFiltering: 'फ़िल्टर लागू करने में त्रुटि। न्यूनतम गति अधिकतम से अधिक नहीं हो सकती।',
                    },
                    title: {
                        settings: 'सेटिंग्स',
                        roadTypes: 'सड़क प्रकार',
                        filter: 'फ़िल्टर',
                        appearence: 'रूप',
                        showOnRoadType: 'निम्नलिखित सड़क प्रकारों पर दिखाएं:',
                        seeOnDiscuss: 'Discuss पर स्क्रिप्ट का थ्रेड देखें',
                        seeOnDiscord: 'Waze Script के Discord सर्वर पर स्क्रिप्ट चैनल देखें',
                        scriptDevBy: 'मेरा संपादक उपयोगकर्ता प्रोफ़ाइल देखें',
                    },
                    label: {
                        displaySpeeds: 'गति दिखाएं',
                        showNoSpeed: 'गति के बिना सेगमेंट में दिखाएं',
                        showOnRoundabout: 'राउंडअबाउट में दिखाएं',
                        showOnUnpaved: 'अस्फल्टेड सड़कों पर दिखाएं',
                        maxZoom: 'जूम स्तर तक रेंडर करें:',
                        toggleLayerShortcut: 'स्पीड डिस्प्ले लेयर बदलें',
                        zIndex: 'लेयर की गहराई',
                        hideShorterSegments: 'छोटे सेगमेंट को छुपाएं',
                        speedFilter: 'गति के अनुसार फ़िल्टर करें',
                        min: 'न्यूनतम:',
                        max: 'अधिकतम:',
                        opacity: 'फ़िल्टर अपारदर्शिता स्तर:',
                    },
                    alertMessage: {
                        zIndex: `यदि आप <b>${this.zIndex.warn}</b> से बड़ी लेयर गहराई का उपयोग करते हैं, तो स्पीड डिस्प्ले अन्य लेयर्स के ऊपर आ जाएगा और स्क्रीन पर कोई भी तत्व चयनित नहीं किया जा सकेगा, जब तक कि आप इसे <b>${this.zIndex.warn}</b> से नीचे नहीं कर देते।`,
                        toggleVisibility: `स्पीड डिस्प्ले लेयर को सक्रिय/निष्क्रिय करने के लिए, बस <b>मैप लेयर्स > सड़कें > स्पीड डिस्प्ले</b> पर जाएं या शॉर्टकट का उपयोग करें <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'डिफ़ॉल्ट',
                        altViennaConv: 'वैकल्पिक वियना सम्मेलन',
                        japan: 'जापान',
                        northAmerica: 'उत्तरी अमेरिका',
                        sacu: 'दक्षिण अफ़्रीकी सीमा शुल्क संघ',
                        uk: 'संयुक्त राज्य',
                    },
                    btn: {
                        updateSettingsTab: 'मेन्‍यू अपडेट करें',
                        resetSettings: 'सेटिंग्स रीसेट करें',
                        seeOnDiscuss: 'चर्चा में देखें',
                        seeOnDiscord: 'Discord पर देखें',
                        scriptDevBy: 'स्क्रिप्ट लुआन तावरेस द्वारा विकसित की गई है',
                    }
                };
            
                break;
            
            case 'hr':
                // Croatian
                translations = {
                    name: 'Prikaz brzine',
                    zoomOut: 'Više udaljeno',
                    zoomIn: 'Više približeno',
                    changelog: 'Povijest izmjena',
                    changelogMessage: `<h6>Novosti:</h6><ul style="margin-bottom: 10px"><li>Poruka "ažuriranje ikona s odgodom" konačno je uklonjena i zamijenjena jednostavnom ikonom koja jasno pokazuje kada skripta još uvijek obrađuje ikone brzine (imajte na umu da postoji namjerno kašnjenje).</li></ul><h6>Ispravci i poboljšanja:</h6><ul><li>Kartica "opće" preimenovana je u "postavke" radi boljeg razumijevanja funkcije svake kartice;</li><li>Filtar veličine segmenta više ne uklanja ikonu, već sada primjenjuje istu neprozirnost kao i filtar brzine.</li></ul>`,
                    close: 'Zatvori',
                    viewScript: 'Pogledaj skriptu',
                    viewChangelog: 'Pogledaj povijest izmjena',
                    reset: 'Poništi',
                    snackbar: {
                        errorFiltering: 'Pogreška pri primjeni filtra. Minimalna brzina ne može biti veća od maksimalne.',
                    },
                    title: {
                        settings: 'Postavke',
                        roadTypes: 'Vrste cesta',
                        filter: 'Filtriraj',
                        appearence: 'Izgled',
                        showOnRoadType: 'Prikaži na sljedećim vrstama cesta:',
                        seeOnDiscuss: 'Pogledajte temu skripte na Discussu',
                        seeOnDiscord: 'Pogledajte kanal skripte na Waze Script Discord poslužitelju',
                        scriptDevBy: 'Pogledajte moj korisnički profil uređivača',
                    },
                    label: {
                        displaySpeeds: 'Prikazuj brzine',
                        showNoSpeed: 'Prikazuj na segmentima bez brzine',
                        showOnRoundabout: 'Prikazuj na kružnim tokovima',
                        showOnUnpaved: 'Prikazuj na neasfaltiranim cestama',
                        maxZoom: 'Prikazuj do razine zooma:',
                        toggleLayerShortcut: 'Prekidač sloja za prikaz brzine',
                        zIndex: 'Dubina sloja',
                        hideShorterSegments: 'Sakrij segmente kraće od',
                        speedFilter: 'Filtriraj prema brzini',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Razina neprozirnosti filtra:',
                    },
                    alertMessage: {
                        zIndex: `Ako koristite dubinu sloja veću od <b>${this.zIndex.warn}</b>, sloj brzine će se preklapati s drugim slojevima i neće biti moguće odabrati bilo koji element na ekranu dok ga ne spustite ispod <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Za aktiviranje/deaktiviranje sloja prikaza brzine, jednostavno idite na <b>Slabe mape > Ceste > Prikaz brzina</b> ili koristite prečac <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Zadano',
                        altViennaConv: 'Alternativna Bečka konvencija',
                        japan: 'Japan',
                        northAmerica: 'Sjeverna Amerika',
                        sacu: 'Južnoafrička unija carina',
                        uk: 'Ujedinjeno Kraljevstvo',
                    },
                    btn: {
                        updateSettingsTab: 'Ažuriraj izbornik',
                        resetSettings: 'Vrati postavke na zadane',
                        seeOnDiscuss: 'Pogledajte u raspravi',
                        seeOnDiscord: 'Pogledajte na Discordu',
                        scriptDevBy: 'Skript je razvio Luan Tavares',
                    }
                };
            
                break;
            
            case 'hu':
                // Hungarian
                translations = {
                    name: 'Sebesség megjelenítése',
                    zoomOut: 'Távolabb',
                    zoomIn: 'Közelebb',
                    changelog: 'Változási napló',
                    changelogMessage: `<h6>Újdonságok:</h6><ul style="margin-bottom: 10px"><li>Végre eltávolítottuk a "késleltetett ikonfrissítés" üzenetet, és egyszerű ikonnal helyettesítettük, jelezve, hogy a szkript még dolgozik a sebességikonokon (ne feledje, a frissítés szándékosan késik).</li></ul><h6>Javítások és fejlesztések:</h6><ul><li>Az "általános" fül neve mostantól "beállítások" a jobb érthetőség érdekében;</li><li>A szegmensméret szűrő mostantól nem távolítja el az ikont, hanem ugyanolyan áttetszőséget alkalmaz, mint a sebességszűrő.</li></ul>`,
                    close: 'Bezárás',
                    viewScript: 'Szkript megtekintése',
                    viewChangelog: 'Változási napló megtekintése',
                    reset: 'Visszaállítás',
                    snackbar: {
                        errorFiltering: 'Hiba a szűrő alkalmazásakor. A minimális sebesség nem lehet nagyobb, mint a maximális.',
                    },
                    title: {
                        settings: 'Beállítások',
                        roadTypes: 'Úttípusok',
                        filter: 'Szűrő',
                        appearence: 'Megjelenés',
                        showOnRoadType: 'Megjelenítés a következő úttípusokon:',
                        seeOnDiscuss: 'Nézd meg a script témáját a Discuss-on',
                        seeOnDiscord: 'Nézd meg a script csatornát a Waze Script Discord szerverén',
                        scriptDevBy: 'Tekintse meg a szerkesztői felhasználói profilomat',
                    },
                    label: {
                        displaySpeeds: 'Sebességek megjelenítése',
                        showNoSpeed: 'Megjelenítés sebesség nélküli szegmenseken',
                        showOnRoundabout: 'Megjelenítés körforgalomban',
                        showOnUnpaved: 'Megjelenítés burkolatlan úton',
                        maxZoom: 'Renderelés legnagyobb zoom szinten:',
                        toggleLayerShortcut: 'Sebesség réteg átváltása',
                        zIndex: 'Réteg mélysége',
                        hideShorterSegments: 'Rejtsd el a rövidebb szegmenseket, mint',
                        speedFilter: 'Szűrés sebesség szerint',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Szűrő átlátszóság szintje:',
                    },
                    alertMessage: {
                        zIndex: `Ha a réteg mélysége nagyobb, mint <b>${this.zIndex.warn}</b>, a sebesség réteg átfedheti a többi réteget, és nem lesz lehetőség semmilyen elem kiválasztására a képernyőn, amíg le nem csökkenti <b>${this.zIndex.warn}</b> alá.`,
                        toggleVisibility: `A sebesség réteg láthatóságának aktiválásához/deaktiválásához egyszerűen menjen ide: <b>Térképrétegek > Út > Sebesség megjelenítése</b>, vagy használja a gyorsbillentyűt <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Alapértelmezett',
                        altViennaConv: 'Alternatív Bécsi Egyezmény',
                        japan: 'Japán',
                        northAmerica: 'Észak-Amerika',
                        sacu: 'Dél-afrikai Vámunió',
                        uk: 'Egyesült Királyság',
                    },
                    btn: {
                        updateSettingsTab: 'Menü frissítése',
                        resetSettings: 'Beállítások visszaállítása',
                        seeOnDiscuss: 'Nézd meg a vitában',
                        seeOnDiscord: 'Nézd meg Discordon',
                        scriptDevBy: 'A scriptet Luan Tavares fejlesztette',
                    }
                };
            
                break;
            
            case 'id':
                // Indonesian
                translations = {
                    name: 'Tampilan Kecepatan',
                    zoomOut: 'Lebih jauh',
                    zoomIn: 'Lebih dekat',
                    changelog: 'Catatan perubahan',
                    changelogMessage: `<h6>Baru:</h6><ul style="margin-bottom: 10px"><li>Pesan "memperbarui ikon dengan jeda" akhirnya dihapus dan diganti dengan ikon sederhana, memperjelas saat skrip masih memproses ikon kecepatan (ingat bahwa ada penundaan yang disengaja).</li></ul><h6>Perbaikan dan peningkatan:</h6><ul><li>Nama tab "umum" diubah menjadi "pengaturan" untuk memperjelas fungsi setiap tab dalam skrip;</li><li>Filter ukuran segmen sekarang tidak lagi menghapus ikon, tetapi menerapkan tingkat transparansi yang sama dengan filter kecepatan.</li></ul>`,
                    close: 'Tutup',
                    viewScript: 'Lihat skrip',
                    viewChangelog: 'Lihat catatan perubahan',
                    reset: 'Atur ulang',
                    snackbar: {
                        errorFiltering: 'Kesalahan dalam menerapkan filter. Kecepatan minimum tidak boleh lebih besar dari maksimum.',
                    },
                    title: {
                        settings: 'Pengaturan',
                        roadTypes: 'Tipe Jalan',
                        filter: 'Filter',
                        appearence: 'Tampilan',
                        showOnRoadType: 'Tampilkan pada tipe jalan berikut:',
                        seeOnDiscuss: 'Lihat thread skrip di Discuss',
                        seeOnDiscord: 'Lihat saluran skrip di server Discord Waze Script',
                        scriptDevBy: 'Lihat profil pengguna editor saya',
                    },
                    label: {
                        displaySpeeds: 'Tampilkan kecepatan',
                        showNoSpeed: 'Tampilkan pada segmen tanpa kecepatan',
                        showOnRoundabout: 'Tampilkan di bundaran',
                        showOnUnpaved: 'Tampilkan di jalan tidak beraspal',
                        maxZoom: 'Render hingga zoom:',
                        toggleLayerShortcut: 'Toggling lapisan tampilan kecepatan',
                        zIndex: 'Kedalaman lapisan',
                        hideShorterSegments: 'Sembunyikan pada segmen yang lebih pendek dari',
                        speedFilter: 'Filter berdasarkan kecepatan',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Tingkat opasitas filter:',
                    },
                    alertMessage: {
                        zIndex: `Jika kedalaman lapisan lebih besar dari <b>${this.zIndex.warn}</b>, lapisan kecepatan akan menutupi lapisan lain dan Anda tidak akan dapat memilih elemen di layar sampai Anda menurunkan kedalaman lapisan kembali di bawah <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Untuk mengaktifkan/mematikan lapisan tampilan kecepatan, cukup pergi ke <b>Lapisan Peta > Jalan > Tampilkan kecepatan</b> atau gunakan pintasan <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Default',
                        altViennaConv: 'Konvensi Wina Alternatif',
                        japan: 'Jepang',
                        northAmerica: 'Amerika Utara',
                        sacu: 'Uni Bea Cukai Afrika Selatan',
                        uk: 'Inggris Raya',
                    },
                    btn: {
                        updateSettingsTab: 'Perbarui menu',
                        resetSettings: 'Atur ulang pengaturan',
                        seeOnDiscuss: 'Lihat di diskusi',
                        seeOnDiscord: 'Lihat di Discord',
                        scriptDevBy: 'Script dikembangkan oleh Luan Tavares',
                    }
                };
            
                break;
            
            case 'it':
                // Italian
                translations = {
                    name: 'Visualizzazione Velocità',
                    zoomOut: 'Zoom indietro',
                    zoomIn: 'Zoom avanti',
                    changelog: 'Registro modifiche',
                    changelogMessage: `<h6>Novità:</h6><ul style="margin-bottom: 10px"><li>Il messaggio "aggiornamento icone con ritardo" è stato finalmente rimosso e sostituito con una semplice icona, che chiarisce quando lo script sta ancora elaborando le icone di velocità (ricorda che c'è un ritardo intenzionale).</li></ul><h6>Correzioni e miglioramenti:</h6><ul><li>La scheda "generale" è stata rinominata in "impostazioni" per chiarire meglio la funzione di ciascuna scheda;</li><li>Il filtro per la dimensione del segmento non rimuove più l'icona, ma applica la stessa opacità del filtro di velocità.</li></ul>`,
                    close: 'Chiudi',
                    viewScript: 'Vedi script',
                    viewChangelog: 'Visualizza registro modifiche',
                    reset: 'Ripristina',
                    snackbar: {
                        errorFiltering: 'Errore nell’applicazione del filtro. La velocità minima non può essere maggiore della massima.',
                    },
                    title: {
                        settings: 'Impostazioni',
                        roadTypes: 'Tipi di strada',
                        filter: 'Filtro',
                        appearence: 'Aspetto',
                        showOnRoadType: 'Mostra sui seguenti tipi di strada:',
                        seeOnDiscuss: 'Vedi il thread dello script su Discuss',
                        seeOnDiscord: 'Vedi il canale dello script sul server Discord di Waze Script',
                        scriptDevBy: 'Visualizza il profilo utente del mio editor',
                    },
                    label: {
                        displaySpeeds: 'Visualizza velocità',
                        showNoSpeed: 'Visualizza su segmento senza velocità',
                        showOnRoundabout: 'Visualizza su rotatoria',
                        showOnUnpaved: 'Visualizza su strada non asfaltata',
                        maxZoom: 'Renderizza fino allo zoom:',
                        toggleLayerShortcut: 'Alterna il layer di visualizzazione velocità',
                        zIndex: 'Profondità del layer',
                        hideShorterSegments: 'Nascondi nei segmenti più corti di',
                        speedFilter: 'Filtra per velocità',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Livello opacità filtro:',
                    },
                    alertMessage: {
                        zIndex: `Se utilizzi una profondità del layer maggiore di <b>${this.zIndex.warn}</b>, il layer della velocità coprirà gli altri e non sarà possibile selezionare alcun elemento sulla mappa finché non riduci nuovamente la profondità sotto <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Per attivare/disattivare il layer di visualizzazione delle velocità, vai su <b>Layer Mappa > Strade > Visualizza velocità</b> o usa la scorciatoia <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Predefinito',
                        altViennaConv: 'Convenzione di Vienna Alternativa',
                        japan: 'Giappone',
                        northAmerica: 'Nord America',
                        sacu: 'Unione Doganale dell’Africa Australe',
                        uk: 'Regno Unito',
                    },
                    btn: {
                        updateSettingsTab: 'Aggiorna menu',
                        resetSettings: 'Ripristina impostazioni',
                        seeOnDiscuss: 'Vedi nella discussione',
                        seeOnDiscord: 'Vedi su Discord',
                        scriptDevBy: 'Script sviluppato da Luan Tavares',
                    }
                };
            
                break;

            case 'ja':
                // Japanese
                translations = {
                    name: '速度表示',
                    zoomOut: 'ズームアウト',
                    zoomIn: 'ズームイン',
                    changelog: '変更履歴',
                    changelogMessage: `<h6>新機能:</h6><ul style="margin-bottom: 10px"><li>「遅延付きアイコン更新」のメッセージを削除し、シンプルなアイコンに置き換えました。これにより、スクリプトが速度アイコンを処理中であることが明確になりました(更新には意図的な遅延があります)。</li></ul><h6>修正と改善:</h6><ul><li>「一般」タブの名前を「設定」に変更し、各タブの機能をより明確にしました。</li><li>セグメントサイズによるフィルターは、アイコンを非表示にする代わりに、速度フィルターと同じ透明度を適用するようになりました。</li></ul>`,
                    close: '閉じる',
                    viewScript: 'スクリプトを見る',
                    viewChangelog: '変更履歴を見る',
                    reset: 'リセット',
                    snackbar: {
                        errorFiltering: 'フィルター適用エラー。最小速度が最大速度を超えることはできません。',
                    },
                    title: {
                        settings: '設定',
                        roadTypes: '道路タイプ',
                        filter: 'フィルター',
                        appearence: '外観',
                        showOnRoadType: '次の道路タイプに表示:',
                        seeOnDiscuss: 'Discussでスクリプトのスレッドを見る',
                        seeOnDiscord: 'Waze ScriptのDiscordサーバーでスクリプトのチャンネルを見る',
                        scriptDevBy: 'エディターのユーザープロフィールを表示',
                    },
                    label: {
                        displaySpeeds: '速度を表示',
                        showNoSpeed: '速度なしのセグメントに表示',
                        showOnRoundabout: 'ラウンドアバウトに表示',
                        showOnUnpaved: '未舗装道路に表示',
                        maxZoom: 'ズーム最大:',
                        toggleLayerShortcut: '速度表示レイヤーを切り替える',
                        zIndex: 'レイヤーの深度',
                        hideShorterSegments: '次の長さ未満のセグメントを非表示:',
                        speedFilter: '速度でフィルタリング',
                        min: '最小:',
                        max: '最大:',
                        opacity: 'フィルターの不透明度レベル:',
                    },
                    alertMessage: {
                        zIndex: `レイヤーの深度を <b>${this.zIndex.warn}</b> より大きく設定すると、速度レイヤーが他のレイヤーを覆い、再び <b>${this.zIndex.warn}</b> 以下に戻すまでマップ上の要素を選択できなくなります。`,
                        toggleVisibility: `速度表示レイヤーをオン/オフにするには、<b>マップレイヤー > 道路 > 速度表示</b>に移動するか、ショートカット <b>ALT</b> + <b>SHIFT</b> + <b>S</b> を使用してください。`
                    },
                    theme: {
                        default: 'デフォルト',
                        altViennaConv: '代替ウィーン条約',
                        japan: '日本',
                        northAmerica: '北アメリカ',
                        sacu: '南部アフリカ関税同盟',
                        uk: 'イギリス',
                    },
                    btn: {
                        updateSettingsTab: 'メニューを更新',
                        resetSettings: '設定をリセット',
                        seeOnDiscuss: 'ディスカッションで見る',
                        seeOnDiscord: 'Discordで見る',
                        scriptDevBy: 'スクリプトはLuan Tavaresによって開発されました',
                    }
                };
            
                break;
            
            case 'ka':
                // Georgian
                translations = {
                    name: 'სისწრაფის ჩვენება',
                    zoomOut: 'გამორთვა',
                    zoomIn: 'ზუმი',
                    changelog: 'შეცდომების ჩანაწერი',
                    changelogMessage: `<h6>ახალი ფუნქციები:</h6><ul style="margin-bottom: 10px"><li>შეტყობინება "ხატულების განახლება შეფერხებით" საბოლოოდ მოიხსნა და ჩანაცვლდა მარტივი ხატულით, რაც ცხადყოფს, რომ სკრიპტი ჯერ კიდევ ამუშავებს სიჩქარის ხატულებს (დაიმახსოვრეთ, რომ განახლებას მიზანმიმართული დაგვიანება აქვს).</li></ul><h6>შეცდომების გამოსწორება და გაუმჯობესება:</h6><ul><li>ჩანართის სახელი "ზოგადი" გადაკეთდა "პარამეტრებად", რათა უკეთ აღიქმულიყო თითოეული ჩანართის ფუნქცია;</li><li>სეგმენტის ზომის ფილტრი აღარ აშორებს ხატულას, არამედ ახლა იყენებს იმავე გამჭვირვალობას, როგორც სიჩქარის ფილტრში.</li></ul>`,
                    close: 'დახურვა',
                    viewScript: 'იხილეთ სქრიპტი',
                    viewChangelog: 'შეცდომების ჩანაწერის ნახვა',
                    reset: 'გადატვირთვა',
                    snackbar: {
                        errorFiltering: 'შეცდომა ფილტრის გამოყენებისას. მინიმალური სიჩქარე არ შეიძლება იყოს მაქსიმალურ სიჩქარეზე მეტი.',
                    },
                    title: {
                        settings: 'პარამეტრები',
                        roadTypes: 'გზების ტიპები',
                        filter: 'ფილტრი',
                        appearence: 'გამოცხადება',
                        showOnRoadType: 'აჩვენე შემდეგი გზის ტიპებზე:',
                        seeOnDiscuss: 'იხილეთ სკრიპტის თემა Discuss-ზე',
                        seeOnDiscord: 'იხილეთ სკრიპტის არხი Waze Script-ის Discord სერვერზე',
                        scriptDevBy: 'ნახეთ ჩემი რედაქტორის მომხმარებლის პროფილი',
                    },
                    label: {
                        displaySpeeds: 'სისწრაფის ჩვენება',
                        showNoSpeed: 'სიჩქარის გარეშე სეგმენტებზე ჩვენება',
                        showOnRoundabout: 'საკვანძო გზებზე ჩვენება',
                        showOnUnpaved: 'აუცილებელი გზების გამოფენა',
                        maxZoom: 'ზუმის მაქსიმალური დონე:',
                        toggleLayerShortcut: 'სისწრაფის ფენის გადართვის ცხელი კლავიშები',
                        zIndex: 'ფენის სიღრმე',
                        hideShorterSegments: 'დამალვა, თუ სეგმენტი არის ამცირებელი:',
                        speedFilter: 'ფილტრი სიჩქარის მიხედვით',
                        min: 'მინიმუმ:',
                        max: 'მაქსიმუმ:',
                        opacity: 'ფილტრის გამჭვირვალობის დონე:',
                    },
                    alertMessage: {
                        zIndex: `თუ გამოიყენეთ ფენის სიღრმე <b>${this.zIndex.warn}</b> მეტი, სისწრაფის ფენა დახურავს ყველა სხვა ფენებს და ვერ შეარჩიეთ ელემენტები ეკრანზე სანამ არ დააბრუნებთ სიღრმე <b>${this.zIndex.warn}</b> ქვემოთ.`,
                        toggleVisibility: `სიჩქარის ფენის ჩართვის/გამორთვისთვის, გთხოვთ გადადით <b>მოგზაურობა > გზები > სიჩქარის ნახვა</b> ან გამოიყენეთ ცხელი კლავიშები <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'ნაგულისხმები',
                        altViennaConv: 'ძველი ვენის კონვენცია',
                        japan: 'იაპონია',
                        northAmerica: 'ჩრდილოეთ ამერიკა',
                        sacu: 'სამხრეთ აფრიკის საბაჟო კავშირი',
                        uk: 'დიდი ბრიტანეთი',
                    },
                    btn: {
                        updateSettingsTab: 'მენიუს განახლება',
                        resetSettings: 'პარამეტრების გადატვირთვა',
                        seeOnDiscuss: 'იხილეთ განხილვაში',
                        seeOnDiscord: 'იხილეთ Discord-ზე',
                        scriptDevBy: 'სკრიპტი გამოიმუშავა ლუან ტავარესმა',
                    }
                };
            
                break;

            case 'ko':
                // Korean
                translations = {
                    name: '속도 표시',
                    zoomOut: '줌 아웃',
                    zoomIn: '줌 인',
                    changelog: '변경 기록',
                    changelogMessage: `<h6>새로운 기능:</h6><ul style="margin-bottom: 10px"><li>"지연된 아이콘 업데이트" 메시지가 제거되고 간단한 아이콘으로 교체되어 스크립트가 여전히 속도 아이콘을 처리 중임을 명확하게 보여줍니다 (업데이트에는 의도적인 지연이 있습니다).</li></ul><h6>수정 및 개선 사항:</h6><ul><li>"일반" 탭의 이름이 "설정"으로 변경되어 각 탭의 기능을 더 잘 이해할 수 있습니다;</li><li>세그먼트 크기 필터는 이제 아이콘을 제거하지 않고 속도 필터와 동일한 불투명도를 적용합니다.</li></ul>`,
                    close: '닫기',
                    viewScript: '스크립트 보기',
                    viewChangelog: '변경 기록 보기',
                    reset: '초기화',
                    snackbar: {
                        errorFiltering: '필터 적용 오류. 최소 속도가 최대 속도보다 클 수 없습니다.',
                    },
                    title: {
                        settings: '설정',
                        roadTypes: '도로 유형',
                        filter: '필터',
                        appearence: '모양',
                        showOnRoadType: '다음 도로 유형에서 표시:',
                        seeOnDiscuss: 'Discuss에서 스크립트 스레드 보기',
                        seeOnDiscord: 'Waze Script Discord 서버에서 스크립트 채널 보기',
                        scriptDevBy: '내 편집기 사용자 프로필 보기',
                    },
                    label: {
                        displaySpeeds: '속도 표시',
                        showNoSpeed: '속도 없는 세그먼트에 표시',
                        showOnRoundabout: '로터리에서 표시',
                        showOnUnpaved: '비포장 도로에서 표시',
                        maxZoom: '최대 줌:',
                        toggleLayerShortcut: '속도 레이어 전환 단축키',
                        zIndex: '레이어 깊이',
                        hideShorterSegments: '이보다 짧은 세그먼트 숨기기',
                        speedFilter: '속도 필터',
                        min: '최소:',
                        max: '최대:',
                        opacity: '필터 불투명도 수준:',
                    },
                    alertMessage: {
                        zIndex: `레이어 깊이가 <b>${this.zIndex.warn}</b>를 초과하면 속도 레이어가 다른 레이어 위에 표시되어 화면에서 다른 요소를 선택할 수 없게 됩니다. 다시 <b>${this.zIndex.warn}</b> 이하로 낮추기 전까지 선택할 수 없습니다.`,
                        toggleVisibility: `속도 레이어의 가시성을 활성화/비활성화하려면 <b>맵 레이어 > 도로 > 속도 표시</b>로 이동하거나 단축키 <b>ALT</b> + <b>SHIFT</b> + <b>S</b>를 사용하세요.`
                    },
                    theme: {
                        default: '기본',
                        altViennaConv: '대체 비엔나 협약',
                        japan: '일본',
                        northAmerica: '북아메리카',
                        sacu: '남부 아프리카 세관 동맹',
                        uk: '영국',
                    },
                    btn: {
                        updateSettingsTab: '메뉴 업데이트',
                        resetSettings: '설정 초기화',
                        seeOnDiscuss: '토론에서 보기',
                        seeOnDiscord: 'Discord에서 보기',
                        scriptDevBy: '스크립트는 Luan Tavares에 의해 개발되었습니다',
                    }
                };
            
                break;

            case 'lt':
                // Lithuanian
                translations = {
                    name: 'Greitis Rodymas',
                    zoomOut: 'Toliau',
                    zoomIn: 'Arčiau',
                    changelog: 'Pakeitimų Žurnalas',
                    changelogMessage: `<h6>Nauja:</h6><ul style="margin-bottom: 10px"><li>Pranešimas „atnaujinami piktogramos su uždelsimu“ pagaliau pašalintas ir pakeistas paprasta piktograma, aiškiai rodanti, kada scenarijus vis dar apdoroja greičio piktogramas (atnaujinimas vyksta su tyčiniu uždelsimu).</li></ul><h6>Taisymai ir patobulinimai:</h6><ul><li>Skirtuko "Bendra" pavadinimas pakeistas į "Nustatymai", kad būtų aiškiau suprantama kiekvieno skirtuko paskirtis;</li><li>Segmento dydžio filtras dabar nepanaikina piktogramos, bet taiko tokią pat skaidrumo reikšmę kaip ir greičio filtras.</li></ul>`,
                    close: 'Uždaryti',
                    viewScript: 'Peržiūrėti scenarijų',
                    viewChangelog: 'Peržiūrėti pakeitimų žurnalą',
                    reset: 'Atstatyti',
                    snackbar: {
                        errorFiltering: 'Klaida taikant filtrą. Mažiausias greitis negali būti didesnis nei didžiausias.',
                    },
                    title: {
                        settings: 'Nustatymai',
                        roadTypes: 'Kelio tipai',
                        filter: 'Filtras',
                        appearence: 'Išvaizda',
                        showOnRoadType: 'Rodyti šiuose kelio tipuose:',
                        seeOnDiscuss: 'Žiūrėkite scenarijaus temą Discuss',
                        seeOnDiscord: 'Žiūrėkite scenarijaus kanalą Waze Script Discord serveryje',
                        scriptDevBy: 'Peržiūrėkite mano redaktoriaus naudotojo profilį',
                    },
                    label: {
                        displaySpeeds: 'Rodyti greičius',
                        showNoSpeed: 'Rodyti segmentuose be greičio',
                        showOnRoundabout: 'Rodyti žiedinėse sankryžose',
                        showOnUnpaved: 'Rodyti neasfaltuotose keliose',
                        maxZoom: 'Rodyti iki židinio lygio:',
                        toggleLayerShortcut: 'Perjungti greičio sluoksnį',
                        zIndex: 'Sluoksnio gylis',
                        hideShorterSegments: 'Slėpti trumpesnius nei',
                        speedFilter: 'Filtruoti pagal greitį',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Filtravimo nepralaidumo lygis:',
                    },
                    alertMessage: {
                        zIndex: `Naudojant sluoksnio gylį, kuris yra didesnis nei <b>${this.zIndex.warn}</b>, greičio sluoksnis uždengs kitus sluoksnius ir negalėsite pasirinkti jokio elemento ekrane, kol vėl nepasieksite gylio žemiau <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Norėdami įjungti/išjungti greičio sluoksnio matomumą, eikite į <b>Žemėlapių sluoksniai > Keliai > Rodyti greičius</b> arba naudokite klaviatūros kombinaciją <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Numatytas',
                        altViennaConv: 'Alternatyvi Vienos konvencija',
                        japan: 'Japonija',
                        northAmerica: 'Šiaurės Amerika',
                        sacu: 'Pietų Afrikos muitų sąjunga',
                        uk: 'Jungtinė Karalystė',
                    },
                    btn: {
                        updateSettingsTab: 'Atnaujinti meniu',
                        resetSettings: 'Atstatyti nustatymus',
                        seeOnDiscuss: 'Žiūrėti diskusijoje',
                        seeOnDiscord: 'Žiūrėti Discorde',
                        scriptDevBy: 'Skriptą sukūrė Luan Tavares',
                    }
                };
            
                break;
                
            case 'lv':
                // Latvian
                translations = {
                    name: 'Ātruma Rādīšana',
                    zoomOut: 'Tālāk',
                    zoomIn: 'Tuvāk',
                    changelog: 'Izmaiņu Žurnāls',
                    changelogMessage: `<h6>Jauns:</h6><ul style="margin-bottom: 10px"><li>Ziņojums "atjaunina ikonas ar aizturi" beidzot tika noņemts un aizstāts ar vienkāršu ikonu, kas skaidri norāda, kad skripts vēl apstrādā ātruma ikonas (atcerieties, ka atjaunināšanai ir tīša aizture).</li></ul><h6>Labošanas un uzlabojumi:</h6><ul><li>Cilne "Vispārīgi" tika pārdēvēta par "Iestatījumi", lai labāk saprastu katras cilnes funkciju;</li><li>Sadaļas izmēra filtrs vairs neizņem ikonu, bet piemēro tādu pašu caurspīdīgumu kā ātruma filtrā.</li></ul>`,
                    close: 'Aizvērt',
                    viewScript: 'Skatīt Skriptu',
                    viewChangelog: 'Skatīt Izmaiņu Žurnālu',
                    reset: 'Atiestatīt',
                    snackbar: {
                        errorFiltering: 'Kļūda, lietojot filtru. Minimālais ātrums nevar būt lielāks par maksimālo.',
                    },
                    title: {
                        settings: 'Iestatījumi',
                        roadTypes: 'Ceļa tipi',
                        filter: 'Filtrs',
                        appearence: 'Izskats',
                        showOnRoadType: 'Rādīt šādos ceļa tipos:',
                        seeOnDiscuss: 'Skatīt skripta pavedienu Discuss',
                        seeOnDiscord: 'Skatīt skripta kanālu Waze Script Discord serverī',
                        scriptDevBy: 'Skatīt manu redaktora lietotāja profilu',
                    },
                    label: {
                        displaySpeeds: 'Rādīt ātrumus',
                        showNoSpeed: 'Rādīt segmentos bez ātruma',
                        showOnRoundabout: 'Rādīt aplī',
                        showOnUnpaved: 'Rādīt neasfaltētajos ceļos',
                        maxZoom: 'Rādīt līdz palielināšanas līmenim:',
                        toggleLayerShortcut: 'Pārslēgt ātruma slāni',
                        zIndex: 'Slāņa dziļums',
                        hideShorterSegments: 'Paslēpt īsākus nekā',
                        speedFilter: 'Filtrēt pēc ātruma',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Filtra līmeņa caurspīdīgums:',
                    },
                    alertMessage: {
                        zIndex: `Izmantojot slāņa dziļumu, kas pārsniedz <b>${this.zIndex.warn}</b>, ātruma slānis pārklās citus slāņus, un nebūs iespējams izvēlēties nevienu elementu ekrānā līdz tā atkārtotai pazemināšanai zem <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Lai aktivizētu/deaktivizētu ātruma slāņa redzamību, vienkārši dodieties uz <b>Kartes Slāņi > Ceļi > Rādīt ātrumus</b> vai izmantojiet īsinājumtaustiņu <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Noklus',
                        altViennaConv: 'Alternatīvā Vīnes konvencija',
                        japan: 'Japāna',
                        northAmerica: 'Ziemeļamerika',
                        sacu: 'Dienvidu Āfrikas muitas savienība',
                        uk: 'Apvienotā Karaliste',
                    },
                    btn: {
                        updateSettingsTab: 'Atjaunināt Meni',
                        resetSettings: 'Atiestatīt Iestatījumus',
                        seeOnDiscuss: 'Skatīt diskusijā',
                        seeOnDiscord: 'Skatīt Discordā',
                        scriptDevBy: 'Skriptu izstrādājis Luan Tavares',
                    }
                };
            
                break;
            
            case 'ms':
                // Malay
                translations = {
                    name: 'Papar Kelajuan',
                    zoomOut: 'Jauhkan',
                    zoomIn: 'Dekatkan',
                    changelog: 'Log Perubahan',
                    changelogMessage: `<h6>Ciri Baharu:</h6><ul style="margin-bottom: 10px"><li>Mesej "mengemas kini ikon dengan kelewatan" akhirnya dialih keluar dan digantikan dengan ikon ringkas, menjelaskan bila skrip masih memproses ikon kelajuan (ingat, terdapat kelewatan yang disengajakan).</li></ul><h6>Pembaikan dan penambahbaikan:</h6><ul><li>Tab "umum" dinamakan semula kepada "tetapan" untuk kejelasan yang lebih baik tentang fungsi setiap tab;</li><li>Penapis saiz segmen kini tidak lagi membuang ikon, tetapi menggunakan ketelusan yang sama seperti penapis kelajuan.</li></ul>`,
                    close: 'Tutup',
                    viewScript: 'Lihat Skrip',
                    viewChangelog: 'Lihat Log Perubahan',
                    reset: 'Tetapkan Semula',
                    snackbar: {
                        errorFiltering: 'Ralat semasa menggunakan penapis. Kelajuan minimum tidak boleh melebihi maksimum.',
                    },
                    title: {
                        settings: 'Tetapan',
                        roadTypes: 'Jenis Jalan',
                        filter: 'Penapis',
                        appearence: 'Penampilan',
                        showOnRoadType: 'Tunjukkan pada jenis jalan berikut:',
                        seeOnDiscuss: 'Lihat thread skrip di Discuss',
                        seeOnDiscord: 'Lihat saluran skrip di pelayan Discord Waze Script',
                        scriptDevBy: 'Lihat profil pengguna editor saya',
                    },
                    label: {
                        displaySpeeds: 'Papar kelajuan',
                        showNoSpeed: 'Papar pada segmen tiada kelajuan',
                        showOnRoundabout: 'Papar pada bulatan',
                        showOnUnpaved: 'Papar pada jalan tidak berturap',
                        maxZoom: 'Paparkan sehingga tahap zum:',
                        toggleLayerShortcut: 'Tukar lapisan paparan kelajuan',
                        zIndex: 'Kedalaman lapisan',
                        hideShorterSegments: 'Sembunyikan pada segmen lebih pendek daripada',
                        speedFilter: 'Tapis mengikut kelajuan',
                        min: 'Min.:',
                        max: 'Maks.:',
                        opacity: 'Tahap kelegapan penapis:',
                    },
                    alertMessage: {
                        zIndex: `Apabila menggunakan kedalaman lapisan lebih besar daripada <b>${this.zIndex.warn}</b>, lapisan kelajuan akan menutupi lapisan lain dan tiada elemen boleh dipilih pada skrin sehingga ia diturunkan di bawah <b>${this.zIndex.warn}</b> semula.`,
                        toggleVisibility: `Untuk menghidupkan/mematikan kebolehtampilan lapisan kelajuan, pergi ke <b>Lapisan Peta > Jalan > Papar kelajuan</b> atau gunakan pintasan <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Lalai',
                        altViennaConv: 'Konvensi Vienna Alternatif',
                        japan: 'Jepun',
                        northAmerica: 'Amerika Utara',
                        sacu: 'Kesatuan Kastam Afrika Selatan',
                        uk: 'United Kingdom',
                    },
                    btn: {
                        updateSettingsTab: 'Kemas kini Menu',
                        resetSettings: 'Tetapkan Semula Tetapan',
                        seeOnDiscuss: 'Lihat dalam perbincangan',
                        seeOnDiscord: 'Lihat di Discord',
                        scriptDevBy: 'Skrip dibangunkan oleh Luan Tavares',
                    }
                };
            
                break;
            
            case 'nl':
                // Dutch
                translations = {
                    name: 'Snelheidsweergave',
                    zoomOut: 'Uitzoomen',
                    zoomIn: 'Inzoomen',
                    changelog: 'Wijzigingslog',
                    changelogMessage: `<h6>Nieuw:</h6><ul style="margin-bottom: 10px"><li>Het bericht "iconen bijwerken met vertraging" is eindelijk verwijderd en vervangen door een eenvoudig pictogram, dat duidelijk aangeeft wanneer het script nog steeds snelheidsiconen verwerkt (onthoud dat er een opzettelijke vertraging is).</li></ul><h6>Oplossingen en verbeteringen:</h6><ul><li>De tab "algemeen" is hernoemd naar "instellingen" voor een betere duidelijkheid over de functie van elke tab;</li><li>Het segmentgroottefilter verwijdert het pictogram niet meer, maar past nu dezelfde transparantie toe als het snelheidsfilter.</li></ul>`,
                    close: 'Sluiten',
                    viewScript: 'Bekijk script',
                    viewChangelog: 'Bekijk wijzigingslog',
                    reset: 'Reset',
                    snackbar: {
                        errorFiltering: 'Fout bij het toepassen van het filter. De minimale snelheid kan niet groter zijn dan de maximale.',
                    },
                    title: {
                        settings: 'Instellingen',
                        roadTypes: 'Wegtypen',
                        filter: 'Filter',
                        appearence: 'Uiterlijk',
                        showOnRoadType: 'Weergeven op de volgende wegtypen:',
                        seeOnDiscuss: 'Bekijk de thread van het script op Discuss',
                        seeOnDiscord: 'Bekijk het kanaal van het script op de Waze Script Discord-server',
                        scriptDevBy: 'Bekijk mijn editor gebruikersprofiel',
                    },
                    label: {
                        displaySpeeds: 'Toon snelheden',
                        showNoSpeed: 'Weergeven op segmenten zonder snelheid',
                        showOnRoundabout: 'Weergeven op rotonde',
                        showOnUnpaved: 'Weergeven op onverharde wegen',
                        maxZoom: 'Weergeven tot zoomniveau:',
                        toggleLayerShortcut: 'Wissel snel de snelheidsweergavelayer',
                        zIndex: 'Laagdiepte',
                        hideShorterSegments: 'Verberg segmenten korter dan',
                        speedFilter: 'Filter op snelheid',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Filterdekking niveau:',
                    },
                    alertMessage: {
                        zIndex: `Wanneer je een laagdiepte groter dan <b>${this.zIndex.warn}</b> gebruikt, zal de snelheidslaag de andere lagen bedekken en kun je geen elementen meer selecteren op het scherm totdat je de laagdiepte weer onder <b>${this.zIndex.warn}</b> brengt.`,
                        toggleVisibility: `Om de snelheidsweergavelayer in of uit te schakelen, ga naar <b>Kaartlagen > Wegen > Weergave van snelheden</b> of gebruik de sneltoets <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Standaard',
                        altViennaConv: 'Alternatieve Wenen Conventie',
                        japan: 'Japan',
                        northAmerica: 'Noord-Amerika',
                        sacu: 'Zuidelijke Afrikaanse Douane Unie',
                        uk: 'Verenigd Koninkrijk',
                    },
                    btn: {
                        updateSettingsTab: 'Werk menu bij',
                        resetSettings: 'Herstel instellingen',
                        seeOnDiscuss: 'Zie op Discussie',
                        seeOnDiscord: 'Zie op Discord',
                        scriptDevBy: 'Script ontwikkeld door Luan Tavares',
                    }
                };
            
                break;

            case 'no':
                // Norwegian
                translations = {
                    name: 'Hastighetsvisning',
                    zoomOut: 'Zoom ut',
                    zoomIn: 'Zoom inn',
                    changelog: 'Endringslogg',
                    changelogMessage: `<h6>Nytt:</h6><ul style="margin-bottom: 10px"><li>Meldingen "oppdaterer ikoner med forsinkelse" er endelig fjernet og erstattet med et enkelt ikon som tydelig viser når skriptet fortsatt behandler fartsikonene (merk at det er en tilsiktet forsinkelse).</li></ul><h6>Fikser og forbedringer:</h6><ul><li>Fanen "Generelt" har fått nytt navn til "Innstillinger" for å klargjøre funksjonen til hver fane;</li><li>Filteret for segmentstørrelse fjerner ikke lenger ikonet, men bruker samme gjennomsiktighet som fartsfilteret.</li></ul>`,
                    close: 'Lukk',
                    viewScript: 'Se skript',
                    viewChangelog: 'Vis endringslogg',
                    reset: 'Tilbakestill',
                    snackbar: {
                        errorFiltering: 'Feil ved bruk av filter. Minimumshastighet kan ikke være større enn maksimum.',
                    },
                    title: {
                        settings: 'Innstillinger',
                        roadTypes: 'Vei typer',
                        filter: 'Filter',
                        appearence: 'Utseende',
                        showOnRoadType: 'Vis på følgende veit typer:',
                        seeOnDiscuss: 'Se tråden for skriptet på Discuss',
                        seeOnDiscord: 'Se skriptets kanal på Waze Script Discord-serveren',
                        scriptDevBy: 'Se min editor brukerprofil',
                    },
                    label: {
                        displaySpeeds: 'Vis hastigheter',
                        showNoSpeed: 'Vis på segmenter uten hastighet',
                        showOnRoundabout: 'Vis på rundkjøring',
                        showOnUnpaved: 'Vis på uasfalterte veier',
                        maxZoom: 'Vis til zoomnivå:',
                        toggleLayerShortcut: 'Bytt visningslag for hastighet',
                        zIndex: 'Lagdybde',
                        hideShorterSegments: 'Skjul segmenter kortere enn',
                        speedFilter: 'Filtrer etter hastighet',
                        min: 'Min.:',
                        max: 'Maks.:',
                        opacity: 'Filter gjennomsiktighetsnivå:',
                    },
                    alertMessage: {
                        zIndex: `Når du bruker en lagdybde større enn <b>${this.zIndex.warn}</b>, vil hastighetslaget overlappe de andre, og du vil ikke kunne velge noen elementer på skjermen før du senker lagdybden til under <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `For å aktivere/deaktivere visningslaget for hastigheter, kan du gå til <b>Kartlag > Veier > Vis hastigheter</b> eller bruke snarveien <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Standard',
                        altViennaConv: 'Alternativ Wien-konvensjon',
                        japan: 'Japan',
                        northAmerica: 'Nord-Amerika',
                        sacu: 'Sydlige afrikanske tollunionen',
                        uk: 'Storbritannia',
                    },
                    btn: {
                        updateSettingsTab: 'Oppdater meny',
                        resetSettings: 'Tilbakestill innstillinger',
                        seeOnDiscuss: 'Se på diskusjonen',
                        seeOnDiscord: 'Se på Discord',
                        scriptDevBy: 'Script utviklet av Luan Tavares',
                    }
                };
            
                break;

            case 'pl':
                // Polish
                translations = {
                    name: 'Wyświetlanie prędkości',
                    zoomOut: 'Oddal',
                    zoomIn: 'Przybliż',
                    changelog: 'Dziennik zmian',
                    changelogMessage: `<h6>Nowości:</h6><ul style="margin-bottom: 10px"><li>Wiadomość „aktualizowanie ikon z opóźnieniem” została usunięta i zastąpiona prostą ikoną, pokazującą, kiedy skrypt nadal przetwarza ikony prędkości (pamiętaj, że aktualizacja odbywa się z zamierzonym opóźnieniem).</li></ul><h6>Poprawki i ulepszenia:</h6><ul><li>Nazwa zakładki „Ogólne” została zmieniona na „Ustawienia” dla lepszego zrozumienia funkcji zakładek;</li><li>Filtr długości segmentu nie usuwa już ikony, lecz stosuje taką samą przezroczystość jak filtr prędkości.</li></ul>`,
                    close: 'Zamknij',
                    viewScript: 'Zobacz skrypt',
                    viewChangelog: 'Zobacz dziennik zmian',
                    reset: 'Resetuj',
                    snackbar: {
                        errorFiltering: 'Błąd podczas stosowania filtra. Minimalna prędkość nie może być większa niż maksymalna.',
                    },
                    title: {
                        settings: 'Ustawienia',
                        roadTypes: 'Rodzaje dróg',
                        filter: 'Filtr',
                        appearence: 'Wygląd',
                        showOnRoadType: 'Pokaż na następujących rodzajach dróg:',
                        seeOnDiscuss: 'Zobacz wątek skryptu na Discuss',
                        seeOnDiscord: 'Zobacz kanał skryptu na serwerze Discord Waze Script',
                        scriptDevBy: 'Zobacz mój profil użytkownika edytora',
                    },
                    label: {
                        displaySpeeds: 'Pokaż prędkości',
                        showNoSpeed: 'Pokaż na segmentach bez prędkości',
                        showOnRoundabout: 'Pokaż na rondzie',
                        showOnUnpaved: 'Pokaż na drodze nieutwardzonej',
                        maxZoom: 'Renderuj do zoomu:',
                        toggleLayerShortcut: 'Przełącz warstwę wyświetlania prędkości',
                        zIndex: 'Głębokość warstwy',
                        hideShorterSegments: 'Ukryj na segmentach krótszych niż',
                        speedFilter: 'Filtruj według prędkości',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Poziom przezroczystości filtra:',
                    },
                    alertMessage: {
                        zIndex: `Używając głębokości warstwy większej niż <b>${this.zIndex.warn}</b>, warstwa prędkości będzie nakładać się na inne, uniemożliwiając wybór jakichkolwiek elementów na ekranie, dopóki nie zmniejszysz jej poniżej <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Aby włączyć/wyłączyć warstwę wyświetlania prędkości, przejdź do <b>Warstwy mapy > Drogi > Wyświetl prędkości</b> lub użyj skrótu <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Domyślny',
                        altViennaConv: 'Alternatywna konwencja wiedeńska',
                        japan: 'Japonia',
                        northAmerica: 'Ameryka Północna',
                        sacu: 'Południowoafrykańska Unia Celna',
                        uk: 'Wielka Brytania',
                    },
                    btn: {
                        updateSettingsTab: 'Aktualizuj menu',
                        resetSettings: 'Resetuj ustawienia',
                        seeOnDiscuss: 'Zobacz na Dyskusji',
                        seeOnDiscord: 'Zobacz na Discordzie',
                        scriptDevBy: 'Skrypt opracowany przez Luana Tavaresa',
                    }
                };
                
                break;

            case 'pt-BR':
                // Portuguese (Brazil)
                translations = {
                    name: 'Exibição de Velocidade',
                    zoomOut: 'Mais afastado',
                    zoomIn: 'Mais aproximado',
                    changelog: 'Registro de alterações',
                    changelogMessage: `<h6>Novidades:</h6><ul style="margin-bottom: 10px"><li>Enfim removido a mensagem de "atualizando os ícones com delay" por um ícone simplório, ficando claro quando o script ainda está processando os ícones de velocidades (lembre-se sempre que existe um delay proposital na atualização dos ícones).</li></ul><h6>Correções e melhorias:</h6><ul><li>Alterado o nome da aba "geral" para "ajustes" para uma melhor clareza no entendimento da função de cada aba do script;</li><li>Modificado o funcionamento da filtragem por tamanho de segmento, onde antes removia o ícone da visualização, agora ele apenas aplica a opcaidade igual a aplicada no filtro de velocidade.</li></ul>`,
                    close: 'Fechar',
                    viewScript: 'Ver script',
                    viewChangelog: 'Visualizar o registro de alterações',
                    reset: 'Redefinir',
                    snackbar: {
                        errorFiltering: 'Erro ao aplicar o filtro. A velocidade mínima não pode ser maior que a máxima.',
                    },
                    title: {
                        settings: 'Ajustes',
                        roadTypes: 'Tipos de via',
                        filter: 'Filtro',
                        appearence: 'Aparência',
                        showOnRoadType: 'Exibir nos seguintes tipos de via:',
                        seeOnDiscuss: 'Veja o tópico do script no Discuss',
                        seeOnDiscord: 'Veja o canal do script no servidor Discord do Waze Script',
                        scriptDevBy: 'Veja o meu perfil de usuário do editor',
                    },
                    label: {
                        displaySpeeds: 'Exibir velocidades',
                        showNoSpeed: 'Exibir em segmento sem velocidade',
                        showOnRoundabout: 'Exibir em rotatória',
                        showOnUnpaved: 'Exibir em via não pavimentada',
                        maxZoom: 'Renderizar até o zoom:',
                        toggleLayerShortcut: 'Alternar a camada de exibição da velocidade',
                        zIndex: 'Profundidade da camada',
                        hideShorterSegments: 'Ocultar em segmentos menores que',
                        speedFilter: 'Filtrar por velocidade',
                        min: 'Mín.:',
                        max: 'Máx.:',
                        opacity: 'Nível de opacidade do filtro:',
                    },
                    alertMessage: {
                        zIndex: `Ao utilizar uma profundidade de camada maior que <b>${this.zIndex.warn}</b>, a camada de velocidade se sobreponhará as outras e não será possível selecionar nenhum elemento na tela até baixa-la novamente abaixo de <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Para ativar/desativar a camada de exibição de velocidades, basta ir até <b>Camadas do Mapa > Vias > Exibir velocidades</b> ou utilizar o atalho <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Padrão',
                        altViennaConv: 'Convenção de Viena Alternativa',
                        japan: 'Japão',
                        northAmerica: 'América do Norte',
                        sacu: 'União Aduaneira da África Austral',
                        uk: 'Reino Unido',
                    },
                    btn: {
                        updateSettingsTab: 'Atualizar menu',
                        resetSettings: 'Redefinir configurações',
                        seeOnDiscuss: 'Ver no Discuss',
                        seeOnDiscord: 'Ver no Discord',
                        scriptDevBy: 'Script desenvolvido por Luan Tavares',
                    }
                };

                break;

            case 'pt-PT':
                // Portuguese (Portugal)
                translations = {
                    name: 'Exibição de Velocidade',
                    zoomOut: 'Mais afastado',
                    zoomIn: 'Mais aproximado',
                    changelog: 'Registo de alterações',
                    changelogMessage: `<h6>Novidades:</h6><ul style="margin-bottom: 10px"><li>A mensagem "a atualizar ícones com atraso" foi finalmente removida e substituída por um ícone simples, indicando quando o script ainda está a processar os ícones de velocidade (existe um atraso propositado na atualização).</li></ul><h6>Correções e melhorias:</h6><ul><li>O separador "Geral" foi renomeado para "Ajustes", para melhor compreensão da função de cada separador;</li><li>O filtro de tamanho de segmento agora aplica opacidade em vez de remover o ícone, tal como o filtro de velocidade.</li></ul>`,
                    close: 'Fechar',
                    viewScript: 'Ver script',
                    viewChangelog: 'Visualizar o registo de alterações',
                    reset: 'Redefinir',
                    snackbar: {
                        errorFiltering: 'Erro ao aplicar o filtro. A velocidade mínima não pode ser superior à máxima.',
                    },
                    title: {
                        settings: 'Ajustes',
                        roadTypes: 'Tipos de estrada',
                        filter: 'Filtro',
                        appearence: 'Aparência',
                        showOnRoadType: 'Mostrar nos seguintes tipos de estrada:',
                        seeOnDiscuss: 'Veja o tópico do script no Discuss',
                        seeOnDiscord: 'Veja o canal do script no servidor Discord do Waze Script',
                        scriptDevBy: 'Veja o meu perfil de utilizador do editor',
                    },
                    label: {
                        displaySpeeds: 'Exibir velocidades',
                        showNoSpeed: 'Exibir em segmento sem velocidade',
                        showOnRoundabout: 'Exibir em rotunda',
                        showOnUnpaved: 'Exibir em via não pavimentada',
                        maxZoom: 'Renderizar até o zoom:',
                        toggleLayerShortcut: 'Alternar a camada de exibição da velocidade',
                        zIndex: 'Profundidade da camada',
                        hideShorterSegments: 'Ocultar em segmentos menores que',
                        speedFilter: 'Filtrar por velocidade',
                        min: 'Mín.:',
                        max: 'Máx.:',
                        opacity: 'Nível de opacidade do filtro:',
                    },
                    alertMessage: {
                        zIndex: `Ao utilizar uma profundidade de camada maior que <b>${this.zIndex.warn}</b>, a camada de velocidade se sobreporá a outras e não será possível selecionar nenhum elemento na tela até que a profundidade seja reduzida abaixo de <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Para ativar/desativar a camada de exibição de velocidades, basta ir a <b>Camadas do Mapa > Vias > Exibir velocidades</b> ou utilizar o atalho <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Padrão',
                        altViennaConv: 'Convenção de Viena Alternativa',
                        japan: 'Japão',
                        northAmerica: 'América do Norte',
                        sacu: 'União Aduaneira da África Austral',
                        uk: 'Reino Unido',
                    },
                    btn: {
                        updateSettingsTab: 'Atualizar menu',
                        resetSettings: 'Redefinir configurações',
                        seeOnDiscuss: 'Ver no Discuss',
                        seeOnDiscord: 'Ver no Discord',
                        scriptDevBy: 'Script desenvolvido por Luan Tavares',
                    }
                };
            
                break;

            case 'ro':
                // Romanian
                translations = {
                    name: 'Afișarea vitezei',
                    zoomOut: 'Mai departe',
                    zoomIn: 'Mai aproape',
                    changelog: 'Jurnalul modificărilor',
                    changelogMessage: `<h6>Noutăți:</h6><ul style="margin-bottom: 10px"><li>Mesajul „actualizare pictograme cu întârziere” a fost în sfârșit eliminat și înlocuit cu o pictogramă simplă, clarificând că scriptul încă procesează pictogramele de viteză (rețineți că există o întârziere intenționată).</li></ul><h6>Corecturi și îmbunătățiri:</h6><ul><li>Fila „General” a fost redenumită „Setări” pentru o mai bună înțelegere a funcției fiecărei file;</li><li>Filtrul după dimensiunea segmentului nu mai elimină pictograma, ci aplică aceeași opacitate ca filtrul de viteză.</li></ul>`,
                    close: 'Închide',
                    viewScript: 'Vezi scriptul',
                    viewChangelog: 'Vizualizează jurnalul modificărilor',
                    reset: 'Resetează',
                    snackbar: {
                        errorFiltering: 'Eroare la aplicarea filtrului. Viteza minimă nu poate fi mai mare decât viteza maximă.',
                    },
                    title: {
                        settings: 'Setări',
                        roadTypes: 'Tipuri de drumuri',
                        filter: 'Filtru',
                        appearence: 'Aparență',
                        showOnRoadType: 'Afișați pe următoarele tipuri de drumuri:',
                        seeOnDiscuss: 'Vezi firul scriptului pe Discuss',
                        seeOnDiscord: 'Vezi canalul scriptului pe serverul Discord Waze Script',
                        scriptDevBy: 'Vizualizează profilul meu de utilizator editor',
                    },
                    label: {
                        displaySpeeds: 'Afișează vitezele',
                        showNoSpeed: 'Afișează pe segmente fără viteză',
                        showOnRoundabout: 'Afișează pe sens giratoriu',
                        showOnUnpaved: 'Afișează pe drumuri neasfaltate',
                        maxZoom: 'Renderizează până la zoom:',
                        toggleLayerShortcut: 'Comută stratul de afișare a vitezei',
                        zIndex: 'Adâncimea stratului',
                        hideShorterSegments: 'Ascunde segmentele mai mici de',
                        speedFilter: 'Filtrează după viteză',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Nivelul de opacitate al filtrului:',
                    },
                    alertMessage: {
                        zIndex: `Atunci când folosești o adâncime a stratului mai mare de <b>${this.zIndex.warn}</b>, stratul de viteză se va suprapune altor straturi și nu vei putea selecta niciun element pe ecran până când nu scazi adâncimea sub <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Pentru a activa/dezactiva stratul de afișare a vitezelor, mergi la <b>Straturi Mapa > Drumuri > Afișează viteze</b> sau folosește scurtătura <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Implicit',
                        altViennaConv: 'Alternativă Convenția de la Viena',
                        japan: 'Japonia',
                        northAmerica: 'America de Nord',
                        sacu: 'Uniunea Vamală a Africii de Sud',
                        uk: 'Regatul Unit',
                    },
                    btn: {
                        updateSettingsTab: 'Actualizează meniul',
                        resetSettings: 'Resetare setări',
                        seeOnDiscuss: 'Vezi în discuție',
                        seeOnDiscord: 'Vezi pe Discord',
                        scriptDevBy: 'Script dezvoltat de Luan Tavares',
                    }
                };
            
                break;

            case 'ru':
                // Russian
                translations = {
                    name: 'Отображение скорости',
                    zoomOut: 'Удалить',
                    zoomIn: 'Приблизить',
                    changelog: 'Журнал изменений',
                    changelogMessage: `<h6>Новое:</h6><ul style="margin-bottom: 10px"><li>Сообщение «обновление иконок с задержкой» наконец удалено и заменено на простой значок, указывающий, что скрипт всё ещё обрабатывает значки скорости (обратите внимание, что обновление происходит с намеренной задержкой).</li></ul><h6>Исправления и улучшения:</h6><ul><li>Вкладка «Общие» переименована в «Настройки» для лучшего понимания функций вкладок;</li><li>Фильтр по длине сегмента теперь не удаляет иконку, а применяет ту же прозрачность, что и фильтр скорости.</li></ul>`,
                    close: 'Закрыть',
                    viewScript: 'Посмотреть скрипт',
                    viewChangelog: 'Просмотреть журнал изменений',
                    reset: 'Сбросить',
                    snackbar: {
                        errorFiltering: 'Ошибка применения фильтра. Минимальная скорость не может быть больше максимальной.',
                    },
                    title: {
                        settings: 'Настройки',
                        roadTypes: 'Типы дорог',
                        filter: 'Фильтр',
                        appearence: 'Внешний вид',
                        showOnRoadType: 'Показать на следующих типах дорог:',
                        seeOnDiscuss: 'Посмотреть тему скрипта на Discuss',
                        seeOnDiscord: 'Посмотреть канал скрипта на сервере Discord Waze Script',
                        scriptDevBy: 'Посмотреть мой профиль пользователя редактора',
                    },
                    label: {
                        displaySpeeds: 'Отображать скорости',
                        showNoSpeed: 'Отображать на сегментах без скорости',
                        showOnRoundabout: 'Отображать на круговых развязках',
                        showOnUnpaved: 'Отображать на неасфальтированных дорогах',
                        maxZoom: 'Отображать до уровня зума:',
                        toggleLayerShortcut: 'Переключить слой отображения скорости',
                        zIndex: 'Глубина слоя',
                        hideShorterSegments: 'Скрывать на сегментах короче чем',
                        speedFilter: 'Фильтровать по скорости',
                        min: 'Мин.:',
                        max: 'Макс.:',
                        opacity: 'Уровень непрозрачности фильтра:',
                    },
                    alertMessage: {
                        zIndex: `При использовании глубины слоя больше <b>${this.zIndex.warn}</b> слой скорости будет перекрывать другие и вы не сможете выбрать элементы на экране, пока не уменьшите глубину слоя ниже <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Для активации/деактивации слоя отображения скорости перейдите в <b>Слои карты > Дороги > Отображать скорости</b> или используйте комбинацию клавиш <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'По умолчанию',
                        altViennaConv: 'Альтернативная Венская конвенция',
                        japan: 'Япония',
                        northAmerica: 'Северная Америка',
                        sacu: 'Южноафриканский таможенный союз',
                        uk: 'Соединенное Королевство',
                    },
                    btn: {
                        updateSettingsTab: 'Обновить меню',
                        resetSettings: 'Сбросить настройки',
                        seeOnDiscuss: 'Смотреть на обсуждении',
                        seeOnDiscord: 'Смотреть в Discord',
                        scriptDevBy: 'Скрипт разработан Луаном Таваресом',
                    }
                };
                
                break;
                
            case 'sk':
                // Slovak
                translations = {
                    name: 'Zobrazenie rýchlosti',
                    zoomOut: 'Ďalej',
                    zoomIn: 'Priblížiť',
                    changelog: 'Záznam zmien',
                    changelogMessage: `<h6>Novinky:</h6><ul style="margin-bottom: 10px"><li>Zpráva „aktualizace ikon se zpožděním“ byla konečně odstraněna a nahrazena jednoduchou ikonou, která ukazuje, že skript stále zpracovává ikony rychlosti (mějte na paměti, že aktualizace má záměrné zpoždění).</li></ul><h6>Opravy a vylepšení:</h6><ul><li>Záložka „Obecné“ byla přejmenována na „Nastavení“ pro lepší porozumění funkcím záložek;</li><li>Filtr velikosti segmentu již neodstraňuje ikonu, ale aplikuje stejnou průhlednost jako filtr rychlosti.</li></ul>`,
                    close: 'Zatvoriť',
                    viewScript: 'Zobraziť skript',
                    viewChangelog: 'Zobraziť záznam zmien',
                    reset: 'Obnoviť',
                    snackbar: {
                        errorFiltering: 'Chyba pri použití filtra. Minimálna rýchlosť nemôže byť väčšia ako maximálna.',
                    },
                    title: {
                        settings: 'Nastavenia',
                        roadTypes: 'Typy ciest',
                        filter: 'Filtrovať',
                        appearence: 'Vzhľad',
                        showOnRoadType: 'Zobraziť na nasledujúcich typoch ciest:',
                        seeOnDiscuss: 'Pozrite si vlákno skriptu na Discuss',
                        seeOnDiscord: 'Pozrite si kanál skriptu na Waze Script Discord serveri',
                        scriptDevBy: 'Zobraziť môj profil používateľa editora',
                    },
                    label: {
                        displaySpeeds: 'Zobraziť rýchlosti',
                        showNoSpeed: 'Zobraziť na segmentoch bez rýchlosti',
                        showOnRoundabout: 'Zobraziť na okruhu',
                        showOnUnpaved: 'Zobraziť na nezpevnených cestách',
                        maxZoom: 'Zobraziť do úrovne priblíženia:',
                        toggleLayerShortcut: 'Prepínať vrstvu zobrazenia rýchlosti',
                        zIndex: 'Hĺbka vrstvy',
                        hideShorterSegments: 'Skryť na segmentoch kratších ako',
                        speedFilter: 'Filtrovať podľa rýchlosti',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Úroveň priezračnosti filtra:',
                    },
                    alertMessage: {
                        zIndex: `Pri použití hĺbky vrstvy väčšej ako <b>${this.zIndex.warn}</b> bude vrstva rýchlosti prekryť ostatné a nebude možné vybrať žiadny prvok na obrazovke, kým ju neznížite pod <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Ak chcete zapnúť/vypnúť vrstvu zobrazenia rýchlosti, prejdite na <b>Mapové vrstvy > Cesty > Zobraziť rýchlosti</b> alebo použite skratku <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Predvolený',
                        altViennaConv: 'Alternatívna Viedenská konvencia',
                        japan: 'Japonsko',
                        northAmerica: 'Severná Amerika',
                        sacu: 'Juhoafrická colná únia',
                        uk: 'Spojené kráľovstvo',
                    },
                    btn: {
                        updateSettingsTab: 'Aktualizovať menu',
                        resetSettings: 'Obnoviť nastavenia',
                        seeOnDiscuss: 'Pozrieť na diskusii',
                        seeOnDiscord: 'Pozrieť na Discord',
                        scriptDevBy: 'Skript vyvinul Luan Tavares',
                    }
                };
            
                break;

            case 'sl':
                // Slovenian
                translations = {
                    name: 'Prikaz hitrosti',
                    zoomOut: 'Povečaj',
                    zoomIn: 'Pomanjšaj',
                    changelog: 'Zgodovina sprememb',
                    changelogMessage: `<h6>Novice:</h6><ul style="margin-bottom: 10px"><li>Sporočilo »posodabljanje ikon z zamikom« je bilo končno odstranjeno in zamenjano s preprosto ikono, ki jasno pokaže, kdaj skripta še vedno obdeluje ikone hitrosti (posodobitev ima nameren zamik).</li></ul><h6>Popravki in izboljšave:</h6><ul><li>Zavihek »Splošno« je preimenovan v »Nastavitve« za boljše razumevanje funkcije zavihkov;</li><li>Filter velikosti segmenta zdaj ne odstrani več ikone, temveč uporabi enako prosojnost kot filter hitrosti.</li></ul>`,
                    close: 'Zapri',
                    viewScript: 'Poglej skript',
                    viewChangelog: 'Poglej zgodovino sprememb',
                    reset: 'Ponastavi',
                    snackbar: {
                        errorFiltering: 'Napaka pri uporabi filtra. Najmanjša hitrost ne more biti večja od največje.',
                    },
                    title: {
                        settings: 'Nastavitve',
                        roadTypes: 'Vrste cest',
                        filter: 'Filtriraj',
                        appearence: 'Izgled',
                        showOnRoadType: 'Prikaži na naslednjih vrstah cest:',
                        seeOnDiscuss: 'Oglejte si temo skripte na Discuss',
                        seeOnDiscord: 'Oglejte si kanal skripte na Waze Script Discord strežniku',
                        scriptDevBy: 'Ogled mojega uporabniškega profila urejevalnika',
                    },
                    label: {
                        displaySpeeds: 'Prikaz hitrosti',
                        showNoSpeed: 'Prikaz na segmentih brez hitrosti',
                        showOnRoundabout: 'Prikaz na krožišču',
                        showOnUnpaved: 'Prikaz na nepovoznih cestah',
                        maxZoom: 'Prikaz do stopnje povečave:',
                        toggleLayerShortcut: 'Preklopi plast prikaza hitrosti',
                        zIndex: 'Globina plasti',
                        hideShorterSegments: 'Skrij na segmentih krajši od',
                        speedFilter: 'Filtriraj po hitrosti',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Raven filtriranja prosojnosti:',
                    },
                    alertMessage: {
                        zIndex: `Pri uporabi globine plasti večje od <b>${this.zIndex.warn}</b> se bo plast hitrosti prekrila z drugimi in ne bo mogoče izbrati nobenega elementa na zaslonu, dokler je ne zmanjšate pod <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Za aktiviranje/izklop plasti prikaza hitrosti, pojdite na <b>Plasti zemljevida > Ceste > Prikaz hitrosti</b> ali uporabite bližnjico <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Privzeto',
                        altViennaConv: 'Alternativna Dunajska konvencija',
                        japan: 'Japonska',
                        northAmerica: 'Severna Amerika',
                        sacu: 'Južnoafriška carinska unija',
                        uk: 'Združeno kraljestvo',
                    },
                    btn: {
                        updateSettingsTab: 'Posodobi meni',
                        resetSettings: 'Ponastavi nastavitve',
                        seeOnDiscuss: 'Oglej si na razpravi',
                        seeOnDiscord: 'Oglej si na Discordu',
                        scriptDevBy: 'Skript je razvijal Luan Tavares',
                    }
                };
            
                break;
            
            case 'sv':
                // Swedish
                translations = {
                    name: 'Hastighetsvisning',
                    zoomOut: 'Zooma ut',
                    zoomIn: 'Zooma in',
                    changelog: 'Ändringslogg',
                    changelogMessage: `<h6>Nytt:</h6><ul style="margin-bottom: 10px"><li>Meddelandet "uppdaterar ikoner med fördröjning" har äntligen tagits bort och ersatts med en enkel ikon som visar när skriptet fortfarande behandlar hastighetsikoner (det finns en avsiktlig fördröjning).</li></ul><h6>Fixar och förbättringar:</h6><ul><li>Fliken "Allmänt" har bytt namn till "Inställningar" för att tydliggöra flikarnas funktion;</li><li>Segmentstorleksfiltret tar inte längre bort ikonen utan tillämpar samma transparens som hastighetsfiltret.</li></ul>`,
                    close: 'Stäng',
                    viewScript: 'Visa skript',
                    viewChangelog: 'Visa ändringslogg',
                    reset: 'Återställ',
                    snackbar: {
                        errorFiltering: 'Fel vid tillämpning av filtret. Minsta hastighet kan inte vara större än den maximala.',
                    },
                    title: {
                        settings: 'Inställningar',
                        roadTypes: 'Väglagstyper',
                        filter: 'Filter',
                        appearence: 'Utseende',
                        showOnRoadType: 'Visa på följande vägtyper:',
                        seeOnDiscuss: 'Se skriptets tråd på Discuss',
                        seeOnDiscord: 'Se skriptets kanal på Waze Script Discord-servern',
                        scriptDevBy: 'Visa min redigerarprofil',
                    },
                    label: {
                        displaySpeeds: 'Visa hastigheter',
                        showNoSpeed: 'Visa på segment utan hastighet',
                        showOnRoundabout: 'Visa på rondell',
                        showOnUnpaved: 'Visa på oasfalterad väg',
                        maxZoom: 'Rendera upp till zoomnivå:',
                        toggleLayerShortcut: 'Växla hastighetslagrets synlighet',
                        zIndex: 'Lagerdjup',
                        hideShorterSegments: 'Dölj på segment som är kortare än',
                        speedFilter: 'Filtrera efter hastighet',
                        min: 'Min.:',
                        max: 'Max.:',
                        opacity: 'Filter opacitetsnivå:',
                    },
                    alertMessage: {
                        zIndex: `När du använder ett lagerdjup större än <b>${this.zIndex.warn}</b> kommer hastighetslagret att täcka andra lager och du kommer inte att kunna välja några objekt på skärmen förrän du sänker det under <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `För att aktivera/inaktivera hastighetsvisningslagret, gå till <b>Map Layers > Roads > Visa hastigheter</b> eller använd genvägen <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Standard',
                        altViennaConv: 'Alternativ Wienkonvention',
                        japan: 'Japan',
                        northAmerica: 'Nordamerika',
                        sacu: 'Sydafrikanska tullunionen',
                        uk: 'Storbritannien',
                    },
                    btn: {
                        updateSettingsTab: 'Uppdatera meny',
                        resetSettings: 'Återställ inställningar',
                        seeOnDiscuss: 'Se på diskussionen',
                        seeOnDiscord: 'Se på Discord',
                        scriptDevBy: 'Script utvecklat av Luan Tavares',
                    }
                };
                
                break;

            case 'th':
                // Thai
                translations = {
                    name: 'การแสดงความเร็ว',
                    zoomOut: 'ซูมออก',
                    zoomIn: 'ซูมเข้า',
                    changelog: 'บันทึกการเปลี่ยนแปลง',
                    changelogMessage: `<h6>ใหม่:</h6><ul style="margin-bottom: 10px"><li>ข้อความ "กำลังอัปเดตไอคอนพร้อมดีเลย์" ถูกลบออกในที่สุดและแทนที่ด้วยไอคอนธรรมดา ทำให้เห็นชัดเจนว่า script กำลังประมวลผลไอคอนความเร็วอยู่ (การอัปเดตมีการหน่วงเวลาโดยเจตนา)</li></ul><h6>การแก้ไขและปรับปรุง:</h6><ul><li>เปลี่ยนชื่อแท็บ "ทั่วไป" เป็น "การตั้งค่า" เพื่อให้เข้าใจหน้าที่ของแต่ละแท็บชัดเจนยิ่งขึ้น</li><li>ฟิลเตอร์ขนาดเซ็กเมนต์จะไม่ลบไอคอนอีกต่อไป แต่จะใช้ความโปร่งใสแบบเดียวกับฟิลเตอร์ความเร็ว</li></ul>`,
                    close: 'ปิด',
                    viewScript: 'ดูสคริปต์',
                    viewChangelog: 'ดูบันทึกการเปลี่ยนแปลง',
                    reset: 'คืนค่า',
                    snackbar: {
                        errorFiltering: 'ข้อผิดพลาดในการใช้ตัวกรอง ความเร็วขั้นต่ำต้องไม่มากกว่าค่าสูงสุด.',
                    },
                    title: {
                        settings: 'การตั้งค่า',
                        roadTypes: 'ประเภทถนน',
                        filter: 'ตัวกรอง',
                        appearence: 'ลักษณะ',
                        showOnRoadType: 'แสดงในประเภทถนนต่อไปนี้:',
                        seeOnDiscuss: 'ดูเธรดของสคริปต์ใน Discuss',
                        seeOnDiscord: 'ดูช่องของสคริปต์ในเซิร์ฟเวอร์ Discord ของ Waze Script',
                        scriptDevBy: 'ดูโปรไฟล์ผู้ใช้ของฉันในตัวแก้ไข',
                    },
                    label: {
                        displaySpeeds: 'แสดงความเร็ว',
                        showNoSpeed: 'แสดงในเซกเมนต์ที่ไม่มีความเร็ว',
                        showOnRoundabout: 'แสดงในทางแยกรอบ',
                        showOnUnpaved: 'แสดงในถนนที่ไม่ได้ปู',
                        maxZoom: 'แสดงได้จนถึงการซูมระดับ:',
                        toggleLayerShortcut: 'สลับการแสดงผลเลเยอร์ความเร็ว',
                        zIndex: 'ความลึกของเลเยอร์',
                        hideShorterSegments: 'ซ่อนในเซกเมนต์ที่สั้นกว่า',
                        speedFilter: 'กรองตามความเร็ว',
                        min: 'ขั้นต่ำ:',
                        max: 'สูงสุด:',
                        opacity: 'ระดับความทึบของฟิลเตอร์:',
                    },
                    alertMessage: {
                        zIndex: `เมื่อใช้ความลึกของเลเยอร์ที่มากกว่า <b>${this.zIndex.warn}</b> เลเยอร์ความเร็วจะทับซ้อนกับเลเยอร์อื่นและคุณจะไม่สามารถเลือกองค์ประกอบใดๆ บนหน้าจอจนกว่าจะลดความลึกลงต่ำกว่า <b>${this.zIndex.warn}</b> อีกครั้ง.`,
                        toggleVisibility: `เพื่อเปิด/ปิดการแสดงผลของเลเยอร์ความเร็ว ให้ไปที่ <b>แผนที่ > ถนน > แสดงความเร็ว</b> หรือใช้ทางลัด <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'ค่าเริ่มต้น',
                        altViennaConv: 'อนุสัญญาวีนนาทางเลือก',
                        japan: 'ญี่ปุ่น',
                        northAmerica: 'อเมริกาเหนือ',
                        sacu: 'สหภาพศุลกากรแอฟริกาตอนใต้',
                        uk: 'สหราชอาณาจักร',
                    },
                    btn: {
                        updateSettingsTab: 'อัปเดตเมนู',
                        resetSettings: 'ตั้งค่าคืนค่า',
                        seeOnDiscuss: 'ดูในกระทู้สนทนา',
                        seeOnDiscord: 'ดูใน Discord',
                        scriptDevBy: 'สคริปต์พัฒนาโดย Luan Tavares',
                    }
                };
            
                break;

            case 'tr':
                // Turkish
                translations = {
                    name: 'Hız Gösterimi',
                    zoomOut: 'Daha uzak',
                    zoomIn: 'Daha yakın',
                    changelog: 'Değişiklikler',
                    changelogMessage: `<h6>Yenilikler:</h6><ul style="margin-bottom: 10px"><li>"Gecikmeli simge güncelleniyor" mesajı nihayet kaldırıldı ve hız simgeleri hâlâ işlenirken bunu belirten basit bir simgeyle değiştirildi (bilerek yapılan bir gecikme vardır).</li></ul><h6>Düzeltmeler ve iyileştirmeler:</h6><ul><li>"Genel" sekmesinin adı, her sekmenin işlevinin daha iyi anlaşılması için "Ayarlar" olarak değiştirildi;</li><li>Segment boyutu filtresi artık simgeyi kaldırmaz, hız filtresinde olduğu gibi opaklık uygular.</li></ul>`,
                    close: 'Kapat',
                    viewScript: 'Scripti Görüntüle',
                    viewChangelog: 'Değişiklikler Kaydını Görüntüle',
                    reset: 'Sıfırla',
                    snackbar: {
                        errorFiltering: 'Filtre uygulanırken hata oluştu. Minimum hız, maksimum hızdan büyük olamaz.',
                    },
                    title: {
                        settings: 'Ayarlar',
                        roadTypes: 'Yol Tipleri',
                        filter: 'Filtre',
                        appearence: 'Görünüm',
                        showOnRoadType: 'Aşağıdaki yol tiplerinde göster:',
                        seeOnDiscuss: 'Script\'in thread\'ini Discuss\'ta görün',
                        seeOnDiscord: 'Waze Script Discord sunucusunda script kanalını görün',
                        scriptDevBy: 'Editör kullanıcı profilimi görüntüle',
                    },
                    label: {
                        displaySpeeds: 'Hızları Göster',
                        showNoSpeed: 'Hızsız segmentlerde göster',
                        showOnRoundabout: 'Dönel kavşakta göster',
                        showOnUnpaved: 'Asfaltlanmamış yolda göster',
                        maxZoom: 'Zoom seviyesine kadar render yap:',
                        toggleLayerShortcut: 'Hız katmanını açıp kapatma kısayolu',
                        zIndex: 'Katman derinliği',
                        hideShorterSegments: 'Daha kısa segmentlerde gizle',
                        speedFilter: 'Hız filtresi',
                        min: 'Min:',
                        max: 'Maks:',
                        opacity: 'Filtre opaklık seviyesi:',
                    },
                    alertMessage: {
                        zIndex: `Katman derinliğini <b>${this.zIndex.warn}</b> değerinden daha yüksek kullanırsanız, hız katmanı diğerlerinin üst üste binmesine neden olur ve herhangi bir öğe seçilemez. Bu durumu düzeltmek için katman derinliğini <b>${this.zIndex.warn}</b> değerinin altına indirin.`,
                        toggleVisibility: `Hız katmanını açıp kapatmak için <b>Harita Katmanları > Yollar > Hızları Göster</b> yolunu izleyebilir veya <b>ALT</b> + <b>SHIFT</b> + <b>S</b> kısayolunu kullanabilirsiniz.`
                    },
                    theme: {
                        default: 'Varsayılan',
                        altViennaConv: 'Alternatif Viyana Sözleşmesi',
                        japan: 'Japonya',
                        northAmerica: 'Kuzey Amerika',
                        sacu: 'Güney Afrika Gümrük Birliği',
                        uk: 'Birleşik Krallık',
                    },
                    btn: {
                        updateSettingsTab: 'Menüyü Güncelle',
                        resetSettings: 'Ayarları Sıfırla',
                        seeOnDiscuss: 'Tartışmada gör',
                        seeOnDiscord: 'Discord’da gör',
                        scriptDevBy: 'Script, Luan Tavares tarafından geliştirilmiştir',
                    }
                };
            
                break;

            case 'uk':
                // Ukrainian
                translations = {
                    name: 'Відображення швидкості',
                    zoomOut: 'Далі',
                    zoomIn: 'Ближче',
                    changelog: 'Журнал змін',
                    changelogMessage: `<h6>Що нового:</h6><ul style="margin-bottom: 10px"><li>Повідомлення «оновлення значків із затримкою» остаточно видалено й замінено простим значком, який чітко вказує, що скрипт ще обробляє значки швидкості (оновлення виконується із запланованою затримкою).</li></ul><h6>Виправлення та покращення:</h6><ul><li>Вкладку «Загальні» перейменовано на «Налаштування» для кращого розуміння функцій;</li><li>Фільтр довжини сегментів більше не видаляє значки, а застосовує ту ж прозорість, що й фільтр швидкості.</li></ul>`,
                    close: 'Закрити',
                    viewScript: 'Переглянути сценарій',
                    viewChangelog: 'Переглянути журнал змін',
                    reset: 'Скинути',
                    snackbar: {
                        errorFiltering: 'Помилка застосування фільтра. Мінімальна швидкість не може бути більшою за максимальну.',
                    },
                    title: {
                        settings: 'Налаштування',
                        roadTypes: 'Типи доріг',
                        filter: 'Фільтр',
                        appearence: 'Зовнішній вигляд',
                        showOnRoadType: 'Показати на наступних типах доріг:',
                        seeOnDiscuss: 'Переглянути тему скрипта на Discuss',
                        seeOnDiscord: 'Переглянути канал скрипта на сервері Discord Waze Script',
                        scriptDevBy: 'Переглянути мій профіль користувача редактора',
                    },
                    label: {
                        displaySpeeds: 'Показувати швидкості',
                        showNoSpeed: 'Показувати на сегментах без швидкості',
                        showOnRoundabout: 'Показувати на кільцевих перехрестях',
                        showOnUnpaved: 'Показувати на непокритих дорогах',
                        maxZoom: 'Рендерити до рівня zoom:',
                        toggleLayerShortcut: 'Перемикання шару відображення швидкості',
                        zIndex: 'Глибина шару',
                        hideShorterSegments: 'Приховувати на сегментах менше ніж',
                        speedFilter: 'Фільтрувати за швидкістю',
                        min: 'Мін.:',
                        max: 'Макс.:',
                        opacity: 'Рівень прозорості фільтра:',
                    },
                    alertMessage: {
                        zIndex: `Якщо використовувати глибину шару більше за <b>${this.zIndex.warn}</b>, шар швидкості накладатиметься на інші, і ви не зможете вибрати жоден елемент на екрані, поки не зменшите глибину до <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Щоб увімкнути/вимкнути шар швидкості, просто перейдіть до <b>Шари карти > Дороги > Відображати швидкості</b> або використовуйте гарячу клавішу <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'За замовчуванням',
                        altViennaConv: 'Альтернативна Віденська конвенція',
                        japan: 'Японія',
                        northAmerica: 'Північна Америка',
                        sacu: 'Південноафриканський митний союз',
                        uk: 'Сполучене Королівство',
                    },
                    btn: {
                        updateSettingsTab: 'Оновити меню',
                        resetSettings: 'Скинути налаштування',
                        seeOnDiscuss: 'Дивитися в обговоренні',
                        seeOnDiscord: 'Дивитися в Discord',
                        scriptDevBy: 'Сценарій розроблений Луаном Таваресом',
                    }
                };
                
                break;

            case 'vi':
                // Vietnamese
                translations = {
                    name: 'Hiển thị tốc độ',
                    zoomOut: 'Zoom ra',
                    zoomIn: 'Zoom vào',
                    changelog: 'Lịch sử thay đổi',
                    changelogMessage: `<h6>Mới:</h6><ul style="margin-bottom: 10px"><li>Thông báo "đang cập nhật biểu tượng với độ trễ" cuối cùng đã được gỡ bỏ và thay thế bằng biểu tượng đơn giản, thể hiện rõ khi nào script vẫn đang xử lý biểu tượng tốc độ (có độ trễ có chủ đích).</li></ul><h6>Sửa lỗi và cải tiến:</h6><ul><li>Tab "Chung" được đổi tên thành "Cài đặt" để dễ hiểu hơn về chức năng của từng tab;</li><li>Bộ lọc kích thước đoạn đường không còn xóa biểu tượng mà áp dụng độ mờ giống như bộ lọc tốc độ.</li></ul>`,
                    close: 'Đóng',
                    viewScript: 'Xem kịch bản',
                    viewChangelog: 'Xem lịch sử thay đổi',
                    reset: 'Đặt lại',
                    snackbar: {
                        errorFiltering: 'Lỗi áp dụng bộ lọc. Tốc độ tối thiểu không thể lớn hơn tốc độ tối đa.',
                    },
                    title: {
                        settings: 'Cài đặt',
                        roadTypes: 'Loại đường',
                        filter: 'Lọc',
                        appearence: 'Ngoại hình',
                        showOnRoadType: 'Hiển thị trên các loại đường sau:',
                        seeOnDiscuss: 'Xem chủ đề của script trên Discuss',
                        seeOnDiscord: 'Xem kênh của script trên máy chủ Discord của Waze Script',
                        scriptDevBy: 'Xem hồ sơ người dùng biên tập viên của tôi',
                    },
                    label: {
                        displaySpeeds: 'Hiển thị tốc độ',
                        showNoSpeed: 'Hiển thị trên đoạn đường không có tốc độ',
                        showOnRoundabout: 'Hiển thị trên vòng xuyến',
                        showOnUnpaved: 'Hiển thị trên đường không trải nhựa',
                        maxZoom: 'Hiển thị đến mức zoom:',
                        toggleLayerShortcut: 'Chuyển đổi lớp hiển thị tốc độ',
                        zIndex: 'Độ sâu lớp',
                        hideShorterSegments: 'Ẩn trên các đoạn đường ngắn hơn',
                        speedFilter: 'Lọc theo tốc độ',
                        min: 'Tối thiểu:',
                        max: 'Tối đa:',
                        opacity: 'Mức độ trong suốt bộ lọc:',
                    },
                    alertMessage: {
                        zIndex: `Khi sử dụng độ sâu lớp lớn hơn <b>${this.zIndex.warn}</b>, lớp tốc độ sẽ chồng lên các lớp khác và bạn sẽ không thể chọn bất kỳ đối tượng nào trên màn hình cho đến khi giảm độ sâu lớp xuống dưới <b>${this.zIndex.warn}</b>.`,
                        toggleVisibility: `Để bật/tắt lớp hiển thị tốc độ, chỉ cần vào <b>Lớp bản đồ > Đoạn đường > Hiển thị tốc độ</b> hoặc sử dụng phím tắt <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Mặc định',
                        altViennaConv: 'Hiệp định Vienna thay thế',
                        japan: 'Nhật Bản',
                        northAmerica: 'Bắc Mỹ',
                        sacu: 'Liên minh Hải quan Nam Phi',
                        uk: 'Vương quốc Anh',
                    },
                    btn: {
                        updateSettingsTab: 'Cập nhật menu',
                        resetSettings: 'Đặt lại cài đặt',
                        seeOnDiscuss: 'Xem trên thảo luận',
                        seeOnDiscord: 'Xem trên Discord',
                        scriptDevBy: 'Script được phát triển bởi Luan Tavares',
                    }
                };
            
                break;

            case 'zh':
                // Chinese
                translations = {
                    name: '速度显示',
                    zoomOut: '缩小',
                    zoomIn: '放大',
                    changelog: '更新日志',
                    changelogMessage: `<h6>新增內容:</h6><ul style="margin-bottom: 10px"><li>「延遲更新圖示」的訊息已被簡單圖示取代,明確顯示腳本正在處理速度圖示(請注意,有刻意延遲)。</li></ul><h6>修正與改進:</h6><ul><li>將「一般」分頁名稱修改為「設定」,更清楚表達各分頁的功能;</li><li>依據路段長度的篩選條件不再移除圖示,而是套用與速度篩選相同的透明度。</li></ul>`,
                    close: '关闭',
                    viewScript: '查看脚本',
                    viewChangelog: '查看更新日志',
                    reset: '重置',
                    snackbar: {
                        errorFiltering: '应用过滤器时出错。最低速度不能大于最高速度。',
                    },
                    title: {
                        settings: '设置',
                        roadTypes: '道路类型',
                        filter: '过滤器',
                        appearence: '外观',
                        showOnRoadType: '在以下道路类型上显示:',
                        seeOnDiscuss: '查看脚本的主题讨论',
                        seeOnDiscord: '查看Waze Script Discord服务器上的脚本频道',
                        scriptDevBy: '查看我的编辑器用户档案',
                    },
                    label: {
                        displaySpeeds: '显示速度',
                        showNoSpeed: '显示无速度段落',
                        showOnRoundabout: '显示在环形交叉口',
                        showOnUnpaved: '显示在未铺砌的道路上',
                        maxZoom: '最大缩放:',
                        toggleLayerShortcut: '切换速度显示图层',
                        zIndex: '图层深度',
                        hideShorterSegments: '隐藏小于以下长度的段落',
                        speedFilter: '按速度过滤',
                        min: '最小:',
                        max: '最大:',
                        opacity: '过滤器透明度级别:',
                    },
                    alertMessage: {
                        zIndex: `当使用大于 <b>${this.zIndex.warn}</b> 的图层深度时,速度图层将覆盖其他图层,您将无法选择任何屏幕上的元素,直到将其深度降低到 <b>${this.zIndex.warn}</b> 以下。`,
                        toggleVisibility: `要启用/禁用速度显示图层,只需转到 <b>地图图层 > 道路 > 显示速度</b> 或使用快捷键 <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: '默认',
                        altViennaConv: '替代维也纳公约',
                        japan: '日本',
                        northAmerica: '北美洲',
                        sacu: '南部非洲关税联盟',
                        uk: '英国',
                    },
                    btn: {
                        updateSettingsTab: '更新菜单',
                        resetSettings: '重置设置',
                        seeOnDiscuss: '在讨论中查看',
                        seeOnDiscord: '在 Discord 上查看',
                        scriptDevBy: '脚本由Luan Tavares开发',
                    }
                };
            
                break;

            case 'zh-TW':
                // Traditional Chinese
                translations = {
                    name: '速度顯示',
                    zoomOut: '縮小',
                    zoomIn: '放大',
                    changelog: '更新日誌',
                    changelogMessage: `<h6>新增內容:</h6><ul style="margin-bottom: 10px"><li>最終移除「延遲更新圖示」的訊息,改為使用簡單圖示,清楚顯示腳本仍在處理速度圖示(請記得更新有刻意延遲)。</li></ul><h6>修正與改善:</h6><ul><li>將「一般」頁籤名稱改為「設定」,以更清楚說明各頁籤功能;</li><li>路段長度過濾器不再移除圖示,而是套用與速度過濾器相同的透明度。</li></ul>`,
                    close: '關閉',
                    viewScript: '查看腳本',
                    viewChangelog: '查看更新日誌',
                    reset: '重設',
                    snackbar: {
                        errorFiltering: '應用篩選器時出錯。最低速度不能大於最高速度。',
                    },
                    title: {
                        settings: '設定',
                        roadTypes: '道路類型',
                        filter: '過濾器',
                        appearence: '外觀',
                        showOnRoadType: '在以下道路類型上顯示:',
                        seeOnDiscuss: '查看腳本的主題討論',
                        seeOnDiscord: '查看Waze Script Discord伺服器上的腳本頻道',
                        scriptDevBy: '查看我的編輯器用戶檔案',
                    },
                    label: {
                        displaySpeeds: '顯示速度',
                        showNoSpeed: '顯示無速度段落',
                        showOnRoundabout: '顯示在環形交叉口',
                        showOnUnpaved: '顯示在未鋪設的道路上',
                        maxZoom: '最大縮放:',
                        toggleLayerShortcut: '切換速度顯示圖層',
                        zIndex: '圖層深度',
                        hideShorterSegments: '隱藏小於以下長度的段落',
                        speedFilter: '按速度過濾',
                        min: '最小:',
                        max: '最大:',
                        opacity: '濾鏡透明度級別:',
                    },
                    alertMessage: {
                        zIndex: `當使用大於 <b>${this.zIndex.warn}</b> 的圖層深度時,速度圖層將覆蓋其他圖層,您將無法選擇任何螢幕上的元素,直到將其深度降低到 <b>${this.zIndex.warn}</b> 以下。`,
                        toggleVisibility: `要啟用/禁用速度顯示圖層,只需前往 <b>地圖圖層 > 道路 > 顯示速度</b> 或使用快捷鍵 <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: '預設',
                        altViennaConv: '替代維也納公約',
                        japan: '日本',
                        northAmerica: '北美洲',
                        sacu: '南部非洲關稅聯盟',
                        uk: '英國',
                    },
                    btn: {
                        updateSettingsTab: '更新菜單',
                        resetSettings: '重設設定',
                        seeOnDiscuss: '在討論中查看',
                        seeOnDiscord: '在 Discord 上查看',
                        scriptDevBy: '腳本由Luan Tavares開發',
                    }
                };
            
                break;

            default:
                // Default language (english)
                translations = {
                    name: 'Speed Display',
                    zoomOut: 'Zoom Out',
                    zoomIn: 'Zoom In',
                    changelog: 'Changelog',
                    changelogMessage: `<h6>What's New:</h6><ul style="margin-bottom: 10px"><li>Finally removed the "updating icons with delay" message and replaced it with a simple icon, making it clear when the script is still processing speed icons (remember there’s an intentional delay).</li></ul><h6>Fixes and improvements:</h6><ul><li>Renamed the "general" tab to "settings" for better clarity on each tab's function;</li><li>The segment size filter no longer removes the icon but now applies the same opacity as the speed filter.</li></ul>`,
                    close: 'Close',
                    viewScript: 'View Script',
                    viewChangelog: 'View Changelog',
                    reset: 'Reset',
                    snackbar: {
                        errorFiltering: 'Error applying filter. Minimum speed cannot be greater than maximum.',
                    },
                    title: {
                        settings: 'General',
                        roadTypes: 'Road Types',
                        filter: 'Filter',
                        appearence: 'Appearence',
                        showOnRoadType: 'Show on the following road types:',
                        seeOnDiscuss: 'See the script\'s thread on the Discuss',
                        seeOnDiscord: 'See the script\'s channel on the Waze Script\'s Discord server',
                        scriptDevBy: 'View my editor user profile',
                    },
                    label: {
                        displaySpeeds: 'Display speeds',
                        showNoSpeed: 'Show on segments with no speed',
                        showOnRoundabout: 'Show on roundabout',
                        showOnUnpaved: 'Show on unpaved road',
                        maxZoom: 'Render until zoom:',
                        toggleLayerShortcut: 'Toggle speed display layer',
                        zIndex: 'Layer depth',
                        hideShorterSegments: 'Hide segments shorter than',
                        speedFilter: 'Filter by speed',
                        min: 'Min:',
                        max: 'Max:',
                        opacity: 'Filter opacity level:',
                    },
                    alertMessage: {
                        zIndex: `When using a layer depth greater than <b>${this.zIndex.warn}</b>, the speed layer will overlap others, and you will not be able to select any element on the screen until you lower the depth below <b>${this.zIndex.warn}</b>.`,
                        // TODO: Adjust translation of the toggleVisibility when finishing SDK migration
                        toggleVisibility: `To toggle the visibility of the speed display layer, just go to <b>Map Layers > Roads > Show speeds</b> or use the shortcut <b>ALT</b> + <b>SHIFT</b> + <b>S</b>`
                    },
                    theme: {
                        default: 'Default',
                        altViennaConv: 'Alternative Vienna Convention',
                        japan: 'Japan',
                        northAmerica: 'North America',
                        sacu: 'Southern African Customs Union',
                        uk: 'United Kingdom',
                    },
                    btn: {
                        updateSettingsTab: 'Update Menu',
                        resetSettings: 'Reset Settings',
                        seeOnDiscuss: 'See on the Discuss',
                        seeOnDiscord: 'See on the Discord',
                        scriptDevBy: 'Script developed by Luan Tavares',
                    }
                };                

                break;
        };

        Object.assign(this.spdTranslations, translations);

        this.logDebug('Languages defined.');
    }
}

window.SDK_INITIALIZED.then(() => new WmeSpeedDisplay());