Greasy Fork is available in English.

WME Route Speeds (MapOMatic fork)

Shows segment speeds in a route.

// ==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             2020.09.10.001
// @grant               none
// @namespace           https://greasyfork.org/en/scripts/369630-wme-route-speeds-mapomatic-fork
// @require             https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @author              wlodek76 (forked by MapOMatic)
// @copyright           2014, 2015 wlodek76
// @contributor         2014, 2015 FZ69617
// ==/UserScript==

/* global $ */
/* global W */
/* global OpenLayers */

/* eslint-disable */
/*Version history:
 * 2020.09.10.001
    - Updating viewPortDiv references and minor code cleanup
 * 2020.07.27.001
    - WME map object references.
 * 2020.05.22.001
    - Change OL to OpenLayers.
 * 2019.12.11.001
    - Minor update to fix bug for WazeWrap not being ready before full init.
 * 2019.11.21.001
    - Minor update to support change in WME v2.43-40-gf367bffa4
 * 2019.10.30.001
 *  - Minor update to support change to WazeWrap registration of events in W.map.
 * 2019.07.25.001
 *  - Minor update to support change from W.model.countries.top to W.model.getTopCountry()
 * 2019.03.05.001
 *  - New: support for vehicle types
 *  - New: pass/permit support
 *  - Remove shortest route option- no longer supported by routing server
 *  - Include server support for Puerto Rico
 *
 * 2017.12.09.001 (20171209)
 * - New: Added Avoid Ferries option (tonestertm)
 *   Fixed longstanding bug which never allowed U-turns
 *
 * 2017.11.15.001 (20171115)
 * - New: Added Avoid Difficult Turns option (tonestertm)
 * - New: Added a checkbox and functionality to place results in Livemap's natural sort order, to help with troubleshooting around penalties.
 *   Note that when Routing Order is checked, the Try More option is ignored, to better approximate Livemap results (tonestertm)
 *
 * 2017.11.10.001 (20171011)
 * - Changing to use WazeWrap.Interface.Tab to load the Route Speeds tab so it can recover from changing the map units (Imperial/Metric)
 *   and coming back from event mode
 *
 * 1.4.8-momfork (20170123)
 * - Updated formatting in options panel to fix checkbox alignment issues.
 *
 * 1.4.7-momfork (20170118)
 * - [FIXED] International translations set to english by default. Causing critical failures.
 *
 * 1.4.6-momfork (20170104)
 * - Restored original marker functionality (*HUGE* thanks to JustinS83 for doing ALL the work to figure out a workaround).
 *
 * 1.4.5-momfork (20161228)
 * - Fixed @include so script will work in WME Beta again.
 *
 * 1.4.4-momfork (20161228)
 * - This is a temporary fix to restore some basic functionality.  The original script may work again if Waze
 *   devs fix an issue on their end.  Had to remove markers for now.
 *
 *
 * 1.4.3 (20150505)
 * - script didn't work when only WME Route Speeds was installed (removed unnecessary unsafeWindow definition)
 *
 * 1.4.2 (20150424)
 * - Added: Paste LiveMap permalink into A/B marker input.
 * - Improved: 'Reset to Livemap Route' did not reset correctly when Shortest route was selected.
 * - Fixed: Layout of A-B button accordingly to browser's size.
 * - Fixed: Translation of internal AVOID_TRAILS option to Dirt roads.
 *
 * 1.4.1.1 (20150424)
 * - Improvement: Fixed options styling minor issues.
 * - Minor Fix: Drawing two-way traffic on segments with sharp angle
 *
 * 1.4.0 (20150424)
 * - New: Displaying traffic on two-way segments
 * - Improved: Faster display of recalculated routes
 * - Improved: First route is selected always by default
 * - Gui: Reorganize of options
 * - Fixed: Waiting for WME initialization
 * - Added: Option for ALLOW_UTURNS
 *
 * 1.3.0 (20150421)
 * - Fix: ALLOW UTURNS in routes
 *
 * 1.2.9.1 (20150419)
 * - Fix: Road Closures icons hides Route Speeds speed/time info
 *
 * 1.2.9 (20150114)
 * - New: Centering the map on the A/B markers.
 * - New: Routing options reset to the Livemap Route equivalents.
 * - New: Experimental "Fastest (no history)" route type option.
 *
 * 1.2.8.1 (20150102)
 * - Bug fix: Routing request options fixed.
 *
 * 1.2.8 (20150101)
 * - New: Added "Route type" routing option.
 * - New: Added "Try more" alternative routes option.
 * - Change: Controls' styling.
 * - Improvement: Route list sorting.
 * - Improvement: Route list selection can now be toggled.
 * - Improvement: Better handling of route request errors.
 * - Other: Minor code changes.
 *
 * 1.2.7.1 (20141211)
 * - Bug fix: Routing request options corrected.
 *
 * 1.2.7 (20141210)
 * - New: Added "Avoid toll roads" routing option.
 * - New: Added "Avoid freeways" routing option.
 * - New: Added "Allow dirt roads" routing option.
 * - Other: Minor code improvements.
 *
 * 1.2.6.1 (20141001)
 * - Fix for new WME by tkr85.
 */

