WME UR Color Selector

(NL) Hiermee geef je elke UR-taal een eigen kleur, zodat je snel ziet in welke taal de melder zijn app gebruikt. Slim gebruik: Kies Gebruik kleur voor overige talen zodat je gemakkelijk kunt zien of het bijvoorbeeld niet je eigen taal is. (EN) This script lets you assign a custom color per UR reporter language, so you can quickly see what language the Waze app is set to. Smart usage: choose Use color for other languages making it easy to see if it's not your native language.

// ==UserScript==
// @name         WME UR Color Selector
// @author       DeKoerier
// @namespace    https://greasyfork.org/users/1499279
// @version      4.0
// @description  (NL) Hiermee geef je elke UR-taal een eigen kleur, zodat je snel ziet in welke taal de melder zijn app gebruikt. Slim gebruik: Kies Gebruik kleur voor overige talen zodat je gemakkelijk kunt zien of het bijvoorbeeld niet je eigen taal is. (EN) This script lets you assign a custom color per UR reporter language, so you can quickly see what language the Waze app is set to. Smart usage: choose Use color for other languages making it easy to see if it's not your native language.
// @match        https://*.waze.com/*editor*
// @grant        none
// @license      GPL3
// @icon         
// @connect      greasyfork.org
// ==/UserScript==

/* global W, I18n, OpenLayers */

const SCRIPT_NAME = "WME UR Color Selector";
const SCRIPT_VERSION = '4.0';
const VERSION_STORAGE_KEY = 'LCVersionCheckerKey';

const WHATS_NEW = {
    "4.0": "- Script name has been changed from WME Language Color Selector to WME UR Color Selector.\n- New collapsible sections: Language color settings, OS colors, and Vehicle type colors. Open/closed state is remembered per section.\n- Saved colors moved to a dedicated panel at the very bottom (outside the collapsibles).\n- UI polish for narrow sidebars: checkboxes aligned with labels, tighter grid layout, thicker card borders, and clearer visual grouping.\n- OS and Vehicle badges in the Update Request popup now use the same coloring logic as Language, with automatic black/white text selection for readability (WCAG contrast based).\n- UR overlay rework: small colored badges (Language → OS → Vehicle) are vertically stacked under the UR icon; refined glyphs; overlay ignores pointer events so map interactions aren’t blocked.\n- Canonicalization & auto-migration: OS keys normalized (e.g., ANDROID → ANDROID_MOBILE; aliases for CarPlay/Android Auto) and Vehicle types normalized/localized (PRIVATE, EV, and new MOTORCYCLE). Existing preferences are migrated automatically.\n- Swatch application now targets the currently focused field (Language/OS/Vehicle). Fixed a bug where applying a saved color to OS accidentally added the item under Languages.\n- Robustness/performance tweaks: safer badge detection, wait for UR icons to be visible before drawing, hide overlay while panning/zooming, and refresh after map moves.",
    "3.3": "- Added expandable color detail menu (▼/▲) for each language.\n- New HEX field with Copy, Paste, and Save options.\n- Saved colors palette: quickly reuse previously chosen colors.\n- Apply saved colors to any language with a single click.\n- Remove saved colors via Alt+Click or right-click.",
    "3.2": "- The save button has been removed. With each update, it became increasingly useless, only saving the color change. Now, everything saves automatically when a change is made.",
    "3.1": "- Bugfix: When loading the page or moving the map the UR badges didn't reload. This has been resolved.",
    "3.0": "- REWORK LANGUAGES & UR-ICON OVERLAY: The language system has been changed and improved (future-proofing). We recommend removing languages if they no longer work and re-adding them from the dropdown menu! Additionally, at the request of several members via Discuss, the option to place a colored circle on/over the UR has been added. Due to Waze's functionality, the UR icon itself cannot be customized. This option can be enabled via the sidebar menu. Feedback is appreciated; any necessary adjustments can then be made to the new UR overlay.",
    "2.9": "- Changed français_BE to francais_BE. If you added français_BE, you can remove it and add the new one.",
    "2.8": "- UI rework to neaten up the language and color buttons and to position the Save button more neatly (thank you YanisKyr). Also, a bug fix: added an option to the dropdown menu to prevent reselecting a language once it's already selected.",
    "2.7": "- Languages added: nl-BE and français_BE. An option has been added to color all other unselected languages, making it easy to see if it's not your native language, for example. Small bugfixes.",
    "2.6": "- Small bugfixes",
    "2.3": "- Removed default colors\n- Added support for 20+ languages\n- Improved UI layout\n- WME SDK compatibility fix",
    "2.2": "- Added support for URComments-Enhanced badges",
    "2.1": "- Multiple bugfixes\n- Better performance in large UR lists",
    "2.0": "- Added full sidebar for configuring language colors",
    "1.0": "- Initial version"
};

const DEBUG_LOGGING = false;
function log(...args) { if (DEBUG_LOGGING) console.log('[LC]', ...args); }
function warn(...args) { if (DEBUG_LOGGING) console.warn('[LC]', ...args); }

let LCSmoveEndTimeout = null;

