WME WazeMY

WME script for WazeMY editing moderation

目前為 2023-04-25 提交的版本,檢視 最新版本

// ==UserScript==
// @name         WME WazeMY
// @namespace    https://greasyfork.org/en/scripts/404584-wazemy
// @version      2023.04.25.01
// @description  WME script for WazeMY editing moderation
// @author       junyianl
// @match        https://beta.waze.com/*
// @match        https://www.waze.com/editor*
// @match        https://www.waze.com/*/editor*
// @exclude      https://www.waze.com/user/editor*
// @require      https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require      https://greasyfork.org/scripts/449165-wme-wazemy-trafcamlist/code/wme-wazemy-trafcamlist.js
// @grant        GM_xmlhttpRequest
// @license      MIT
// ==/UserScript==

// @connect      p3.fgies.com
// @connect      p4.fgies.com
// @connect      t2.fgies.com
// @connect      jalanow.com
// @connect      llm.gov.my

/* global W */
/* global WazeWrap */
/* global $ */
/* global OL */
/* global OpenLayers */

/**
 * All mentions of script names and links in this script is my way of giving
 * credit to where it's due.
 * Without those scripts, I would have no place to start.
 * Big thanks to the original script authors.
 *
 * Huge thanks to the following contributors for cam locations around Malaysia.
 * :: epailxi | dckj | firman_bakti | rickylo103 | kweeheng ::
 */

// var trafficCamsData = [
//     { desc: 'Jalan Sultan Ismail / Jalan Imbi near Berjaya Times Square KL', lat: 3.14369, lon: 101.71245, url: { Jalanow: 'https://p4.fgies.com/kl8/img/K012W.jpg' } },
//     { desc: 'PLS CAM C2 SLIM RIVER IC KM372.6 NB', lat: 3.84943, lon: 101.39765, url: {LLM: 'https://www.llm.gov.my/assets/ajax.vigroot.php?h=PLS|PLS CAM C2 SLIM RIVER IC KM372.6 NB'} },
//     { desc: 'PLUS-Utara near Tapah KM324.4 SB', lat: 4.236409, lon: 101.255623, url: {Jalanow: 'https://p3.fgies.com/bucket-e1e2/E1E2-22.jpg',LLM: 'https://www.llm.gov.my/assets/ajax.vigroot.php?h=PLS|PLS CAM C1 TAPAH KM324.4 SB'} },
//     { desc: 'DUKE CAM 10 SENTUL PASAR KM0.7 NB', lat: 3.19891, lon: 101.69867, url: {Jalanow: 'https://p3.fgies.com/bucket-duke/DUKE-10.jpg',LLM: 'https://www.llm.gov.my/assets/ajax.vigroot.php?h=DUKE'} },
//     { desc: 'PLUS-Utara near Slim River KM373.9 NB', lat: 3.841385, lon: 101.407039, url: {Jalanow: 'https://p3.fgies.com/bucket-e1e2/E1E2-06.jpg',LLM: 'https://www.llm.gov.my/assets/ajax.vigroot.php?h=PLS|PLS CAM C2 SLIM RIVER KM373.9 NB'} },
// ];

const updateMessage = `
Fixed tooltips after latest WME update.
NEW FEATURE!! PP/PA images enlarger. Gets the non-thumbnail version in a new tab window.
`;

var staticUpdateID;