(function () {
	"use strict";

	const SCRIPT_STORE = 'wme_routespeeds';
	const DEFAULT_SETTINGS = {
		passes: []
	}

	let _settings;
	let _modelPasses = [];

	var wmech_version = GM_info.script.version;

	var epsg900913 = new OpenLayers.Projection("EPSG:900913");
	var epsg4326 = new OpenLayers.Projection("EPSG:4326");

	var selected = 0;
	var routewsp1 = [];
	var routewsp2 = [];
	var routewsp3 = [];
	var routewsp4 = [];
	var routewsp5 = [];
	var routeodc1 = [];
	var routeodc2 = [];
	var routeodc3 = [];
	var routeodc4 = [];
	var routeodc5 = [];

	var routewait = 0;
	var routeSelected = 1;
	var routeSelectedLast = 0;

	var markerA;
	var markerB;
	var markerA_offset_click = [0, 0];
	var markerB_offset_click = [0, 0];

	var routespeedsoption1 = false;
	var routespeedsoption2 = false;
	var routespeedsoption3 = false;
	var routespeedsoption4 = false;
	var routespeedsoption5 = false;
	var routespeedsoption6 = 3;
	var routespeedsoption7 = true;
	var routespeedsoption8 = false;
	var routespeedsoption9 = false;
	var routespeedsoption10 = false;
	var routespeedsoption11 = false;
	var routespeedsoption12 = false;
	var routespeedsoption13 = 1;
	var routespeedsoption14 = true;
	var routespeedsoption15 = false;
	var routespeedsoption16 = false;
	var routespeedsoption17 = false;
	var routespeedsoption18 = 'PRIVATE';

	var lastmapcenter = [0, 0];
	var panningX = 0;
	var panningY = 0;
	var acceleration = 1.6;
	var accelerationmin = 10;
	var accelerationmax = 200;
	var accelerationmargin = 30;
	var accelerationbackstop = 3;

	var koloractive = [
		"#808080",  // 0 km/h
		"#271308",  // 10 km/h
		"#813b27",  // 20 km/h
		"#e22700",  // 30 km/h
		"#ef7200",  // 40 km/h
		"#ffd307",  // 50 km/h
		"#6cf104",  // 60 km/h
		"#2fa035",  // 70 km/h
		"#0bbbe9",  // 80 km/h
		"#0f77e0",  // 90 km/h
		"#0346fc",  // 100 km/h
		"#3918d7",  // 110 km/h
		"#8c07f7",  // 120 km/h
		"#ea0ae7",  // 130 km/h
		"#b00094",  // 140 km/h
		"#670055"   // 200 km/h
	];

	var jqueryinfo = 0;
	var tabswitched = 0;
	var closurelayer = null;
	var closurelayerZINDEX = [];
	var leftHand = false;

	function log(msg) {
		console.log('WME Route Speeds:', msg);
	}
	//------------------------------------------------------------------------------------------------
	function bootstrapWMERouteSpeeds(tries = 1) {
		// Need to wait for countries to load, otherwise restrictionSubscriptions are not available yet.
		if (W && W.loginManager && W.map && W.loginManager.user && W.model && W.model.countries.getObjectArray().length && WazeWrap.Ready) {
			log('Initializing...');
			initialiseWMERouteSpeeds();
			log(wmech_version + " loaded.");
		} else {
			if (tries === 100) {
				log('Bootstrap timeout. Script has failed to load.');
				return;
			} else if (tries % 10 === 0) {
				log('Bootstrap failed. Trying again...');
			}
			setTimeout(() => bootstrapWMERouteSpeeds(++tries), 300);
		}
	}
	//------------------------------------------------------------------------------------------------
	function panningWMERouteSpeeds() {
		var WM = window.W.map;

		//var operationPending = W.vent._events.listeners.operationPending[0];
		//if (operationPending == undefined) return;
		//var pending = operationPending.obj.pending[0];

		//var lastcenterX = lastmapcenter[0];
		//var lastcenterY = lastmapcenter[1];
		//var centerX = WM.getCenter().lon;
		//var centerY = WM.getCenter().lat;

		//if (lastcenterX == 0) lastcenterX = centerX;
		//if (lastcenterY == 0) lastcenterY = centerY;

		//if ( lastcenterX==centerX && lastcenterY==centerY && pending == undefined ) {
		if (panningX || panningY) {
			var accelX = panningX;
			var accelY = panningY;

			if (accelX < 0) accelX = -Math.pow(Math.abs(accelX), acceleration) - accelerationmin;
			if (accelX > 0) accelX = Math.pow(Math.abs(accelX), acceleration) + accelerationmin;

			if (accelY < 0) accelY = -Math.pow(Math.abs(accelY), acceleration) - accelerationmin;
			if (accelY > 0) accelY = Math.pow(Math.abs(accelY), acceleration) + accelerationmin;

			if (accelX < -accelerationmax) accelX = -accelerationmax;
			if (accelY < -accelerationmax) accelY = -accelerationmax;
			if (accelX > accelerationmax) accelX = accelerationmax;
			if (accelY > accelerationmax) accelY = accelerationmax;

			WM.pan(accelX, accelY);
		}
		//}
	}
	//------------------------------------------------------------------------------------------------
	function saveRouteSpeedsOptions() {

		var obj1 = getId('routespeeds-option1');
		var obj2 = getId('routespeeds-option2');
		var obj3 = getId('routespeeds-option3');
		var obj4 = getId('routespeeds-option4');
		var obj5 = getId('routespeeds-option5');
		var obj6 = getId('routespeeds-option6');
		var obj7 = getId('routespeeds-option7');
		var obj8 = getId('routespeeds-option8');
		var obj9 = getId('routespeeds-option9');
		var obj10 = getId('routespeeds-option10');
		var obj11 = getId('routespeeds-option11');
		var obj12 = getId('routespeeds-option12');
		var obj13 = getId('routespeeds-option13');
		var obj14 = getId('routespeeds-option14');
		var obj15 = getId('routespeeds-option15');
		var obj16 = getId('routespeeds-option16');
		var obj17 = getId('routespeeds-option17');
		var obj18 = getId('routespeeds-option18');

		if (obj1 !== undefined) {
			localStorage.setItem("RouteSpeedsOption1", obj1.checked);
			localStorage.setItem("RouteSpeedsOption2", obj2.checked);
			localStorage.setItem("RouteSpeedsOption3", obj3.checked);
			localStorage.setItem("RouteSpeedsOption4", obj4.checked);
			localStorage.setItem("RouteSpeedsOption5", obj5.checked);
			localStorage.setItem("RouteSpeedsOption6", obj6.value);
			localStorage.setItem("RouteSpeedsOption7", obj7.checked);
			localStorage.setItem("RouteSpeedsOption8", obj8.checked);
			localStorage.setItem("RouteSpeedsOption9", obj9.checked);
			localStorage.setItem("RouteSpeedsOption10", obj10.checked);
			localStorage.setItem("RouteSpeedsOption11", obj11.checked);
			localStorage.setItem("RouteSpeedsOption12", obj12.checked);
			localStorage.setItem("RouteSpeedsOption13", obj13.value);
			localStorage.setItem("RouteSpeedsOption14", true);  // ALLOW_UTURNS is by default always true
			localStorage.setItem("RouteSpeedsOption15", obj15.checked);
			localStorage.setItem("RouteSpeedsOption16", obj16.checked);
			localStorage.setItem("RouteSpeedsOption17", obj17.checked);
			localStorage.setItem("RouteSpeedsOption18", obj18.value);
		}

		localStorage.setItem(SCRIPT_STORE, JSON.stringify(_settings));
	}
	//---------------------------------------------------------------------------------------
	function loadRouteSpeedsOptions() {

		if (localStorage.RouteSpeedsOption1) routespeedsoption1 = (localStorage.RouteSpeedsOption1 == "true");
		if (localStorage.RouteSpeedsOption2) routespeedsoption2 = (localStorage.RouteSpeedsOption2 == "true");
		if (localStorage.RouteSpeedsOption3) routespeedsoption3 = (localStorage.RouteSpeedsOption3 == "true");
		if (localStorage.RouteSpeedsOption4) routespeedsoption4 = (localStorage.RouteSpeedsOption4 == "true");
		if (localStorage.RouteSpeedsOption5) routespeedsoption5 = (localStorage.RouteSpeedsOption5 == "true");
		if (localStorage.RouteSpeedsOption6) routespeedsoption6 = (localStorage.RouteSpeedsOption6);
		if (localStorage.RouteSpeedsOption7) routespeedsoption7 = (localStorage.RouteSpeedsOption7 == "true");
		if (localStorage.RouteSpeedsOption8) routespeedsoption8 = (localStorage.RouteSpeedsOption8 == "true");
		if (localStorage.RouteSpeedsOption9) routespeedsoption9 = (localStorage.RouteSpeedsOption9 == "true");
		if (localStorage.RouteSpeedsOption10) routespeedsoption10 = (localStorage.RouteSpeedsOption10 == "true");
		if (localStorage.RouteSpeedsOption11) routespeedsoption11 = (localStorage.RouteSpeedsOption11 == "true");
		if (localStorage.RouteSpeedsOption12) routespeedsoption12 = (localStorage.RouteSpeedsOption12 == "true");
		if (localStorage.RouteSpeedsOption13) routespeedsoption13 = (localStorage.RouteSpeedsOption13);
		if (localStorage.RouteSpeedsOption14) routespeedsoption14 = (localStorage.RouteSpeedsOption14 == "true");
		if (localStorage.RouteSpeedsOption15) routespeedsoption15 = (localStorage.RouteSpeedsOption15 == "true");
		if (localStorage.RouteSpeedsOption16) routespeedsoption16 = (localStorage.RouteSpeedsOption16 == "true");
		if (localStorage.RouteSpeedsOption17) routespeedsoption17 = (localStorage.RouteSpeedsOption17 == "true");
		if (localStorage.RouteSpeedsOption18) routespeedsoption18 = (localStorage.RouteSpeedsOption18);


		getId('routespeeds-option1').checked = routespeedsoption1;
		getId('routespeeds-option2').checked = routespeedsoption2;
		getId('routespeeds-option3').checked = routespeedsoption3;
		getId('routespeeds-option4').checked = routespeedsoption4;
		getId('routespeeds-option5').checked = routespeedsoption5;
		getId('routespeeds-option6').value = routespeedsoption6;
		getId('routespeeds-option7').checked = routespeedsoption7;
		getId('routespeeds-option8').checked = routespeedsoption8;
		getId('routespeeds-option9').checked = routespeedsoption9;
		getId('routespeeds-option10').checked = routespeedsoption10;
		getId('routespeeds-option11').checked = routespeedsoption11;
		getId('routespeeds-option12').checked = routespeedsoption12;
		getId('routespeeds-option13').value = routespeedsoption13;
		getId('routespeeds-option14').checked = routespeedsoption14;
		getId('routespeeds-option15').checked = routespeedsoption15;
		getId('routespeeds-option16').checked = routespeedsoption16;
		getId('routespeeds-option17').checked = routespeedsoption17;
		getId('routespeeds-option18').value = routespeedsoption18;

		// Create the global object where settings will be stored in memory.
		_settings = $.parseJSON(localStorage.getItem(SCRIPT_STORE)) || {};

		// Fill in any missing settings from the DEFAULT_SETTINGS object
		for (let prop in DEFAULT_SETTINGS) {
			if (!_settings.hasOwnProperty(prop)) {
				_settings[prop] = DEFAULT_SETTINGS[prop];
			}
		}

		update_adv_switches();
	}
	//---------------------------------------------------------------------------------------
	function update_adv_switches() {

		getId('routespeeds-option5-span').style.display = routespeedsoption5 ? 'inline' : 'none';
		//    getId('routespeeds-option5-span').style.display = routespeedsoption15 ? 'none' : 'inline';  // FIXIT
		getId('routespeeds-option10-span').style.display = routespeedsoption10 ? 'inline' : 'inline';
	}
	//---------------------------------------------------------------------------------------
	function getRoutingManager() {
		if (W.model.countries.getObjectById(235) || W.model.countries.getObjectById(40) || W.model.countries.getObjectById(182)) { // US, Canada, & PR
			return '/RoutingManager/routingRequest';
		} else if (W.model.countries.getObjectById(106)) { // Israel
			return '/il-RoutingManager/routingRequest';
		} else {
			return '/row-RoutingManager/routingRequest'; // ROW
		}
	}
	//------------------------------------------------------------------------------------------------
	function getSegmentMidPoint(seg, end) {

		var points, p1, p2, dx, dy, x, y;
		points = seg.geometry.components.length;

		if (points == 2) {
			p1 = seg.geometry.components[0];
			p2 = seg.geometry.components[1];

			x = p1.x + (p2.x - p1.x) * 0.5;
			y = p1.y + (p2.y - p1.y) * 0.5;
			return OpenLayers.Layer.SphericalMercator.inverseMercator(x, y);
		}

		var length = 0;
		for (var i = 0; i < points - 1; i++) {
			p1 = seg.geometry.components[i + 0];
			p2 = seg.geometry.components[i + 1];
			dx = p2.x - p1.x;
			dy = p2.y - p1.y;
			length += Math.sqrt(dx * dx + dy * dy);
		}
		var midlen = length / 2.0;

		var length1 = 0;
		var length2 = 0;
		for (i = 0; i < points - 1; i++) {
			p1 = seg.geometry.components[i + 0];
			p2 = seg.geometry.components[i + 1];
			dx = p2.x - p1.x;
			dy = p2.y - p1.y;
			length1 = length2;
			length2 = length2 + Math.sqrt(dx * dx + dy * dy);

			if (midlen >= length1 && midlen < length2) {
				var proc = (midlen - length1) / (length2 - length1);
				x = p1.x + (p2.x - p1.x) * proc;
				y = p1.y + (p2.y - p1.y) * proc;
				return OpenLayers.Layer.SphericalMercator.inverseMercator(x, y);
			}
		}

		if (end === 0) {
			p1 = seg.geometry.components[0];
			p2 = seg.geometry.components[1];
		}
		else {
			p1 = seg.geometry.components[points - 2];
			p2 = seg.geometry.components[points - 1];
		}

		x = p1.x + (p2.x - p1.x) * 0.5;
		y = p1.y + (p2.y - p1.y) * 0.5;
		return OpenLayers.Layer.SphericalMercator.inverseMercator(x, y);
	}
	//------------------------------------------------------------------------------------------------
	function getColor(speed) {
		if (speed === 0) return koloractive[0];
		var k = parseInt(speed / 10.0) + 1;
		if (k > 15) k = 15;
		return koloractive[k];
	}
	//-----------------------------------------------------------------------------------------------
	function updatePassesLabel() {
		let count = _modelPasses.filter(pass => _settings.passes.indexOf(pass.key) > -1).length;
		$('#routespeeds-passes-label').text(`Passes & Permits (${count} of ${_modelPasses.length})`);
	}
	//------------------------------------------------------------------------------------------------
	function addLabel(lines, speedtekst, routespeedsoption2, odctime, odclen, routespeedsoption4, id) {

		var speed = parseInt(speedtekst);

		var kolor1 = '#F0F0F0';
		var kolor2 = '#000000';
		var p1, p2, pt, textFeature, k, sx, sy;
		if (speed >= 40 && speed < 50) { kolor1 = '#404040'; kolor2 = '#FFFFFF'; }
		if (speed >= 50 && speed < 60) { kolor1 = '#404040'; kolor2 = '#FFFFFF'; }

		if (routespeedsoption4) speedtekst = parseInt(speedtekst / 1.609 + 0.5);
		if (speedtekst === 0) speedtekst = "?";

		var numlines = lines.length;
		if (numlines >= 2) {
			var line;
			var ps = parseInt(numlines) >> 1;
			p1 = lines[ps].components[0];
			p2 = lines[ps].components[1];
			var proc = 0.5;

			var dist = [];
			var dsum = 0;
			for (k = 0; k < numlines; k++) {
				line = lines[k];
				var d = line.getGeodesicLength(epsg900913);
				dsum += d;
				dist.push(d);
			}
			var dmid = dsum / 2.0;
			var d1 = 0;
			var d2 = 0;
			for (k = 0; k < numlines; k++) {
				line = lines[k];
				d1 = d2;
				d2 += dist[k];
				if (dmid >= d1 && dmid < d2) {
					p1 = lines[k].components[0];
					p2 = lines[k].components[1];
					proc = (dmid - d1) / (d2 - d1);
				}
			}

			sx = p1.x + (p2.x - p1.x) * proc;
			sy = p1.y + (p2.y - p1.y) * proc;

			if (routespeedsoption2) speedtekst = odctime + "s ";

			pt = new OpenLayers.Geometry.Point(sx, sy);
			textFeature = new OpenLayers.Feature.Vector(pt, { labelText: speedtekst, fontColor: kolor1, pointRadius: 0 });
			return textFeature;
		}
		else if (numlines == 1) {
			p1 = lines[0].components[0];
			p2 = lines[0].components[1];

			sx = (p1.x + p2.x) * 0.5;
			sy = (p1.y + p2.y) * 0.5;

			if (routespeedsoption2) speedtekst = odctime + "s ";

			pt = new OpenLayers.Geometry.Point(sx, sy);
			textFeature = new OpenLayers.Feature.Vector(pt, { labelText: speedtekst, fontColor: kolor1, pointRadius: 0 });
			return textFeature;
		}
		else return null;
	}
	//------------------------------------------------------------------------------------------------
	function panmap(draggingobj, x, y) {
        let viewPortDiv = draggingobj.map.getViewport();
		let maxX = viewPortDiv.clientWidth;
		let maxY = viewPortDiv.clientHeight;
		let lastx = draggingobj.last.x;
		let lasty = draggingobj.last.y;
		let mx = x - lastx;
		let my = y - lasty;

		if (x < accelerationmargin) {
			if (mx < 0) panningX = x - accelerationmargin;
			if (mx > accelerationbackstop) panningX = 0;
		}
		else if (x > maxX - accelerationmargin) {
			if (mx > 0) panningX = x - (maxX - accelerationmargin);
			if (mx < -accelerationbackstop) panningX = 0;
		}
		else panningX = 0;

		if (y < accelerationmargin) {
			if (my < 0) panningY = y - (accelerationmargin);
			if (my > accelerationbackstop) panningY = 0;
		}
		else if (y > maxY - accelerationmargin - 25) {
			if (my > 0) panningY = y - (maxY - accelerationmargin - 25);
			if (my < -accelerationbackstop) panningY = 0;
		}
		else panningY = 0;
	}
	//------------------------------------------------------------------------------------------------
	function createMarkers(lon1, lat1, lon2, lat2, disp) {

		var WM = window.W.map;

		var mlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeedsMarkers");
		var markerLayer = mlayers[0];
		var p1, p2, lonlat;

		if (markerA === undefined && markerB === undefined) {
			var di = WazeWrap.Require.DivIcon;
			var iconA = new di("routespeedsmarkerA");
			var iconB = new di("routespeedsmarkerB");

			p1 = new OpenLayers.Geometry.Point(lon1, lat1).transform(epsg4326, epsg900913);
			p2 = new OpenLayers.Geometry.Point(lon2, lat2).transform(epsg4326, epsg900913);

			var lonlatA = new OpenLayers.LonLat(p1.x, p1.y);
			var lonlatB = new OpenLayers.LonLat(p2.x, p2.y);

			markerA = new OpenLayers.Marker(lonlatA, iconA);
			markerB = new OpenLayers.Marker(lonlatB, iconB);

			var wh = WazeWrap.Require.DragElement();//require("Waze/Handler/DragElement");
			markerA.dragging = new wh(WM);
			markerB.dragging = new wh(WM);

			markerA.dragging.down = function (e) {
				lonlat = this.map.getLonLatFromViewPortPx(e.xy);
				if (lonlat === null) return;
				markerA_offset_click[0] = markerA.lonlat.lon - lonlat.lon;
				markerA_offset_click[1] = markerA.lonlat.lat - lonlat.lat;
			};
			markerB.dragging.down = function (e) {
				lonlat = this.map.getLonLatFromViewPortPx(e.xy);
				if (lonlat === null) return;
				markerB_offset_click[0] = markerB.lonlat.lon - lonlat.lon;
				markerB_offset_click[1] = markerB.lonlat.lat - lonlat.lat;
			};

			markerA.dragging.move = function (e) {
				lonlat = this.map.getLonLatFromViewPortPx(e.xy);
				markerA.lonlat.lon = lonlat.lon + markerA_offset_click[0];
				markerA.lonlat.lat = lonlat.lat + markerA_offset_click[1];
				markerLayer.drawMarker(markerA);
				panmap(this, e.xy.x, e.xy.y);
			};
			markerB.dragging.move = function (e) {
				lonlat = this.map.getLonLatFromViewPortPx(e.xy);
				markerB.lonlat.lon = lonlat.lon + markerB_offset_click[0];
				markerB.lonlat.lat = lonlat.lat + markerB_offset_click[1];
				markerLayer.drawMarker(markerB);
				panmap(this, e.xy.x, e.xy.y);
			};

			markerA.dragging.done = function (e) {

				if (routespeedsoption1) return;

				panningX = 0;
				panningY = 0;

				var lonlatA = new OpenLayers.LonLat(markerA.lonlat.lon, markerA.lonlat.lat).transform(epsg900913, epsg4326);
				var lonlatB = new OpenLayers.LonLat(markerB.lonlat.lon, markerB.lonlat.lat).transform(epsg900913, epsg4326);

				lon1 = parseInt(lonlatA.lon * 1000000.0 + 0.5) / 1000000.0;
				lat1 = parseInt(lonlatA.lat * 1000000.0 + 0.5) / 1000000.0;
				lon2 = parseInt(lonlatB.lon * 1000000.0 + 0.5) / 1000000.0;
				lat2 = parseInt(lonlatB.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(lon1, lat1, lon2, lat2);
			};
			markerB.dragging.done = function (e) {

				if (routespeedsoption1) return;

				panningX = 0;
				panningY = 0;

				var lonlatA = new OpenLayers.LonLat(markerA.lonlat.lon, markerA.lonlat.lat).transform(epsg900913, epsg4326);
				var lonlatB = new OpenLayers.LonLat(markerB.lonlat.lon, markerB.lonlat.lat).transform(epsg900913, epsg4326);

				lon1 = parseInt(lonlatA.lon * 1000000.0 + 0.5) / 1000000.0;
				lat1 = parseInt(lonlatA.lat * 1000000.0 + 0.5) / 1000000.0;
				lon2 = parseInt(lonlatB.lon * 1000000.0 + 0.5) / 1000000.0;
				lat2 = parseInt(lonlatB.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(lon1, lat1, lon2, lat2);
			};

			markerA.dragging.activate(iconA.$div);
			markerB.dragging.activate(iconB.$div);

			markerA.display(disp);
			markerB.display(disp);

			markerLayer.addMarker(markerA);
			markerLayer.addMarker(markerB);
		}
		else {
			p1 = new OpenLayers.Geometry.Point(lon1, lat1).transform(epsg4326, epsg900913);
			p2 = new OpenLayers.Geometry.Point(lon2, lat2).transform(epsg4326, epsg900913);

			markerA.lonlat.lon = p1.x;
			markerA.lonlat.lat = p1.y;
			markerB.lonlat.lon = p2.x;
			markerB.lonlat.lat = p2.y;

			markerA.display(disp);
			markerB.display(disp);

			markerLayer.drawMarker(markerA);
			markerLayer.drawMarker(markerB);
		}

		markerA.created = disp;
		markerB.created = disp;
	}
	//------------------------------------------------------------------------------------------------
	function showLayers(disp) {
		var WM = window.W.map;

		var rlayers1 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds1");
		var rlayers2 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds2");
		var rlayers3 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds3");
		var rlayers4 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds4");
		var rlayers5 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds5");
		var routeLayer1 = rlayers1[0];
		var routeLayer2 = rlayers2[0];
		var routeLayer3 = rlayers3[0];
		var routeLayer4 = rlayers4[0];
		var routeLayer5 = rlayers5[0];

		if (routeLayer1 === undefined) return;
		if (routeLayer2 === undefined) return;
		if (routeLayer3 === undefined) return;
		if (routeLayer4 === undefined) return;
		if (routeLayer5 === undefined) return;

		routeLayer1.setVisibility(disp);
		routeLayer2.setVisibility(disp);
		routeLayer3.setVisibility(disp);
		routeLayer4.setVisibility(disp);
		routeLayer5.setVisibility(disp);
	}
	//--------------------------------------------------------------------------------------------------------
	function showMarkers(disp) {
		var WM = window.W.map;

		var mlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeedsMarkers");
		var markerLayer = mlayers[0];

		if (markerLayer === undefined) return false;
		if (markerA === undefined) return false;
		if (markerB === undefined) return false;

		if (markerA.created) markerA.display(disp);
		if (markerB.created) markerB.display(disp);

		return (markerA.created && markerB.created);
	}
	//------------------------------------------------------------------------------------------------
	function reverseMarkers() {
		var WM = window.W.map;

		var mlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeedsMarkers");
		var markerLayer = mlayers[0];

		if (markerLayer === undefined) return;
		if (markerA === undefined || !markerA.created) return;
		if (markerB === undefined || !markerB.created) return;

		var copy = markerA.lonlat;
		markerA.lonlat = markerB.lonlat;
		markerB.lonlat = copy;

		markerLayer.drawMarker(markerA);
		markerLayer.drawMarker(markerB);
	}
	//------------------------------------------------------------------------------------------------
	function loopWMERouteSpeeds() {

		if (routespeedsoption1) return;

		var tabOpen = $('#user-tabs > .nav-tabs > li > a[href="#sidepanel-routespeeds"]').attr('aria-expanded') == "true";
		if (!tabOpen) {
			if (tabswitched !== 1) {
				tabswitched = 1;
				showLayers(false);
				showMarkers(false);
				showClosures(0);
			}
			return;
		}
		else {
			if (tabswitched !== 2) {
				tabswitched = 2;
				showLayers(true);
				showMarkers(true);
				showClosures(1);
			}
		}

		//var routespeedsbutton = getId('routespeeds-button-livemap');
		//if (routespeedsbutton == 'undefined') return;
		//var routespeedsbutton_ofsW = routespeedsbutton.offsetWidth;
		//var routespeedsbutton_ofsH = routespeedsbutton.offsetHeight;
		//console.log(routespeedsbutton_ofsW, routespeedsbutton_ofsH);
		//if (routespeedsbutton_ofsW == 0 || routespeedsbutton_ofsH==0) return;


		var WM = window.W.map;

		var rlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds1");
		if (rlayers.length === 0) {

			var drc_style1 = new OpenLayers.Style({
				strokeDashstyle: 'solid',
				strokeColor: "${strokeColor}",
				strokeOpacity: 1.0,
				strokeWidth: "${strokeWidth}",
				fillColor: '#0040FF',
				fillOpacity: 1.0,
				pointRadius: "${pointRadius}",
				label: "${labelText}",
				fontFamily: "Tahoma, Courier New",
				labelOutlineColor: '#FFFFFF',
				labelOutlineWidth: 0,
				fontColor: "${fontColor}",
				fontOpacity: 1.0,
				fontSize: "10px",
				display: 'block'
			});

			var drc_style2 = new OpenLayers.Style({
				strokeDashstyle: 'solid',
				strokeColor: "${strokeColor}",
				strokeOpacity: 1.0,
				strokeWidth: "${strokeWidth}",
				fillColor: '#0040FF',
				fillOpacity: 1.0,
				pointRadius: "${pointRadius}",
				label: "${labelText}",
				fontFamily: "Tahoma, Courier New",
				labelOutlineColor: '#FFFFFF',
				labelOutlineWidth: 0,
				fontColor: "${fontColor}",
				fontOpacity: 1.0,
				fontSize: "10px",
				display: 'block'
			});

			var drc_style3 = new OpenLayers.Style({
				strokeDashstyle: 'solid',
				strokeColor: "${strokeColor}",
				strokeOpacity: 1.0,
				strokeWidth: "${strokeWidth}",
				fillColor: '#0040FF',
				fillOpacity: 1.0,
				pointRadius: "${pointRadius}",
				label: "${labelText}",
				fontFamily: "Tahoma, Courier New",
				labelOutlineColor: '#FFFFFF',
				labelOutlineWidth: 0,
				fontColor: "${fontColor}",
				fontOpacity: 1.0,
				fontSize: "10px",
				display: 'block'
			});

			var drc_style4 = new OpenLayers.Style({
				strokeDashstyle: 'solid',
				strokeColor: "${strokeColor}",
				strokeOpacity: 1.0,
				strokeWidth: "${strokeWidth}",
				fillColor: '#0040FF',
				fillOpacity: 1.0,
				pointRadius: "${pointRadius}",
				label: "${labelText}",
				fontFamily: "Tahoma, Courier New",
				labelOutlineColor: '#FFFFFF',
				labelOutlineWidth: 0,
				fontColor: "${fontColor}",
				fontOpacity: 1.0,
				fontSize: "10px",
				display: 'block'
			});

			var drc_style5 = new OpenLayers.Style({
				strokeDashstyle: 'solid',
				strokeColor: "${strokeColor}",
				strokeOpacity: 1.0,
				strokeWidth: "${strokeWidth}",
				fillColor: '#0040FF',
				fillOpacity: 1.0,
				pointRadius: "${pointRadius}",
				label: "${labelText}",
				fontFamily: "Tahoma, Courier New",
				labelOutlineColor: '#FFFFFF',
				labelOutlineWidth: 0,
				fontColor: "${fontColor}",
				fontOpacity: 1.0,
				fontSize: "10px",
				display: 'block'
			});

			var drc_mapLayer1 = new OpenLayers.Layer.Vector("Route Speeds", {
				displayInLayerSwitcher: true,
				uniqueName: "__DrawRouteSpeeds1",
				styleMap: new OpenLayers.StyleMap(drc_style1)
			});

			var drc_mapLayer2 = new OpenLayers.Layer.Vector("Route Speeds 2", {
				displayInLayerSwitcher: false,
				uniqueName: "__DrawRouteSpeeds2",
				styleMap: new OpenLayers.StyleMap(drc_style2)
			});

			var drc_mapLayer3 = new OpenLayers.Layer.Vector("Route Speeds 3", {
				displayInLayerSwitcher: false,
				uniqueName: "__DrawRouteSpeeds3",
				styleMap: new OpenLayers.StyleMap(drc_style3)
			});

			var drc_mapLayer4 = new OpenLayers.Layer.Vector("Route Speeds 4", {
				displayInLayerSwitcher: false,
				uniqueName: "__DrawRouteSpeeds4",
				styleMap: new OpenLayers.StyleMap(drc_style4)
			});

			var drc_mapLayer5 = new OpenLayers.Layer.Vector("Route Speeds 5", {
				displayInLayerSwitcher: false,
				uniqueName: "__DrawRouteSpeeds5",
				styleMap: new OpenLayers.StyleMap(drc_style5)
			});

			I18n.translations[I18n.currentLocale()].layers.name["__DrawRouteSpeeds1"] = "Route Speeds";
			I18n.translations[I18n.currentLocale()].layers.name["__DrawRouteSpeeds2"] = "Route Speeds 2";
			I18n.translations[I18n.currentLocale()].layers.name["__DrawRouteSpeeds3"] = "Route Speeds 3";
			I18n.translations[I18n.currentLocale()].layers.name["__DrawRouteSpeeds4"] = "Route Speeds 4";
			I18n.translations[I18n.currentLocale()].layers.name["__DrawRouteSpeeds5"] = "Route Speeds 5";

			drc_mapLayer1.setVisibility(true);
			drc_mapLayer2.setVisibility(true);
			drc_mapLayer3.setVisibility(true);
			drc_mapLayer4.setVisibility(true);
			drc_mapLayer5.setVisibility(true);

			WM.addLayer(drc_mapLayer1);
			WM.addLayer(drc_mapLayer2);
			WM.addLayer(drc_mapLayer3);
			WM.addLayer(drc_mapLayer4);
			WM.addLayer(drc_mapLayer5);

			return;
		}

		var mlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeedsMarkers");
		if (mlayers.length === 0) {

			var drc_mapLayer = new OpenLayers.Layer.Markers("Route Speeds Markers", {
				displayInLayerSwitcher: false,
				uniqueName: "__DrawRouteSpeedsMarkers"
			});

			I18n.translations[I18n.currentLocale()].layers.name["__DrawRouteSpeedsMarkers"] = "Route Speeds Markers";
			WM.addLayer(drc_mapLayer);
			drc_mapLayer.setVisibility(true);

			createMarkers(16, 52, 17, 53, false);

			return;
		}

		if (jqueryinfo === 1) {
			jqueryinfo = 2;
			console.log('WME Route Speeds: jQuery reloaded ver. ' + jQuery.fn.jquery);
		}
		if (jqueryinfo === 0) {
			if (typeof jQuery === 'undefined') {
				console.log('WME Route Speeds: jQuery current ver. ' + jQuery.fn.jquery);

				var 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);
				jqueryinfo = 1;
			}
		}


		var rlayers1 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds1");
		var rlayers2 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds2");
		var rlayers3 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds3");
		var rlayers4 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds4");
		var rlayers5 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds5");
		var routeLayer1 = rlayers1[0];
		var routeLayer2 = rlayers2[0];
		var routeLayer3 = rlayers3[0];
		var routeLayer4 = rlayers4[0];
		var routeLayer5 = rlayers5[0];
		if (routeLayer1 === undefined) return;
		if (routeLayer2 === undefined) return;
		if (routeLayer3 === undefined) return;
		if (routeLayer4 === undefined) return;
		if (routeLayer5 === undefined) return;

		if (routeLayer1.getVisibility() === false) {
			if (routeLayer2.getVisibility() === true) {
				routeLayer2.setVisibility(false);
				routeLayer3.setVisibility(false);
				routeLayer4.setVisibility(false);
				routeLayer5.setVisibility(false);
			}
		}
		else {
			if (routeLayer2.getVisibility() === false) {
				routeLayer2.setVisibility(true);
				routeLayer3.setVisibility(true);
				routeLayer4.setVisibility(true);
				routeLayer5.setVisibility(true);
			}
		}

		var numSelected = WazeWrap.getSelectedFeatures().length;
		var seg1 = WazeWrap.getSelectedFeatures()[0];
		var seg2 = WazeWrap.getSelectedFeatures()[1];

		if (seg1 !== undefined && seg2 !== undefined) {
			if (!selected) {
				selected = 1;

				var coords1 = getSegmentMidPoint(seg1, 0);
				var coords2 = getSegmentMidPoint(seg2, 1);

				var lon1 = parseInt(coords1.lon * 1000000.0 + 0.5) / 1000000.0;
				var lat1 = parseInt(coords1.lat * 1000000.0 + 0.5) / 1000000.0;
				var lon2 = parseInt(coords2.lon * 1000000.0 + 0.5) / 1000000.0;
				var lat2 = parseInt(coords2.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;
				}

				createMarkers(lon1, lat1, lon2, lat2, true);

				leftHand = false;
				if (seg1.model.model.isLeftHand) leftHand = true;
				if (seg2.model.model.isLeftHand) leftHand = true;

				requestRouteFromLiveMap(lon1, lat1, lon2, lat2);
			}
		}
		else {
			if (seg1 !== undefined || seg2 !== undefined) {
				if (selected) {
					selected = 0;

					routeLayer1.removeAllFeatures();
					routeLayer2.removeAllFeatures();
					routeLayer3.removeAllFeatures();
					routeLayer4.removeAllFeatures();
					routeLayer5.removeAllFeatures();

					getId('routespeeds-summary1').style.visibility = 'hidden';
					getId('routespeeds-summary2').style.visibility = 'hidden';
					getId('routespeeds-summary3').style.visibility = 'hidden';
					getId('routespeeds-summary4').style.visibility = 'hidden';
					getId('routespeeds-summary5').style.visibility = 'hidden';

					getId('routespeeds-summary1').className = 'routespeeds_summary_classA';
					getId('routespeeds-summary2').className = 'routespeeds_summary_classA';
					getId('routespeeds-summary3').className = 'routespeeds_summary_classA';
					getId('routespeeds-summary4').className = 'routespeeds_summary_classA';
					getId('routespeeds-summary5').className = 'routespeeds_summary_classA';
				}
			}
		}
	}
	//--------------------------------------------------------------------------------------------------------
	function createRouteFeatures(id, routewsp, routeodc) {

		var WM = window.W.map;

		var rlayers;
		if (id == 1) rlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds1");
		if (id == 2) rlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds2");
		if (id == 3) rlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds3");
		if (id == 4) rlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds4");
		if (id == 5) rlayers = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds5");
		var routeLayer = rlayers[0];
		if (routeLayer === undefined) return;

		var lineFeatures = [];
		var labelFeatures = [];
		var lines = [];
		var outlinepoints = [];

		var odc = 0;
		var odclen = routeodc[odc].length;
		var odctime = 0;
		if (routespeedsoption7) odctime = routeodc[odc].crossTime;
		else odctime = routeodc[odc].crossTimeWithoutRealTime;
		var odcx = 0;
		var odcy = 0;
		if (odc + 1 < routeodc.length) {
			odcx = routeodc[odc + 1].path.x;
			odcy = routeodc[odc + 1].path.y;
		}

		var speed = 0;
		if (odctime > 0) speed = 3.6 * odclen / odctime;
		var speedtekst = parseInt(speed);
		var kolor = getColor(speed);

		var ptA = new OpenLayers.Geometry.Point(0, 0);
		var ptB = new OpenLayers.Geometry.Point(0, 0);

		var doubletraffic = false;
		var p1 = null;
		var p2 = null;
		var doublepoints = {};
		var wsp1, wsp2, dlon, dlat, dx, dy, label, len, i;


		//wykrycie czy trasa przechodzi dwa razy przez te same punkty, jeżeli tak to rysowanie trasy z odstępem, aby był widoczny przebieg trasy
		//(detection whether the route passes through the same points twice, if so drawing the route with a gap to make the route visible)
		for (i = 0; i < routewsp.length - 1; i++) {
			wsp1 = routewsp[i + 0];
			wsp2 = routewsp[i + 1];

			dlon = Math.abs(wsp1.x - wsp2.x);
			dlat = Math.abs(wsp1.y - wsp2.y);

			if (dlon < 0.0000001 && dlat < 0.0000001) continue;

			var s1 = parseInt(wsp1.x * 10000000 + 0.5) + ',' + parseInt(wsp1.y * 10000000 + 0.5);
			var s2 = parseInt(wsp2.x * 10000000 + 0.5) + ',' + parseInt(wsp2.y * 10000000 + 0.5);

			if (s1 === s2) continue;

			if (doublepoints[s1] === undefined) doublepoints[s1] = 0;
			if (doublepoints[s2] === undefined) doublepoints[s2] = 0;
			doublepoints[s1]++;
			doublepoints[s2]++;

			if (doublepoints[s2] >= 2) {
				doubletraffic = true;
				break;
			}
		}

		var doubletrafficoffset = 0;
		if (doubletraffic) {
			doubletrafficoffset = 11 * Math.pow(2.0, 5 - W.map.getZoom());
		}


		for (i = 0; i < routewsp.length - 1; i++) {
			wsp1 = routewsp[i + 0];
			wsp2 = routewsp[i + 1];

			if (i === 0) {
				ptA.x = wsp1.x;
				ptA.y = wsp1.y;
				ptA = ptA.transform(epsg4326, epsg900913);
				//var p = new drc_OpenLayers.Geometry.Point(wsp1.x, wsp1.y).transform(epsg4326, epsg900913);
				//var pt = new drc_OpenLayers.Geometry.Point(p.x, p.y);
				//var textFeature = new drc_OpenLayers.Feature.Vector( ptA, {labelText: "A", pointRadius: 8, fontColor: '#FFFFFF' } );
				//labelFeatures.push(textFeature);
			}
			if (i === routewsp.length - 2) {
				ptB.x = wsp2.x;
				ptB.y = wsp2.y;
				ptB = ptB.transform(epsg4326, epsg900913);
				//var p = new drc_OpenLayers.Geometry.Point(wsp2.x, wsp2.y).transform(epsg4326, epsg900913);
				//var pt = new drc_OpenLayers.Geometry.Point(p.x, p.y);
				//var textFeature = new drc_OpenLayers.Feature.Vector( ptB, {labelText: "B", pointRadius: 8, fontColor: '#FFFFFF' } );
				//labelFeatures.push(textFeature);
			}

			dx = Math.abs(wsp1.x - odcx);
			dy = Math.abs(wsp1.y - odcy);

			//console.log(wsp1, odcx, odcy, dx, dy);

			if (dx < 0.000001 && dy < 0.000001) {

				if (!routespeedsoption3) {
					label = addLabel(lines, speedtekst, routespeedsoption2, odctime, odclen, routespeedsoption4, id);
					if (label !== null) labelFeatures.push(label);
				}
				while (lines.length > 0) lines.pop();

				if (odc + 1 < routeodc.length) {
					odc++;
					odclen = routeodc[odc].length;
					if (routespeedsoption7) odctime = routeodc[odc].crossTime;
					else odctime = routeodc[odc].crossTimeWithoutRealTime;
					if (odc + 1 < routeodc.length) {
						odcx = routeodc[odc + 1].path.x;
						odcy = routeodc[odc + 1].path.y;
					}

					speed = 0;
					if (odctime > 0) speed = 3.6 * odclen / odctime;
					speedtekst = parseInt(speed);
					kolor = getColor(speed);

				}
			}

			dlon = Math.abs(wsp1.x - wsp2.x);
			dlat = Math.abs(wsp1.y - wsp2.y);

			if (dlon < 0.0000001 && dlat < 0.0000001) continue;

			var p3 = new OpenLayers.Geometry.Point(wsp1.x, wsp1.y).transform(epsg4326, epsg900913);
			var p4 = new OpenLayers.Geometry.Point(wsp2.x, wsp2.y).transform(epsg4326, epsg900913);

			if (doubletraffic) {
				dx = p4.x - p3.x;
				dy = p4.y - p3.y;
				var r = Math.sqrt(dx * dx + dy * dy);
				var angle = Math.acos(dx / r);
				if (dy < 0) angle = -angle;
				angle = angle - 0.5 * Math.PI;
				if (leftHand) angle += Math.PI;

				p3.x += doubletrafficoffset * Math.cos(angle) * 0.6;
				p3.y += doubletrafficoffset * Math.sin(angle) * 0.6;
				p4.x += doubletrafficoffset * Math.cos(angle) * 0.6;
				p4.y += doubletrafficoffset * Math.sin(angle) * 0.6;

				if (p1 !== null && p2 !== null) {

					var Ax = p1.x;
					var Ay = p1.y;
					var Bx = p2.x;
					var By = p2.y;
					var Cx = p3.x;
					var Cy = p3.y;
					var Dx = p4.x;
					var Dy = p4.y;

					dx = Cx - Bx;
					dy = Cy - By;

					var delta = Math.sqrt(dx * dx + dy * dy);

					var mx = ((By - Ay) * (Dx - Cx) - (Dy - Cy) * (Bx - Ax));
					var my = ((Dy - Cy) * (Bx - Ax) - (By - Ay) * (Dx - Cx));

					if (Math.abs(mx) > 0.000000001 && Math.abs(my) > 0.000000001 && delta > 0.1) {

						var x = ((Bx - Ax) * (Dx * Cy - Dy * Cx) - (Dx - Cx) * (Bx * Ay - By * Ax)) / mx;
						var y = ((Dy - Cy) * (Bx * Ay - By * Ax) - (By - Ay) * (Dx * Cy - Dy * Cx)) / my;

						var dx2 = x - Bx;
						var dy2 = y - By;
						var delta2 = Math.sqrt(dx2 * dx2 + dy2 * dy2);

						if (delta2 < 1000) {      // checking if point of crossing is close to segments

							len = lineFeatures.length;
							if (len > 0) {
								var lf = lineFeatures[len - 1];
								lf.geometry.components[1].x = x;
								lf.geometry.components[1].y = y;
							}

							p3.x = x;
							p3.y = y;
						}
					}
				}
			}

			outlinepoints.push(p3);
			outlinepoints.push(p4);

			let points = [];
			points.push(p3);
			points.push(p4);

			let line = new OpenLayers.Geometry.LineString(points);
			lines.push(line);

			let lineFeature = new OpenLayers.Feature.Vector(line, { strokeColor: kolor, labelText: '', strokeWidth: 10 });

			lineFeatures.push(lineFeature);

			p1 = p3;
			p2 = p4;
		}

		if (!routespeedsoption3) {
			label = addLabel(lines, speedtekst, routespeedsoption2, odctime, odclen, routespeedsoption4, id);
			if (label !== null) labelFeatures.push(label);
		}
		while (lines.length > 0) lines.pop();

		let outlinestring = new OpenLayers.Geometry.LineString(outlinepoints);
		let outlineFeature = new OpenLayers.Feature.Vector(outlinestring, { strokeColor: '#404040', labelText: '', strokeWidth: 12 });
		routeLayer.addFeatures(outlineFeature);

		routeLayer.addFeatures(lineFeatures);
		routeLayer.addFeatures(labelFeatures);



		let summarylen = 0;
		let summarysec = 0;
		if (routespeedsoption7) {
			for (let i = 0; i < routeodc.length; i++) {
				summarylen += routeodc[i].length;
				summarysec += routeodc[i].crossTime;
			}
		}
		else {
			for (let i = 0; i < routeodc.length; i++) {
				summarylen += routeodc[i].length;
				summarysec += routeodc[i].crossTimeWithoutRealTime;
			}
		}
		len = summarylen / 1000.0;
		let sec = summarysec % 60;
		let min = (summarysec - sec) % 3600;
		let hou = (summarysec - sec - min) % 216000;
		min = (min / 60) % 60;
		hou = (hou / 3600);

		let t = '';
		if (hou < 10) t += '0' + hou + ":"; else t += hou + ":";
		if (min < 10) t += '0' + min + ":"; else t += min + ":";
		if (sec < 10) t += '0' + sec;
		else t += sec;

		let lenmph = len / 1.609;

		let avgspeed = (summarylen / 1000.0) / (summarysec / 3600.0);
		if (routespeedsoption4) avgspeed = avgspeed / 1.609;

		let summaryobj;
		if (id == 1) summaryobj = getId('routespeeds-summary1');
		if (id == 2) summaryobj = getId('routespeeds-summary2');
		if (id == 3) summaryobj = getId('routespeeds-summary3');
		if (id == 4) summaryobj = getId('routespeeds-summary4');
		if (id == 5) summaryobj = getId('routespeeds-summary5');

		let html;
		if (id == 1) html = '<div class=routespeeds_header style="background: #4d4dcd; color: #e0e0e0; "></div>' + '<span style="color: #404040;">Route 1</span> ';
		if (id == 2) html = '<div class=routespeeds_header style="background: #d34f8a; color: #e0e0e0; "></div>' + '<span style="color: #404040;">Route 2</span> ';
		if (id == 3) html = '<div class=routespeeds_header style="background: #188984; color: #e0e0e0; "></div>' + '<span style="color: #404040;">Route 3</span> ';
		if (id == 4) html = '<div class=routespeeds_header style="background: #cafa27; color: #404040; "></div>' + '<span style="color: #404040;">Route 4</span> ';
		if (id == 5) html = '<div class=routespeeds_header style="background: #ffca3f; color: #e0e0e0; "></div>' + '<span style="color: #404040;">Route 5</span> ';

		let lenstr = precFloat(len, 2);
		let u1 = 'km';
		let u2 = 'km&#47;h';
		if (routespeedsoption4) {
			lenstr = precFloat(lenmph, 2);
			u1 = 'miles';
			u2 = 'mph';
		}

		html += '<div style="min-width:50px; display:inline-block; text-align:right;" ><b>' + lenstr + '</b></div>';
		html += '<span style="font-size:11px;"> ' + u1 + '</span> &nbsp;<b>' + t + '</b>';
		html += '<div style="display:inline-block; min-width:40px; text-align:right; color:#404040" >' + precFloat(avgspeed, 1) + '</div> <span style="font-size:11px;">' + u2 + '</span>';

		summaryobj.innerHTML = html;

		if (id === routeSelected) summaryobj.className = 'routespeeds_summary_classB';
		summaryobj.style.visibility = 'visible';
	}
	//--------------------------------------------------------------------------------------------------------
	function precFloat(f, prec) {
		if (!isFinite(f)) return "&mdash;";

		if (f < 0)
			f -= Math.pow(0.1, prec) * 0.5;
		else
			f += Math.pow(0.1, prec) * 0.5;

		let ipart = parseInt(f);
		let fpart = Math.abs(f - ipart);
		f = ipart;

		if (fpart === '0') fpart = '0.0';
		fpart += '0000000000000000';
		if (prec) f += fpart.substr(1, prec + 1);

		return f;
	}
	//--------------------------------------------------------------------------------------------------------
	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(x1, y1, x2, y2) {
		var atTime = getnowtoday();

		var numRoutes = 1;
		if (routespeedsoption5) numRoutes = parseInt(routespeedsoption6);
		var numPaths = (routespeedsoption5 && routespeedsoption12 && !routespeedsoption15) ? numRoutes * 10 : numRoutes; //Routing Order - last condition disables Try More option

		var routeType = (routespeedsoption13 === 3) ? "TIME" : "HISTORIC_TIME";

		var avoidTollRoads = routespeedsoption8;
		var avoidPrimaries = routespeedsoption9;
		var avoidTrails = routespeedsoption10;
		var avoidLongTrails = routespeedsoption11;
		var allowUTurns = routespeedsoption14;
		var avoidDifficult = routespeedsoption16;
		var avoidFerries = routespeedsoption17;
		var vehType = routespeedsoption18;

		var options = {
			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(",");
			}
		};

		options.add("AVOID_TOLL_ROADS", avoidTollRoads, false);
		options.add("AVOID_PRIMARIES", avoidPrimaries, false);
		options.add("AVOID_DANGEROUS_TURNS", avoidDifficult, false);
		options.add("AVOID_FERRIES", avoidFerries, false);
		options.add("ALLOW_UTURNS", allowUTurns, true);

		if (avoidLongTrails) { options.put("AVOID_LONG_TRAILS", true); }
		else if (avoidTrails) { options.put("AVOID_TRAILS", true); }
		else { options.put("AVOID_LONG_TRAILS", false); }


		var url = getRoutingManager();
		let expressPass = _settings.passes.map(key => key);
		var data = {
			from: "x:" + x1 + " y:" + y1,
			to: "x:" + x2 + " y:" + y2,
			returnJSON: true,
			returnGeometries: true,
			returnInstructions: true,
			timeout: 60000,
			at: atTime,
			type: routeType,
			nPaths: numPaths,
			clientVersion: '4.0.0',
			options: options.get(),
			vehicleType: vehType,
			subscription: expressPass
		};

		routewait = 1;

		getId('routespeeds-error').innerHTML = "";

		console.time('WME Route Speeds: routing time');

		$.ajax({
			dataType: "json",
			cache: false,
			url: url,
			data: data,
			traditional: true,
			dataFilter: function (data, dataType) {
				return data.replace(/NaN/g, '0');
			},
			error: function (req, textStatus, errorThrown) {
				let str = "Route request failed" + (textStatus !== null ? " with " + textStatus : "") + "!";
				str += "<br>" + errorThrown;
				handleRouteRequestError(str);
			},
			success: function (json) {
				if (json.error !== undefined) {
					let str = json.error;
					str = str.replace("|", "<br>");
					handleRouteRequestError(str);
				}
				else {

					routewsp1 = [];
					routeodc1 = [];
					routewsp2 = [];
					routeodc2 = [];
					routewsp3 = [];
					routeodc3 = [];
					routewsp4 = [];
					routeodc4 = [];
					routewsp5 = [];
					routeodc5 = [];



					if (json.coords !== undefined) {
						console.log("WME Route Speeds: 1 route received" + " (" + numPaths + " requested)");

						if (routeSelected > 1) routeSelected = 1;

						routewsp1 = json.coords;
						routeodc1 = json.response.results;

					}
					if (json.alternatives !== undefined) {
						console.log("WME Route Speeds: " + json.alternatives.length + " routes received" + " (" + numPaths + " requested)");

						var sortByField = (routespeedsoption13 === 2) ? "length" : routespeedsoption7 ? "crossTime" : "crossTimeWithoutRealTime";

						if (!routespeedsoption15) {                                    // Routing Order
							json.alternatives.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 (json.alternatives.length > numRoutes) {
							json.alternatives = json.alternatives.slice(0, numRoutes);
						}

						if (routeSelectedLast) routeSelected = routeSelectedLast;
						if (routeSelected > json.alternatives.length) routeSelected = json.alternatives.length;

						for (let n = 0; n < json.alternatives.length; n++) {

							if (n === 0) routewsp1 = json.alternatives[n].coords;
							if (n === 0) routeodc1 = json.alternatives[n].response.results;


							if (n === 1) routewsp2 = json.alternatives[n].coords;
							if (n === 1) routeodc2 = json.alternatives[n].response.results;

							if (n === 2) routewsp3 = json.alternatives[n].coords;
							if (n === 2) routeodc3 = json.alternatives[n].response.results;

							if (n === 3) routewsp4 = json.alternatives[n].coords;
							if (n === 3) routeodc4 = json.alternatives[n].response.results;

							if (n === 4) routewsp5 = json.alternatives[n].coords;
							if (n === 4) routeodc5 = json.alternatives[n].response.results;
						}
					}

					rezoom();
				}

				getId('routespeeds-button-livemap').style.backgroundColor = '';
				getId('routespeeds-button-reverse').style.backgroundColor = '';
			},
			complete: function () {
				console.timeEnd('WME Route Speeds: routing time');
				routewait = 0;
			}
		});
	}
	//--------------------------------------------------------------------------------------------------------
	function handleRouteRequestError(message) {
		console.log("WME Route Speeds: route request error: " + message.replace("<br>", "\n"));

		getId('routespeeds-button-livemap').style.backgroundColor = '';
		getId('routespeeds-button-reverse').style.backgroundColor = '';

		getId('routespeeds-summary1').innerHTML = '';
		getId('routespeeds-summary2').innerHTML = '';
		getId('routespeeds-summary3').innerHTML = '';
		getId('routespeeds-summary4').innerHTML = '';
		getId('routespeeds-summary5').innerHTML = '';

		getId('routespeeds-summary1').style.visibility = 'hidden';
		getId('routespeeds-summary2').style.visibility = 'hidden';
		getId('routespeeds-summary3').style.visibility = 'hidden';
		getId('routespeeds-summary4').style.visibility = 'hidden';
		getId('routespeeds-summary5').style.visibility = 'hidden';

		let WM = window.W.map;
		let rlayers1 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds1");
		let rlayers2 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds2");
		let rlayers3 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds3");
		let rlayers4 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds4");
		let rlayers5 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds5");
		let routeLayer1 = rlayers1[0];
		let routeLayer2 = rlayers2[0];
		let routeLayer3 = rlayers3[0];
		let routeLayer4 = rlayers4[0];
		let routeLayer5 = rlayers5[0];
		if (routeLayer1 !== undefined) routeLayer1.removeAllFeatures();
		if (routeLayer2 !== undefined) routeLayer2.removeAllFeatures();
		if (routeLayer3 !== undefined) routeLayer3.removeAllFeatures();
		if (routeLayer4 !== undefined) routeLayer4.removeAllFeatures();
		if (routeLayer5 !== undefined) routeLayer5.removeAllFeatures();

		getId('routespeeds-error').innerHTML = "<br>" + message;
	}
	//--------------------------------------------------------------------------------------------------------
	function livemapRouteClick() {
		routeSelected = 1;
		routeSelectedLast = 0;

		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 (routespeedsoption1) return;
		if (routewait) return;

		routewsp1 = [];
		routeodc1 = [];
		routewsp2 = [];
		routeodc2 = [];
		routewsp3 = [];
		routeodc3 = [];
		routewsp4 = [];
		routeodc4 = [];
		routewsp5 = [];
		routeodc5 = [];

		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, true);

		if (pastedlink) {
			clickA();
		}

		requestRouteFromLiveMap(x1, y1, x2, y2);
	}
	//--------------------------------------------------------------------------------------------------------
	function reverseRoute() {

		if (routespeedsoption1) return;
		if (routewait) return;

		routewsp1 = [];
		routeodc1 = [];
		routewsp2 = [];
		routeodc2 = [];
		routewsp3 = [];
		routeodc3 = [];
		routewsp4 = [];
		routeodc4 = [];
		routewsp5 = [];
		routeodc5 = [];

		let stra = getId('sidepanel-routespeeds-b').value;
		let strb = getId('sidepanel-routespeeds-a').value;
		if (stra === "") return;
		if (strb === "") return;

		getId('sidepanel-routespeeds-a').value = stra;
		getId('sidepanel-routespeeds-b').value = strb;

		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 objprog2 = getId('routespeeds-button-reverse');
		objprog2.style.backgroundColor = '#FF8000';

		createMarkers(x1, y1, x2, y2, true);

		requestRouteFromLiveMap(x1, y1, x2, y2);
	}
	//--------------------------------------------------------------------------------------------------------
	function resetOptions() {

		getId('routespeeds-option5').checked = routespeedsoption5 = true;
		getId('routespeeds-option6').value = routespeedsoption6 = 3;
		getId('routespeeds-option12').checked = routespeedsoption12 = false;

		getId('routespeeds-option7').checked = routespeedsoption7 = false;

		getId('routespeeds-option13').value = routespeedsoption13 = 1;

		getId('routespeeds-option8').checked = routespeedsoption8 = false;
		getId('routespeeds-option9').checked = routespeedsoption9 = false;
		getId('routespeeds-option10').checked = routespeedsoption10 = true;
		getId('routespeeds-option11').checked = routespeedsoption11 = false;
		getId('routespeeds-option14').checked = routespeedsoption14 = true;
		getId('routespeeds-option15').checked = routespeedsoption15 = true;
		getId('routespeeds-option16').checked = routespeedsoption16 = false;
		getId('routespeeds-option17').checked = routespeedsoption17 = false;
		getId('routespeeds-option18').value = routespeedsoption18 = 'PRIVATE';

		update_adv_switches();
	}
	//--------------------------------------------------------------------------------------------------------
	function resetOptionsToLivemapRouteClick() {
		if (routewait) return;

		resetOptions();

		_modelPasses.forEach(pass => (getId(`routespeeds-pass-${pass.key}`).checked = false));
		_settings.passes = [];

		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function hourChange() {

		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function dayChange() {

		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickA() { gotoMarker(markerA); }
	function clickB() { gotoMarker(markerB); }
	//--------------------------------------------------------------------------------------------------------
	function gotoMarker(marker) {

		if (routespeedsoption1 || marker === undefined || !marker.created) return;

		let pt = marker.lonlat.toPoint();
		let zoom = window.W.map.getZoom();

		window.W.map.setCenter([pt.x, pt.y], zoom);
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption1() {
		var WM = window.W.map;

		routespeedsoption1 = (getId('routespeeds-option1').checked === true);

		if (routespeedsoption1) {
			getId('sidepanel-routespeeds').style.color = "#A0A0A0";

			getId('routespeeds-summary1').innerHTML = '';
			getId('routespeeds-summary2').innerHTML = '';
			getId('routespeeds-summary3').innerHTML = '';
			getId('routespeeds-summary4').innerHTML = '';
			getId('routespeeds-summary5').innerHTML = '';

			getId('routespeeds-summary1').style.visibility = 'hidden';
			getId('routespeeds-summary2').style.visibility = 'hidden';
			getId('routespeeds-summary3').style.visibility = 'hidden';
			getId('routespeeds-summary4').style.visibility = 'hidden';
			getId('routespeeds-summary5').style.visibility = 'hidden';

			let rlayers1 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds1");
			let rlayers2 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds2");
			let rlayers3 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds3");
			let rlayers4 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds4");
			let rlayers5 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds5");
			let routeLayer1 = rlayers1[0];
			let routeLayer2 = rlayers2[0];
			let routeLayer3 = rlayers3[0];
			let routeLayer4 = rlayers4[0];
			let routeLayer5 = rlayers5[0];

			if (routeLayer1 !== undefined) routeLayer1.removeAllFeatures();
			if (routeLayer2 !== undefined) routeLayer2.removeAllFeatures();
			if (routeLayer3 !== undefined) routeLayer3.removeAllFeatures();
			if (routeLayer4 !== undefined) routeLayer4.removeAllFeatures();
			if (routeLayer5 !== undefined) routeLayer5.removeAllFeatures();

			showMarkers(false);
			showClosures(0);
		}
		else {
			getId('sidepanel-routespeeds').style.color = "";

			if (showMarkers(true)) livemapRoute();
			showClosures(1);
		}
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption2() {
		routespeedsoption2 = (getId('routespeeds-option2').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption3() {
		routespeedsoption3 = (getId('routespeeds-option3').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption4() {
		routespeedsoption4 = (getId('routespeeds-option4').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption5() {
		routeSelected = 1;
		routeSelectedLast = 0;

		routespeedsoption5 = (getId('routespeeds-option5').checked === true);
		getId('routespeeds-option5-span').style.display = routespeedsoption5 ? 'inline' : 'none';
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption6() {
		routespeedsoption5 = (getId('routespeeds-option5').checked === true);
		update_adv_switches();

		routespeedsoption6 = parseInt(getId('routespeeds-option6').value);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption7() {
		routespeedsoption7 = (getId('routespeeds-option7').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption8() {
		routespeedsoption8 = (getId('routespeeds-option8').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption9() {
		routespeedsoption9 = (getId('routespeeds-option9').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption10() {
		routespeedsoption10 = (getId('routespeeds-option10').checked === true);

		routespeedsoption11 = false;
		getId('routespeeds-option11').checked = false;

		update_adv_switches();
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption11() {
		routespeedsoption11 = (getId('routespeeds-option11').checked === true);

		routespeedsoption10 = false;
		getId('routespeeds-option10').checked = false;

		update_adv_switches();
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption12() {
		routespeedsoption12 = (getId('routespeeds-option12').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption13() {
		routespeedsoption13 = parseInt(getId('routespeeds-option13').value);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption14() {
		routespeedsoption14 = (getId('routespeeds-option14').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption15() {
		routespeedsoption15 = (getId('routespeeds-option15').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption16() {
		routespeedsoption16 = (getId('routespeeds-option16').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption17() {
		routespeedsoption17 = (getId('routespeeds-option17').checked === true);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickOption18() {
		routespeedsoption18 = (getId('routespeeds-option18').value);
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickPassOption() {
		let passKey = this.id.match(/^routespeeds-pass-(.*)/i)[1];
		if (this.checked) {
			_settings.passes.push(passKey);
		} else {
			_settings.passes = _settings.passes.filter(key => key !== passKey)
		}
		updatePassesLabel();
		livemapRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function clickRoute1() { toggleRoute(1); }
	function clickRoute2() { toggleRoute(2); }
	function clickRoute3() { toggleRoute(3); }
	function clickRoute4() { toggleRoute(4); }
	function clickRoute5() { toggleRoute(5); }
	//--------------------------------------------------------------------------------------------------------
	function toggleRoute(routeNo) {
		if (routeSelected === routeNo) routeNo = 0;
		routeSelectedLast = routeSelected = routeNo;
		switchRoute();
	}
	//--------------------------------------------------------------------------------------------------------
	function switchRoute() {
		var WM = window.W.map;

		if (routeSelected == 1) getId('routespeeds-summary1').className = 'routespeeds_summary_classB';
		else getId('routespeeds-summary1').className = 'routespeeds_summary_classA';
		if (routeSelected == 2) getId('routespeeds-summary2').className = 'routespeeds_summary_classB';
		else getId('routespeeds-summary2').className = 'routespeeds_summary_classA';
		if (routeSelected == 3) getId('routespeeds-summary3').className = 'routespeeds_summary_classB';
		else getId('routespeeds-summary3').className = 'routespeeds_summary_classA';
		if (routeSelected == 4) getId('routespeeds-summary4').className = 'routespeeds_summary_classB';
		else getId('routespeeds-summary4').className = 'routespeeds_summary_classA';
		if (routeSelected == 5) getId('routespeeds-summary5').className = 'routespeeds_summary_classB';
		else getId('routespeeds-summary5').className = 'routespeeds_summary_classA';

		let rlayers1 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds1");
		let rlayers2 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds2");
		let rlayers3 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds3");
		let rlayers4 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds4");
		let rlayers5 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds5");
		let routeLayer1 = rlayers1[0];
		let routeLayer2 = rlayers2[0];
		let routeLayer3 = rlayers3[0];
		let routeLayer4 = rlayers4[0];
		let routeLayer5 = rlayers5[0];
		if (routeLayer1 === undefined) return;
		if (routeLayer2 === undefined) return;
		if (routeLayer3 === undefined) return;
		if (routeLayer4 === undefined) return;
		if (routeLayer5 === undefined) return;

		let style1 = routeLayer1.styleMap.styles.default.defaultStyle;
		let style2 = routeLayer2.styleMap.styles.default.defaultStyle;
		let style3 = routeLayer3.styleMap.styles.default.defaultStyle;
		let style4 = routeLayer4.styleMap.styles.default.defaultStyle;
		let style5 = routeLayer5.styleMap.styles.default.defaultStyle;

		let s1 = style1.strokeColor;
		let s2 = style2.strokeColor;
		let s3 = style3.strokeColor;
		let s4 = style4.strokeColor;
		let s5 = style5.strokeColor;

		let t1 = style1.label;
		let t2 = style2.label;
		let t3 = style3.label;
		let t4 = style4.label;
		let t5 = style5.label;

		style1.strokeColor = '#4d4dcd';
		style2.strokeColor = '#d34f8a';
		style3.strokeColor = '#188984';
		style4.strokeColor = '#cafa27';
		style5.strokeColor = '#ffca3f';
		//style1.strokeColor = '#76768f';
		//style2.strokeColor = '#917682';
		//style3.strokeColor = '#6b8a88';
		//style4.strokeColor = '#998f73';
		//style5.strokeColor = '#769178';
		//style1.strokeColor = '#7070a0';
		//style2.strokeColor = '#a07070';
		//style3.strokeColor = '#70a070';
		//style4.strokeColor = '#a0a070';
		//style5.strokeColor = '#a070a0';
		style1.strokeWidth = 5;
		style2.strokeWidth = 5;
		style3.strokeWidth = 5;
		style4.strokeWidth = 5;
		style5.strokeWidth = 5;
		style1.label = '';
		style2.label = '';
		style3.label = '';
		style4.label = '';
		style5.label = '';

		if (routeSelected === 0 || routeSelected === 1) { style1.strokeColor = '${strokeColor}'; style1.strokeWidth = '${strokeWidth}'; style1.label = '${labelText}'; }
		if (routeSelected === 0 || routeSelected === 2) { style2.strokeColor = '${strokeColor}'; style2.strokeWidth = '${strokeWidth}'; style2.label = '${labelText}'; }
		if (routeSelected === 0 || routeSelected === 3) { style3.strokeColor = '${strokeColor}'; style3.strokeWidth = '${strokeWidth}'; style3.label = '${labelText}'; }
		if (routeSelected === 0 || routeSelected === 4) { style4.strokeColor = '${strokeColor}'; style4.strokeWidth = '${strokeWidth}'; style4.label = '${labelText}'; }
		if (routeSelected === 0 || routeSelected === 5) { style5.strokeColor = '${strokeColor}'; style5.strokeWidth = '${strokeWidth}'; style5.label = '${labelText}'; }

		let z1 = parseInt(routeLayer1.getZIndex());
		let z2 = parseInt(routeLayer2.getZIndex());
		let z3 = parseInt(routeLayer3.getZIndex());
		let z4 = parseInt(routeLayer4.getZIndex());
		let z5 = parseInt(routeLayer5.getZIndex());
		let z;

		if (z1 > z2) { z = z1; z1 = z2; z2 = z; }
		if (z1 > z3) { z = z1; z1 = z3; z3 = z; }
		if (z1 > z4) { z = z1; z1 = z4; z4 = z; }
		if (z1 > z5) { z = z1; z1 = z5; z5 = z; }
		if (z2 > z3) { z = z2; z2 = z3; z3 = z; }
		if (z2 > z4) { z = z2; z2 = z4; z4 = z; }
		if (z2 > z5) { z = z2; z2 = z5; z5 = z; }
		if (z3 > z4) { z = z3; z3 = z4; z4 = z; }
		if (z3 > z5) { z = z3; z3 = z5; z5 = z; }
		if (z4 > z5) { z = z4; z4 = z5; z5 = z; }

		//wlodek76: finding closure layer and changing its zindex to hide it under Route Speeds layer
		//          we cannot easily set route speed layer over markers because it will block access to elements on these layers
		let clayers = WM.getLayersBy("uniqueName", "closures");
		if (clayers[0] !== undefined && closurelayer === null) {

			closurelayer = clayers[0];
			closurelayerZINDEX[0] = clayers[0].getZIndex();
			closurelayerZINDEX[1] = z1 - 5;

			closurelayer.setZIndex(closurelayerZINDEX[1]);
			closurelayer.redraw();
		}

		if (routeSelected <= 1) { routeLayer1.setZIndex(z5); routeLayer2.setZIndex(z4); routeLayer3.setZIndex(z3); routeLayer4.setZIndex(z2); routeLayer5.setZIndex(z1); }
		if (routeSelected === 2) { routeLayer1.setZIndex(z4); routeLayer2.setZIndex(z5); routeLayer3.setZIndex(z3); routeLayer4.setZIndex(z2); routeLayer5.setZIndex(z1); }
		if (routeSelected === 3) { routeLayer1.setZIndex(z4); routeLayer2.setZIndex(z3); routeLayer3.setZIndex(z5); routeLayer4.setZIndex(z2); routeLayer5.setZIndex(z1); }
		if (routeSelected === 4) { routeLayer1.setZIndex(z4); routeLayer2.setZIndex(z3); routeLayer3.setZIndex(z2); routeLayer4.setZIndex(z5); routeLayer5.setZIndex(z1); }
		if (routeSelected === 5) { routeLayer1.setZIndex(z4); routeLayer2.setZIndex(z3); routeLayer3.setZIndex(z2); routeLayer4.setZIndex(z1); routeLayer5.setZIndex(z5); }

		if (t1 !== style1.label || s1 !== style1.strokeColor) routeLayer1.redraw();
		if (t2 !== style2.label || s2 !== style2.strokeColor) routeLayer2.redraw();
		if (t3 !== style3.label || s3 !== style3.strokeColor) routeLayer3.redraw();
		if (t4 !== style4.label || s4 !== style4.strokeColor) routeLayer4.redraw();
		if (t5 !== style5.label || s5 !== style5.strokeColor) routeLayer5.redraw();
	}
	//--------------------------------------------------------------------------------------------------------
	function showClosures(mode) {
		if (closurelayer !== null && closurelayerZINDEX.length == 2) {
			closurelayer.setZIndex(closurelayerZINDEX[mode]);
			closurelayer.redraw();
		}
	}
	//--------------------------------------------------------------------------------------------------------
	function rezoom() {

		var WM = window.W.map;

		var rlayers1 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds1");
		var rlayers2 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds2");
		var rlayers3 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds3");
		var rlayers4 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds4");
		var rlayers5 = WM.getLayersBy("uniqueName", "__DrawRouteSpeeds5");

		var routeLayer1 = rlayers1[0];
		var routeLayer2 = rlayers2[0];
		var routeLayer3 = rlayers3[0];
		var routeLayer4 = rlayers4[0];
		var routeLayer5 = rlayers5[0];

		if (routeLayer1 !== undefined) routeLayer1.removeAllFeatures();
		if (routeLayer2 !== undefined) routeLayer2.removeAllFeatures();
		if (routeLayer3 !== undefined) routeLayer3.removeAllFeatures();
		if (routeLayer4 !== undefined) routeLayer4.removeAllFeatures();
		if (routeLayer5 !== undefined) routeLayer5.removeAllFeatures();

		getId('routespeeds-summary1').innerHTML = '';
		getId('routespeeds-summary2').innerHTML = '';
		getId('routespeeds-summary3').innerHTML = '';
		getId('routespeeds-summary4').innerHTML = '';
		getId('routespeeds-summary5').innerHTML = '';

		getId('routespeeds-summary1').className = 'routespeeds_summary_classA';
		getId('routespeeds-summary2').className = 'routespeeds_summary_classA';
		getId('routespeeds-summary3').className = 'routespeeds_summary_classA';
		getId('routespeeds-summary4').className = 'routespeeds_summary_classA';
		getId('routespeeds-summary5').className = 'routespeeds_summary_classA';

		getId('routespeeds-summary1').style.visibility = 'hidden';
		getId('routespeeds-summary2').style.visibility = 'hidden';
		getId('routespeeds-summary3').style.visibility = 'hidden';
		getId('routespeeds-summary4').style.visibility = 'hidden';
		getId('routespeeds-summary5').style.visibility = 'hidden';

		switchRoute();

		if (routewsp1.length >= 2 && routeodc1.length >= 1) {
			createRouteFeatures(1, routewsp1, routeodc1);
		}

		if (routewsp2.length >= 2 && routeodc2.length >= 1) {
			createRouteFeatures(2, routewsp2, routeodc2);
		}

		if (routewsp3.length >= 2 && routeodc3.length >= 1) {
			createRouteFeatures(3, routewsp3, routeodc3);
		}

		if (routewsp4.length >= 2 && routeodc4.length >= 1) {
			createRouteFeatures(4, routewsp4, routeodc4);
		}

		if (routewsp5.length >= 2 && routeodc5.length >= 1) {
			createRouteFeatures(5, routewsp5, routeodc5);

		}
	}
	//--------------------------------------------------------------------------------------------------------
	function enterAB(ev) {
		if (ev.keyCode === 13) {
			livemapRoute();
		}
	}
	//--------------------------------------------------------------------------------------------------------
	function getCheckboxHtml(idSuffix, text, title, divCss = {}, labelCss = {}) {
		let id = 'routespeeds-' + idSuffix;
		return $('<div>', { class: 'controls-container' }).append(
			$('<input>', { id: id, type: 'checkbox', title: title }),
			$('<label>', { for: id }).text(text).css(labelCss)
		).css(divCss)[0].outerHTML;
	}
	//--------------------------------------------------------------------------------------------------------
	function initialiseWMERouteSpeeds() {
		var line_div_break = '<br>';
		line_div_break += '</div>';
		line_div_break += '<div style="margin-left:55px">';

		if (typeof W === 'undefined') W = window.W;
		if (typeof W.loginManager === 'undefined') W.loginManager = window.W.loginManager;
		if (typeof W.loginManager === 'undefined') W.loginManager = window.loginManager;
		if (W.loginManager !== null && W.loginManager.user) {
			var user = W.loginManager.user;
			//console.log(user);

			//wlodek76: I prefer more condensed lines for avoid options, so I've added a personal switch here
			if (user !== null) {
				if (user.userName === "wlodek76" && user.id === 203457007) {
					line_div_break = '';
				}
			}
		}

		var addon = document.createElement('section');
		addon.id = "routespeeds-addon";
		addon.innerHTML = '' +
			'<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' + wmech_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 &#8596; B</button></div>' +
			'<div style="clear:both; "></div>' +

			'<div style="margin-top:8px;">' +
			'<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-summary1 class=routespeeds_summary_classA></div>' +
			'<div id=routespeeds-summary2 class=routespeeds_summary_classA></div>' +
			'<div id=routespeeds-summary3 class=routespeeds_summary_classA></div>' +
			'<div id=routespeeds-summary4 class=routespeeds_summary_classA></div>' +
			'<div id=routespeeds-summary5 class=routespeeds_summary_classA></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('option1', 'Disable script') +
			getCheckboxHtml('option3', 'Hide labels') +
			getCheckboxHtml('option2', 'Show cross-times through segments') +
			getCheckboxHtml('option4', 'Speed in mph') +

			'<div>' +
			getCheckboxHtml('option5', 'Alternative routes', '', { display: 'inline-block' }) +
			'<select id=routespeeds-option6 style="margin-left:10px; display:inline-block; height:18px;" >' +
			'<option id=routespeeds-option6 value="1">1</option>' +
			'<option id=routespeeds-option6 value="2">2</option>' +
			'<option id=routespeeds-option6 value="3">3</option>' +
			'<option id=routespeeds-option6 value="4">4</option>' +
			'<option id=routespeeds-option6 value="5">5</option>' +
			'</select>' +

			'<span id="routespeeds-option5-span" style="display:none;">' +
			getCheckboxHtml('option12', 'Try more', 'When enabled, ten times more alternative routes are requested from the routing server.&#13;This usually increases the request time, but sometimes provides interesting routes...',
				{ display: 'inline-block' }, { marginLeft: '10px', paddingLeft: '17px' }) +
			'</span>' +
			'</div>' +

			getCheckboxHtml('option7', 'Real-Time Traffic', 'note: this only seems to affect routes within the last 30-60 minutes, up to Now') +
			getCheckboxHtml('option15', 'Use Routing Order', 'Sorts routes in the same order they would appear in the app or livemap') +

			'<div>' +
			'<label class="" style="display:inline-block;">' +
			'Route type:<select id=routespeeds-option13 style="margin-left:10px;" >' +
			'<option value="1">Fastest</option>' +
			'<option value="3">Fastest (no history)</option>' +
			'</select>' +
			'<br>' +
			'Vehicle type:<select id=routespeeds-option18 style="margin-left:10px;" >' +
			'<option id=routespeeds-option18 value="PRIVATE">Private</option>' +
			'<option id=routespeeds-option18 value="TAXI">Taxi</option>' +
			'<option id=routespeeds-option18 value="MOTORCYCLE">Motorcycle</option>' +
			'</select>' +
			'</div>' +

			'<table><tbody><tr><td style="vertical-align:top; padding-right:4px;"><b>Avoid:</b></td><td>' +
			getCheckboxHtml('option8', 'Tolls') +
			getCheckboxHtml('option9', 'Freeways') +
			getCheckboxHtml('option16', 'Difficult turns') +
			getCheckboxHtml('option17', 'Ferries') +
			getCheckboxHtml('option10', 'Unpaved') +
			'<div id="routespeeds-option10-span" style="display:none;">' +
			getCheckboxHtml('option11', '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('option14', '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>';

        /*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'));
		new WazeWrap.Interface.Tab('Route Speeds', addon.innerHTML, init);

		window.addEventListener("beforeunload", saveRouteSpeedsOptions, true);
	}

	let _lastTopCountryId;
	function buildPassesDiv() {
		$('#routespeeds-passes-container').empty();
		let passesObj = W.model.getTopCountry().restrictionSubscriptions;
		if (passesObj) {
			_modelPasses = Object.keys(passesObj).map(key => { return { key: key, name: passesObj[key] } }).sort((a, b) => {
				if (a.name > b.name) {
					return 1;
				} else if (a.name < b.name) {
					return -1;
				}
				return 0;
			});
		} else {
			_modelPasses = [];
		}

		if (_modelPasses.length) {
			$('#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;">' +
				_modelPasses.map(pass => {
					let id = 'routespeeds-pass-' + pass.key;
					return '    <div class="controls-container" style="padding-top:2px;display:block;">' +
						'      <input id="' + id + '" type="checkbox" class="routespeeds-pass-checkbox">' +
						'      <label for="' + id + '" style="white-space:pre-line">' + pass.name + '</label>' +
						'    </div>';
				}).join(' ') +
				'  </div>' +
				'</fieldset>'
			);

			_modelPasses.forEach(pass => {
				$(`#routespeeds-pass-${pass.key}`).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.addClass('hidden');
				} else {
					checkboxDivs.removeClass('hidden');
				}
				// $($this.children()[0])
				// 	.toggleClass('fa fa-fw fa-chevron-down')
				// 	.toggleClass('fa fa-fw fa-chevron-right');
				// $($this.siblings()[0]).toggleClass('collapse');
			})

			_modelPasses.forEach(pass => $(`#routespeeds-pass-${pass.key}`).prop('checked', _settings.passes.indexOf(pass.key) > -1));
			updatePassesLabel();
		}
	}


	function onModelMergeEnd() {
		// Detect when the "top" country changes and update the list of passes.
		try {
			const topCountry = W.model.getTopCountry();
			if (topCountry && topCountry.id !== _lastTopCountryId) {
				_lastTopCountryId = topCountry.id;
				buildPassesDiv();
			}
		} catch (ex) {
			console.error('WME Route Speeds error: ', ex);
		}
	}

	function init() {
		resetOptions();
		loadRouteSpeedsOptions();

		if (routespeedsoption1) getId('sidepanel-routespeeds').style.color = "#A0A0A0";
		else getId('sidepanel-routespeeds').style.color = "";

		getId('routespeeds-option6').value = routespeedsoption6;
		getId('routespeeds-option13').value = routespeedsoption13;
		getId('routespeeds-option18').value = routespeedsoption18;

		getId('routespeeds-option1').onclick = clickOption1;
		getId('routespeeds-option2').onclick = clickOption2;
		getId('routespeeds-option3').onclick = clickOption3;
		getId('routespeeds-option4').onclick = clickOption4;
		getId('routespeeds-option5').onclick = clickOption5;
		getId('routespeeds-option6').onchange = clickOption6;
		getId('routespeeds-option7').onclick = clickOption7;
		getId('routespeeds-option8').onclick = clickOption8;
		getId('routespeeds-option9').onclick = clickOption9;
		getId('routespeeds-option10').onclick = clickOption10;
		getId('routespeeds-option11').onclick = clickOption11;
		getId('routespeeds-option12').onclick = clickOption12;
		getId('routespeeds-option13').onchange = clickOption13;
		getId('routespeeds-option14').onclick = clickOption14;
		getId('routespeeds-option15').onclick = clickOption15;
		getId('routespeeds-option16').onclick = clickOption16;
		getId('routespeeds-option17').onclick = clickOption17;
		getId('routespeeds-option18').onchange = clickOption18;

		getId('routespeeds-summary1').onclick = clickRoute1;
		getId('routespeeds-summary2').onclick = clickRoute2;
		getId('routespeeds-summary3').onclick = clickRoute3;
		getId('routespeeds-summary4').onclick = clickRoute4;
		getId('routespeeds-summary5').onclick = clickRoute5;

		getId('sidepanel-routespeeds-a').onkeydown = enterAB;
		getId('sidepanel-routespeeds-b').onkeydown = enterAB;

		getId('routespeeds-button-livemap').onclick = livemapRouteClick;
		getId('routespeeds-button-reverse').onclick = reverseRoute;
		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;

		const topCountry = W.model.getTopCountry();
		if (topCountry) {
			_lastTopCountryId = topCountry.id;
			buildPassesDiv();
		}

		// window.W.map.events.register("zoomend", null, rezoom);
		WazeWrap.Events.register('zoomend', null, rezoom);
		W.model.events.register('mergeend', null, onModelMergeEnd);

		window.setInterval(loopWMERouteSpeeds, 500);
		window.setInterval(panningWMERouteSpeeds, 100);
	}
	//--------------------------------------------------------------------------------------------------------------
	bootstrapWMERouteSpeeds();
})();