(function () {
    'use strict';

    const LANG_STORAGE_KEY = 'languageColorPrefs';

    // I wasn't able to find the shortcode of the languages below. Due to this, these are not yet available via WME LCS! Sorry!
    // 'अंग्रेजी ', 'ქართული', 'Tiếng Việt',

    const LANGUAGE_MAP = {
        dutch: 'Nederlands',
        dutch_be: 'nl-BE',
        francais_BE: 'francais_BE',
        francais: 'Français',
        eng: 'English',
        eng_uk: 'English (UK)',
        eng_ireland: 'eng_ireland',
        eng_us: 'US English',
        eng_au: 'eng_au',
        deutsch: 'Deutsch',
        espanol: 'Español',
        espanol_latam: 'Español (A. Latina)',
        portuguese_br: 'Português (Brasil)',
        portuguese_pt: 'Português (Portugal)',
        galician: 'Galego',
        italiano: 'Italiano',
        polski: 'Polski',
        russian: 'Русский',
        ukrainian: 'Українська',
        arabic: 'العربية',
        romanian: 'Română',
        hungarian: 'Magyar',
        slovak: 'Slovenčina',
        slovenian: 'Slovenščina',
        croatian: 'Hrvatski',
        czech: 'Čeština',
        heb: 'עברית',
        danish: 'Dansk',
        swedish: 'svenska',
        norwegian: 'Norsk',
        finnish: 'Suomi',
        estonian: 'Eesti',
        latvian: 'Latviešu',
        lithuanian: 'Lietuviškai',
        catalan: 'Català',
        turkish: 'Türkçe',
        japanese: '日本語',
        korean: '한국어',
        chinese: '中文',
        chinese_tw: '繁體中文',
        chinese_hk: 'zh-HK',
        filipino: 'fil',
        malay: 'Bahasa Malaysia',
        indonesian: 'Bahasa Indonesia',
        vietnamese: 'vietnamese',
        afrikaans: 'Afrikaans',
        bulgarian: 'Български',
        greek: 'Ελληνικά',
        thai: 'ภาษาไทย',
        persian: 'پارسی',
        onbekend: 'Unknown',
        default: 'default'
    };

    const OS_KEYS = ['IOS','ANDROID_MOBILE','CARPLAY','ANDROID_AUTO'];
    const OS_LABEL = { IOS:'iOS', ANDROID_MOBILE:'Android', CARPLAY:'CarPlay', ANDROID_AUTO:'Android Auto' };
    const OS_ALIASES = {
        IOS: ['iOS','IOS'],
        ANDROID_MOBILE: ['ANDROID_MOBILE','Android','ANDROID','Android Mobile','Android (mobile)'],
        CARPLAY: ['CarPlay','Apple CarPlay'],
        ANDROID_AUTO: ['Android Auto','AndroidAuto','Android-Auto']
    };
    function normalizeToOSKey(label) {
        const t = (label||'').trim().toLowerCase();
        for (const k of OS_KEYS) if (OS_ALIASES[k].some(a => a.toLowerCase() === t)) return k;
        return null;
    }

    const VEH_KEYS = ['PRIVATE','EV', 'MOTORCYCLE', 'TAXI'];
    const VEH_LABEL = {
        PRIVATE: { nl:'Personenauto', en:'Private' },
        EV: { nl:'EV', en:'EV' },
        MOTORCYCLE: { nl: 'Motor', en: 'Motorcycle'},
        TAXI: { nl: 'Taxi', en: 'Taxi'},
    };
    const VEH_ALIASES = {
        PRIVATE: ['PRIVATE','Personenauto','Private'],
        EV: ['EV','Elektrisch','Electric','Electric vehicle','Elektrische auto'],
        MOTORCYCLE: ['Motor', 'Motorcycle', 'Motorbike'],
        TAXI: ['Taxi']
    };
    const uiLang = () => (I18n?.locale?.startsWith('nl') ? 'nl' : 'en');
    const vehicleDisplay = (key) => (VEH_LABEL[key]?.[uiLang()] || key);
    function normalizeToVehicleKey(label) {
        const t = (label||'').trim().toLowerCase();
        for (const k of VEH_KEYS) if (VEH_ALIASES[k].some(a => a.toLowerCase() === t)) return k;
        return null;
    }

    const isDutch = I18n?.locale === 'nl';
    const UI = {
        tab: isDutch ? 'UR Kleuren' : 'UR Colors',
        tabTitle: isDutch ? 'Configureer melderskleuren per taal' : 'Configure reporter language colors',
        panelTitle: isDutch ? 'TaalKleur-instellingen' : 'LanguageColor settings',
        remove: isDutch ? 'Verwijder taal' : 'Remove language',
        save: isDutch ? 'Opslaan' : 'Save',
        addLanguage: isDutch ? 'Taal toevoegen...' : 'Add language...'
    };

    let colorPrefs = JSON.parse(localStorage.getItem(LANG_STORAGE_KEY)) || {};

    const OTHER_PREFS_KEY = 'languageColorOther';
    let colorOtherPrefs = JSON.parse(localStorage.getItem(OTHER_PREFS_KEY)) || { enabled: false, color: '#ff0000' };
    const SHOW_BADGES_KEY = 'languageColorShowURBadge';
    let showBadgeOnUR = JSON.parse(localStorage.getItem(SHOW_BADGES_KEY)) ?? false;

    const OS_PREFS_KEY = 'languageColorOSPrefs';
    const OS_OTHER_PREFS_KEY = 'languageColorOSOther';
    const SHOW_OS_BADGE_KEY = 'languageColorShowOSBadge';
    let osColorPrefs = JSON.parse(localStorage.getItem(OS_PREFS_KEY)) || {};
    let osOtherPrefs = JSON.parse(localStorage.getItem(OS_OTHER_PREFS_KEY)) || { enabled: false, color: '#7c4dff' };
    let showOSBadge = JSON.parse(localStorage.getItem(SHOW_OS_BADGE_KEY)) ?? false;

    const VEH_PREFS_KEY = 'languageColorVehiclePrefs';
    const VEH_OTHER_PREFS_KEY = 'languageColorVehicleOther';
    const SHOW_VEH_BADGE_KEY = 'languageColorShowVehicleBadge';
    let vehicleColorPrefs = JSON.parse(localStorage.getItem(VEH_PREFS_KEY)) || {};
    let vehicleOtherPrefs = JSON.parse(localStorage.getItem(VEH_OTHER_PREFS_KEY)) || { enabled: false, color: '#00bcd4' };
    let showVehicleBadge = JSON.parse(localStorage.getItem(SHOW_VEH_BADGE_KEY)) ?? false;

    const DEFAULT_BADGE_BG = 'rgb(204 204 204 / 0%)';
    const DEFAULT_BADGE_TEXT = '#55595e';
    const FALLBACK_COLOR_INPUT = '#cccccc';

    const SWATCHES_KEY = 'languageColorSwatches';
    let lastFocusTarget = null;
    let swatchDeleteMode = false;

    (function migratePrefsToCanonical() {
        let changed = false;
        const vNew = { ...vehicleColorPrefs };
        for (const [k,v] of Object.entries(vehicleColorPrefs)) {
            const key = normalizeToVehicleKey(k) || k.toUpperCase();
            if (key !== k) {
                delete vNew[k];
                if (!vNew[key]) vNew[key] = v;
                changed = true;
            }
        }
        if (changed) {
            vehicleColorPrefs = vNew;
            localStorage.setItem(VEH_PREFS_KEY, JSON.stringify(vehicleColorPrefs));
        }

        changed = false;
        const oNew = { ...osColorPrefs };
        for (const [k,v] of Object.entries(osColorPrefs)) {
            const k2 = normalizeToOSKey(k) || k.toUpperCase();
            if (k2 !== k) {
                delete oNew[k];
                if (!oNew[k2]) oNew[k2] = v;
                changed = true;
            }
        }
        if (changed) {
            osColorPrefs = oNew;
            localStorage.setItem(OS_PREFS_KEY, JSON.stringify(osColorPrefs));
        }
    })();

    const observer = new MutationObserver(() => applyLanguageColor());

    function hexToRgb(hex) {
        if (!hex) return null;
        let h = hex.trim().replace(/^#/, '');
        if (h.length === 3) h = h.split('').map(ch => ch + ch).join('');
        if (h.length !== 6) return null;
        const n = parseInt(h, 16);
        return { r: (n >> 16) & 255, g: (n >> 8) & 255, b: n & 255 };
    }
    function srgbToLinear(c) {
        const v = c / 255;
        return v <= 0.04045 ? v / 12.92 : Math.pow((v + 0.055) / 1.055, 2.4);
    }
    function relLuminanceFromHex(hex) {
        const rgb = hexToRgb(hex);
        if (!rgb) return null;
        const R = srgbToLinear(rgb.r);
        const G = srgbToLinear(rgb.g);
        const B = srgbToLinear(rgb.b);
        return 0.2126 * R + 0.7152 * G + 0.0722 * B;
    }
    function pickTextColorOnBg(hex) {
        const Lbg = relLuminanceFromHex(hex);
        if (Lbg == null) return DEFAULT_BADGE_TEXT;
        const contrastWhite = (1.0 + 0.05) / (Lbg + 0.05);
        const contrastBlack = (Lbg + 0.05) / 0.05;
        return contrastWhite >= contrastBlack ? '#ffffff' : '#000000';
    }

    function applyLanguageColor() {
        const root = document.querySelector('.reporter-preferences');
        if (!root) return;

        const paint = (wzBadgeEl, hex) => {
            const chip = wzBadgeEl?.shadowRoot?.querySelector('.wz-badge');
            if (!chip) return;
            if (hex && /^#[0-9A-F]{6}$/i.test(hex)) {
                chip.style.backgroundColor = hex;
                chip.style.color = pickTextColorOnBg(hex);
            } else {
                chip.style.backgroundColor = DEFAULT_BADGE_BG;
                chip.style.color = DEFAULT_BADGE_TEXT;
            }
        };

        const langBadge = root.querySelector('wz-badge.reporterLanguage');
        if (langBadge) {
            const label = langBadge.textContent.trim();
            const isKnown = label in colorPrefs;
            const c = colorPrefs[label];
            const hex = (isKnown && /^#[0-9A-F]{6}$/i.test(c)) ? c :
            (!isKnown && colorOtherPrefs.enabled ? colorOtherPrefs.color : null);
            paint(langBadge, hex);
        }

        let osBadge = root.querySelector('wz-badge.reporterOs, wz-badge.reporterOS, wz-badge.os, wz-badge.reporter-os') || null;
        if (!osBadge) {
            osBadge = Array.from(root.querySelectorAll('wz-badge'))
                .filter(b => b !== langBadge)
                .find(b => normalizeToOSKey(b.textContent.trim())) || null;
        }
        if (osBadge) {
            const key = normalizeToOSKey(osBadge.textContent.trim());
            const hex = (key && /^#[0-9A-F]{6}$/i.test(osColorPrefs[key])) ? osColorPrefs[key]
            : (key && osOtherPrefs.enabled ? osOtherPrefs.color : null);
            paint(osBadge, hex);
        }

        let vehBadge = root.querySelector('wz-badge.reporterVehicle, wz-badge.reporterVehicleType, wz-badge.vehicle, wz-badge.reporter-vehicle') || null;
        if (!vehBadge) {
            vehBadge = Array.from(root.querySelectorAll('wz-badge'))
                .filter(b => b !== langBadge && b !== osBadge)
                .find(b => normalizeToVehicleKey(b.textContent.trim())) || null;
        }
        if (vehBadge) {
            const key = normalizeToVehicleKey(vehBadge.textContent.trim());
            const hex = (key && /^#[0-9A-F]{6}$/i.test(vehicleColorPrefs[key])) ? vehicleColorPrefs[key]
            : (key && vehicleOtherPrefs.enabled ? vehicleOtherPrefs.color : null);
            paint(vehBadge, hex);
        }
    }

    function removeSwatch(hex) {
        const list = getSwatches().filter(c => c.toLowerCase() !== hex.toLowerCase());
        setSwatches(list);
    }
    function isHex6(v){ return /^#?[0-9a-fA-F]{6}$/.test(v?.trim()); }
    function toHex6(v){
        if(!v) return null;
        let s = v.trim();
        if(s[0] !== '#') s = '#'+s;
        if(isHex6(s)) return s.toLowerCase();
        return null;
    }
    function getSwatches(){
        try { return JSON.parse(localStorage.getItem(SWATCHES_KEY)) || []; }
        catch { return []; }
    }
    function setSwatches(arr){
        localStorage.setItem(SWATCHES_KEY, JSON.stringify(arr.slice(0, 30)));
    }
    function addSwatchUnique(hex){
        const h = toHex6(hex);
        if(!h) return;
        const list = getSwatches();
        if(!list.includes(h)){
            list.unshift(h);
            setSwatches(list);
        }
    }

    function ensureSidebarStyles() {
        if (document.getElementById('lcs-compact-css')) return;
        const css = `
#languageColorConfig.lcs-ui{
  --lcs-gap:6px; --lcs-border:#cfd8e3; --lcs-radius:12px;
  font:12.5px/1.35 system-ui,-apple-system,Segoe UI,Roboto,Arial,sans-serif; color:#1f2937;
}

/* Titels */
#languageColorConfig.lcs-ui h3{ margin:10px 0 6px; font-size:14px; font-weight:700; }

/* Algemene “card”-sectie */
#languageColorConfig.lcs-ui .lcs-section{
  background:#fff; border:1.6px solid var(--lcs-border); border-radius:var(--lcs-radius);
  padding:8px; margin:8px 0; box-shadow:0 1px 0 rgba(0,0,0,.03);
}

/* Accordion (collapsible) */
#languageColorConfig.lcs-ui .lcs-accordion details{
  background:#fff; border:1.6px solid var(--lcs-border); border-radius:var(--lcs-radius);
  overflow:hidden; margin:8px 0;
}
#languageColorConfig.lcs-ui .lcs-accordion summary{
  display:flex; align-items:center; gap:8px; padding:10px; cursor:pointer; user-select:none;
  font-weight:700; list-style:none; outline:none;
}
#languageColorConfig.lcs-ui .lcs-accordion summary::-webkit-details-marker{ display:none; }
#languageColorConfig.lcs-ui .lcs-caret{ margin-left:auto; transition:transform .15s ease; }
#languageColorConfig.lcs-ui details[open] .lcs-caret{ transform:rotate(180deg); }
#languageColorConfig.lcs-ui .lcs-acc-body{ padding:10px; border-top:1px dashed #e5e7eb; }

/* Rij met label + controls */
#languageColorConfig.lcs-ui .lcs-row{
  display:grid; grid-template-columns:1fr auto auto auto; gap:var(--lcs-gap); align-items:center;
}
#languageColorConfig.lcs-ui .lcs-label{ font-weight:600; overflow:hidden; white-space:nowrap; text-overflow:ellipsis; }

/* Checkbox + tekst op één lijn (en optioneel extra control rechts) */
#languageColorConfig.lcs-ui .lcs-check{
  display:grid; grid-template-columns:auto 1fr auto; gap:8px; align-items:center;
}
#languageColorConfig.lcs-ui .lcs-check input[type="checkbox"]{ margin:0; transform:translateY(0); }

/* Inputs/knoppen */
#languageColorConfig.lcs-ui input[type="color"]{
  width:28px; height:22px; border:none; border-radius:6px; padding:0; cursor:pointer;
}
#languageColorConfig.lcs-ui button{
  height:22px; padding:0 6px; border-radius:6px; border:1px solid #d8dee9; background:#f8fafc; cursor:pointer;
}
#languageColorConfig.lcs-ui button:hover{ background:#eef2f7; }
#languageColorConfig.lcs-ui select{
  width:100%; padding:6px; border:1px solid #d8dee9; border-radius:8px; background:#fff;
}

/* Detailpaneeltje per item */
#languageColorConfig.lcs-ui .lcs-details{
  margin-top:6px; padding:6px; border:1px dashed #e5e7eb; border-radius:8px; background:#f9fafb;
}
#languageColorConfig.lcs-ui .lcs-hexrow{ display:flex; align-items:center; gap:6px; flex-wrap:wrap; }
#languageColorConfig.lcs-ui .lcs-hexrow input[type="text"]{
  width:96px; padding:4px 6px; border:1px solid #d1d5db; border-radius:6px;
}

/* Swatches onderaan */
#languageColorConfig.lcs-ui .lcs-swatches{ display:grid; grid-template-columns:repeat(auto-fill, minmax(18px, 1fr)); gap:6px; }
#languageColorConfig.lcs-ui .lcs-swatch-btn{ width:18px; height:18px; border-radius:4px; border:1px solid #aaa; cursor:pointer; }
#languageColorConfig.lcs-ui .lcs-muted{ font-size:12px; color:#6b7280; }
`;
        const style = document.createElement('style');
        style.id = 'lcs-compact-css';
        style.textContent = css;
        document.head.appendChild(style);
    }

    function buildCustomTab(sdk) {
        sdk.Sidebar.registerScriptTab().then(({ tabLabel, tabPane }) => {
            tabLabel.innerHTML = `<img src="" style="height: 1em; vertical-align: middle; margin-right: 4px;">${UI.tab}`;
            tabLabel.title = UI.tabTitle;

            tabPane.id = "language-tab";
            tabPane.innerHTML = `<div id="languageColorConfig" style="padding:10px;"></div>
                         <div style="padding:10px;"><small>${SCRIPT_NAME} v${SCRIPT_VERSION}</small></div>`;

            ensureSidebarStyles();
            buildSidebar(document.getElementById('languageColorConfig'));
        });
    }

    function buildSidebar(container) {
        if (!container) return;
        container.classList.add('lcs-ui');
        container.innerHTML = '';

        const acc = document.createElement('div'); acc.className = 'lcs-accordion';
        container.appendChild(acc);
        const accKey = k => `LC_ACC_${k}`;
        const makeAccordion = (title, key) => {
            const d = document.createElement('details');
            d.open = (localStorage.getItem(accKey(key)) ?? '1') === '1';
            const s = document.createElement('summary');
            s.innerHTML = `<span>${title}</span><span class="lcs-caret">▾</span>`;
            const body = document.createElement('div'); body.className = 'lcs-acc-body';
            d.append(s, body);
            d.addEventListener('toggle', () => localStorage.setItem(accKey(key), d.open ? '1' : '0'));
            acc.appendChild(d);
            return body;
        };

        const langBody = makeAccordion(isDutch ? 'TaalKleur-instellingen' : 'Language color settings', 'LANG');
        const osBody = makeAccordion(isDutch ? 'Besturingssysteem-kleuren' : 'Operating system colors', 'OS');
        const vehBody = makeAccordion(isDutch ? 'Voertuigtype-kleuren' : 'Vehicle type colors', 'VEH');

        const langs = new Set(Object.keys(colorPrefs));
        langs.forEach(lang => {
            if (!lang) return;
            const color = colorPrefs[lang];
            const displayColor = (color && /^#[0-9A-F]{6}$/i.test(color)) ? color : FALLBACK_COLOR_INPUT;

            const row = document.createElement('div'); row.className = 'lcs-section';
            const header = document.createElement('div'); header.className = 'lcs-row';

            const label = document.createElement('label'); label.className='lcs-label'; label.textContent = `${lang}:`;
            const colorInput = document.createElement('input'); colorInput.type='color'; colorInput.value = displayColor; colorInput.dataset.lang = lang;

            const toggleBtn = document.createElement('button'); toggleBtn.textContent='▼'; toggleBtn.title = isDutch?'Toon/Verberg kleurdetails':'Show/Hide color details'; toggleBtn.setAttribute('aria-expanded','false');
            const resetBtn = document.createElement('button'); resetBtn.textContent='⟳'; resetBtn.title = UI.remove;

            const panel = document.createElement('div'); panel.className='lcs-details'; panel.style.display='none';
            const hexRow = document.createElement('div'); hexRow.className='lcs-hexrow';

            const hexInput = document.createElement('input'); hexInput.type='text'; hexInput.placeholder='#rrggbb'; hexInput.value = color ? color : '';
            const copyBtn = document.createElement('button'); copyBtn.textContent='⧉'; copyBtn.title = isDutch?'Kopieer HEX':'Copy HEX';
            const pasteBtn = document.createElement('button'); pasteBtn.textContent='📋'; pasteBtn.title = isDutch?'Plak HEX':'Paste HEX';
            const saveSwatchBtn = document.createElement('button'); saveSwatchBtn.textContent='★'; saveSwatchBtn.title = isDutch?'Kleur opslaan':'Save color';

            const markFocused = () => { lastFocusTarget = { kind: 'LANG', key: lang }; };
            colorInput.addEventListener('focus', markFocused);
            hexInput.addEventListener('focus', markFocused);

            toggleBtn.onclick = () => {
                const open = panel.style.display !== 'none';
                panel.style.display = open ? 'none' : 'block';
                toggleBtn.setAttribute('aria-expanded', String(!open));
                toggleBtn.textContent = open ? '▼' : '▲';
            };

            colorInput.oninput = () => {
                const v = colorInput.value.toLowerCase();
                if (v === FALLBACK_COLOR_INPUT.toLowerCase()) { colorPrefs[lang]=''; hexInput.value=''; }
                else if (/^#[0-9a-f]{6}$/.test(v)) { colorPrefs[lang]=v; hexInput.value=v; }
                localStorage.setItem(LANG_STORAGE_KEY, JSON.stringify(colorPrefs));
                applyLanguageColor(); renderLanguageLabelsAsUROStyle();
            };
            hexInput.addEventListener('change', () => {
                const h = toHex6(hexInput.value);
                if (h) { hexInput.value=h; colorInput.value=h; colorPrefs[lang]=h; }
                else if (hexInput.value.trim()==='') { colorPrefs[lang]=''; }
                else { hexInput.value = colorPrefs[lang] || ''; }
                localStorage.setItem(LANG_STORAGE_KEY, JSON.stringify(colorPrefs));
                applyLanguageColor(); renderLanguageLabelsAsUROStyle();
            });
            copyBtn.onclick = async () => { const h = toHex6(hexInput.value) || toHex6(colorInput.value); if (!h) return; try{ await navigator.clipboard.writeText(h);}catch{} };
            pasteBtn.onclick = async () => {
                try{ const txt = await navigator.clipboard.readText(); const h = toHex6(txt); if(!h) return;
                    hexInput.value=h; colorInput.value=h; colorPrefs[lang]=h; localStorage.setItem(LANG_STORAGE_KEY, JSON.stringify(colorPrefs));
                    applyLanguageColor(); renderLanguageLabelsAsUROStyle();
                   }catch{}
            };
            saveSwatchBtn.onclick = () => { const h = toHex6(hexInput.value) || toHex6(colorInput.value); if(!h) return; addSwatchUnique(h); renderSwatches(); };
            resetBtn.onclick = () => {
                delete colorPrefs[lang]; localStorage.setItem(LANG_STORAGE_KEY, JSON.stringify(colorPrefs));
                const badge = Array.from(document.querySelectorAll('.reporter-preferences wz-badge.reporterLanguage')).find(b=>b.textContent.trim()===lang);
                if (badge?.shadowRoot) badge.shadowRoot.querySelector('.wz-badge')?.removeAttribute('style');
                applyLanguageColor(); renderLanguageLabelsAsUROStyle(); buildSidebar(container);
            };

            header.append(label, colorInput, toggleBtn, resetBtn);
            hexRow.append(hexInput, copyBtn, pasteBtn, saveSwatchBtn);
            panel.append(hexRow);
            row.append(header, panel);
            langBody.appendChild(row);
        });

        (function(){
            const ddWrap = document.createElement('div'); ddWrap.className='lcs-section';
            const dropdown = document.createElement('select');
            dropdown.innerHTML = `<option value="">${UI.addLanguage}</option>`;
            Object.entries(LANGUAGE_MAP).forEach(([_, displayLang]) => {
                const opt = document.createElement('option');
                opt.value = displayLang;
                const isAdded = displayLang in colorPrefs;
                opt.textContent = isAdded ? `${displayLang} ${isDutch ? '(toegevoegd)' : '(added)'}` : displayLang;
                opt.disabled = isAdded; if (isAdded) opt.style.color='#888';
                dropdown.appendChild(opt);
            });
            dropdown.onchange = () => {
                const val = dropdown.value;
                if (val && !(val in colorPrefs)) {
                    colorPrefs[val] = ''; localStorage.setItem(LANG_STORAGE_KEY, JSON.stringify(colorPrefs));
                    applyLanguageColor(); renderLanguageLabelsAsUROStyle(); buildSidebar(container);
                }
            };
            ddWrap.appendChild(dropdown);
            langBody.appendChild(ddWrap);
        })();

        (function(){
            const otherWrap = document.createElement('div'); otherWrap.className='lcs-section';
            const row = document.createElement('div'); row.className='lcs-check';
            const otherToggle = document.createElement('input'); otherToggle.type='checkbox'; otherToggle.checked = colorOtherPrefs.enabled; otherToggle.id='toggleOtherLangColor';
            const otherLabel = document.createElement('label'); otherLabel.setAttribute('for','toggleOtherLangColor'); otherLabel.textContent = isDutch ? 'Overige talen' : 'Other languages';
            const otherColorInput = document.createElement('input'); otherColorInput.type='color'; otherColorInput.value = colorOtherPrefs.color;
            otherToggle.onchange = () => { colorOtherPrefs.enabled = otherToggle.checked; localStorage.setItem(OTHER_PREFS_KEY, JSON.stringify(colorOtherPrefs)); applyLanguageColor(); renderLanguageLabelsAsUROStyle(); };
            otherColorInput.oninput = () => { colorOtherPrefs.color = otherColorInput.value; localStorage.setItem(OTHER_PREFS_KEY, JSON.stringify(colorOtherPrefs)); applyLanguageColor(); renderLanguageLabelsAsUROStyle(); };
            row.append(otherToggle, otherLabel, otherColorInput); otherWrap.append(row); langBody.appendChild(otherWrap);

            const badgeWrap = document.createElement('div'); badgeWrap.className='lcs-section';
            const badgeRow = document.createElement('div'); badgeRow.className='lcs-check';
            const badgeToggle = document.createElement('input'); badgeToggle.type='checkbox'; badgeToggle.checked=showBadgeOnUR; badgeToggle.id='toggleBadgeCircle';
            const badgeLabel = document.createElement('label'); badgeLabel.setAttribute('for','toggleBadgeCircle'); badgeLabel.textContent = isDutch ? 'Kleurcirkel op UR tonen' : 'Show color circle on UR';
            badgeToggle.onchange = () => { showBadgeOnUR = badgeToggle.checked; localStorage.setItem(SHOW_BADGES_KEY, JSON.stringify(showBadgeOnUR)); renderLanguageLabelsAsUROStyle(); };
            badgeRow.append(badgeToggle, badgeLabel, document.createElement('span'));
            badgeWrap.append(badgeRow); langBody.appendChild(badgeWrap);
        })();

        (function buildOSSection() {
            new Set(Object.keys(osColorPrefs)).forEach(osKey => makeRow(osKey, 'OS'));

            const ddWrap = document.createElement('div'); ddWrap.className='lcs-section';
            const dd = document.createElement('select');
            dd.innerHTML = `<option value="">${isDutch ? 'OS toevoegen...' : 'Add OS...'}</option>`;
            OS_KEYS.forEach(key => {
                const label = OS_LABEL[key] || key;
                const isAdded = key in osColorPrefs;
                const opt = document.createElement('option');
                opt.value = key; opt.textContent = isAdded ? `${label} ${isDutch?'(toegevoegd)':'(added)'}` : label;
                opt.disabled = isAdded; if (isAdded) opt.style.color='#888';
                dd.appendChild(opt);
            });
            dd.onchange = () => {
                const key = dd.value;
                if (key && !(key in osColorPrefs)) { osColorPrefs[key] = ''; localStorage.setItem(OS_PREFS_KEY, JSON.stringify(osColorPrefs)); renderLanguageLabelsAsUROStyle(); buildSidebar(container); }
            };
            ddWrap.appendChild(dd); osBody.appendChild(ddWrap);

            const otherWrap = document.createElement('div'); otherWrap.className='lcs-section';
            const row = document.createElement('div'); row.className='lcs-check';
            const toggle = document.createElement('input'); toggle.type='checkbox'; toggle.checked=osOtherPrefs.enabled; toggle.id='toggleOtherOS';
            const label = document.createElement('label'); label.setAttribute('for','toggleOtherOS'); label.textContent = isDutch ? 'Overige OS' : 'Other OS';
            const picker = document.createElement('input'); picker.type='color'; picker.value = osOtherPrefs.color;
            toggle.onchange = () => { osOtherPrefs.enabled = toggle.checked; localStorage.setItem(OS_OTHER_PREFS_KEY, JSON.stringify(osOtherPrefs)); renderLanguageLabelsAsUROStyle(); };
            picker.oninput = () => { osOtherPrefs.color = picker.value; localStorage.setItem(OS_OTHER_PREFS_KEY, JSON.stringify(osOtherPrefs)); renderLanguageLabelsAsUROStyle(); };
            row.append(toggle, label, picker); otherWrap.append(row); osBody.appendChild(otherWrap);

            const bWrap = document.createElement('div'); bWrap.className='lcs-section';
            const bRow = document.createElement('div'); bRow.className='lcs-check';
            const bToggle = document.createElement('input'); bToggle.type='checkbox'; bToggle.checked=showOSBadge; bToggle.id='toggleOSBadge';
            const bLabel = document.createElement('label'); bLabel.setAttribute('for','toggleOSBadge'); bLabel.textContent = isDutch ? 'Kleuricoon voor OS op UR' : 'Show OS icon on UR';
            bToggle.onchange = () => { showOSBadge = bToggle.checked; localStorage.setItem(SHOW_OS_BADGE_KEY, JSON.stringify(showOSBadge)); renderLanguageLabelsAsUROStyle(); };
            bRow.append(bToggle, bLabel, document.createElement('span')); bWrap.append(bRow); osBody.appendChild(bWrap);
        })();

        (function buildVehicleSection() {
            new Set(Object.keys(vehicleColorPrefs)).forEach(vKey => makeRow(vKey, 'VEH'));

            const ddWrap = document.createElement('div'); ddWrap.className='lcs-section';
            const dd = document.createElement('select');
            dd.innerHTML = `<option value="">${isDutch ? 'Voertuigtype toevoegen...' : 'Add vehicle type...'}</option>`;
            VEH_KEYS.forEach(key => {
                const label = vehicleDisplay(key);
                const isAdded = key in vehicleColorPrefs;
                const opt = document.createElement('option');
                opt.value = key; opt.textContent = isAdded ? `${label} ${isDutch?'(toegevoegd)':'(added)'}` : label;
                opt.disabled = isAdded; if (isAdded) opt.style.color='#888';
                dd.appendChild(opt);
            });
            dd.onchange = () => {
                const key = dd.value;
                if (key && !(key in vehicleColorPrefs)) { vehicleColorPrefs[key] = ''; localStorage.setItem(VEH_PREFS_KEY, JSON.stringify(vehicleColorPrefs)); renderLanguageLabelsAsUROStyle(); buildSidebar(container); }
            };
            ddWrap.appendChild(dd); vehBody.appendChild(ddWrap);

            const otherWrap = document.createElement('div'); otherWrap.className='lcs-section';
            const row = document.createElement('div'); row.className='lcs-check';
            const toggle = document.createElement('input'); toggle.type='checkbox'; toggle.checked=vehicleOtherPrefs.enabled; toggle.id='toggleOtherVehicle';
            const label = document.createElement('label'); label.setAttribute('for','toggleOtherVehicle'); label.textContent = isDutch ? 'Overige voertuigtypes' : 'Other vehicle types';
            const picker = document.createElement('input'); picker.type='color'; picker.value = vehicleOtherPrefs.color;
            toggle.onchange = () => { vehicleOtherPrefs.enabled = toggle.checked; localStorage.setItem(VEH_OTHER_PREFS_KEY, JSON.stringify(vehicleOtherPrefs)); renderLanguageLabelsAsUROStyle(); };
            picker.oninput = () => { vehicleOtherPrefs.color = picker.value; localStorage.setItem(VEH_OTHER_PREFS_KEY, JSON.stringify(vehicleOtherPrefs)); renderLanguageLabelsAsUROStyle(); };
            row.append(toggle, label, picker); otherWrap.append(row); vehBody.appendChild(otherWrap);

            const bWrap = document.createElement('div'); bWrap.className='lcs-section';
            const bRow = document.createElement('div'); bRow.className='lcs-check';
            const bToggle = document.createElement('input'); bToggle.type='checkbox'; bToggle.checked=showVehicleBadge; bToggle.id='toggleVehicleBadge';
            const bLabel = document.createElement('label'); bLabel.setAttribute('for','toggleVehicleBadge'); bLabel.textContent = isDutch ? 'Kleuricoon voor voertuig op UR' : 'Show vehicle icon on UR';
            bToggle.onchange = () => { showVehicleBadge = bToggle.checked; localStorage.setItem(SHOW_VEH_BADGE_KEY, JSON.stringify(showVehicleBadge)); renderLanguageLabelsAsUROStyle(); };
            bRow.append(bToggle, bLabel, document.createElement('span')); bWrap.append(bRow); vehBody.appendChild(bWrap);
        })();

        function makeRow(key, kind) {
            const colorDict = kind === 'OS' ? osColorPrefs : vehicleColorPrefs;
            const STORAGE = kind === 'OS' ? OS_PREFS_KEY : VEH_PREFS_KEY;
            const targetBody = (kind === 'OS') ? osBody : vehBody;

            const row = document.createElement('div'); row.className='lcs-section';
            const header = document.createElement('div'); header.className='lcs-row';

            const label = document.createElement('label'); label.className='lcs-label';
            const display = kind === 'OS' ? (OS_LABEL[key] || key) : vehicleDisplay(key);
            label.textContent = `${display}:`;

            const colorInput = document.createElement('input'); colorInput.type='color';
            const current = colorDict[key];
            const displayColor = (current && /^#[0-9A-F]{6}$/i.test(current)) ? current : FALLBACK_COLOR_INPUT;
            colorInput.value = displayColor; colorInput.dataset.key = key;

            const toggleBtn = document.createElement('button'); toggleBtn.textContent='▼'; toggleBtn.title = isDutch?'Toon/Verberg kleurdetails':'Show/Hide color details'; toggleBtn.setAttribute('aria-expanded','false');
            const resetBtn = document.createElement('button'); resetBtn.textContent='⟳'; resetBtn.title = isDutch?'Verwijder item':'Remove item';

            const panel = document.createElement('div'); panel.className='lcs-details'; panel.style.display='none';
            const hexRow = document.createElement('div'); hexRow.className='lcs-hexrow';

            const hexInput = document.createElement('input'); hexInput.type='text'; hexInput.placeholder='#rrggbb'; hexInput.value = current ? current : '';
            const copyBtn = document.createElement('button'); copyBtn.textContent='⧉';
            const pasteBtn = document.createElement('button'); pasteBtn.textContent='📋';
            const saveSwatchBtn = document.createElement('button'); saveSwatchBtn.textContent='★';

            const markFocused = () => { lastFocusTarget = { kind: (kind === 'OS' ? 'OS' : 'VEH'), key }; };
            colorInput.addEventListener('focus', markFocused);
            hexInput.addEventListener('focus', markFocused);

            toggleBtn.onclick = () => {
                const open = panel.style.display !== 'none';
                panel.style.display = open ? 'none' : 'block';
                toggleBtn.setAttribute('aria-expanded', String(!open));
                toggleBtn.textContent = open ? '▼' : '▲';
            };
            colorInput.oninput = () => {
                const v = colorInput.value.toLowerCase();
                if (v === FALLBACK_COLOR_INPUT.toLowerCase()) { colorDict[key]=''; hexInput.value=''; }
                else if (/^#[0-9a-f]{6}$/.test(v)) { colorDict[key]=v; hexInput.value=v; }
                localStorage.setItem(STORAGE, JSON.stringify(colorDict)); renderLanguageLabelsAsUROStyle();
            };
            hexInput.addEventListener('change', () => {
                const h = toHex6(hexInput.value);
                if (h) { hexInput.value=h; colorInput.value=h; colorDict[key]=h; }
                else if (hexInput.value.trim()==='') { colorDict[key]=''; }
                else { hexInput.value = colorDict[key] || ''; }
                localStorage.setItem(STORAGE, JSON.stringify(colorDict)); renderLanguageLabelsAsUROStyle();
            });
            copyBtn.onclick = async () => { const h = toHex6(hexInput.value) || toHex6(colorInput.value); if(!h)return; try{await navigator.clipboard.writeText(h);}catch{} };
            pasteBtn.onclick = async () => { try{ const txt = await navigator.clipboard.readText(); const h = toHex6(txt); if(!h)return; hexInput.value=h; colorInput.value=h; colorDict[key]=h; localStorage.setItem(STORAGE, JSON.stringify(colorDict)); renderLanguageLabelsAsUROStyle(); }catch{} };
            saveSwatchBtn.onclick = () => { const h = toHex6(hexInput.value) || toHex6(colorInput.value); if(!h)return; addSwatchUnique(h); window.renderSwatches?.(); };

            resetBtn.onclick = () => { delete colorDict[key]; localStorage.setItem(STORAGE, JSON.stringify(colorDict)); renderLanguageLabelsAsUROStyle(); buildSidebar(container); };

            header.append(label, colorInput, toggleBtn, resetBtn);
            hexRow.append(hexInput, copyBtn, pasteBtn, saveSwatchBtn);
            panel.append(hexRow);
            row.append(header, panel);
            targetBody.appendChild(row);
        }

        const swWrap = document.createElement('div'); swWrap.className='lcs-section';
        const swTitle = document.createElement('div'); swTitle.className='lcs-muted';
        swTitle.textContent = isDutch ? 'Opgeslagen kleuren:' : 'Saved colors:';
        const swGrid = document.createElement('div'); swGrid.className='lcs-swatches';
        swWrap.append(swTitle, swGrid);
        container.appendChild(swWrap);

        function renderSwatches(){
            swGrid.innerHTML = '';
            const list = getSwatches();
            if(!list.length){
                const empty = document.createElement('div');
                empty.textContent = isDutch ? '(nog geen opgeslagen kleuren)' : '(no saved colors yet)';
                empty.className = 'lcs-muted';
                swGrid.appendChild(empty);
                return;
            }
            list.forEach(hex => {
                const b = document.createElement('button');
                b.className = 'lcs-swatch-btn';
                b.title = `${hex}\n${isDutch ? 'Klik: toepassen • Alt+klik of rechtsklik: verwijderen' : 'Click: apply • Alt+click or right-click: delete'}`;
                b.style.background = hex;
                b.dataset.hex = hex;

                b.addEventListener('click', (e) => {
                    const t = lastFocusTarget; if (!t) return;
                    if (e.altKey) { removeSwatch(hex); renderSwatches(); return; }
                    let dict, storageKey;
                    if (t.kind === 'LANG') { dict = colorPrefs; storageKey = LANG_STORAGE_KEY; }
                    else if (t.kind === 'OS') { dict = osColorPrefs; storageKey = OS_PREFS_KEY; }
                    else { dict = vehicleColorPrefs; storageKey = VEH_PREFS_KEY; }
                    dict[t.key] = hex; localStorage.setItem(storageKey, JSON.stringify(dict));
                    applyLanguageColor(); renderLanguageLabelsAsUROStyle(); buildSidebar(container);
                });
                b.addEventListener('contextmenu', (e) => { e.preventDefault(); removeSwatch(hex); renderSwatches(); });
                swGrid.appendChild(b);
            });
        }
        window.renderSwatches = renderSwatches;
        renderSwatches();
    }

    function waitForSdk() {
        const wait = setInterval(() => {
            if (window.SDK_INITIALIZED && typeof window.getWmeSdk === 'function') {
                clearInterval(wait);
                window.SDK_INITIALIZED.then(() => {
                    const sdk = window.getWmeSdk({ scriptId: 'language_highlighter', scriptName: 'Language Highlighter' });
                    sdk.Events.once({ eventName: 'wme-ready' }).then(() => {
                        buildCustomTab(sdk);
                        observer.observe(document.querySelector('#panel-container'), { childList: true, subtree: true });
                        console.log("✅ Language Color Selector loaded. Thank you for using my script!");
                        checkVersionAndShowChangelog();
                        init();
                    });
                });
            }
        }, 500);
    }
    waitForSdk();

    function checkVersionAndShowChangelog() {
        const lastVersion = localStorage.getItem(VERSION_STORAGE_KEY);
        if (lastVersion !== SCRIPT_VERSION) {
            localStorage.setItem(VERSION_STORAGE_KEY, SCRIPT_VERSION);
            showChangelogPopup(SCRIPT_VERSION);
        }
    }

    function showChangelogPopup(version) {
        const changelog = WHATS_NEW[version];
        if (!changelog) return;

        const box = document.createElement('div');
        box.style = 'position:fixed;top:15px;right:15px;background:#4caf50;color:#fff;padding:12px 16px;border-radius:8px;z-index:99999;max-width:320px;font-family:sans-serif;box-shadow:0 2px 8px rgba(0,0,0,0.3);';

        box.innerHTML = `
      <strong>✅ Language Color Selector updated to v${version}</strong><br>
      <div style="margin-top:6px;white-space:pre-wrap;font-size:90%">${changelog}</div>
    `;

        document.body.appendChild(box);
        setTimeout(() => box.remove(), 15000);
    }

    function tryGetSdk(timeout = 1000) {
        return new Promise(resolve => {
            if (W?.userscripts?.sdk) return resolve(W.userscripts.sdk);
            const timer = setTimeout(() => resolve(null), timeout);
            document.addEventListener("wme-sdk-ready", () => {
                clearTimeout(timer);
                resolve(W.userscripts.sdk);
            }, { once: true });
        });
    }

    async function init() {
        if (!W?.userscripts?.state?.isReady) {
            await new Promise(resolve => {
                document.addEventListener("wme-ready", resolve, { once: true });
            });
        }

        log("🚀 WME klaar, initialisatie gestart...");

        async function tellURs() {
            const urList = Object.values(W.model.mapUpdateRequests?.objects || {});
            log(`📦 Aantal URs zichtbaar op de kaart: ${urList.length}`);
            const results = await Promise.all(urList.map(async ur => {
                const id = ur.attributes?.id ?? ur.id ?? 'onbekend';

                let taal = ur.attributes?.reporterLanguage;
                if (!taal) {
                    try {
                        const upref = W.model.UpdateRequestUserPreferences.get(ur.userID);
                        taal = upref?.attributes?.language;
                    } catch {}
                }

                const osRaw = (ur.attributes?.os || '').toString().toUpperCase();
                const osLabel = OS_LABEL[osRaw] || osRaw || 'Unknown';

                const vehRaw = (ur.attributes?.hasEv ? 'EV' : (ur.attributes?.vehicleType || 'UNKNOWN')).toString().toUpperCase();
                const vehLabel = vehicleDisplay(vehRaw) || vehRaw;

                return `• ID ${id} → taal: ${taal ?? 'onbekend'} | OS: ${osLabel} | voertuig: ${vehLabel}`;
            }));
            log("🧾 UR’s:", results.join("\n"));
        }

        tellURs();
        W.map.events.register("moveend", null, () => setTimeout(tellURs, 300));
    }

    const LABEL_CLASS = 'ur-language-label';
    function removeLanguageLabels() {
        document.querySelectorAll(`.${LABEL_CLASS}, [id^="urLang-"], [id^="urOS-"], [id^="urVehicle-"]`).forEach(e => e.remove());
    }

    function appendColoredIcon({ layer, id, x, y, kind, color, stackIndex = 0 }) {
        const NS = "http://www.w3.org/2000/svg";
        const g = document.createElementNS(NS, "g");
        g.setAttribute("id", id);
        g.setAttribute("class", LABEL_CLASS);

        const BASE_DX = -25;
        const BASE_DY = -20;
        const STEP_Y = 18;

        const dx = BASE_DX;
        const dy = BASE_DY + stackIndex * STEP_Y;

        g.setAttribute("transform", `translate(${x + 3 + dx}, ${y + 3 + dy})`);
        g.style.pointerEvents = "none";

        const bg = document.createElementNS(NS, "circle");
        bg.setAttribute("cx", 0);
        bg.setAttribute("cy", 0);
        bg.setAttribute("r", 8);
        bg.setAttribute("fill", color);
        bg.setAttribute("stroke", "#222");
        bg.setAttribute("stroke-width", "1.2");
        g.appendChild(bg);

        if (kind === "lang") {
            const body = document.createElementNS(NS, "rect");
            body.setAttribute("x", -5.2);
            body.setAttribute("y", -4.2);
            body.setAttribute("width", 10.4);
            body.setAttribute("height", 7.4);
            body.setAttribute("rx", 2);
            body.setAttribute("ry", 2);
            body.setAttribute("fill", "white");
            body.setAttribute("stroke", "#222");
            body.setAttribute("stroke-width", "0.8");
            g.appendChild(body);

            const tail = document.createElementNS(NS, "path");
            tail.setAttribute("d", "M -2.2 3.4 L -5.0 5.8 L -4.4 3.4 Z");
            tail.setAttribute("fill", "white");
            tail.setAttribute("stroke", "#222");
            tail.setAttribute("stroke-width", "0.8");
            g.appendChild(tail);

        } else if (kind === "os") {
            const phone = document.createElementNS(NS, "rect");
            phone.setAttribute("x", -4.2);
            phone.setAttribute("y", -6);
            phone.setAttribute("width", 8.4);
            phone.setAttribute("height", 12);
            phone.setAttribute("rx", 2.2);
            phone.setAttribute("ry", 2.2);
            phone.setAttribute("fill", "white");
            phone.setAttribute("stroke", "#222");
            phone.setAttribute("stroke-width", "0.8");
            g.appendChild(phone);

            const slot = document.createElementNS(NS, "rect");
            slot.setAttribute("x", -1.6);
            slot.setAttribute("y", -4.9);
            slot.setAttribute("width", 3.2);
            slot.setAttribute("height", 1);
            slot.setAttribute("rx", 0.5);
            slot.setAttribute("ry", 0.5);
            slot.setAttribute("fill", "#222");
            g.appendChild(slot);

            const home = document.createElementNS(NS, "circle");
            home.setAttribute("cx", 0);
            home.setAttribute("cy", 7.1);
            home.setAttribute("r", 0.9);
            home.setAttribute("fill", "#222");
            g.appendChild(home);

        } else if (kind === "veh") {
            const car = document.createElementNS(NS, "path");
            car.setAttribute("d", "M -6 3.5 L -6.8 1.2 L -4 -4 H 4 L 6.8 1.2 L 6 3.5 Z");
            car.setAttribute("fill", "white");
            car.setAttribute("stroke", "#222");
            car.setAttribute("stroke-width", "0.8");
            g.appendChild(car);

            const win = document.createElementNS(NS, "rect");
            win.setAttribute("x", -3.2);
            win.setAttribute("y", -3);
            win.setAttribute("width", 6.4);
            win.setAttribute("height", 2.3);
            win.setAttribute("rx", 0.6);
            win.setAttribute("ry", 0.6);
            win.setAttribute("fill", "white");
            win.setAttribute("stroke", "#222");
            win.setAttribute("stroke-width", "0.6");
            g.appendChild(win);

            const hl1 = document.createElementNS(NS, "circle");
            hl1.setAttribute("cx", -4.2);
            hl1.setAttribute("cy", 2.2);
            hl1.setAttribute("r", 0.9);
            hl1.setAttribute("fill", "#222");
            g.appendChild(hl1);

            const hl2 = document.createElementNS(NS, "circle");
            hl2.setAttribute("cx", 4.2);
            hl2.setAttribute("cy", 2.2);
            hl2.setAttribute("r", 0.9);
            hl2.setAttribute("fill", "#222");
            g.appendChild(hl2);
        }

        layer.appendChild(g);
    }

    function renderLanguageLabelsAsUROStyle() {
        log("🎨 [renderLanguageLabelsAsUROStyle] Start");

        removeLanguageLabels();

        const urList = Object.values(W?.model?.mapUpdateRequests?.objects || {});
        if (!urList?.length) {
            log("❌ Geen UR’s gevonden op de kaart.");
            return;
        }
        log(`🔍 ${urList.length} URs in model`);

        let badgeLayer = document.getElementById('LCBadgeLayer');
        if (!badgeLayer) {
            const svgRoot = document.querySelector('svg');
            if (!svgRoot) {
                warn("❌ Geen SVG-root gevonden");
                return;
            }
            badgeLayer = document.createElementNS("http://www.w3.org/2000/svg", "g");
            badgeLayer.setAttribute('id', 'LCBadgeLayer');
            svgRoot.appendChild(badgeLayer);
        }

        urList.forEach(ur => {
            const urId = ur.attributes?.id ?? ur.id;
            if (!urId) {
                warn("⚠️ Geen geldig UR-id gevonden, sla deze over.");
                return;
            }

            let marker;
            try {
                marker = W.userscripts.getMapElementByDataModel?.(ur);
            } catch (e) {
                warn(`⚠️ Fout bij ophalen marker voor UR ${urId}:`, e);
            }
            if (!marker) {
                marker = document.querySelector(`g[id^="OpenLayers_Layer_Vector_"] use[*|href$="/${urId}"]`);
                if (!marker) {
                    warn(`❌ Geen marker gevonden voor UR ${urId} (ook niet via fallback)`);
                    return;
                }
                log(`🔁 Marker gevonden via fallback-DOM voor UR ${urId}`);
            }

            const lon = ur.attributes?.geoJSONGeometry?.coordinates?.[0];
            const lat = ur.attributes?.geoJSONGeometry?.coordinates?.[1];
            if (typeof lon !== "number" || typeof lat !== "number") {
                warn(`⚠️ Geen geldige coordinaten voor UR ${urId}`);
                return;
            }
            const olLonLat = new OpenLayers.LonLat(lon, lat);
            const pixel = W.map.getOLMap().getViewPortPxFromLonLat(
                olLonLat.transform("EPSG:4326", W.map.getOLMap().getProjection())
            );
            const x = pixel.x;
            const y = pixel.y;

            const taal = ur.attributes?.reporterLanguage ?? 'onbekend';
            const mappedLang = LANGUAGE_MAP[taal?.toLowerCase()] || taal;
            const langHex =
                  (mappedLang in colorPrefs && /^#[0-9A-F]{6}$/i.test(colorPrefs[mappedLang]))
            ? colorPrefs[mappedLang]
            : (!(mappedLang in colorPrefs) && colorOtherPrefs.enabled ? colorOtherPrefs.color : null);

            const osKey = (ur.attributes?.os || '').toString().toUpperCase();
            const osHex =
                  (osKey && /^#[0-9A-F]{6}$/i.test(osColorPrefs[osKey])) ? osColorPrefs[osKey]
            : (osKey && osOtherPrefs.enabled ? osOtherPrefs.color : null);

            const isEV = !!ur.attributes?.hasEv;
            const vehKey = isEV ? 'EV' : (ur.attributes?.vehicleType || 'UNKNOWN').toString().toUpperCase();
            const vehHex =
                  (vehKey && /^#[0-9A-F]{6}$/i.test(vehicleColorPrefs[vehKey])) ? vehicleColorPrefs[vehKey]
            : (vehKey && vehicleOtherPrefs.enabled ? vehicleOtherPrefs.color : null);

            const stack = [];
            if (showBadgeOnUR && langHex) stack.push({ id: `urLang-${urId}`, kind: 'lang', color: langHex });
            if (showOSBadge && osHex) stack.push({ id: `urOS-${urId}`, kind: 'os', color: osHex });
            if (showVehicleBadge && vehHex) stack.push({ id: `urVehicle-${urId}`, kind: 'veh', color: vehHex });

            stack.forEach((item, idx) => {
                appendColoredIcon({
                    layer: badgeLayer,
                    id: item.id,
                    x, y,
                    kind: item.kind,
                    color: item.color,
                    stackIndex: idx
                });
            });
        });

        log("✅ [renderLanguageLabelsAsUROStyle] Voltooid");
    }

    function waitForVisibleURs(maxWaitMs = 4000) {
        return new Promise(resolve => {
            const startTime = Date.now();
            const check = () => {
                const urCount = Object.keys(W?.model?.mapUpdateRequests?.objects || {}).length;
                const icons = document.querySelectorAll('g[id^="OpenLayers_Layer_Vector_"] image');
                const visibleURIcons = Array.from(icons).filter(img =>
                                                                img.getAttribute('href')?.startsWith('blob:') || img.getAttribute('xlink:href')?.startsWith('blob:')
                                                               );
                const enoughLoaded = urCount > 0 && visibleURIcons.length >= urCount;
                if (enoughLoaded || Date.now() - startTime > maxWaitMs) resolve();
                else setTimeout(check, 250);
            };
            check();
        });
    }

    function waitForMapAndBindMoveEnd() {
        log("🔁 Controleren op W.map...");
        const check = setInterval(() => {
            if (W?.map?.events?.register && W?.model?.mapUpdateRequests?.objects) {
                clearInterval(check);
                log("✅ W.map gevonden, move-events geregistreerd");

                W.map.events.register("movestart", null, () => {
                    const badgeLayer = document.getElementById('LCBadgeLayer');
                    if (badgeLayer) badgeLayer.style.display = 'none';
                });

                W.map.events.register("moveend", null, async () => {
                    if (LCSmoveEndTimeout) clearTimeout(LCSmoveEndTimeout);
                    LCSmoveEndTimeout = setTimeout(async () => {
                        await waitForVisibleURs();
                        renderLanguageLabelsAsUROStyle();
                        const badgeLayer = document.getElementById('LCBadgeLayer');
                        if (badgeLayer) badgeLayer.style.display = 'block';
                        LCSmoveEndTimeout = null;
                    }, 500);
                });

                renderLanguageLabelsAsUROStyle();
            }
        }, 250);
    }
    waitForMapAndBindMoveEnd();

    LCSmoveEndTimeout = setTimeout(() => {
        renderLanguageLabelsAsUROStyle();
        LCSmoveEndTimeout = null;
    }, 5000);

})();