WME Colored Map Comments

Change the color of Map Comment Points based on HEX Color Code.

// ==UserScript==
// @name         WME Colored Map Comments
// @namespace    Dude495
// @version      2025.05.30.02
// @author       Dude495
// @description  Change the color of Map Comment Points based on HEX Color Code.
// @include      /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @license      GNU GPLv3
// @connect      greasyfork.org
// @grant        GM_xmlhttpRequest
// @grant        GM_addElement
// @require      https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require      https://update.greasyfork.org/scripts/509664/WME%20Utils%20-%20Bootstrap.js
/* global W */
/* global WazeWrap */
/* global bootstrap */
// ==/UserScript==

(function main() {
    'use strict';

    let originalPointStyle = null;
    let originalHoverPointStyle = null;
    let originalSelectedPointStyle = null;
    let originalPolygonStyle = null;
    let originalHoverPolygonStyle = null;
    let originalSelectedPolygonStyle = null;

    const updateMessage = 'Improved: MC Points and MC Areas now properly revert to Waze’s default styles when their options are unchecked.<br><br>Fixed:<br>- VCheckbox and slider states are reliably loaded from saved settings.<br>- Removed redundant code and improved maintainability.<br><br>';
    const scriptName = GM_info.script.name;
    const scriptVersion = GM_info.script.version;
    const downloadUrl = 'https://update.greasyfork.org/scripts/380974/WME%20Colored%20Map%20Comments.user.js';
    const forumUrl = 'https://www.waze.com/forum/viewtopic.php?f=819&t=279838';

    function ChangeColor() {
        const defaultRules = W.map.getLayerByUniqueName('mapComments').styleMap.styles.default.rules;
        const highlightRules = W.map.getLayerByUniqueName('mapComments').styleMap.styles.highlight.rules;
        const selectRules = W.map.getLayerByUniqueName('mapComments').styleMap.styles.highlightselected.rules;
        const mcStyle = defaultRules[0];
        const hoverStyle = highlightRules[1];
        const selectedStyle = selectRules[1];

        if (!originalPointStyle && mcStyle && mcStyle.symbolizer.Point) {
            originalPointStyle = { ...mcStyle.symbolizer.Point };
            originalHoverPointStyle = { ...hoverStyle.symbolizer.Point };
            originalSelectedPointStyle = { ...selectedStyle.symbolizer.Point };
        }
        if (!originalPolygonStyle && mcStyle && mcStyle.symbolizer.Polygon) {
            originalPolygonStyle = { ...mcStyle.symbolizer.Polygon };
            originalHoverPolygonStyle = { ...hoverStyle.symbolizer.Polygon };
            originalSelectedPolygonStyle = { ...selectedStyle.symbolizer.Polygon };
        }

        if (mcStyle) {
        // MC Points
            if ($('#MCP').is(':checked')) {
                mcStyle.symbolizer.Point.fillColor = localStorage.getItem('CMC-Point');
                mcStyle.symbolizer.Point.fillOpacity = localStorage.getItem('CMC-MCPOpacity');
                hoverStyle.symbolizer.Point.fillColor = localStorage.getItem('CMC-Point');
                hoverStyle.symbolizer.Point.fillOpacity = localStorage.getItem('CMC-MCPOpacity') - 0.1;
                selectedStyle.symbolizer.Point.fillColor = localStorage.getItem('CMC-Point');
            } else if (originalPointStyle) {
                Object.assign(mcStyle.symbolizer.Point, originalPointStyle);
                Object.assign(hoverStyle.symbolizer.Point, originalHoverPointStyle);
                Object.assign(selectedStyle.symbolizer.Point, originalSelectedPointStyle);
            }

            // MC Areas
            if ($('#MCA').is(':checked')) {
                mcStyle.symbolizer.Polygon.fillColor = localStorage.getItem('CMC-Area');
                mcStyle.symbolizer.Polygon.strokeColor = localStorage.getItem('CMC-Area');
                mcStyle.symbolizer.Polygon.fillOpacity = localStorage.getItem('CMC-MCAOpacity');
                hoverStyle.symbolizer.Polygon.fillColor = localStorage.getItem('CMC-Area');
                hoverStyle.symbolizer.Polygon.strokeColor = localStorage.getItem('CMC-Area');
                hoverStyle.symbolizer.Polygon.fillOpacity = localStorage.getItem('CMC-MCAOpacity') - 0.1;
                selectedStyle.symbolizer.Polygon.fillColor = localStorage.getItem('CMC-Area');
                selectedStyle.symbolizer.Polygon.strokeColor = localStorage.getItem('CMC-Area');
            } else if (originalPolygonStyle) {
                Object.assign(mcStyle.symbolizer.Polygon, originalPolygonStyle);
                Object.assign(hoverStyle.symbolizer.Polygon, originalHoverPolygonStyle);
                Object.assign(selectedStyle.symbolizer.Polygon, originalSelectedPolygonStyle);
            }
            W.map.getLayersByName('mapComments')[0].redraw();
        }
    }
    function Save() {
        localStorage.setItem('CMC-Point', $('#CMC-Point').val());
        localStorage.setItem('CMC-Area', $('#CMC-Area').val());
        localStorage.setItem('CMC-MCPOpacity', $('#MCP-Slider').val() / 100);
        localStorage.setItem('CMC-MCAOpacity', $('#MCA-Slider').val() / 100);
        ChangeColor();
        $('#CMC-colorWheelPoint')[0].value = localStorage.getItem('CMC-Point');
        $('#CMC-colorWheelArea')[0].value = localStorage.getItem('CMC-Area');
        localStorage.setItem('MCP', $('#MCP').is(':checked'));
        localStorage.setItem('MCA', $('#MCA').is(':checked'));
        /* console.log(['CMC Settings Saved:',
                     'CMC-Point: '+localStorage.getItem('CMC-Point'),
                     'CMC-Area: '+localStorage.getItem('CMC-Area'),
                     'CMC-MCPOpacity: '+localStorage.getItem('CMC-MCPOpacity'),
                     'CMC-MCAOpacity: '+localStorage.getItem('CMC-MCAOpacity'),
                    ].join('\n')); */
    }
    function init() {
        const $section = $('<div id="CMC-Panel">');
        $section.html([
            `<h5 align="left" style="color:black">Colored Map Comments v${GM_info.script.version},</h5>`,
            '<input data-toggle="tooltip" type="checkbox" id="MCP" title="Enable Color Highlighting for MC Points">Colored MC Points</input>',
            '<br>',
            '<textarea style="resize: none; width: 85px; height: 25px;" rows="1" cols="10" maxlength="7" id="CMC-Point" data-toggle="tooltip" title="Enter the HEX Color Code for the color you want Map Comments to appear. (Include the #)" maxlength="7"></textarea>',
            `<input style="width: 25px; margin-left: 5px; top: -7px; position: relative;" id="CMC-colorWheelPoint" type="color" value="${localStorage.getItem('CMC-Point')}"></input>`,
            '<div class="rangeslider" title="Set opacity level" style="resize: none; width: 100px;"><input type="range" min="1" max="100" value="10" class="myslider" id="MCP-Slider"></div>',
            '<br>',
            '<input id="MCA" type="checkbox" title="Enable Color Highlighting for MC Area Polygons" data-toggle="tooltip">Colored MC Areas</input><br>',
            '<textarea style="resize: none; width: 85px; height: 25px;" rows="1" cols="10" maxlength="7" id="CMC-Area" data-toggle="tooltip" title="Enter the HEX Color Code for the color you want Map Comment Areas to appear. (Include the #)"></textarea>',
            `<input style="width: 25px; margin-left: 5px; top: -7px; position: relative;" id="CMC-colorWheelArea" type="color" value="${localStorage.getItem('CMC-Area')}">`,
            '<div class="rangeslider" title="Set opacity level" style="resize: none; width: 100px;"><input type="range" min="1" max="100" value="10" id="MCA-Slider" class="myslider"></input></div>',
            '<button class="btn btn-success" style="margin-bottom: 25px;" id="CMC-Save" data-toggle="tooltip" title="This button is only needed if you manually change the HEX Code">Save</button>'
        ].join(' '));

        WazeWrap.Interface.Tab('MC Colors', $section.html(), () => {
            $('#MCP')[0].checked = localStorage.getItem('MCP') === 'true';
            $('#MCA')[0].checked = localStorage.getItem('MCA') === 'true';
            $('#CMC-colorWheelPoint')[0].onchange = function onCmcColorWheelPointChange() {
                $('#CMC-Point')[0].value = $('#CMC-colorWheelPoint')[0].value;
                Save();
            };
            $('#CMC-Point').val(localStorage.getItem('CMC-Point'));
            $('#MCP-Slider').val(localStorage.getItem('CMC-MCPOpacity') * 100);
            $('#MCA-Slider').val(localStorage.getItem('CMC-MCAOpacity') * 100);
            $('#CMC-colorWheelArea')[0].onchange = function onCmcColorWheelAreaChange() {
                $('#CMC-Area')[0].value = $('#CMC-colorWheelArea')[0].value;
                Save();
            };
            $('#MCP-Slider')[0].onchange = function onMcpSliderChange() {
                Save();
            };
            $('#MCA-Slider')[0].onchange = function onMcaSliderChange() {
                Save();
            };
            $('#MCP')[0].onchange = function onMcpChange() {
                Save();
            };
            $('#MCA')[0].onchange = function onMcaChange() {
                Save();
            };
            $('#CMC-Save')[0].onclick = function onCmcSaveClick() {
                Save();
            };
            $('#CMC-Area').val(localStorage.getItem('CMC-Area'));
            ChangeColor();
            $('[data-toggle="tooltip"]').tooltip();
        }, 'MC Colors');
    }

    function sandboxBootstrap() {
        if (WazeWrap?.Ready) {
            bootstrap({
                scriptUpdateMonitor: { downloadUrl }
            });
            init();
            WazeWrap.Interface.ShowScriptUpdate(scriptName, scriptVersion, updateMessage, downloadUrl, forumUrl);
        } else {
            setTimeout(sandboxBootstrap, 250);
        }
    }

    // Start the "sandboxed" code.
    sandboxBootstrap();
    console.log(`${scriptName} initialized.`);
})();