// ==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();
};
})();