WME WazeMY

WME script for WazeMY editing moderation

当前为 2023-12-13 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         WME WazeMY
// @namespace    https://greasyfork.org/en/scripts/404584-wazemy
// @version      2023.12.13.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 = `
Cam updates
`;

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.wazeFeature._wmeObject.attributes.name + '</b><br>';
            let address = landmark.attributes.wazeFeature._wmeObject.getAddress();
            try {
                result += address.getHouseNumber() ? (address.getHouseNumber() + ', ') : ''
                result += (address.getStreetName() ? address.getStreetName() : 'No street') + '<br>';
                result += address.getCityName() + ', ';
                result += address.getStateName() + '<br>';
            }
            catch {
                result += 'No address<br>';
            }
            result += '<b>Lock:</b> ' + (landmark.attributes.wazeFeature._wmeObject.getLockRank() + 1);
            showTooltip = true;
        } else if (segment != null) {
            let segmentId = segment.attributes.wazeFeature.id;
            // let primaryStreetId = WazeWrap.Model.getPrimaryStreetId(segmentId);
            let address = segment.attributes.wazeFeature._wmeObject.getAddress();
            result = '<b>';
            result += address.getStreetName();
            result += '</b><br>';
            result += address.getCityName() + ', ' + address.getStateName();
            result += '<br>';
            result += '<b>ID:</b> ' + segmentId + '<br>';
            result += '<b>Lock:</b> ' + (segment.attributes.wazeFeature._wmeObject.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();
    };

})();