(function () {
    'use strict';

   var settings = {};

    console.log("[WazeMY] Bootstrapping...");

    if (W?.userscripts?.state.isInitialized) {
        onInitialized();
    } else {
        document.addEventListener("wme-initialized", onInitialized, { once:true });
    }
    
    function onInitialized() {
        if (W?.userscripts?.state.isReady) {
            onReady();
        } else {
            document.addEventListener("wme-ready", onReady, { once:true });
        }
    }

    function onReady() {
        console.log("[WazeMY] Initializing...");
        const { tabLabel, tabPane } = W.userscripts.registerSidebarTab("wazemy");
        tabLabel.innerText = "WazeMY";
        tabLabel.title = "WazeMY";

        tabPane.innerHTML = `
            <div>
                <h4>WazeMY</h4>
                <b>${GM_info.script.version}</b>
            </div>
            <br>
            <div id="wazemySettings">
                <h6>Settings</h6>
                <input type="checkbox" id="wazemySettings_tooltip">
                <label for="wazemySettings_tooltip">Map tooltip</label>
                <br>
                <input type="checkbox" id="wazemySettings_trafcam">
                <label for="wazemySettings_trafcam">Traffic cameras</label>
                <br>
                <br>
            </div>
            <div>
                <h6>Shortcuts</h6>
                Ctrl+Alt+C: <i>Copy lat/lon of mouse position to clipboard.</i><br>
            </div>
        `;

        wazemyTooltip_init();
        wazemyTrafcam_init();
        wazemyImgEnlarge_init();

        WazeWrap.Interface.ShowScriptUpdate(
            "WME WazeMY", 
            GM_info.script.version,
            updateMessage,
            "https://greasyfork.org/en/scripts/404584-wazemy",
            null);
        
        new WazeWrap.Interface.Shortcut(
            "WazeMY_latloncopy",
            "Copies lat/lon of mouse position",
            "wazemy",
            "WazeMY",
            "CA+c",
            wazemyCopyLatLon,
            null).add();

        initializeSettings();

        console.log("[WazeMY] Initialized.");
    }

    // function onInitialized2() {
    //     console.log("[WazeMY] Initializing...");
        
    //     // Initialize features of WME WazeMY
    //     wazemyTooltip_init();
    //     wazemyTrafcam_init();

    //     WazeWrap.Interface.ShowScriptUpdate("WME WazeMY", GM_info.script.version,
    //     updateMessage, "https://greasyfork.org/en/scripts/404584-wazemy", null);
        
    //     // Initialize keyboard shortcuts
    //     new WazeWrap.Interface.Shortcut('WazeMY_latloncopy',
    //     'Copies lat/lon of mouse position', 'wazemy', 'WazeMY', 'CA+c',
    //     wazemyCopyLatLon, null).add();
        
    //     // Initialize Koi later because of dependency on the Settings page.
    //     // wazemyKoi_init();

    //     console.log("[WazeMY] Init complete.")
    // }

    /* ******* */
    /* Tooltip */
    /* ******* */
    function wazemyTooltip_init() {
        let $tooltip = $('<div/>');
        $tooltip.attr('id', 'wazemyTooltip');
        $tooltip.css({
            'height': 'auto',
            'width': 'auto',
            'background': 'rgba(0,0,0,0.5)',
            'color': 'white',
            'borderRadius': '5px',
            'padding': '5px',
            'position': 'absolute',
            'top': '0px',
            'left': '0px',
            'visibility': 'hidden',
            'zIndex': '10000'
        })
        $tooltip.appendTo('body');
    }

    function wazemyTooltip_initSettings() {
        setChecked('wazemySettings_tooltip', settings.tooltip);
        if (settings.tooltip) {
            WazeWrap.Events.register('mousemove', null, wazemyTooltip_show);
        }
        $('#wazemySettings_tooltip').change(function () {
            var settingName = $(this)[0].id.substr(15); // strip off the "wazemySettings_" prefix
            settings[settingName] = this.checked;
            saveSettings();
            if (this.checked) {
                WazeWrap.Events.register('mousemove', null, wazemyTooltip_show);
            }
            else {
                $('#wazemyTooltip').css('visibility', 'hidden');
                WazeWrap.Events.unregister('mousemove', null, wazemyTooltip_show);
            }
        });
    }

    function wazemyTooltip_show(e) { // from URO+
        var result = '';
        var showTooltip = false;

        let landmark = W.map.venueLayer.getFeatureBy('renderIntent', 'highlight');
        let segment = W.map.segmentLayer.getFeatureBy('renderIntent', 'highlight');

        if (landmark != null) {
            result = '<b>' + landmark.attributes.repositoryObject.attributes.name + '</b><br>';
            let address = landmark.attributes.repositoryObject.getAddress();
            try {
                result += address.attributes.houseNumber ? (address.attributes.houseNumber + ', ') : ''
                result += (address.attributes.street.name ? address.attributes.street.name : 'No street') + '<br>';
                result += address.attributes.city.attributes.name + ', ';
                result += address.attributes.state.name + '<br>';
            }
            catch {
                result += 'No address<br>';
            }
            result += '<b>Lock:</b> ' + (landmark.attributes.repositoryObject.getLockRank() + 1);
            showTooltip = true;
        } else if (segment != null) {
            let segmentId = segment.attributes.repositoryObject.attributes.id;
            // let primaryStreetId = WazeWrap.Model.getPrimaryStreetId(segmentId);
            let address = segment.attributes.repositoryObject.getAddress();
            result = '<b>';
            result += address.attributes.street.name ? address.attributes.street.name : 'No street';
            result += '</b><br>';
            result += address.attributes.city.attributes.name + ', ' + address.attributes.state.name;
            result += '<br>';
            result += '<b>ID:</b> ' + segmentId + '<br>';
            result += '<b>Lock:</b> ' + (segment.attributes.repositoryObject.getLockRank() + 1);
            showTooltip = true;
        }

        if (showTooltip == true) {
            let tw = $('#wazemyTooltip').width();
            let th = $('#wazemyTooltip').height();
            var tooltipX = e.clientX + window.scrollX + 15;
            var tooltipY = e.clientY + window.scrollY + 15;
            // Handle cases where tooltip is too near the edge.
            if ((tooltipX + tw) > W.map.$map.innerWidth()) {
                tooltipX -= tw + 20; // 20 = scroll bar size
                if (tooltipX < 0) tooltipX = 0;
            }
            if ((tooltipY + th) > W.map.$map.innerHeight()) {
                tooltipY -= th + 20;
                if (tooltipY < 0) tooltipY = 0;
            }
            $('#wazemyTooltip').html(result);
            $('#wazemyTooltip').css({
                'top': tooltipY + 'px',
                'left': tooltipX + 'px',
                'visibility': 'visible'
            });
        } else {
            $('#wazemyTooltip').css('visibility', 'hidden');
        }
    }

    /* *************** */
    /* Traffic cameras */
    /* *************** */
    // Adapted from WME DOT Cameras
    var wazemyTrafcamLayer;

    function wazemyTrafcam_init() {
        if (!OpenLayers.Icon) {
            wazemyTrafcam_installIconClass();
        }
        wazemyTrafcamLayer = new OpenLayers.Layer.Markers("wazemyTrafcamLayer");
        W.map.addLayer(wazemyTrafcamLayer);
        wazemyTrafcam_showIcons();
        wazemyTrafcamLayer.setVisibility(false);
    }

    function wazemyTrafcam_initSettings() {
        setChecked('wazemySettings_trafcam', settings.trafcam);
        if (settings.trafcam) {
            wazemyTrafcamLayer.setVisibility(true);
            // WazeWrap.Events.register('moveend', null, wazemyTrafcam_show);
        }
        $('#wazemySettings_trafcam').change(function () {
            var settingName = $(this)[0].id.substr(15); // strip off the "wazemySettings_" prefix
            settings[settingName] = this.checked;
            saveSettings();
            if (this.checked) {
                wazemyTrafcamLayer.setVisibility(true);
                // WazeWrap.Events.register('moveend', null, wazemyTrafcam_show);
            } else {
                // WazeWrap.Events.unregister('moveend', null, wazemyTrafcam_show);
                wazemyTrafcamLayer.setVisibility(false);
            }
        });
    }

    function wazemyTrafcam_showIcons() {
        trafficCamsData.forEach((e, idx) => {
            wazemyTrafcam_drawCamIcon({
                idx: idx,
                desc: e.desc,
                src: e.url,
                width: 20,
                height: 20,
                lat: e.lat,
                lon: e.lon
            });
        });
    }

    function wazemyTrafcam_drawCamIcon(spec) {
        const camIcon = '';
        let size = new OpenLayers.Size(20, 20);
        let icon = new OpenLayers.Icon(camIcon, size);
        let epsg4326 = new OpenLayers.Projection("EPSG:4326"); // WGS 1984 projection. Malaysia uses EPSG:900913
        let projectTo = W.map.getProjectionObject();
        let lonLat = new OpenLayers.LonLat(spec.lon, spec.lat).transform(epsg4326, projectTo);
        var newMarker = new OpenLayers.Marker(lonLat, icon);
        newMarker.idx = spec.idx;
        newMarker.title = spec.desc;
        newMarker.url = spec.src;
        newMarker.width = spec.width;
        newMarker.height = spec.height;
        newMarker.location = lonLat;
        newMarker.events.register('click', newMarker, wazemyTrafcam_popupCam);
        wazemyTrafcamLayer.addMarker(newMarker);
    }

    function wazemyTrafcam_popupCam(e) {
        // console.log("wazemyTrafcam_popupCam");

        clearInterval(staticUpdateID);
        $("#gmPopupContainerCam").remove();
        $("#gmPopupContainerCam").hide();

        var popupHTML = ([`
            <div id="gmPopupContainerCam" style="margin: 1;text-align: center;padding: 5px;z-index: 1100; position: absolute; color: white; background: rgba(0,0,0,0.5)">
            <table border=0>
                <tr>
                    <td><div id="mycamdivheader" style="min-height: 20px;white-space: pre-wrap;white-space: -moz-pre-wrap; white-space: -pre-wrap;white-space: -o-pre-wrap;word-wrap: break-word;width:380px">${this.title}</div></td>
                    <td align="right"><a href="#close" id="gmCloseCamDlgBtn" title="Close" style="color:red">X</a></td>
                </tr>
                <tr><td colspan=2>Select source:
                    <select id="wazemy_camSource">
                    </select>
                    <div hidden id="mycamid">${this.idx}</div>
                </td></tr>
                <tr><td colspan=2><img style="width:400px" id="staticimage"></td></tr>
                <tr><td colspan=2><div id="mycamstatus"></div></td></tr>
            </table></div>
        `]);
        $("body").append(popupHTML);

        for (var urlsrc in this.url) {
            if (urlsrc == 'LLM') {
                let sp = this.url['LLM'].split('|');
                if (sp.length == 2) {
                    $('#wazemy_camSource').append($('<option>').val(urlsrc).text(urlsrc));
                }
            } else {
                $('#wazemy_camSource').append($('<option>').val(urlsrc).text(urlsrc));
            }
        }

        $("#wazemy_camSource").change(function (e) {
            switch (this.value) {
                case 'Jalanow':
                    wazemyTrafcam_getJalanowImage(trafficCamsData[$('#mycamid').text()]['url']['Jalanow']);
                    break;
                case 'LLM':
                    wazemyTrafcam_getLLMImage(trafficCamsData[$('#mycamid').text()]['url']['LLM']);
                    break;
            }
        });

        // Get image for the first time when popup is displayed.
        switch (Object.keys(this.url)[0]) {
            case 'Jalanow':
                wazemyTrafcam_getJalanowImage(this.url['Jalanow']);
                break;
            case 'LLM':
                wazemyTrafcam_getLLMImage(this.url['LLM']);
                break;
        }

        // Handle cases where popup is too near the edge.
        let tw = $('#gmPopupContainerCam').width();
        let th = $('#gmPopupContainerCam').height() + 200;
        var tooltipX = e.clientX + window.scrollX + 15;
        var tooltipY = e.clientY + window.scrollY + 15;
        if ((tooltipX + tw) > W.map.$map.innerWidth()) {
            tooltipX -= tw + 20; // 20 = scroll bar size
            if (tooltipX < 0) tooltipX = 0;
        }
        if ((tooltipY + th) > W.map.$map.innerHeight()) {
            tooltipY -= th + 20;
            if (tooltipY < 0) tooltipY = 0;
        }

        $("#gmPopupContainerCam").css({ left: tooltipX });
        $("#gmPopupContainerCam").css({ top: tooltipY });

        //Add listener for popup's "Close" button
        $("#gmCloseCamDlgBtn").click(function () {
            clearInterval(staticUpdateID);
            $("#gmPopupContainerCam").remove();
            $("#gmPopupContainerCam").hide();
        });
        wazemyTrafcam_dragElement(document.getElementById("gmPopupContainerCam"));

    }

    function wazemyTrafcam_getJalanowImage(url) {
        GM_xmlhttpRequest({
            method: 'GET',
            responseType: 'blob',
            headers: {
                "authority": "p4.fgies.com",
                "referer": 'https://www.jalanow.com/',
                'accept': 'image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8'
            },
            url: url,
            onload: function (response) {
                document.getElementById('staticimage').src = URL.createObjectURL(response.response);
                document.getElementById('mycamstatus').innerHTML = "";
            },
            onerror: function (response) {
                document.getElementById('mycamstatus').innerHTML = "Error loading image.";
            },
            onprogress: function (response) {
                document.getElementById('mycamstatus').innerHTML = "Loading image...";
            }
        });
    }

    function wazemyTrafcam_getLLMImage(url) {
        let camImg = url.split("|");

        GM_xmlhttpRequest({
            method: 'GET',
            responseType: 'document',
            url: camImg[0],
            onload: function (response) {
                const re = new RegExp('src="data:image\/png;base64, ([A-Za-z0-9/+=]*)" title="' + camImg[1] + '"');
                const m = response.responseText.match(re);
                document.getElementById('staticimage').src = "data:image/png;base64," + m[1];
                document.getElementById('mycamstatus').innerHTML = "";
            },
            onerror: function (response) {
                document.getElementById('mycamstatus').innerHTML = "Error loading image.";
            },
            onprogress: function (response) {
                document.getElementById('mycamstatus').innerHTML = "Loading image...";
            }
        })
    }

    function wazemyTrafcam_installIconClass() {
        OpenLayers.Icon = OpenLayers.Class({
            url: null,
            size: null,
            offset: null,
            calculateOffset: null,
            imageDiv: null,
            px: null,
            initialize: function (a, b, c, d) {
                this.url = a;
                this.size = b || {
                    w: 20,
                    h: 20
                };
                this.offset = c || {
                    x: -(this.size.w / 2),
                    y: -(this.size.h / 2)
                };
                this.calculateOffset = d;
                a = OpenLayers.Util.createUniqueID("OL_Icon_");
                let div = this.imageDiv = OpenLayers.Util.createAlphaImageDiv(a);
                $(div.firstChild).removeClass('olAlphaImg'); // LEAVE THIS LINE TO PREVENT WME-HARDHATS SCRIPT FROM TURNING ALL ICONS INTO HARDHAT WAZERS --MAPOMATIC
            },
            destroy: function () {
                this.erase();
                OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild);
                this.imageDiv.innerHTML = "";
                this.imageDiv = null;
            },
            clone: function () {
                return new OpenLayers.Icon(this.url, this.size, this.offset, this.calculateOffset);
            },
            setSize: function (a) {
                null !== a && (this.size = a);
                this.draw();
            },
            setUrl: function (a) {
                null !== a && (this.url = a);
                this.draw();
            },
            draw: function (a) {
                OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, this.size, this.url, "absolute");
                this.moveTo(a);
                return this.imageDiv;
            },
            erase: function () {
                null !== this.imageDiv && null !== this.imageDiv.parentNode && OpenLayers.Element.remove(this.imageDiv);
            },
            setOpacity: function (a) {
                OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, null, null, null, null, a);
            },
            moveTo: function (a) {
                null !== a && (this.px = a);
                null !== this.imageDiv && (null === this.px ? this.display(!1) : (
                    this.calculateOffset && (this.offset = this.calculateOffset(this.size)),
                    OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, {
                        x: this.px.x + this.offset.x,
                        y: this.px.y + this.offset.y
                    })
                ));
            },
            display: function (a) {
                this.imageDiv.style.display = a ? "" : "none";
            },
            isDrawn: function () {
                return this.imageDiv && this.imageDiv.parentNode && 11 != this.imageDiv.parentNode.nodeType;
            },
            CLASS_NAME: "OpenLayers.Icon"
        });
    }

    // Make the DIV element draggable:
    function wazemyTrafcam_dragElement(elmnt) {
        var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
        if (document.getElementById("mycamdivheader")) {
            // if present, the header is where you move the DIV from:
            document.getElementById("mycamdivheader").onmousedown = dragMouseDown;
        } else {
            // otherwise, move the DIV from anywhere inside the DIV:
            elmnt.onmousedown = dragMouseDown;
        }

        function dragMouseDown(e) {
            e = e || window.event;
            e.preventDefault();
            // get the mouse cursor position at startup:
            pos3 = e.clientX;
            pos4 = e.clientY;
            document.onmouseup = closeDragElement;
            // call a function whenever the cursor moves:
            document.onmousemove = elementDrag;
        }

        function elementDrag(e) {
            e = e || window.event;
            e.preventDefault();
            // calculate the new cursor position:
            pos1 = pos3 - e.clientX;
            pos2 = pos4 - e.clientY;
            pos3 = e.clientX;
            pos4 = e.clientY;
            // set the element's new position:
            elmnt.style.top = (elmnt.offsetTop - pos2) + "px";
            elmnt.style.left = (elmnt.offsetLeft - pos1) + "px";
        }

        function closeDragElement() {
            // stop moving when mouse button is released:
            document.onmouseup = null;
            document.onmousemove = null;
        }
    }

    /* ************ */
    /* Copy lat/lon */
    /* ************ */
    function wazemyCopyLatLon() {
        copyToClipboard($('.wz-map-ol-control-span-mouse-position').text());
    }

    /* ********************* */
    /* PP/PA photos enlarger */
    /* ********************* */ 
    function wazemyImgEnlarge_init() {
        document.body.addEventListener("click", (e) => {
            let venueimg = document.getElementsByClassName("modal-dialog venue-image-dialog");
            if (venueimg.length > 0) {
                let modaldialog = venueimg[0];
                let imglink = modaldialog.getElementsByTagName("img")[0].getAttribute("src");
                let newimglink = imglink.replace("thumbs/thumb700_", "");
                
                let venuename = modaldialog.getElementsByClassName("venue-name")[0];
                let linktext = modaldialog.getElementsByTagName("a");
                if (linktext.length > 0) { // delete old one if exists
                    linktext[0].remove();
                }
                venuename.insertAdjacentHTML("afterend", `<a href='${newimglink}' target="_blank" rel="noopener noreferrer">Enlarge</a>`);
            }
        })
    }

    function initializeSettings() {
        loadSettings();

        wazemyTooltip_initSettings();
        wazemyTrafcam_initSettings();
    }

    function saveSettings() {
        if (localStorage) {
            var localsettings = {
                tooltip: settings.tooltip,
                trafcam: settings.trafcam
            };

            localStorage.setItem('WME_wazemySettings', JSON.stringify(localsettings));
        }
    }

    function loadSettings() {
        var loadedSettings = $.parseJSON(localStorage.getItem("WME_wazemySettings"));
        var defaultSettings = {
            tooltip: false,
        };
        settings = loadedSettings ? loadedSettings : defaultSettings;
        for (var prop in defaultSettings) {
            if (!settings.hasOwnProperty(prop)) {
                settings[prop] = defaultSettings[prop];
            }
        }
    }

    function setChecked(checkboxId, checked) {
        $('#' + checkboxId).prop('checked', checked);
    }

    // utility functions
    var copyToClipboard = function (str) { // from PIE
        var $temp = $('<input>');
        $('body').append($temp);
        $temp.val(str).select();
        document.execCommand('copy');
        $temp.remove();
    };

})();