// ==UserScript==
// @name WME Route Speeds (MapOMatic fork)
// @description Shows segment speeds in a route.
// @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @version 2025.10.17.0
// @grant GM_xmlhttpRequest
// @grant unsafeWindow
// @namespace https://greasyfork.org/en/scripts/369630-wme-route-speeds-mapomatic-fork
// @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require https://cdn.jsdelivr.net/npm/@turf/[email protected]/turf.min.js
// @author wlodek76 (forked by MapOMatic)
// @copyright 2014, 2015 wlodek76
// @contributor 2014, 2015 FZ69617
// @connect greasyfork.org
// @connect waze.com
// ==/UserScript==
/* global getWmeSdk, $, jQuery, WazeWrap, turf */
(function () {
"use strict";
//--------------------------------------------------------------------------
// Constants and global variables
const DOWNLOAD_URL = 'https://update.greasyfork.org/scripts/369630/WME%20Route%20Speeds%20%28MapOMatic%20fork%29.user.js';
const SCRIPT_NAME = GM_info.script.name;
const SCRIPT_VERSION = GM_info.script.version.toString();
const SCRIPT_SHORT_NAME = "Route Speeds";
const MARKER_LAYER_NAME = SCRIPT_SHORT_NAME + ": Markers";
const MARKER_A_IMAGE = "";
const MARKER_B_IMAGE = "";
const ROUTE_LAYER_NAME = SCRIPT_SHORT_NAME + ": Routes";
const ROUTE_COLORS = [
'#4d4dcd', // route 1
'#d34f8a', // route 2
'#188984', // route 3
'#cafa27', // route 4
'#ffca3f', // route 5
'#39e440', // route 6
'#a848e2', // route 7
'#cbbf00', // route 8
'#2994f3', // route 9
'#ff3d1e', // route 10
'#b0b7f8', // route 11
'#ffb0ba', // route 12
'#71ded2', // route 13
'#86c211', // route 14
'#ff8500', // route 15
'#00a842', // route 16
'#ecd4ff', // route 17
'#7c00ff', // route 18
'#caeeff', // route 19
'#ffdab8', // route 20
];
function getRouteColor(routeIndex) {
return ROUTE_COLORS[routeIndex % ROUTE_COLORS.length];
}
const KM_PER_MILE = 1.609344;
const INVALID_SPEED_COLOR = '#808080';
const METRIC_SPEED_COLORS = [
'#2e131c', // < 5.5 km/h
'#711422', // < 10.5 km/h
'#af0b26', // < 15.5 km/h
'#e9052a', // < 20.5 km/h
'#ff632a', // < 30.5 km/h
'#ffab20', // < 40.5 km/h
'#ffd60f', // < 50.5 km/h
'#9ce30b', // < 60.5 km/h
'#23bf4c', // < 70.5 km/h
'#32c6c2', // < 80.5 km/h
'#09d7ff', // < 90.5 km/h
'#09a9ff', // < 100.5 km/h
'#1555fe', // < 110.5 km/h
'#5e00e0', // < 120.5 km/h
'#a504cd', // < 130.5 km/h
'#851680', // < 140.5 km/h
'#531947', // >= 140.5 km/h
];
const IMPERIAL_SPEED_COLORS = [
'#2e131c', // < 3.5 mph
'#711422', // < 6.5 mph
'#af0b26', // < 9.5 mph
'#e9052a', // < 12.5 mph
'#ff492a', // < 15.5 mph
'#ff7b28', // < 20.5 mph
'#ffab20', // < 25.5 mph
'#ffd60f', // < 30.5 mph
'#9ce30b', // < 35.5 mph
'#04d02e', // < 40.5 mph
'#2cae60', // < 45.5 mph
'#32c6c2', // < 50.5 mph
'#09d7ff', // < 55.5 mph
'#09a9ff', // < 60.5 mph
'#0d75ff', // < 65.5 mph
'#1b2fff', // < 70.5 mph
'#5e00e0', // < 75.5 mph
'#a504cd', // < 80.5 mph
'#851680', // < 85.5 mph
'#531947', // >= 85.5 mph
];
function getSpeedColor(speed) {
if (speed === 0) return INVALID_SPEED_COLOR;
let speedRounded = Math.round(speed);
if (options.useMiles) {
if (speedRounded <= 15) return IMPERIAL_SPEED_COLORS[Math.ceil(speedRounded / 3) - 1];
else return IMPERIAL_SPEED_COLORS[Math.min(Math.ceil(speedRounded / 5) + 1, IMPERIAL_SPEED_COLORS.length - 1)];
} else {
if (speedRounded <= 20) return METRIC_SPEED_COLORS[Math.ceil(speedRounded / 5) - 1];
else return METRIC_SPEED_COLORS[Math.min(Math.ceil(speedRounded / 10) + 1, METRIC_SPEED_COLORS.length - 1)];
}
}
const WME_LAYERS_TO_MOVE = ["closures", "turn_closure", "closure_nodes"];
const SCRIPT_LAYERS_TO_COVER = ["LT Highlights Layer", "LT Names Layer", "LT Lane Graphics"];
const SAVED_OPTIONS_KEY = "RouteSpeedsOptions";
const options = {
enableScript: true,
showLabels: true,
showSpeeds: true,
useMiles: false,
showRouteText: false,
getAlternatives: true,
maxRoutes: 3,
liveTraffic: true,
routingOrder: true,
useRBS: false,
routeType: 1,
vehicleType: 'PRIVATE',
avoidTolls: false,
avoidFreeways: false,
avoidDifficult: false,
avoidFerries: false,
avoidUnpaved: true,
avoidLongUnpaved: false,
allowUTurns: true,
passes: []
};
let sdk;
let topCountry = {id: null};
let markerMoving = "none";
let startFirstClickRegistered = false;
let mouseMoveHandler;
let pointA = {};
let pointB = {};
let tabStatus = 0;
let jQueryStatus = 0;
let z17_reached = false;
let baseZIndex = -1;
let originalZIndices = [];
let alreadyReportedWMELayer = false;
let twoSegmentsSelected = false;
let routesReceived = [];
let routesShown = [];
let waitingForRoute = false;
let routeSelected = 0;
let routeSelectedLast = -1;
let storedFeatures = [];
//--------------------------------------------------------------------------
// Script startup functions
function onSDKInitialized() {
sdk = getWmeSdk({scriptId: "wme-route-speeds", scriptName: SCRIPT_SHORT_NAME});
if (sdk.State.isReady()) {
onWMEReady();
} else {
log("Waiting for WME...");
sdk.Events.once({ eventName: "wme-ready" }).then(onWMEReady);
}
}
function onWMEReady() {
log("Initializing...");
initializeScript();
log(SCRIPT_VERSION + " loaded.");
startScriptUpdateMonitor();
}
function startScriptUpdateMonitor(tries = 0) {
if (WazeWrap && WazeWrap.Ready) {
log("Checking for script updates...");
try {
let updateMonitor = new WazeWrap.Alerts.ScriptUpdateMonitor(SCRIPT_NAME, SCRIPT_VERSION, DOWNLOAD_URL, GM_xmlhttpRequest);
updateMonitor.start();
} catch (ex) {
error(ex);
}
} else {
if (tries == 0) {
log("Waiting for WazeWrap...");
} else if (tries >= 60) {
warn("WazeWrap loading failed after 60 tries. Script updates will not be detected.");
return;
}
setTimeout(startScriptUpdateMonitor, 500, tries + 1);
}
}
function initializeScript() {
let addon = document.createElement('section');
addon.id = "routespeeds-addon";
addon.innerHTML = '<div id="sidepanel-routespeeds" style="margin: 0px 8px; width: auto;">' +
'<div style="margin-bottom:4px; padding:0px;"><a href="https://greasyfork.org/en/scripts/369630-wme-route-speeds-mapomatic-fork" target="_blank">' +
'<span style="font-weight:bold; text-decoration:underline">WME Route Speeds</span></a><span style="margin-left:6px; color:#888; font-size:11px;">v' + SCRIPT_VERSION + '</span>' +
'</div>' +
'<style>\n' +
'#sidepanel-routespeeds select { margin-left:20px; font-size:12px; height:22px; border:1px solid; border-color:rgb(169, 169, 169); border-radius:4px; border: 1px solid; border-color: rgb(169, 169, 169); -webkit-border-radius:4px; -moz-border-radius:4px; }\n' +
'#sidepanel-routespeeds select, #sidepanel-routespeeds input { margin-top:2px; margin-bottom:2px; width:initial; }\n' +
'#sidepanel-routespeeds input[type="checkbox"] { margin-bottom:0px; }\n' +
'#sidepanel-routespeeds label ~ label, #sidepanel-routespeeds span label { margin-left:20px; }\n' +
'#sidepanel-routespeeds .controls-container { padding:0px; }\n' +
'#sidepanel-routespeeds label { font-weight:normal; }\n' +
'</style>' +
'<div style="float:left; display:inline-block;">' +
'<a id="routespeeds-button-A" onclick="return false;" style="cursor:pointer; width:20px; display:inline-block; vertical-align:middle;" title="Center map on A marker">A:</a>' +
'<input id="sidepanel-routespeeds-a" class="form-control" style="width:165px; padding:6px; margin:0px; display:inline; height:24px" type="text" name=""/>' +
'<br><div style="height: 4px;"></div>' +
'<a id="routespeeds-button-B" onclick="return false;" style="cursor:pointer; width:20px; display:inline-block; vertical-align:middle;" title="Center map on B marker">B:</a>' +
'<input id="sidepanel-routespeeds-b" class="form-control" style="width:165px; padding:6px; margin:0px; display:inline; height:24px" type="text" name=""/>' +
'</div>' +
'<div style="float:right; padding-right:20px; padding-top:6%; ">' +
'<button id=routespeeds-button-reverse class="waze-btn waze-btn-blue waze-btn-smaller" style="padding-left:15px; padding-right:15px;" title="Calculate reverse route" >A ↔ B</button></div>' +
'<div style="clear:both; "></div>' +
'<div id="routespeeds-marker-click-explanation" style="font-size:11px; color:#404040; line-height:1.1; display:none;">Click the A or B marker on the map to move it. Click again to finish moving the marker.</div>' +
'<div style="margin-top:5px;">' +
'<select id=routespeeds-hour>' +
'<option value="now">Now</option>' +
'<option value="0" >00:00</option>' +
'<option value="30" >00:30</option>' +
'<option value="60" >01:00</option>' +
'<option value="90" >01:30</option>' +
'<option value="120">02:00</option>' +
'<option value="150">02:30</option>' +
'<option value="180">03:00</option>' +
'<option value="210">03:30</option>' +
'<option value="240">04:00</option>' +
'<option value="270">04:30</option>' +
'<option value="300">05:00</option>' +
'<option value="330">05:30</option>' +
'<option value="360">06:00</option>' +
'<option value="390">06:30</option>' +
'<option value="420">07:00</option>' +
'<option value="450">07:30</option>' +
'<option value="480">08:00</option>' +
'<option value="510">08:30</option>' +
'<option value="540">09:00</option>' +
'<option value="570">09:30</option>' +
'<option value="600">10:00</option>' +
'<option value="630">10:30</option>' +
'<option value="660">11:00</option>' +
'<option value="690">11:30</option>' +
'<option value="720">12:00</option>' +
'<option value="750">12:30</option>' +
'<option value="780">13:00</option>' +
'<option value="810">13:30</option>' +
'<option value="840">14:00</option>' +
'<option value="870">14:30</option>' +
'<option value="900">15:00</option>' +
'<option value="930">15:30</option>' +
'<option value="960">16:00</option>' +
'<option value="990">16:30</option>' +
'<option value="1020">17:00</option>' +
'<option value="1050">17:30</option>' +
'<option value="1080">18:00</option>' +
'<option value="1110">18:30</option>' +
'<option value="1140">19:00</option>' +
'<option value="1170">19:30</option>' +
'<option value="1200">20:00</option>' +
'<option value="1230">20:30</option>' +
'<option value="1260">21:00</option>' +
'<option value="1290">21:30</option>' +
'<option value="1320">22:00</option>' +
'<option value="1350">22:30</option>' +
'<option value="1380">23:00</option>' +
'<option value="1410">23:30</option>' +
'</select>' +
'<select id=routespeeds-day style="margin-left:5px;" >' +
'<option value="today">Today</option>' +
'<option value="1">Monday</option>' +
'<option value="2">Tuesday</option>' +
'<option value="3">Wednesday</option>' +
'<option value="4">Thursday</option>' +
'<option value="5">Friday</option>' +
'<option value="6">Saturday</option>' +
'<option value="0">Sunday</option>' +
'</select>' +
'</div>' +
'<div style="padding-top:8px; padding-bottom:6px;">' +
'<button id=routespeeds-button-livemap class="waze-btn waze-btn-blue waze-btn-smaller" style="width:100%;">Calculate Route</button>' +
'</div>' +
'<b><div id=routespeeds-error style="color:#FF0000"></div></b>' +
'<div id=routespeeds-routecount></div>' +
'<div id=routespeeds-summaries style="font-size:11px; font-variant-numeric:tabular-nums;"></div>' +
'<div style="margin-bottom:4px;">' +
'<b>Options:</b>' +
'<a id="routespeeds-reset-options-to-livemap-route" onclick="return false;" style="cursor:pointer; float:right; margin-right:20px;" title="Reset routing options to the Livemap Route equivalents">Reset to Livemap Route</a>' +
'</div>' +
getCheckboxHtml('enablescript', 'Enable script') +
getCheckboxHtml('showLabels', 'Show segment labels') +
getCheckboxHtml('livetraffic', 'Use real-time traffic', 'Note: this only seems to affect routes within the last 30-60 minutes, up to Now') +
getCheckboxHtml('showSpeeds', 'Show speed on labels') +
getCheckboxHtml('usemiles', 'Use miles and mph') +
getCheckboxHtml('routetext', 'Show route descriptions') +
'<div>' +
getCheckboxHtml('getalternatives', 'Alternative routes: show', '', { display: 'inline-block' }) +
'<select id=routespeeds-maxroutes style="margin-left:-4px; display:inline-block;" >' +
'<option id=routespeeds-maxroutes value="1">1</option>' +
'<option id=routespeeds-maxroutes value="2">2</option>' +
'<option id=routespeeds-maxroutes value="3">3</option>' +
'<option id=routespeeds-maxroutes value="4">4</option>' +
'<option id=routespeeds-maxroutes value="5">5</option>' +
'<option id=routespeeds-maxroutes value="6">6</option>' +
'<option id=routespeeds-maxroutes value="7">7</option>' +
'<option id=routespeeds-maxroutes value="8">8</option>' +
'<option id=routespeeds-maxroutes value="10">10</option>' +
'<option id=routespeeds-maxroutes value="12">12</option>' +
'<option id=routespeeds-maxroutes value="15">15</option>' +
'<option id=routespeeds-maxroutes value="40">all</option>' +
'</select>' +
'</div>' +
getCheckboxHtml('routingorder', 'Use Routing Order', 'Sorts routes in the same order they would appear in the app or livemap (only works if the server returned more routes than requested)') +
getCheckboxHtml('userbs', 'Use Routing Beta Server (RBS)', '', { display: window.location.hostname.includes('beta') ? 'inline' : 'none' }) +
'<div>' +
'<label class="" style="display:inline-block;">' +
'Route type:<select id=routespeeds-routetype style="margin-left:10px;" >' +
'<option value="1">Fastest</option>' +
'<option value="3">Fastest (no history)</option>' +
'</select>' +
'<br>' +
'Vehicle type:<select id=routespeeds-vehicletype style="margin-left:10px;" >' +
'<option id=routespeeds-vehicletype value="PRIVATE">Private</option>' +
'<option id=routespeeds-vehicletype value="EV">Electric</option>' +
'<option id=routespeeds-vehicletype value="TAXI">Taxi</option>' +
'<option id=routespeeds-vehicletype value="MOTORCYCLE">Motorcycle</option>' +
'</select>' +
'</div>' +
'<table><tbody><tr><td style="vertical-align:top; padding-right:4px;"><b>Avoid:</b></td><td>' +
getCheckboxHtml('avoidtolls', 'Tolls') +
getCheckboxHtml('avoidfreeways', 'Freeways') +
getCheckboxHtml('avoiddifficult', 'Difficult turns') +
getCheckboxHtml('avoidferries', 'Ferries') +
getCheckboxHtml('avoidunpaved', 'Unpaved') +
'<div id="routespeeds-avoidunpaved-span" style="display:inline;">' +
getCheckboxHtml('avoidlongunpaved', 'Long unpaved roads', '', { marginLeft: '10px' }) +
'</div>' +
'</td></tr></tbody></table>' +
'<table style="margin-top:3px;"><tbody><tr><td style="vertical-align:top; padding-right:4px;"><b>Allow:</b></td><td>' +
getCheckboxHtml('allowuturns', 'U-Turns') +
'</td></tr></tbody></table>' +
'<div id="routespeeds-passes-container"></div>' +
'<style>' +
'.routespeedsmarkerA { display:block; width:27px; height:36px; margin-left:-13px; margin-top:-34px; }' +
'.routespeedsmarkerB { display:block; width:27px; height:36px; margin-left:-13px; margin-top:-34px; }' +
//+ '.routespeedsmarkerA { background:url("http://341444cc-a-62cb3a1a-s-sites.googlegroups.com/site/wazeaddons/routespeeds_marker_a.png"); }'
//+ '.routespeedsmarkerB { background:url("http://341444cc-a-62cb3a1a-s-sites.googlegroups.com/site/wazeaddons/routespeeds_marker_b.png"); }'
'.routespeedsmarkerA { background-image:url(); }' +
'.routespeedsmarkerB { background-image:url(); }' +
'.routespeedsmarkerA:hover { cursor:move }' +
'.routespeedsmarkerB:hover { cursor:move }' +
'.routespeeds_summary_classA { visibility:hidden; display:inline-block; color:#000000; margin:2px 0px 2px 0px; padding:2px 6px 2px 4px; border:1px solid #c0c0c0; background:#F8F8F8; border-radius:4px; vertical-align:middle; white-space:nowrap; }' +
'.routespeeds_summary_classB { visibility:hidden; display:inline-block; color:#000000; margin:2px 0px 2px 0px; padding:2px 6px 2px 4px; border:1px solid #c0c0c0; background:#d0fffe; border-radius:4px; vertical-align:middle; white-space:nowrap; }' +
'.routespeeds_summary_classA:hover { cursor:pointer; border:1px solid #808080; xbackground:#a0fffd; }' +
'.routespeeds_summary_classB:hover { cursor:pointer; border:1px solid #808080; xbackground:#a0fffd; }' +
'.routespeeds_header { display:inline-block; width:14px; height:14px; text-align:center; border-radius:2px; margin-right:2px; position:relative; top:2px; }' +
'</style>' +
'</div>';
/* var userTabs = getId('user-info');
var navTabs = getElementsByClassName('nav-tabs', userTabs)[0];
var tabContent = getElementsByClassName('tab-content', userTabs)[0];
newtab = document.createElement('li');
newtab.innerHTML = '<a id=sidepanel-routespeeds href="#sidepanel-routespeeds" data-toggle="tab" style="" >Route Speeds</a>';
navTabs.appendChild(newtab);
addon.id = "sidepanel-routespeeds";
addon.className = "tab-pane";
tabContent.appendChild(addon); */
$('head').append([
'<style>',
'label[for^="routespeeds-"] { margin-right: 10px;padding-left: 19px; }',
'.hidden { display:none; }',
'</style>'
].join('\n'));
sdk.Sidebar.registerScriptTab().then((tab) => {
tab.tabLabel.innerHTML = '<span id="routespeeds-tab-label">' + SCRIPT_SHORT_NAME + '</span>';
tab.tabPane.innerHTML = addon.innerHTML;
onTabCreated();
});
window.addEventListener("beforeunload", saveRouteSpeedsOptions, true);
}
function getCheckboxHtml(idSuffix, text, title, divCss = {}, labelCss = {}) {
let id = 'routespeeds-' + idSuffix;
return $('<div>', { class: 'controls-container' }).append(
$('<input>', { id: id, type: 'checkbox' }),
$('<label>', { for: id, title: title }).text(text).css(labelCss)
).css(divCss)[0].outerHTML;
}
function saveRouteSpeedsOptions() {
localStorage.setItem(SAVED_OPTIONS_KEY, JSON.stringify(options));
}
function resetOptions() {
getId('routespeeds-getalternatives').checked = options.getAlternatives = true;
getId('routespeeds-maxroutes').value = options.maxRoutes = 3;
getId('routespeeds-livetraffic').checked = options.liveTraffic = false;
getId('routespeeds-routetype').value = options.routeType = 1;
getId('routespeeds-avoidtolls').checked = options.avoidTolls = false;
getId('routespeeds-avoidfreeways').checked = options.avoidFreeways = false;
getId('routespeeds-avoidunpaved').checked = options.avoidUnpaved = true;
getId('routespeeds-avoidlongunpaved').checked = options.avoidLongUnpaved = false;
getId('routespeeds-allowuturns').checked = options.allowUTurns = true;
getId('routespeeds-routingorder').checked = options.routingOrder = true;
getId('routespeeds-userbs').checked = options.useRBS = false;
getId('routespeeds-avoiddifficult').checked = options.avoidDifficult = false;
getId('routespeeds-avoidferries').checked = options.avoidFerries = false;
getId('routespeeds-vehicletype').value = options.vehicleType = 'PRIVATE';
}
function loadRouteSpeedsOptions() {
try {
Object.assign(options, JSON.parse(localStorage.getItem(SAVED_OPTIONS_KEY)));
} catch {
warn("Saved options could not be loaded. Using defaults.");
}
getId('routespeeds-enablescript').checked = options.enableScript;
getId('routespeeds-showLabels').checked = options.showLabels;
getId('routespeeds-showSpeeds').checked = options.showSpeeds;
getId('routespeeds-usemiles').checked = options.useMiles;
getId('routespeeds-routetext').checked = options.showRouteText;
getId('routespeeds-getalternatives').checked = options.getAlternatives;
getId('routespeeds-maxroutes').value = options.maxRoutes;
getId('routespeeds-livetraffic').checked = options.liveTraffic;
getId('routespeeds-avoidtolls').checked = options.avoidTolls;
getId('routespeeds-avoidfreeways').checked = options.avoidFreeways;
getId('routespeeds-avoidunpaved').checked = options.avoidUnpaved;
getId('routespeeds-avoidlongunpaved').checked = options.avoidLongUnpaved;
getId('routespeeds-routetype').value = options.routeType;
getId('routespeeds-allowuturns').checked = options.allowUTurns;
getId('routespeeds-routingorder').checked = options.routingOrder;
getId('routespeeds-userbs').checked = options.useRBS;
getId('routespeeds-avoiddifficult').checked = options.avoidDifficult;
getId('routespeeds-avoidferries').checked = options.avoidFerries;
getId('routespeeds-vehicletype').value = options.vehicleType;
}
function onTabCreated() {
resetOptions();
loadRouteSpeedsOptions();
if (!options.enableScript) getId('sidepanel-routespeeds').style.color = "#A0A0A0";
else getId('sidepanel-routespeeds').style.color = "";
getId('routespeeds-enablescript').onclick = clickEnableScript;
getId('routespeeds-showLabels').onclick = clickShowLabels;
getId('routespeeds-showSpeeds').onclick = clickShowSpeeds;
getId('routespeeds-usemiles').onclick = clickUseMiles;
getId('routespeeds-routetext').onclick = clickShowRouteText;
getId('routespeeds-getalternatives').onclick = clickGetAlternatives;
getId('routespeeds-maxroutes').onchange = clickMaxRoutes;
getId('routespeeds-livetraffic').onclick = clickLiveTraffic;
getId('routespeeds-avoidtolls').onclick = clickAvoidTolls;
getId('routespeeds-avoidfreeways').onclick = clickAvoidFreeways;
getId('routespeeds-avoidunpaved').onclick = clickAvoidUnpaved;
getId('routespeeds-avoidlongunpaved').onclick = clickAvoidLongUnpaved;
getId('routespeeds-routetype').onchange = clickRouteType;
getId('routespeeds-allowuturns').onclick = clickAllowUTurns;
getId('routespeeds-routingorder').onclick = clickRoutingOrder;
getId('routespeeds-userbs').onclick = clickUseRBS;
getId('routespeeds-avoiddifficult').onclick = clickAvoidDifficult;
getId('routespeeds-avoidferries').onclick = clickAvoidFerries;
getId('routespeeds-vehicletype').onchange = clickVehicleType;
getId('sidepanel-routespeeds-a').onkeydown = enterAB;
getId('sidepanel-routespeeds-b').onkeydown = enterAB;
getId('routespeeds-button-livemap').onclick = livemapRouteClick;
getId('routespeeds-button-reverse').onclick = clickReverseRoute;
getId('routespeeds-reset-options-to-livemap-route').onclick = resetOptionsToLivemapRouteClick;
getId('routespeeds-hour').onchange = hourChange;
getId('routespeeds-day').onchange = dayChange;
getId('routespeeds-button-A').onclick = clickA;
getId('routespeeds-button-B').onclick = clickB;
updateTopCountry();
sdk.Events.on({
eventName: "wme-map-data-loaded",
eventHandler: onMapDataLoaded
});
sdk.Map.addLayer({
layerName: MARKER_LAYER_NAME,
styleRules: [
{
style: {
graphicWidth: 27,
graphicHeight: 36,
graphicXOffset: -13.5,
graphicYOffset: -33.5,
graphicOpacity: 1,
cursor: "pointer"
}
},
{
predicate: (featureProperties) => featureProperties.A,
style: {
externalGraphic: MARKER_A_IMAGE
},
},
{
predicate: (featureProperties) => !featureProperties.A,
style: {
externalGraphic: MARKER_B_IMAGE
},
},
],
});
sdk.Events.trackLayerEvents({layerName: MARKER_LAYER_NAME});
sdk.Events.on({
eventName: "wme-layer-feature-mouse-enter",
eventHandler: mouseEnterHandler
});
sdk.Events.on({
eventName: "wme-layer-feature-mouse-leave",
eventHandler: mouseLeaveHandler
});
sdk.Map.addLayer({
layerName: ROUTE_LAYER_NAME,
styleRules: [{
style: {
strokeWidth: "${getStrokeWidth}",
strokeColor: "${getStrokeColor}",
pointRadius: 0,
label: "${getLabelText}",
fontSize: "10px",
fontFamily: "Tahoma, Courier New",
fontWeight: "${getFontWeight}",
fontColor: "${getFontColor}",
labelOutlineWidth: 2,
labelOutlineColor: "#404040"
}
}],
styleContext: {
getStrokeWidth: ({feature}) => feature.properties.strokeWidth,
getStrokeColor: ({feature}) => feature.properties.strokeColor,
getLabelText: ({feature}) => feature.properties.labelText,
getFontWeight: ({feature}) => feature.properties.fontWeight,
getFontColor: ({feature}) => feature.properties.fontColor
}
});
sdk.Events.on({
eventName: "wme-map-move-end",
eventHandler: onMapMoveEnd
});
window.setInterval(loopWMERouteSpeeds, 500);
}
function updateTopCountry() {
let newTopCountry = sdk.DataModel.Countries.getTopCountry();
if (newTopCountry && newTopCountry.id != topCountry.id) {
topCountry = newTopCountry;
buildPassesDiv();
}
}
function buildPassesDiv() {
$('#routespeeds-passes-container').empty();
if (topCountry.restrictionSubscriptions.length == 0) return;
$('#routespeeds-passes-container').append(
'<fieldset style="border:1px solid silver;padding:8px;border-radius:4px;-webkit-padding-before: 0;">' +
' <legend id="routespeeds-passes-legend" style="margin-bottom:0px;border-bottom-style:none;width:auto;">' +
' <i class="fa fa-fw fa-chevron-down" style="cursor: pointer;font-size: 12px;margin-right: 4px"></i>' +
' <span id="routespeeds-passes-label" style="font-size:14px;font-weight:600; cursor: pointer">Passes & Permits</span>' +
' </legend>' +
' <div id="routespeeds-passes-internal-container" style="padding-top:0px;">' +
topCountry.restrictionSubscriptions.map((pass, i) => {
//let id = 'routespeeds-pass-' + pass.key;
return ' <div class="controls-container" style="padding-top:2px;display:block;">' +
' <input id="routespeeds-pass-' + i + '" type="checkbox" class="routespeeds-pass-checkbox" data-pass-key = "' + pass.id + '">' +
' <label for="routespeeds-pass-' + i + '" style="white-space:pre-line">' + pass.name + '</label>' +
' </div>';
}).join(' ') +
' </div>' +
'</fieldset>'
);
$('.routespeeds-pass-checkbox').click(clickPassOption);
$('#routespeeds-passes-legend').click(function () {
let $this = $(this);
let $chevron = $($this.children()[0]);
$chevron
.toggleClass('fa fa-fw fa-chevron-down')
.toggleClass('fa fa-fw fa-chevron-right');
let collapse = $chevron.hasClass('fa-chevron-right');
let checkboxDivs = $('input.routespeeds-pass-checkbox:not(:checked)').parent();
if (collapse) {
checkboxDivs.css('display', 'none');
} else {
checkboxDivs.css('display', 'block');
}
// $($this.children()[0])
// .toggleClass('fa fa-fw fa-chevron-down')
// .toggleClass('fa fa-fw fa-chevron-right');
// $($this.siblings()[0]).toggleClass('collapse');
})
$('.routespeeds-pass-checkbox').each((i, elem) => {
const $elem = $(elem);
const passKey = $elem.data('pass-key');
$elem.prop('checked', options.passes.includes(passKey));
});
updatePassesLabel();
}
function updatePassesLabel() {
let count = topCountry.restrictionSubscriptions.filter(pass => options.passes.indexOf(pass.id) > -1).length;
$('#routespeeds-passes-label').text(`Passes & Permits (${count} of ${topCountry.restrictionSubscriptions.length})`);
}
//--------------------------------------------------------------------------
// Main loop function
function loopWMERouteSpeeds() {
if (!options.enableScript) return;
let tabOpen = $('#user-tabs #routespeeds-tab-label').parent().parent().attr('aria-expanded') == "true";
if (tabOpen) {
if (tabStatus !== 2) {
tabStatus = 2;
sdk.Map.setLayerVisibility({layerName: MARKER_LAYER_NAME, visibility: true});
sdk.Map.setLayerVisibility({layerName: ROUTE_LAYER_NAME, visibility: true});
reorderLayers(1);
}
} else {
if (tabStatus !== 1) {
tabStatus = 1;
sdk.Map.setLayerVisibility({layerName: MARKER_LAYER_NAME, visibility: false});
sdk.Map.setLayerVisibility({layerName: ROUTE_LAYER_NAME, visibility: false});
reorderLayers(0);
}
return;
}
if (jQueryStatus === 1) {
jQueryStatus = 2;
log('jQuery reloaded ver. ' + jQuery.fn.jquery);
}
if (jQueryStatus === 0) {
if (typeof jQuery === 'undefined') {
log('jQuery current ver. ' + jQuery.fn.jquery);
let script = document.createElement('script');
script.type = "text/javascript";
script.src = "https://ajax.googleapis.com/ajax/libs/jquery/1.9.1/jquery.min.js";
//script.src = "https://ajax.googleapis.com/ajax/libs/jquery/1.11.0/jquery.min.js";
document.getElementsByTagName('head')[0].appendChild(script);
jQueryStatus = 1;
}
}
let selection = sdk.Editing.getSelection();
let selectedIDs = [];
if (selection !== null && selection.objectType == "segment") selectedIDs = selection.ids;
if (selectedIDs.length >= 2) {
if (!twoSegmentsSelected) {
twoSegmentsSelected = true;
let midpointA = getSegmentMidpoint(selectedIDs[0]);
let midpointB = getSegmentMidpoint(selectedIDs[selectedIDs.length - 1]);
if (getId('sidepanel-routespeeds-a') !== undefined) {
getId('sidepanel-routespeeds-a').value = midpointA[0].toFixed(6) + ", " + midpointA[1].toFixed(6);
getId('sidepanel-routespeeds-b').value = midpointB[0].toFixed(6) + ", " + midpointB[1].toFixed(6);
}
createMarkers(midpointA[0], midpointA[1], midpointB[0], midpointB[1]);
requestRouteFromLiveMap(true);
}
} else if (selectedIDs.length == 1) {
if (twoSegmentsSelected) {
twoSegmentsSelected = false;
sdk.Map.removeAllFeaturesFromLayer({layerName: ROUTE_LAYER_NAME});
getId('routespeeds-summaries').style.visibility = 'hidden';
}
}
if (!z17_reached) {
if (sdk.Map.getZoomLevel() >= 17) {
z17_reached = true;
switchRoute();
}
}
}
//--------------------------------------------------------------------------
// Routing and helper functions
function log(msg) {
console.log(SCRIPT_SHORT_NAME + ":", msg);
};
function warn(msg) {
console.warn(SCRIPT_SHORT_NAME + ":", msg);
};
function error(msg) {
console.error(SCRIPT_SHORT_NAME + ":", msg);
};
function getRoutingManager() {
let region = sdk.Settings.getRegionCode();
if (region == "usa") {
return 'https://routing-livemap-am.waze.com/RoutingManager/routingRequest';
} else if (region == "il") {
return 'https://routing-livemap-il.waze.com/RoutingManager/routingRequest';
} else {
return 'https://routing-livemap-row.waze.com/RoutingManager/routingRequest';
}
}
function getSegmentMidpoint(id) {
let geometry = sdk.DataModel.Segments.getById({segmentId: id}).geometry;
return turf.along(geometry, turf.length(geometry) / 2).geometry.coordinates;
}
function getLabelTime(segmentInfo) {
let time = 0;
if (options.liveTraffic) time += segmentInfo.crossTime;
else time += segmentInfo.crossTimeWithoutRealTime;
return time;
}
function getLabelWeight(segmentInfo) {
if (options.liveTraffic && segmentInfo.crossTime != segmentInfo.crossTimeWithoutRealTime) return 'bold';
else return 'normal';
}
function getLabelColor(segmentInfo) {
if (options.liveTraffic) {
if (segmentInfo.crossTime != segmentInfo.crossTimeWithoutRealTime) {
let ratio = segmentInfo.crossTime / segmentInfo.crossTimeWithoutRealTime;
if (ratio > 2) return '#ff0000';
if (ratio > 1.25) return '#ff9900';
if (ratio >= 0.8) return '#ffff00';
if (ratio >= 0.5) return '#99ee00';
else return '#00bb33';
} else {
return '#f8f8f8';
}
} else {
return '#f8f8f8';
}
}
function getSpeed(length_m, time_s) {
if (time_s == 0) return 0;
if (options.useMiles) return 3.6 * length_m / (time_s * KM_PER_MILE);
else return 3.6 * length_m / time_s;
}
function getTimeText(time_s) {
let seconds = time_s % 60;
let minutes = Math.floor((time_s % 3600) / 60);
let hours = Math.floor(time_s / 3600);
return String(hours).padStart(2, '0') + ':' + String(minutes).padStart(2, '0') + ':' + String(seconds).padStart(2, '0');
}
function createMarkers(lon1, lat1, lon2, lat2) {
sdk.Map.removeAllFeaturesFromLayer({layerName: MARKER_LAYER_NAME});
placeMarker("A", lon1, lat1);
placeMarker("B", lon2, lat2);
sdk.Map.setLayerVisibility({layerName: MARKER_LAYER_NAME, visibility: true});
getId("routespeeds-marker-click-explanation").style.display = "block";
}
function placeMarker(id, lon, lat) {
if (id == "A") {
pointA.lon = lon;
pointA.lat = lat;
} else {
pointB.lon = lon;
pointB.lat = lat;
}
sdk.Map.addFeatureToLayer({
layerName: MARKER_LAYER_NAME,
feature: {
id: id,
type: "Feature",
geometry: {
type: "Point",
coordinates: [lon, lat],
},
properties: {
A: id == "A",
},
}
});
}
function moveMarker(id, lon, lat) {
sdk.Map.removeFeatureFromLayer({layerName: MARKER_LAYER_NAME, featureId: id});
placeMarker(id, lon, lat);
}
function routeRevisitsAnyNode(routeIndex) {
let nodes = {};
for (let i = 0; i < routesShown[routeIndex].response.results.length; i++) {
let nodeIDString = routesShown[routeIndex].response.results[i].path.nodeId.toString();
if (nodes[nodeIDString] === undefined) {
nodes[nodeIDString] = 1;
} else {
return true;
}
}
return false;
}
function getRouteFeature(routeIndex, segmentIndex, geometry, mode) {
let feature = {};
if (mode == "label") {
feature = turf.along(geometry, turf.length(geometry) / 2);
let labelText;
if (options.showSpeeds) {
let speed = getSpeed(routesShown[routeIndex].response.results[segmentIndex].length, getLabelTime(routesShown[routeIndex].response.results[segmentIndex]));
if (speed >= 1) labelText = Math.round(speed);
else if (speed == 0) labelText = '?';
else labelText = '<1';
} else {
labelText = getLabelTime(routesShown[routeIndex].response.results[segmentIndex]) + 's';
}
feature.properties = {
labelText: labelText,
fontWeight: getLabelWeight(routesShown[routeIndex].response.results[segmentIndex]),
fontColor: getLabelColor(routesShown[routeIndex].response.results[segmentIndex])
};
} else {
feature.type = "Feature";
feature.geometry = geometry;
feature.properties = {labelText: ""};
if (mode == "simple") {
feature.properties.strokeWidth = 5;
feature.properties.strokeColor = getRouteColor(routeIndex);
} else if (mode == "outline") {
feature.properties.strokeWidth = 12;
feature.properties.strokeColor = "#404040";
} else {
feature.properties.strokeWidth = 10;
feature.properties.strokeColor = getSpeedColor(getSpeed(routesShown[routeIndex].response.results[segmentIndex].length,
getLabelTime(routesShown[routeIndex].response.results[segmentIndex])));
}
}
feature.id = "route " + routeIndex + ", segment " + segmentIndex + ": " + mode;
return feature;
}
function splitGeometryIntoSegments(routeIndex) {
let offset = 0;
if (routeRevisitsAnyNode(routeIndex)) {
offset = topCountry.isLeftHandTraffic ? -0.01 : 0.01;
}
let geometries = [];
let currentSegmentIndex = 0;
let currentSegmentCoords = [[routesShown[routeIndex].coords[0].x, routesShown[routeIndex].coords[0].y]];
let startNextSegment = [0, 0];
if (routesShown[routeIndex].response.results.length > 1) {
startNextSegment = [routesShown[routeIndex].response.results[1].path.x, routesShown[routeIndex].response.results[1].path.y];
}
for (let i = 1; i < routesShown[routeIndex].coords.length; i++) {
let currentPoint = [routesShown[routeIndex].coords[i].x, routesShown[routeIndex].coords[i].y];
currentSegmentCoords.push(currentPoint);
if (Math.abs(currentPoint[0] - startNextSegment[0]) < 10 ** -8 && Math.abs(currentPoint[1] - startNextSegment[1]) < 10 ** -8) {
let currentGeometry = {
type: "LineString",
coordinates: currentSegmentCoords
};
if (offset) {
currentGeometry = turf.lineOffset(currentGeometry, offset).geometry;
}
geometries.push(currentGeometry);
currentSegmentIndex++;
currentSegmentCoords = [];
if (currentSegmentIndex + 1 < routesShown[routeIndex].response.results.length) {
startNextSegment = [routesShown[routeIndex].response.results[currentSegmentIndex + 1].path.x, routesShown[routeIndex].response.results[currentSegmentIndex + 1].path.y];
} else {
startNextSegment = [0, 0];
}
}
}
let lastGeometry = {
type: "LineString",
coordinates: currentSegmentCoords
};
if (offset) {
lastGeometry = turf.lineOffset(lastGeometry, offset).geometry;
}
geometries.push(lastGeometry);
if (offset) {
cleanOffsetGeometries(geometries);
}
return geometries;
}
function cleanOffsetGeometries(geometries) {
if (geometries.length < 2) return;
for (let i = 1; i < geometries.length; i++) {
let intersections = turf.lineIntersect(geometries[i - 1], geometries[i]).features;
for (let j = 0; j < intersections.length; j++) {
if (turf.distance(intersections[j], getFirstPoint(geometries[i])) < 0.0201) {
geometries[i - 1] = turf.cleanCoords(turf.lineSlice(getFirstPoint(geometries[i - 1]), intersections[j].geometry, geometries[i - 1]).geometry);
geometries[i] = turf.cleanCoords(turf.lineSlice(intersections[j].geometry, getLastPoint(geometries[i]), geometries[i]).geometry);
break;
}
}
if (!turf.booleanEqual(getLastPoint(geometries[i - 1]), getFirstPoint(geometries[i]))) {
geometries[i - 1].coordinates.push(getFirstPoint(geometries[i]).coordinates);
}
}
}
function getFirstPoint(geometry) {
return {
type: "Point",
coordinates: geometry.coordinates[0]
};
}
function getLastPoint(geometry) {
return {
type: "Point",
coordinates: geometry.coordinates[geometry.coordinates.length - 1]
};
}
function createRouteFeatures(routeIndex) {
let geometries = splitGeometryIntoSegments(routeIndex);
if (routeSelected == routeIndex || routeSelected == -1) {
for (let i = 0; i < geometries.length; i++) {
storedFeatures.push(getRouteFeature(routeIndex, i, geometries[i], "outline"));
}
for (let i = 0; i < geometries.length; i++) {
storedFeatures.push(getRouteFeature(routeIndex, i, geometries[i], "main"));
}
if (options.showLabels) {
for (let i = 0; i < geometries.length; i++) {
storedFeatures.push(getRouteFeature(routeIndex, i, geometries[i], "label"));
}
}
} else {
for (let i = 0; i < geometries.length; i++) {
storedFeatures.push(getRouteFeature(routeIndex, i, geometries[i], "simple"));
}
}
}
function getElementsByClassName(classname, node) {
if (!node) node = document.getElementsByTagName("body")[0];
var a = [];
var re = new RegExp('\\b' + classname + '\\b');
var els = node.getElementsByTagName("*");
for (var i = 0, j = els.length; i < j; i++) {
if (re.test(els[i].className)) a.push(els[i]);
}
return a;
}
function getId(node) {
return document.getElementById(node);
}
function getnowtoday() {
let hour = getId('routespeeds-hour').value;
let day = getId('routespeeds-day').value;
if (hour === '---') hour = 'now';
if (day === '---') day = 'today';
if (hour === '') hour = 'now';
if (day === '') day = 'today';
let t = new Date();
let thour = (t.getHours() * 60) + t.getMinutes();
let tnow = (t.getDay() * 1440) + thour;
let tsel = tnow;
if (hour === 'now') {
if (day === "0") tsel = (parseInt(day) * 1440) + thour;
if (day === "1") tsel = (parseInt(day) * 1440) + thour;
if (day === "2") tsel = (parseInt(day) * 1440) + thour;
if (day === "3") tsel = (parseInt(day) * 1440) + thour;
if (day === "4") tsel = (parseInt(day) * 1440) + thour;
if (day === "5") tsel = (parseInt(day) * 1440) + thour;
if (day === "6") tsel = (parseInt(day) * 1440) + thour;
}
else {
if (day === "today") tsel = (t.getDay() * 1440) + parseInt(hour);
if (day === "0") tsel = (parseInt(day) * 1440) + parseInt(hour);
if (day === "1") tsel = (parseInt(day) * 1440) + parseInt(hour);
if (day === "2") tsel = (parseInt(day) * 1440) + parseInt(hour);
if (day === "3") tsel = (parseInt(day) * 1440) + parseInt(hour);
if (day === "4") tsel = (parseInt(day) * 1440) + parseInt(hour);
if (day === "5") tsel = (parseInt(day) * 1440) + parseInt(hour);
if (day === "6") tsel = (parseInt(day) * 1440) + parseInt(hour);
}
//console.log(tsel, tnow, tsel-tnow);
return tsel - tnow;
}
function requestRouteFromLiveMap(clearSelection) {
var atTime = getnowtoday();
var numRoutes = Math.min(10, options.getAlternatives ? options.maxRoutes : 1);
var routeType = (options.routeType === 3) ? "TIME" : "HISTORIC_TIME";
var avoidTollRoads = options.avoidTolls;
var avoidPrimaries = options.avoidFreeways;
var avoidTrails = options.avoidUnpaved;
var avoidLongTrails = options.avoidLongUnpaved;
var allowUTurns = options.allowUTurns;
var avoidDifficult = options.avoidDifficult;
var avoidFerries = options.avoidFerries;
var vehType = options.vehicleType;
var opt = {
data: [],
add: function (name, value, defaultValue) {
if (value !== defaultValue) {
this.data.push(name + (value ? ":t" : ":f"));
}
},
put: function (name, value) {
this.data.push(name + (value ? ":t" : ":f"));
},
get: function () {
return this.data.join(",");
}
};
opt.add("AVOID_TOLL_ROADS", avoidTollRoads, false);
opt.add("AVOID_PRIMARIES", avoidPrimaries, false);
opt.add("AVOID_DANGEROUS_TURNS", avoidDifficult, false);
opt.add("AVOID_FERRIES", avoidFerries, false);
opt.add("ALLOW_UTURNS", allowUTurns, true);
if (avoidLongTrails) { opt.put("AVOID_LONG_TRAILS", true); }
else if (avoidTrails) { opt.put("AVOID_TRAILS", true); }
else { opt.put("AVOID_LONG_TRAILS", false); }
var url = getRoutingManager();
let expressPass = options.passes.map(key => key);
var data = {
from: "x:" + pointA.lon + " y:" + pointA.lat,
to: "x:" + pointB.lon + " y:" + pointB.lat,
returnJSON: true,
returnGeometries: true,
returnInstructions: true,
timeout: 60000,
at: atTime,
type: routeType,
nPaths: numRoutes,
clientVersion: '4.0.0',
options: opt.get(),
vehicleType: vehType,
subscription: expressPass
};
if (options.useRBS) data.id = "beta";
waitingForRoute = true;
getId('routespeeds-error').innerHTML = "";
GM_xmlhttpRequest({
method: "GET",
url: url + "?" + jQuery.param(data),
headers: {
"Content-Type": "application/json"
},
nocache: true,
responseType: "json",
onerror: function(response) {
let str = "Route request failed" + (response.status !== null ? " with error " + response.status : "") + "!<br>";
handleRouteRequestError(str);
waitingForRoute = false;
sdk.Editing.clearSelection();
},
onload: function(response) {
if (response.response.error !== undefined) {
let str = response.response.error;
str = str.replace("|", "<br>");
handleRouteRequestError(str);
} else {
if (response.response.coords !== undefined) {
if (routeSelected > 0) routeSelected = 0;
routesReceived = [response.response];
}
if (response.response.alternatives !== undefined) {
routesReceived = response.response.alternatives;
}
getId('routespeeds-routecount').innerHTML = 'Received <b>' + routesReceived.length + '</b> route' + (routesReceived.length == 1 ? '' : "s") + ' from the server';
sortRoutes();
}
getId('routespeeds-button-livemap').style.backgroundColor = '';
getId('routespeeds-button-reverse').style.backgroundColor = '';
switchRoute();
waitingForRoute = false;
sdk.Editing.clearSelection();
},
});
}
function sortRoutes() {
routesShown = [...routesReceived];
if (!options.routingOrder) {
let sortByField = (options.routeType === 2) ? "length" : options.liveTraffic ? "crossTime" : "crossTimeWithoutRealTime";
routesShown.sort(function (a, b) {
let valField = "total_" + sortByField;
let val = function (r) {
if (r[valField] !== undefined) return r[valField];
let val = 0;
for (let i = 0; i < r.results.length; ++i) {
val += r.results[i][sortByField];
}
return r[valField] = val;
};
return val(a.response) - val(b.response);
});
}
if (routesShown.length > options.maxRoutes) {
routesShown = routesShown.slice(0, options.maxRoutes);
}
if (routeSelectedLast != -1) routeSelected = routeSelectedLast;
if (routeSelected >= routesShown.length) routeSelected = routesShown.length - 1;
createSummaries();
drawRoutes(true);
}
function switchRoute() {
for (let i = 0; i < routesShown.length; i++) {
let summary = getId('routespeeds-summary-' + i);
summary.className = (routeSelected == i) ? 'routespeeds_summary_classB' : 'routespeeds_summary_classA';
}
let z;
for (let name of SCRIPT_LAYERS_TO_COVER) {
try {
baseZIndex = Math.max(baseZIndex, sdk.Map.getLayerZIndex({layerName: name}));
} catch (ex) {}
}
let routeLayerZIndex = sdk.Map.getLayerZIndex({layerName: ROUTE_LAYER_NAME});
if (routeLayerZIndex < baseZIndex) {
sdk.Map.setLayerZIndex({layerName: ROUTE_LAYER_NAME, zIndex: baseZIndex + 1});
} else {
baseZIndex = routeLayerZIndex;
}
reorderLayers(1);
drawRoutes(true);
}
function reorderLayers(mode) {
if (baseZIndex == -1) return;
if (originalZIndices.length == 0) {
for (let i = 0; i < WME_LAYERS_TO_MOVE.length; i++) {
originalZIndices[i] = -1;
}
}
for (let i = 0; i < WME_LAYERS_TO_MOVE.length; i++) {
let z = -1;
try {
z = sdk.Map.getLayerZIndex({layerName: WME_LAYERS_TO_MOVE[i]});
if (mode) {
if (originalZIndices[i] == -1) {
originalZIndices[i] = z;
}
if (z != baseZIndex - WME_LAYERS_TO_MOVE.length + i) {
sdk.Map.setLayerZIndex({layerName: WME_LAYERS_TO_MOVE[i], zIndex: baseZIndex - WME_LAYERS_TO_MOVE.length + i});
sdk.Map.redrawLayer({layerName: WME_LAYERS_TO_MOVE[i]});
}
} else {
if (z != originalZIndices[i]) {
sdk.Map.setLayerZIndex({layerName: WME_LAYERS_TO_MOVE[i], zIndex: originalZIndices[i]});
sdk.Map.redrawLayer({layerName: WME_LAYERS_TO_MOVE[i]});
}
}
} catch (ex) {
if (!alreadyReportedWMELayer) {
error("WME layer " + WME_LAYERS_TO_MOVE[i] + " not found: " + ex);
alreadyReportedWMELayer = true;
}
}
}
}
function drawRoutes(recreate) {
sdk.Map.removeAllFeaturesFromLayer({layerName: ROUTE_LAYER_NAME});
if (recreate) {
storedFeatures = [];
for (let i = routesShown.length - 1; i >= 0; i--) {
if (i == routeSelected) continue;
createRouteFeatures(i)
}
if (routeSelected != -1 && routesShown.length) {
createRouteFeatures(routeSelected)
}
}
sdk.Map.addFeaturesToLayer({
layerName: ROUTE_LAYER_NAME,
features: storedFeatures
});
}
function createSummaries() {
var summaryDiv = getId('routespeeds-summaries');
summaryDiv.innerHTML = '';
let lengthUnit = options.useMiles ? "miles" : "km";
let speedUnit = options.useMiles ? "mph" : "km/h";
for (let i = 0; i < routesShown.length; i++) {
summaryDiv.innerHTML += '<div id=routespeeds-summary-' + i + ' class=routespeeds_summary_classA></div>';
}
for (let i = 0; i < routesShown.length; i++) {
let routeDiv = getId('routespeeds-summary-' + i);
routeDiv.onclick = function(){ toggleRoute(i) };
if (routeSelected == i) routeDiv.className = 'routespeeds_summary_classB';
let html = '<div class=routespeeds_header style="background: ' + getRouteColor(i) + '; color:#e0e0e0; "></div>' + '<div style="min-width:24px; display:inline-block; font-size:14px; color:#404040; text-align:right;"><b>' + (i+1) + '.</b></div>';
let lengthM = 0;
for (let s = 0; s < routesShown[i].response.results.length; s++) {
lengthM += routesShown[i].response.results[s].length;
}
let length = lengthM / 1000;
if (options.useMiles) length /= KM_PER_MILE;
let lengthText = length.toFixed(2);
let time = options.liveTraffic ? routesShown[i].response.totalRouteTime : routesShown[i].response.totalRouteTimeWithoutRealtime;
let timeText = getTimeText(time);
html += '<div style="min-width:57px; display:inline-block; font-size:14px; text-align:right;">' + lengthText + '</div>' + '<span style="color:#404040;"> ' + lengthUnit + '</span>';
html += '<div style="min-width:75px; display:inline-block; font-size:14px; text-align:right;"><b>' + timeText + '</b></div>';
let avgSpeed = getSpeed(lengthM, time);
html += '<div style="min-width:48px; display:inline-block; font-size:14px; text-align:right;" >' + avgSpeed.toFixed(1) + '</div><span style="color:#404040;"> ' + speedUnit + '</span>';
if (options.showRouteText) {
let maxWidth = options.useMiles ? 277 : 270;
let laneTypes = [];
if (routesShown[i].response.routeAttr.includes('Toll')) laneTypes.push('Toll');
laneTypes.push(...routesShown[i].response.laneTypes);
let separator = '';
if (routesShown[i].response.minPassengers) separator += " (" + routesShown[i].response.minPassengers + "+)";
if (laneTypes.length) separator += ': ';
html += '<div style="max-width:' + maxWidth + 'px; white-space:normal; line-height:normal; font-variant-numeric:normal;">' + laneTypes.join(', ') + separator + routesShown[i].response.routeName + '</div>';
}
routeDiv.innerHTML = html;
routeDiv.style.visibility = 'visible';
}
summaryDiv.style.visibility = 'visible';
}
function handleRouteRequestError(message) {
warn("route request error: " + message.replace("<br>", "\n"));
getId('routespeeds-button-livemap').style.backgroundColor = '';
getId('routespeeds-button-reverse').style.backgroundColor = '';
getId('routespeeds-summaries').style.visibility = 'hidden';
getId('routespeeds-summaries').innerHTML = '';
routesReceived = [];
sortRoutes();
getId('routespeeds-error').innerHTML = "<br>" + message;
getId('routespeeds-routecount').innerHTML = '';
}
function livemapRouteClick() {
routeSelected = 0;
routeSelectedLast = -1;
livemapRoute();
}
function get_coords_from_livemap_link(link) {
let lon1 = '';
let lat1 = '';
let lon2 = '';
let lat2 = '';
let opt = link.split('&');
for (let i = 0; i < opt.length; i++) {
let o = opt[i];
if (o.indexOf('from_lon=') === 0) lon1 = o.substring(9, 30);
if (o.indexOf('from_lat=') === 0) lat1 = ', ' + o.substring(9, 30);
if (o.indexOf('to_lon=') === 0) lon2 = o.substring(7, 30);
if (o.indexOf('to_lat=') === 0) lat2 = ', ' + o.substring(7, 30);
}
getId('sidepanel-routespeeds-a').value = lon1 + lat1;
getId('sidepanel-routespeeds-b').value = lon2 + lat2;
}
function livemapRoute() {
if (!options.enableScript) return;
if (waitingForRoute) return;
let stra = getId('sidepanel-routespeeds-a').value;
let strb = getId('sidepanel-routespeeds-b').value;
let pastedlink = false;
//sprawdzenie czy wklejono link z LiveMap, jeżeli tak to sparsowanie i przeformatowanie współrzędnych oraz przeniesienie widoku mapy na miejsce wklejonej trasy
//(checking if the link from LiveMap has been pasted, if yes, paring and reformatting the coordinates and moving the map view to the location of the pasted route)
if (stra.indexOf('livemap?') >= 0 || stra.indexOf('livemap/?') >= 0) {
get_coords_from_livemap_link(stra);
stra = getId('sidepanel-routespeeds-a').value;
strb = getId('sidepanel-routespeeds-b').value;
pastedlink = true;
}
else if (strb.indexOf('livemap?') >= 0 || strb.indexOf('livemap/?') >= 0) {
get_coords_from_livemap_link(strb);
stra = getId('sidepanel-routespeeds-a').value;
strb = getId('sidepanel-routespeeds-b').value;
pastedlink = true;
}
stra = getId('sidepanel-routespeeds-a').value;
strb = getId('sidepanel-routespeeds-b').value;
if (stra === "") return;
if (strb === "") return;
let p1 = stra.split(",");
let p2 = strb.split(",");
if (p1.length < 2) return;
if (p2.length < 2) return;
let x1 = p1[0].trim();
let y1 = p1[1].trim();
let x2 = p2[0].trim();
let y2 = p2[1].trim();
x1 = parseFloat(x1);
y1 = parseFloat(y1);
x2 = parseFloat(x2);
y2 = parseFloat(y2);
if (isNaN(x1)) return;
if (isNaN(y1)) return;
if (isNaN(x2)) return;
if (isNaN(y2)) return;
if (x1 < -180 || x1 > 180) x1 = 0;
if (x2 < -180 || x2 > 180) x2 = 0;
if (y1 < -90 || y1 > 90) y1 = 0;
if (y2 < -90 || y2 > 90) y2 = 0;
let objprog1 = getId('routespeeds-button-livemap');
objprog1.style.backgroundColor = '#FF8000';
createMarkers(x1, y1, x2, y2);
if (pastedlink) {
clickA();
}
requestRouteFromLiveMap(false);
}
//--------------------------------------------------------------------------
// Map event handlers
function onMapDataLoaded(event) {
if (!options.enableScript) return;
updateTopCountry();
}
function onMapMoveEnd(event) {
if (!options.enableScript) return;
drawRoutes(false);
}
function mouseEnterHandler(event) {
if (!startFirstClickRegistered) {
sdk.Events.on({
eventName: "wme-map-mouse-down",
eventHandler: startFirstClick
});
startFirstClickRegistered = true;
}
if (markerMoving != event.featureId) {
markerMoving = event.featureId;
}
}
function mouseLeaveHandler(event) {
if (startFirstClickRegistered) {
sdk.Events.off({
eventName: "wme-map-mouse-down",
eventHandler: startFirstClick
});
startFirstClickRegistered = false;
}
markerMoving = "none";
}
function startFirstClick(event) {
sdk.Events.on({
eventName: "wme-map-mouse-up",
eventHandler: onFirstClick
});
sdk.Events.on({
eventName: "wme-map-mouse-move",
eventHandler: cancelFirstClick
});
}
function cancelFirstClick(event) {
sdk.Events.off({
eventName: "wme-map-mouse-up",
eventHandler: onFirstClick
});
sdk.Events.off({
eventName: "wme-map-mouse-move",
eventHandler: cancelFirstClick
});
}
function onFirstClick(event) {
sdk.Events.stopLayerEventsTracking({layerName: MARKER_LAYER_NAME});
sdk.Events.off({
eventName: "wme-map-mouse-down",
eventHandler: startFirstClick
});
startFirstClickRegistered = false;
cancelFirstClick(event);
sdk.Events.on({
eventName: "wme-map-mouse-down",
eventHandler: startSecondClick
});
let markerLocationPixel = sdk.Map.getMapPixelFromLonLat({
lonLat: {
lon: markerMoving == "A" ? pointA.lon : pointB.lon,
lat: markerMoving == "A" ? pointA.lat : pointB.lat
}
});
let offsetX = markerLocationPixel.x - event.x;
let offsetY = markerLocationPixel.y - event.y;
mouseMoveHandler = ({x, y}) => onMouseMoveWithMarker(markerMoving, x + offsetX, y + offsetY);
sdk.Events.on({
eventName: "wme-map-mouse-move",
eventHandler: mouseMoveHandler
});
}
function onMouseMoveWithMarker(id, x, y) {
let newLocation = sdk.Map.getLonLatFromMapPixel({x: x, y: y});
moveMarker(id, newLocation.lon, newLocation.lat);
}
function startSecondClick(event) {
sdk.Events.off({
eventName: "wme-map-mouse-move",
eventHandler: mouseMoveHandler
});
sdk.Events.on({
eventName: "wme-map-mouse-up",
eventHandler: onSecondClick
});
sdk.Events.on({
eventName: "wme-map-mouse-move",
eventHandler: cancelSecondClick
});
}
function cancelSecondClick(event) {
sdk.Events.off({
eventName: "wme-map-mouse-up",
eventHandler: onSecondClick
});
sdk.Events.off({
eventName: "wme-map-mouse-move",
eventHandler: cancelSecondClick
});
sdk.Events.on({
eventName: "wme-map-mouse-move",
eventHandler: mouseMoveHandler
});
}
function onSecondClick(event) {
sdk.Events.off({
eventName: "wme-map-mouse-down",
eventHandler: startSecondClick
});
sdk.Events.off({
eventName: "wme-map-mouse-up",
eventHandler: onSecondClick
});
sdk.Events.off({
eventName: "wme-map-mouse-move",
eventHandler: cancelSecondClick
});
mouseEnterHandler({featureId: markerMoving});
sdk.Events.trackLayerEvents({layerName: MARKER_LAYER_NAME});
let lon1 = parseInt(pointA.lon * 1000000.0 + 0.5) / 1000000.0;
let lat1 = parseInt(pointA.lat * 1000000.0 + 0.5) / 1000000.0;
let lon2 = parseInt(pointB.lon * 1000000.0 + 0.5) / 1000000.0;
let lat2 = parseInt(pointB.lat * 1000000.0 + 0.5) / 1000000.0;
if (getId('sidepanel-routespeeds-a') !== undefined) {
getId('sidepanel-routespeeds-a').value = lon1 + ", " + lat1;
getId('sidepanel-routespeeds-b').value = lon2 + ", " + lat2;
}
var objprog1 = getId('routespeeds-button-livemap');
if (objprog1.style.backgroundColor === '') objprog1.style.backgroundColor = '#FF8000';
requestRouteFromLiveMap(true);
}
//--------------------------------------------------------------------------
// Sidebar event handlers
function resetOptionsToLivemapRouteClick() {
if (waitingForRoute) return;
resetOptions();
$(`.routespeeds-pass-checkbox`).prop( "checked", false );;
options.passes = [];
livemapRoute();
}
function hourChange() {
livemapRoute();
}
function dayChange() {
livemapRoute();
}
function clickA() {
if (pointA.lon !== undefined) sdk.Map.setMapCenter({lonLat: pointA});
}
function clickB() {
if (pointB.lon !== undefined) sdk.Map.setMapCenter({lonLat: pointB});
}
function clickEnableScript() {
options.enableScript = (getId('routespeeds-enablescript').checked === true);
if (!options.enableScript) {
getId('sidepanel-routespeeds').style.color = "#A0A0A0";
getId('routespeeds-summaries').style.visibility = 'hidden';
sdk.Map.setLayerVisibility({layerName: MARKER_LAYER_NAME, visibility: false});
sdk.Map.removeAllFeaturesFromLayer({layerName: ROUTE_LAYER_NAME});
reorderLayers(0);
}
else {
getId('sidepanel-routespeeds').style.color = "";
sdk.Map.setLayerVisibility({layerName: MARKER_LAYER_NAME, visibility: true});
if (routesShown.length > 0) drawRoutes(false);
reorderLayers(1);
}
}
function clickReverseRoute() {
if (!options.enableScript || waitingForRoute) return;
let newA = [pointB.lon, pointB.lat];
let newB = [pointA.lon, pointA.lat];
if (getId('sidepanel-routespeeds-a') !== undefined) {
getId('sidepanel-routespeeds-a').value = newA[0].toFixed(6) + ", " + newA[1].toFixed(6);
getId('sidepanel-routespeeds-b').value = newB[0].toFixed(6) + ", " + newB[1].toFixed(6);
}
createMarkers(newA[0], newA[1], newB[0], newB[1]);
requestRouteFromLiveMap(false);
}
function clickShowLabels() {
options.showLabels = (getId('routespeeds-showLabels').checked === true);
drawRoutes(true);
}
function clickShowSpeeds() {
options.showSpeeds = (getId('routespeeds-showSpeeds').checked === true);
drawRoutes(true);
}
function clickUseMiles() {
options.useMiles = (getId('routespeeds-usemiles').checked === true);
createSummaries();
drawRoutes(options.showLabels && options.showSpeeds);
}
function clickShowRouteText() {
options.showRouteText = (getId('routespeeds-routetext').checked === true);
createSummaries();
}
function clickGetAlternatives() {
routeSelected = 0;
routeSelectedLast = -1;
options.getAlternatives = (getId('routespeeds-getalternatives').checked === true);
if (options.getAlternatives && routesReceived.length < options.maxRoutes) {
livemapRoute();
} else {
sortRoutes();
}
}
function clickMaxRoutes() {
options.getAlternatives = (getId('routespeeds-getalternatives').checked === true);
options.maxRoutes = parseInt(getId('routespeeds-maxroutes').value);
if (options.getAlternatives && routesReceived.length < options.maxRoutes) {
livemapRoute();
} else {
sortRoutes();
}
}
function clickLiveTraffic() {
options.liveTraffic = (getId('routespeeds-livetraffic').checked === true);
sortRoutes();
}
function clickAvoidTolls() {
options.avoidTolls = (getId('routespeeds-avoidtolls').checked === true);
livemapRoute();
}
function clickAvoidFreeways() {
options.avoidFreeways = (getId('routespeeds-avoidfreeways').checked === true);
livemapRoute();
}
function clickAvoidUnpaved() {
options.avoidUnpaved = (getId('routespeeds-avoidunpaved').checked === true);
options.avoidLongUnpaved = false;
getId('routespeeds-avoidlongunpaved').checked = false;
livemapRoute();
}
function clickAvoidLongUnpaved() {
options.avoidLongUnpaved = (getId('routespeeds-avoidlongunpaved').checked === true);
options.avoidUnpaved = false;
getId('routespeeds-avoidunpaved').checked = false;
livemapRoute();
}
function clickRouteType() {
options.routeType = parseInt(getId('routespeeds-routetype').value);
livemapRoute();
}
function clickAllowUTurns() {
options.allowUTurns = (getId('routespeeds-allowuturns').checked === true);
livemapRoute();
}
function clickRoutingOrder() {
options.routingOrder = (getId('routespeeds-routingorder').checked === true);
sortRoutes();
}
function clickUseRBS() {
options.useRBS = (getId('routespeeds-userbs').checked === true);
livemapRoute();
}
function clickAvoidDifficult() {
options.avoidDifficult = (getId('routespeeds-avoiddifficult').checked === true);
livemapRoute();
}
function clickAvoidFerries() {
options.avoidFerries = (getId('routespeeds-avoidferries').checked === true);
livemapRoute();
}
function clickVehicleType() {
options.vehicleType = (getId('routespeeds-vehicletype').value);
livemapRoute();
}
function clickPassOption() {
let passKey = $(this).data('pass-key');
if (this.checked) {
options.passes.push(passKey);
} else {
options.passes = options.passes.filter(key => key !== passKey)
}
updatePassesLabel();
livemapRoute();
}
function toggleRoute(routeIndex) {
if (routeSelected == routeIndex) routeIndex = -1;
routeSelected = routeIndex;
routeSelectedLast = routeIndex;
switchRoute();
}
function enterAB(ev) {
if (ev.keyCode === 13) {
livemapRoute();
}
}
//--------------------------------------------------------------------------
// Code execution starts here
unsafeWindow.SDK_INITIALIZED.then(onSDKInitialized);
})();