// ==UserScript==
// @name WME Advanced Closures
// @version 1.4
// @description Recurrent and imported closures in the Waze Map Editor
// @namespace WMEAC
// @include https://www.waze.com/editor/*
// @include https://www.waze.com/*/editor/*
// @include https://beta.waze.com/*
// @exclude https://www.waze.com/user/*
// @exclude https://www.waze.com/*/user/*
// @icon 
// @grant unsafeWindow
// @grant GM_xmlhttpRequest
// @connect holidayapi.com
// @copyright 2016, dummyd2, seb-d59
// @author dummyd2, seb-d59
// ==/UserScript==
/*******
*
* You are free to:
* Share, copy, and redistribute the script in any medium or format
* under the following terms:
* Attribution - You must give appropriate credit. You may do so in any
* reasonable manner, but not in any way that suggests the licensor
* endorses you or your use.
*
* NonCommercial - You may not use the script for commercial purposes.
*
* NoModifications - You may NOT MODIFY the script.
*
* You are invited to contact authors on waze forum for more details.
*
********/
/*jshint multistr: true */
function WMEAC_downloadHelperInjected() {
window.WMEAC_downloadHelper = {jobs:[], _waitForData:function(id) {
if (this.jobs.length <= id) {
this.jobs[id].callback({url:null, data:null, callback:this.jobs[id].callback, status:"error", error:"Request not found"});
} else {
if (this.jobs[id].status == "success" || this.jobs[id].status == "error") {
this.jobs[id].callback(this.jobs[id]);
} else {
if (this.jobs[id].status == "downloading" && this.jobs[id].progressCallback) {
this.jobs[id].progressCallback(this.jobs[id]);
}
var _this = this;
window.setTimeout(function() {
_this._waitForData(id);
}, 500);
}
}
}, add:function(url, callback, progressCallback) {
this.jobs.push({url:url, data:null, callback:callback, progressCallback:progressCallback, status:"added", progression:0, error:""});
var _this = this;
window.setTimeout(function() {
_this._waitForData(_this.jobs.length - 1);
}, 500);
}};
}
var WMEAC_downloadHelperInjectedScript = document.createElement("script");
WMEAC_downloadHelperInjectedScript.textContent = "" + WMEAC_downloadHelperInjected.toString() + " \n" + "WMEAC_downloadHelperInjected();";
WMEAC_downloadHelperInjectedScript.setAttribute("type", "application/javascript");
document.body.appendChild(WMEAC_downloadHelperInjectedScript);
function lookFordownloadHelperJob() {
for (var i = 0;i < unsafeWindow.WMEAC_downloadHelper.jobs.length;i++) {
if (unsafeWindow.WMEAC_downloadHelper.jobs[i].status == "added") {
unsafeWindow.WMEAC_downloadHelper.jobs[i].status = cloneInto("downloading", unsafeWindow.WMEAC_downloadHelper.jobs[i]);
var f = function() {
var job = i;
GM_xmlhttpRequest({method:"GET", headers:{"User-Agent":"Mozilla/5.0", "Accept":"text/plain"}, synchronous:false, timeout:1E4, url:unsafeWindow.WMEAC_downloadHelper.jobs[job].url, onerror:function(r) {
unsafeWindow.WMEAC_downloadHelper.jobs[job].status = cloneInto("error", unsafeWindow.WMEAC_downloadHelper.jobs[job]);
}, ontimeout:function(r) {
console.debug("TOTO Timeout while getting area from server: ", r);
unsafeWindow.WMEAC_downloadHelper.jobs[job].status = cloneInto("error", unsafeWindow.WMEAC_downloadHelper.jobs[job]);
}, onload:function(r) {
unsafeWindow.WMEAC_downloadHelper.jobs[job].status = cloneInto("success", unsafeWindow.WMEAC_downloadHelper.jobs[job]);
unsafeWindow.WMEAC_downloadHelper.jobs[job].data = cloneInto(r.responseText, unsafeWindow.WMEAC_downloadHelper.jobs[job]);
}, onprogress:function(r) {
unsafeWindow.WMEAC_downloadHelper.jobs[job].progression = cloneInto(r.total == 0 ? 0 : r.loaded / r.total, unsafeWindow.WMEAC_downloadHelper.jobs[job]);
}});
}();
}
}
window.setTimeout(lookFordownloadHelperJob, 2E3);
}
window.setTimeout(lookFordownloadHelperJob);
function WMEAC_Injected() {
var WMEAC = {};
WMEAC.isDebug = false;
WMEAC.ac_version = "1.4";
WMEAC.closureTabTimeout = null;
WMEAC.csv = [];
WMEAC.csvCurrentClosureList = null;
WMEAC.csvCurrentBatchClosureList = null;
WMEAC.pendingOps = false;
WMEAC.pb = null;
WMEAC.daysOfWeek = ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"];
WMEAC.lastGeneratedHolidays = [];
WMEAC.presets = [];
WMEAC.getElementsByClassName = function(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;
};
WMEAC.removeChildElements = function(node) {
while (node.firstChild) {
WMEAC.removeChildElements(node.firstChild);
node.removeChild(node.firstChild);
}
};
WMEAC.createElement = function(options) {
if (options.hasOwnProperty("type") == false) {
return null;
}
var el = document.createElement(options.type);
if (options.hasOwnProperty("id") == true) {
el.id = options.id;
}
if (options.hasOwnProperty("className") == true) {
el.className = options.className;
}
return el;
};
WMEAC.getId = function(node) {
var el = document.getElementById(node);
return el;
};
WMEAC.logBeta = function(msg, obj) {
};
WMEAC.logDebug = function(msg, obj) {
if (WMEAC.isDebug) {
WMEAC.log("DEBUG - " + msg, obj);
}
};
WMEAC.logError = function(msg, obj) {
console.error("Advanced closures v" + WMEAC.ac_version + " - " + msg, obj);
};
WMEAC.log = function(msg, obj) {
if (obj == null) {
console.log("Advanced closures v" + WMEAC.ac_version + " - " + msg);
} else {
console.debug("Advanced closures v" + WMEAC.ac_version + " - " + msg + " ", obj);
}
};
WMEAC.isValidDate = function(d) {
if (Object.prototype.toString.call(d) === "[object Date]") {
if (isNaN(d.getTime())) {
return false;
} else {
return true;
}
} else {
return false;
}
};
WMEAC.dateToClosureStr = function(d) {
var yyyy = d.getUTCFullYear().toString();
var MM = (d.getUTCMonth() + 1).toString();
var dd = d.getUTCDate().toString();
var hh = d.getUTCHours().toString();
var mm = d.getUTCMinutes().toString();
return yyyy + "-" + (MM[1] ? MM : "0" + MM[0]) + "-" + (dd[1] ? dd : "0" + dd[0]) + " " + (hh[1] ? hh : "0" + hh[0]) + ":" + (mm[1] ? mm : "0" + mm[0]);
};
WMEAC.CSVtoArray = function(text) {
var b = [];
var re_valid = /^\s*(?:'[^'\\]*(?:\\[\S\s][^'\\]*)*'|"[^"\\]*(?:\\[\S\s][^"\\]*)*"|[^,'"\s\\]*(?:\s+[^,'"\s\\]+)*)\s*(?:,\s*(?:'[^'\\]*(?:\\[\S\s][^'\\]*)*'|"[^"\\]*(?:\\[\S\s][^"\\]*)*"|[^,'"\s\\]*(?:\s+[^,'"\s\\]+)*)\s*)*$/;
var re_value = /(?!\s*$)\s*(?:'([^'\\]*(?:\\[\S\s][^'\\]*)*)'|"([^"\\]*(?:\\[\S\s][^"\\]*)*)"|([^,'"\s\\]*(?:\s+[^,'"\s\\]+)*))\s*(?:,|$)/g;
var lines = text.split("\n");
lines.forEach(function(line) {
if (!re_valid.test(line)) {
return;
}
var a = [];
line.replace(re_value, function(m0, m1, m2, m3) {
if (m1 !== undefined) {
a.push(m1.replace(/\\'/g, "'"));
} else {
if (m2 !== undefined) {
a.push(m2.replace(/\\"/g, '"'));
} else {
if (m3 !== undefined) {
a.push(m3);
}
}
}
return "";
});
if (/,\s*$/.test(line)) {
a.push("");
}
b.push(a);
});
return b;
};
WMEAC.segmentsIDsToSegments = function(ids) {
return ids.filter(function(e) {
return Waze.model.segments.objects.hasOwnProperty(e);
}).map(function(e) {
return Waze.model.segments.objects[e];
});
};
WMEAC.reloadRoadLayer = function() {
var l = Waze.map.getLayersBy("uniqueName", "roads")[0];
l.redraw({force:!0});
l.removeBackBuffer();
Waze.controller.reload();
};
WMEAC.reloadClosuresLayer = function(endHandler) {
var l = Waze.map.getLayersBy("uniqueName", "closures")[0];
l.redraw({force:!0});
Waze.controller.reload();
if (endHandler) {
var tmp = function reloaded() {
WMEAC.log("Test if reloaded...");
if (WMEAC.pendingOps == true) {
WMEAC.log("Not yet. Waiting for WME...");
window.setTimeout(reloaded, 500);
} else {
endHandler();
}
};
window.setTimeout(tmp, 500);
}
};
WMEAC.showClosuresLayer = function(show) {
var l = Waze.map.getLayersBy("uniqueName", "closures");
if (l.length == 1) {
l[0].setVisibility(show);
}
};
WMEAC.setDraggable = function(element, options) {
if (!options.hasOwnProperty("controller")) {
options.controller = element;
}
if (!options.hasOwnProperty("container")) {
options.container = $("body");
}
options.controller.css({cursor:"move"});
options.controller.on("mousedown", function(e) {
var x = e.pageX - element.offset().left;
var y = e.pageY - element.offset().top;
$("body").on("mouseup", function(e) {
options.container.off("mousemove", elemmousemove);
});
function elemmousemove(e) {
e.preventDefault();
element.offset({top:e.pageY - y, left:e.pageX - x});
}
options.container.on("mousemove", elemmousemove);
});
};
WMEAC.dateTimeOverlaps = function(dt1, dt2) {
return dt1.startDate < dt2.endDate && dt1.endDate > dt2.startDate;
};
WMEAC.solveOverlaps = function(closureToAdd, existingClosureList, mode) {
var ecs = existingClosureList.map(function(e) {
return {isNew:false, ref:e, startDate:e.startDate, endDate:e.endDate};
});
closureToAdd.isNew = true;
ecs.push(closureToAdd);
var changes = true;
while (changes) {
changes = false;
ecs.sort(function(a, b) {
return new Date(a.startDate) - new Date(b.startDate);
});
for (var i = 1;i < ecs.length;i++) {
if (WMEAC.dateTimeOverlaps(ecs[i - 1], ecs[i])) {
var indexOfNew = i - 1;
var indexOfExisting = i;
if (ecs[i].isNew) {
indexOfNew = i;
indexOfExisting = i - 1;
}
var r1 = ecs[indexOfNew];
var r2 = ecs[indexOfExisting];
var range1 = {};
var range2 = {};
switch(mode) {
case 0:
return [];
break;
case 1:
ecs.splice(indexOfExisting, 1);
changes = true;
break;
case 2:
range1.start = new Date(r1.startDate);
range1.end = new Date(r1.endDate);
range2.start = new Date(r2.startDate);
range2.end = new Date(r2.endDate);
changes = true;
if (range1.start >= range2.start && range1.end <= range2.end) {
ecs.splice(indexOfNew, 1);
} else {
if (range1.start < range2.start && range1.end > range2.end) {
ecs.push({isNew:true, startDate:r2.endDate, endDate:r1.endDate});
r1.endDate = r2.startDate;
} else {
if (range1.start < range2.start) {
r1.endDate = r2.startDate;
} else {
r1.startDate = r2.endDate;
}
}
}
break;
case 3:
range1.start = new Date(r1.startDate);
range1.end = new Date(r1.endDate);
range2.start = new Date(r2.startDate);
range2.end = new Date(r2.endDate);
changes = true;
if (range1.start > range2.start && range1.end < range2.end) {
ecs.push({isNew:false, startDate:r1.endDate, endDate:r2.endDate, ref:r2.ref});
r2.endDate = r1.startDate;
} else {
if (range1.start <= range2.start && range1.end >= range2.end) {
ecs.splice(indexOfExisting, 1);
} else {
if (range1.start < range2.start) {
r2.startDate = r1.endDate;
} else {
r2.endDate = r1.startDate;
}
}
}
break;
}
}
}
}
return ecs;
};
WMEAC.getCountriesFromSegmentSet = function(segs) {
var cids = segs.map(function(s) {
if (s.attributes.hasOwnProperty("primaryStreetID") && s.attributes.primaryStreetID != null) {
var stid = s.attributes.primaryStreetID;
if (Waze.model.streets.objects.hasOwnProperty(stid)) {
var st = Waze.model.streets.objects[stid];
if (st.hasOwnProperty("cityID") && st.cityID != null && typeof st.cityID != "undefined") {
var ctid = st.cityID;
if (Waze.model.cities.objects.hasOwnProperty(ctid)) {
return Waze.model.cities.objects[ctid].countryID;
}
}
}
}
return null;
}).filter(function(cid) {
return cid != null;
});
return Waze.model.countries.getObjectArray(function(c) {
return cids.indexOf(c.id) != -1;
});
};
WMEAC.getOppositeClosure = function(closure) {
return Waze.model.roadClosures.getObjectArray(function(c) {
return closure.reason == c.reason && closure.startDate == c.startDate && closure.endDate == c.endDate && closure.segID == c.segID && closure.forward != c.forward;
});
};
WMEAC.getCityStreetsFromSegmentSet = function(segs) {
var r = {};
function add(city, street) {
if (!r.hasOwnProperty(city)) {
r[city] = {};
}
if (!r[city].hasOwnProperty(street)) {
r[city][street] = 0;
}
r[city][street]++;
}
segs.forEach(function(s) {
var city = "noCity";
if (s.attributes.primaryStreetID != null && Waze.model.streets.objects.hasOwnProperty(s.attributes.primaryStreetID)) {
var st = Waze.model.streets.objects[s.attributes.primaryStreetID];
if (st.hasOwnProperty("cityID") && st.cityID != null && typeof st.cityID != "undefined") {
var ctid = st.cityID;
if (Waze.model.cities.objects.hasOwnProperty(ctid)) {
if (!Waze.model.cities.objects[ctid].isEmpty) {
city = Waze.model.cities.objects[ctid].name;
}
}
}
if (Waze.model.streets.objects[s.attributes.primaryStreetID].isEmpty) {
add(city, "noStreet");
} else {
add(city, Waze.model.streets.objects[s.attributes.primaryStreetID].name);
}
}
});
return r;
};
WMEAC.download = function(data, filename) {
var element = document.createElement("a");
element.style.display = "none";
element.setAttribute("href", encodeURI("data:text/plain," + data));
element.setAttribute("download", filename);
document.body.appendChild(element);
element.click();
document.body.removeChild(element);
};
var cssElt = WMEAC.createElement({type:"style"});
cssElt.type = "text/css";
var css = "";
css += ".slashed:after { content: ''; position: relative; width: 140%; height: 1px; display: block; background: red; transform: rotate(-30deg); margin-top: -50%; margin-left: -20%; }";
css += ".wmeac-sidepanel button { border: none; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; background-color: #F1DDDB }";
css += "#wmeac-progressBarInfo { display: none; width: 90%; float: left; position: absolute; border-top-left-radius: 5px; border-top-right-radius: 5px; border-bottom-right-radius: 5px; border-bottom-left-radius: 5px; margin-bottom: -100%; background-color: #c9e1e9; z-index: 999; margin: 5px; margin-right: 20px; }";
css += ".wmeac-progressBarBG { margin-top: 2px; margin-bottom: 2px; margin-left: 2px; margin-right: 2px; padding-bottom: 0px; padding-top: 0px; padding-left: 0px; padding-right: 0px; width: 33%; background-color: #93c4d3; border: 3px rgb(147, 196, 211); border-top-left-radius: 5px; border-top-right-radius: 5px; border-bottom-right-radius: 5px; border-bottom-left-radius: 5px; height: 22px;}";
css += ".wmeac-progressBarFG { float: left; position: relative; bottom: 22px; height: 0px; text-align: center; width: 100% }";
css += ".wmeac-button { border: none; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; background-color: #F1DDDB; display: inline-block; padding: 6px 12px; cursor: pointer; text-align: center; font-weight: bold; }";
css += ".wmeac-closuredialog { border: 2px solid #F1DDDB; width: 100%; float: left; display: none; position: absolute; padding: 0 0px; border-bottom-left-radius: 10px; border-bottom-right-radius: 10px; border-top-left-radius: 10px; border-top-right-radius: 10px; background-color: #FDEDEB; width: 500px; z-index: 9999; left: 80px; top: 10px;}";
css += ".wmeac-closuredialog button { border: none; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; background-color: #F1DDDB; margin: 3px; }";
css += ".wmeac-closuredialog h1 { background-color: #F1DDDB; text-align: center; font-size: medium; margin-top: 0px; padding: 10px;}";
css += ".wmeac-closuredialog .content { padding: 10px;}";
css += ".wmeac-closuredialog .content table { witdh: 100%; border: none; font-size: 10px; text-transform: uppercase;}";
css += ".wmeac-closuredialog .content table tbody tr { vertical-align: top;}";
css += ".wmeac-closuredialog .content table tbody tr td { padding-right: 2px; padding-left: 2px;}";
css += ".wmeac-closuredialog-fromgroup { display: inline-block; }";
css += ".wmeac-nav-tabs>ul { border-bottom: 1px solid #F6C3BE }";
css += ".wmeac-nav-tabs>li { float: left; margin-bottom: -1px; }";
css += ".wmeac-nav-tabs>li>a { border: 1px solid #F6C3BE; border-top-left-radius: 5px; border-top-right-radius: 5px; margin-right: 2px;}";
css += ".wmeac-nav-tabs>li.active>a { background-color: rgba(0, 0, 0, 0); border-bottom: 1px solid #FDEDEB}";
css += ".wmeac-nav-tabs>li:not(.active)>a { background-color: #DADBDC}";
css += ".wmeac-tab-pane {border: 1px solid #F6C3BE; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 0px; border-top-right-radius: 0px; padding: 5px;}";
css += ".wmeac-closuredialog .footer { height: 40px; padding: 0 10px;}";
css += "#wmeac-csv-closures-list ul { list-style-type: none; padding: 0px;}";
css += "#wmeac-csv-closures-list ul li { width: 100%; height: 42px; border-radius: 4px; margin-top: 1px; }";
css += "#wmeac-csv-closures-list ul li > * { display: table-cell; vertical-align: middle;}";
css += ".wmeac-csv-closures-list-add { background-color: #C6DFFF; }";
css += ".wmeac-csv-closures-list-remove { background-color: #FFC65F; }";
css += ".wmeac-csv-closures-list-failed { background-color: #FF8585; }";
css += ".wmeac-csv-closures-list-done { background-color: #B9FAB1; }";
css += ".wmeac-csv-closures-list-col-action { width: 14px; min-width: 14px; }";
css += ".wmeac-csv-closures-list-col-lr { font-size: xx-small; width: 100%; }";
css += ".wmeac-csv-closures-list-col-lr > * { height: 14px; overflow-y: hidden; vertical-align: middle; }";
css += ".wmeac-csv-closures-list-col-dates { width: 75px; min-width: 75px; font-size: xx-small; text-align: center; }";
css += ".wmeac-csv-closures-list-col-dates > * { height: 14px; overflow-y: hidden; vertical-align: center; }";
css += ".wmeac-csv-closures-list-col-dir { width: 35px; min-width: 35px; text-align: center; }";
css += ".wmeac-csv-closures-list-col-it { width: 15px; min-width: 15px; }";
css += ".wmeac-csv-closures-list-col-target { width: 15px; min-width: 15px; }";
css += ".wmeac-csv-closures-list-col-apply { width: 15px; min-width: 15px; }";
css += ".wmeac-csv-closures-minilog { font-size: xx-small; font-family: monospace; border: 2px solid #F6C3BE; border-top: none; border-bottom-right-radius: 4px; border-bottom-left-radius: 4px; background-color: #FFFFFF; }";
css += "#wmeac-csv-closures-log { font-size: xx-small; font-family: monospace; border: 2px solid #F6C3BE; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; padding-top: 8px; position: relative; margin-top: 10px; }";
css += '#wmeac-csv-closures-log:before { content: "Logs"; position: absolute; top: -8px; left: 5px; float: left; background: #F6C3BE; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; padding: 2px; }';
css += "#wmeac-csv-closures-preview { font-size: small; white-space: nowrap; font-family: monospace; border: 2px solid #F6C3BE; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; padding-top: 8px; position: relative; min-height: 20px; }";
css += '#wmeac-csv-closures-preview:before { content: "Preview"; position: absolute; top: -8px; left: 5px; float: left; background: #F6C3BE; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; padding: 2px; }';
css += '.wmeac-hl:after { content: "\\f018"; position: relative; display: block; margin-top: -100%; margin-left: 50%; font-family: FontAwesome; }';
cssElt.innerHTML = css;
document.body.appendChild(cssElt);
WMEAC.bootstrapAC = function() {
window.setTimeout(WMEAC.initialize, 500);
};
WMEAC.initialize = function() {
WMEAC.log("init");
WMEAC.waitForWaze(function() {
WMEAC.load();
WMEAC.log("presets", WMEAC.presets);
WMEAC.initUI();
});
WMEAC.log("init done");
};
WMEAC.waitForWaze = function(handler) {
var APIRequired = [{o:"Waze"}, {o:"Waze.model"}, {o:"Waze.map"}, {o:"Waze.loginManager"}];
for (var i = 0;i < APIRequired.length;i++) {
var path = APIRequired[i].o.split(".");
var object = window;
for (var j = 0;j < path.length;j++) {
object = object[path[j]];
if (typeof object == "undefined" || object == null) {
window.setTimeout(function() {
WMEAC.waitForWaze(handler);
}, 500);
return;
}
}
}
var userInfo = WMEAC.getId("user-info");
if (userInfo == null) {
window.setTimeout(function() {
WMEAC.waitForWaze(handler);
}, 500);
return;
}
var navTabs = userInfo.getElementsByTagName("ul");
if (navTabs.length == 0) {
window.setTimeout(function() {
WMEAC.waitForWaze(handler);
}, 500);
return;
}
if (typeof navTabs[0] == "undefined") {
window.setTimeout(function() {
WMEAC.waitForWaze(handler);
}, 500);
return;
}
var tabContents = userInfo.getElementsByTagName("div");
if (tabContents.length == 0) {
window.setTimeout(function() {
WMEAC.waitForWaze(handler);
}, 500);
return;
}
if (typeof tabContents[0] == "undefined") {
window.setTimeout(function() {
WMEAC.waitForWaze(handler);
}, 500);
return;
}
handler();
};
WMEAC.initUI = function() {
var addon = WMEAC.createElement({type:"section", id:"wmeac-addon"});
WMEAC.pb = new WMEAC.ProgressBar("wmeac-progressBarInfo");
addon.appendChild(WMEAC.pb.divpbi);
var section = WMEAC.createElement({type:"p", id:"wmeac-main-title"});
section.style.paddingTop = "0px";
section.style.marginTop = "-15px";
section.style.textIndent = "8px";
var title = '<b><a target="_blank" href="https://greasyfork.org/scripts/20705-wme-advanced-closures"><u>Advanced Closures</u></a> <a target="_blank" href="https://www.waze.com/forum/viewtopic.php?f=1316&t=193462">Fr</a> <a target="_blank" href="https://www.waze.com/forum/viewtopic.php?f=819&t=193465">En</a> </b> v' + WMEAC.ac_version;
section.innerHTML = title;
addon.appendChild(section);
var divAdvCl = WMEAC.createElement({type:"div", className:"wmeac-sidepanel", id:"wmeac-ac"});
var addACBtn = WMEAC.createElement({type:"div", id:"wmeac-add-advanced-closure-button", className:"wmeac-button"});
addACBtn.style.width = "100%";
addACBtn.innerHTML = '<i class="fa fa-clock-o"></i> Add advanced closure';
addACBtn.addEventListener("click", WMEAC.showAddAdvancedClosure);
divAdvCl.appendChild(addACBtn);
var divCSV = WMEAC.createElement({type:"div", className:"wmeac-sidepanel", id:"wmeac-csv"});
var csvHTML = '<label for="wmeac-csv-file" class="wmeac-button">Parse CSV</label> <input id="wmeac-csv-file" type="file" name="files[]" style="display: none;" />';
csvHTML += ' <div id="wmeac-csv-closures" style="display: none;"> <div id="wmeac-csv-closures-controls"> <input type="checkbox" id="wmeac-csv-closures-controls-check"> | <a href="#" id="wmeac-csv-closures-controls-apply">Apply</a> | <a href="#" id="wmeac-csv-closures-controls-segs">Check segments</a> </div> <div id="wmeac-csv-closures-list"> <ul id="wmeac-csv-closures-list-elts"> </ul> </div> </div> <div id="wmeac-csv-closures-log"> </div>';
divCSV.innerHTML = csvHTML;
addon.appendChild(divAdvCl);
addon.appendChild(WMEAC.createElement({type:"hr"}));
addon.appendChild(divCSV);
var userTabs = WMEAC.getId("user-tabs");
var userInfo = WMEAC.getId("user-info");
var sidePanelPrefs = WMEAC.getId("sidepanel-prefs");
var navTabs = WMEAC.getElementsByClassName("nav-tabs", userTabs)[0];
var tabContent = sidePanelPrefs.parentNode;
newtab = WMEAC.createElement({type:"li"});
newtab.innerHTML = '<a title="Advanced closures" href="#sidepanel-wmeac" data-toggle="tab"><span class="fa fa-road slashed"></span></a>';
navTabs.appendChild(newtab);
addon.id = "sidepanel-wmeac";
addon.className = "tab-pane";
addon.style.marginLeft = "-10px";
tabContent.appendChild(addon);
var observer = new MutationObserver(function(mutations) {
mutations.forEach(function(mutation) {
function rescurse(node) {
if (node.id == "segment-edit-closures") {
WMEAC.installButtonInClosureTab(node);
} else {
if (node.className == "closures-list") {
var target = WMEAC.getId("segment-edit-closures");
if (target) {
WMEAC.installButtonInClosureTab(target);
}
} else {
for (var j = 0;j < node.childNodes.length;j++) {
rescurse(node.childNodes[j]);
}
}
}
}
for (var i = 0;i < mutation.addedNodes.length;i++) {
rescurse(mutation.addedNodes[i]);
}
});
});
observer.observe(WMEAC.getId("edit-panel"), {childList:true, subtree:true});
WMEAC.installButtonInClosureTab();
Waze.vent.on("operationPending", function(e) {
if (e.operation.id != "pending.road_data") {
return;
}
WMEAC.pendingOps = true;
});
Waze.vent.on("operationDone", function(e) {
if (e.operation.id != "pending.road_data") {
return;
}
WMEAC.pendingOps = false;
});
Waze.model.events.register("mergeend", null, WMEAC.refreshHighlight);
WMEAC.refreshHighlight();
window.setTimeout(WMEAC.connectAdvancedClosureTabHandlers);
};
WMEAC.installButtonInClosureTab = function(node) {
if (!node) {
node = WMEAC.getId("segment-edit-closures");
}
if (!node) {
return;
}
if ($(node).find("#wmeac-closuretab-add-advanced-closure-button").length == 0) {
var addACBtn = WMEAC.createElement({type:"div", id:"wmeac-closuretab-add-advanced-closure-button", className:"wmeac-button"});
addACBtn.style.width = "100%";
addACBtn.style.marginBottom = "10px";
addACBtn.innerHTML = '<i class="fa fa-clock-o"></i> Add advanced closure';
addACBtn.addEventListener("click", WMEAC.showAddAdvancedClosure);
$(node).find(".main").prepend(addACBtn);
}
};
WMEAC.showAddAdvancedClosure = function() {
var ACDiv = WMEAC.getId("wmeac-add-advanced-closure-dialog");
if (ACDiv == null) {
ACDiv = WMEAC.createElement({type:"div", id:"wmeac-add-advanced-closure-dialog", className:"wmeac-closuredialog"});
ACDiv.innerHTML = WMEAC.HTMLTemplates.advancedClosureDialog;
Waze.map.div.appendChild(ACDiv);
window.setTimeout(WMEAC.connectAdvancedClosureDialogHandlers);
ACDiv.style.display = "none";
}
if (ACDiv.style.display == "block") {
$(ACDiv).css({left:"80px", top:"10px"});
} else {
ACDiv.style.display = "block";
Waze.selectionManager.events.register("selectionchanged", null, WMEAC.refreshClosureList);
Waze.selectionManager.events.register("selectionchanged", null, WMEAC.refreshClosureListFromSelection);
WMEAC.refreshClosureListFromSelection();
}
WMEAC.showClosuresLayer(true);
};
WMEAC.HTMLTemplates = {};
var rangeStartEndUI = ' <div class="form-group"> <label class="control-label" for="closure_rangestartDate">Range start (included)</label> <div class="controls"> <div style="width: 58%" class="date date-input-group input-group pull-left"> <input id="wmeac-advanced-closure-dialog-rangestartdate" class="form-control start-date" type="text" name="closure_rangestartDate"> <span class="input-group-addon"> <i class="fa fa-calendar"></i> </span> </div> </div> </div> <div class="form-group"> <label class="control-label" for="closure_rangeendDate">Range end (included)</label> <div class="controls"> <div style="width: 58%" class="date date-input-group input-group pull-left"> <input id="wmeac-advanced-closure-dialog-rangeenddate" class="form-control end-date" type="text" name="closure_rangeendDate"> <span class="input-group-addon"> <i class="fa fa-calendar"></i> </span> </div> </div> </div>';
var startTimeAndDurationUI = ' <div class="wmeac-closuredialog-fromgroup"> <label class="control-label" for="closure_startTime">Start</label> <div class="controls"> <div style="width: 58%;" class="bootstrap-timepicker input-group pull-left"> <input id="wmeac-advanced-closure-dialog-starttime" class="form-control start-time" type="text" name="closure_startTime"> <span class="input-group-addon"> <i class="fa fa-clock-o"></i> </span> </div> </div> </div> <div class="wmeac-closuredialog-fromgroup"> <label class="control-label">Duration</label> <div style="width: 58%;" class="bootstrap-timepicker input-group"> <div class="controls" style="display: flex;"> <span class="input-group-addon pull-left"> <i class="fa fa-step-forward"></i> </span> <span class="form-control" style="padding: 1px; display: flex"> <input id="wmeac-advanced-closure-dialog-duration-day" name="value" value=0 size=3/> <span style="padding: 5px;">D</span> </span> </div> <div class="bootstrap-timepicker input-group pull-left"> <input id="wmeac-advanced-closure-dialog-durationtime" class="form-control start-time" type="text" name="closure_durationTime"> <span class="input-group-addon"> <i class="fa fa-clock-o"></i> </span> </div> </div> </div>';
var descriptionUI = ' <div class="form-group"> <label class="control-label" for="closure_reason">Description</label> <div class="controls"> <input id="wmeac-advanced-closure-dialog-reason" class="form-control" type="text" name="closure_reason"> </div> </div>';
var locationUI = ' <div class="form-group"> <label class="control-label" for="closure_location">Location</label> <div class="controls"> <input id="wmeac-advanced-closure-dialog-location" class="form-control" type="text" name="closure_location"> </div> </div>';
var directionUI = ' <div class="form-group"> <label class="control-label" for="closure_direction">Direction</label> <div class="controls"> <select id="wmeac-advanced-closure-dialog-direction" style="font-family:\'FontAwesome\', Arial;" class="form-control" name="closure_direction"> <option value="3">Two way ()</option><option value="1">One way (A → B)</option><option value="2">One way (B → A)</option> </select> </div> </div>';
var ignoreTrafficUI = ' <div class="checkbox"> <label class="control-label" style="font-weight: bold;"> <input id="wmeac-advanced-closure-dialog-ignoretraffic" type="checkbox" name="closure_permanent"> Ignore Traffic </label> </div>';
var MTEUI = ' <div class="form-group"> <label class="control-label control-label-inline" for="closure_MTE">Link to MTE</label> <div class="controls"> <select id="wmeac-advanced-closure-dialog-mteid" class="form-control" name="closure_MTE" disabled><option value="">None</option></select> </div> </div>';
var overlapModeUI = ' <div class="form-group"> <label class="control-label" for="closure_overlap">Overlap action</label> <div class="controls"> <select id="wmeac-advanced-closure-dialog-overlap" style="font-family:\'FontAwesome\', Arial;" class="form-control" name="closure_overlap"> <option value="0">Keep existing</option><option value="1">Delete existing</option><option value="2">Fill with new</option><option value="3">Force new</option> </select> </div> </div>';
var tabRepeatUI = ' <div style="width: 150px;" class="input-group"> <div class="controls"> <div class="input-group pull-left"> <input id="wmeac-advanced-closure-dialog-repeat-ntimes" class="form-control" type="text" name="closure_repeat_ntimes"> <span class="input-group-addon" for="closure_repeat_ntimes">times</span> </div> </div> </div> <div style="width: 150px;" class="input-group"> <div class="controls"> <div style="width: 150px;" class="bootstrap-timepicker input-group"> <span class="input-group-addon"> every </span> <span class="form-control" style="padding: 1px; display: flex"> <input id="wmeac-advanced-closure-dialog-repeat-every-day" name="value" value=0 size=3/> <span style="padding: 5px;">D</span> <input id="wmeac-advanced-closure-dialog-repeat-every-hour" name="value" value=0 size=3/> <span style="padding: 5px;">H</span> <input id="wmeac-advanced-closure-dialog-repeat-every-minute" name="value" value=0 size=2/> <span style="padding: 5px;">M</span> </span> </div> </div> </div>';
var daysOfWeekUI = WMEAC.daysOfWeek.clone();
daysOfWeekUI.push(daysOfWeekUI.shift());
var tabEachUI = '<div class="checkbox"> <label class="control-label" style="font-weight: bold;"> <input id="wmeac-advanced-closure-dialog-each-dayall" type="checkbox" name="closure_each_dayall"> All </label> </div> ' + daysOfWeekUI.map(function(d, i) {
return '<div class="checkbox"> <label class="control-label" style="font-weight: bold;"> <input id="wmeac-advanced-closure-dialog-each-' + (i + 1) % 7 + '" type="checkbox" name="closure_each_' + d + '"> ' + d + " </label> </div>";
}).join("");
var tabHolidayUI = '<div class="content"> <a id="wmeac-advanced-closure-dialog-holiday-refresh" href="#">Refresh holidays</a><br> <i id="wmeac-advanced-closure-dialog-holiday-refresh-spinner" class="fa fa-spinner fa-pulse fa-3x fa-fw" style="display: none;"></i> <div id="wmeac-advanced-closure-dialog-holiday-list" class="form-group" style="overflow-y: scroll; max-height: 200px;"> </div></div>';
var tabPresetsUI = '<div class="content"> <table><tr><td style="width: 50%; border-right: 1px solid #F6C3BE; padding-right: 5px;"> <div class="form-group"> <label class="control-label" for="presets_load">Load preset</label> <div class="controls"> <div class="input-group"> <select style="width: 100%;" id="wmeac-advanced-closure-dialog-presets-list" name="presets_load"> </select> <span id="wmeac-advanced-closure-dialog-presets-load" class="input-group-addon"> <i class="fa fa-folder-open-o"></i> </span> <span id="wmeac-advanced-closure-dialog-presets-delete" class="input-group-addon"> <i class="fa fa-trash"></i> </span> </div> </div> <label class="control-label" for="seg_load">Load from segment</label> <div class="controls"> <div class="input-group"> <select style="width: 100%;" id="wmeac-advanced-closure-dialog-segclosure-list" name="presets_load"> </select> <span id="wmeac-advanced-closure-dialog-presets-load-fromseg" class="input-group-addon"> <i class="fa fa-share"></i> </span> </div> </div> </div> </td><td style="padding-left: 5px;"> <div class="form-group"> <label class="control-label" for="presets_save">Save preset</label> <div class="controls"> <div class="input-group pull-left"> <input id="wmeac-advanced-closure-dialog-presets-name" class="form-control" type="text" name="presets_save"> <span id="wmeac-advanced-closure-dialog-presets-save" class="input-group-addon"> <i class="fa fa-floppy-o"></i> </span> </div> </div> </div> </td></tr></table></div>';
var tabsUI = ' <ul class="nav wmeac-nav-tabs"> <li class="active"> <a id="wmeac-advanced-closure-dialog-repeat" data-toggle="tab" href="#wmeac-advanced-closure-dialog-tabrepeat">Repeat</a> </li> <li> <a id="wmeac-advanced-closure-dialog-each" data-toggle="tab" href="#wmeac-advanced-closure-dialog-tabeach">Each</a> </li> <li> <a id="wmeac-advanced-closure-dialog-holiday" data-toggle="tab" href="#wmeac-advanced-closure-dialog-tabholiday">Holidays</a> </li> <li style="float: right;"> <a id="wmeac-advanced-closure-dialog-presets" data-toggle="tab" href="#wmeac-advanced-closure-dialog-tabpresets"><i class="fa fa-floppy-o"></i></a> </li> </ul> <div class="tab-content"> <div class="tab-pane active wmeac-tab-pane" id="wmeac-advanced-closure-dialog-tabrepeat"> ' +
tabRepeatUI + ' </div> <div class="tab-pane wmeac-tab-pane" id="wmeac-advanced-closure-dialog-tabeach"> ' + tabEachUI + ' </div> <div class="tab-pane wmeac-tab-pane" id="wmeac-advanced-closure-dialog-tabholiday"> ' + tabHolidayUI + ' </div> <div class="tab-pane wmeac-tab-pane" id="wmeac-advanced-closure-dialog-tabpresets"> ' + tabPresetsUI + " </div> </div>";
var footerUI = '<div class="footer"> <div id="wmeac-csv-closures-preview"><div id="wmeac-csv-closures-preview-content" style="overflow: scroll; max-height: 100px;"></div></div> <button style="float: left;" id="wmeac-advanced-closure-dialog-exportCSV-button">Export CSV</button> <button style="float: right;" id="wmeac-advanced-closure-dialog-close-button">Close</button> <button style="float: right;" id="wmeac-advanced-closure-dialog-apply-button">Apply</button></div>';
WMEAC.HTMLTemplates.advancedClosureDialog = '<h1>Advanced closures</h1><div class="content"> <table> <tr> <td style="width: 50%;">' + rangeStartEndUI + startTimeAndDurationUI + " </td> <td>" + descriptionUI + directionUI + ignoreTrafficUI + MTEUI + " </td> </tr> </table>" + tabsUI + "</div>" + footerUI;
WMEAC.connectAdvancedClosureDialogHandlers = function() {
var e = null;
e = WMEAC.getId("wmeac-advanced-closure-dialog-exportCSV-button");
if (e) {
e.addEventListener("click", function() {
var rc = WMEAC.buildClosuresListFromRecurringUI();
if (rc.error != "") {
alert("Can't apply closures.\nPlease, check all parameters.");
return;
}
if (Waze.selectionManager.selectedItems.isEmpty() || Waze.selectionManager.selectedItems[0].model.type != "segment") {
alert("Please, select segment(s) before.");
return;
}
var reason = $("#wmeac-advanced-closure-dialog-reason").val();
var direction = $("#wmeac-advanced-closure-dialog-direction").val();
var isIT = $("#wmeac-advanced-closure-dialog-ignoretraffic").is(":checked");
var mteId = $("#wmeac-advanced-closure-dialog-mteid").val();
var mte = Waze.model.majorTrafficEvents.get(mteId);
closureList = rc.list.map(function(e) {
var details = {reason:reason, direction:direction, startDate:e.start, endDate:e.end, location:"", permanent:isIT};
if (mte) {
details.eventId = mte.id;
}
return details;
});
var selection = _.pluck(Waze.selectionManager.selectedItems, "model");
var selectionReversed = [];
if (direction != "3") {
var rev = Waze.selectionManager.getReversedSegments();
selection = selection.filter(function(e) {
if (rev[e.attributes.id]) {
selectionReversed.push(e);
return false;
}
return true;
});
}
var lonlat = Waze.map.center.transform(Waze.map.projection.projCode, "EPSG:4326");
var csv = "header,reason,start date (yyyy-mm-dd hh:mm),end date (yyyy-mm-dd hh:mm),direction (A to B|B to A|TWO WAY),ignore trafic (Yes|No),segment IDs (id1;id2;...),lon/lat (like in a permalink: lon=xxx&lat=yyy),zoom (2 to 10),MTE id (empty cell if not),comment (optional)\n";
closureList.forEach(function(e) {
csv += 'add,"' + e.reason + '","' + e.startDate + '","' + e.endDate + '","' + (direction == 3 ? "TWO WAY" : direction == 2 ? "B to A" : "A to B") + '",' + (isIT ? "Yes" : "No") + ',"' + selection.map(function(s) {
return s.attributes.id;
}).join(";") + '","lon=' + lonlat.lon + "&lat=" + lonlat.lat + '",' + Waze.map.zoom + "," + mteId + ',"Generated by WMEAC"\n';
});
if (!selectionReversed.isEmpty()) {
closureList.forEach(function(e) {
csv += 'add,"' + e.reason + '","' + e.startDate + '","' + e.endDate + '","' + (direction == 3 ? "TWO WAY" : direction == 2 ? "A to B" : "B to A") + '",' + (isIT ? "Yes" : "No") + ',"' + selectionReversed.map(function(s) {
return s.attributes.id;
}).join(";") + '","lon=' + lonlat.lon + "&lat=" + lonlat.lat + '",' + Waze.map.zoom + "," + mteId + ',"Generated by WMEAC"\n';
});
}
WMEAC.download(csv, "closures.csv");
});
}
e = WMEAC.getId("wmeac-advanced-closure-dialog-close-button");
if (e) {
e.addEventListener("click", function() {
var d = WMEAC.getId("wmeac-add-advanced-closure-dialog");
if (d) {
Waze.selectionManager.events.unregister("selectionchanged", null, WMEAC.refreshClosureList);
Waze.selectionManager.events.unregister("selectionchanged", null, WMEAC.refreshClosureListFromSelection);
d.style.display = "none";
}
});
}
e = WMEAC.getId("wmeac-advanced-closure-dialog-apply-button");
if (e) {
e.addEventListener("click", function() {
var rc = WMEAC.buildClosuresListFromRecurringUI();
if (rc.error != "") {
alert("Can't apply closures.\nPlease, check all parameters.");
return;
}
if (Waze.selectionManager.selectedItems.isEmpty() || Waze.selectionManager.selectedItems[0].model.type != "segment") {
alert("Please, select segment(s) before.");
return;
}
if (Waze.selectionManager.selectedItems.every(function(e) {
return e.model.isAllowed(e.model.PERMISSIONS.EDIT_CLOSURES);
}) == false) {
alert("You don't have permission to edit closures on all those segments.");
return;
}
var reason = $("#wmeac-advanced-closure-dialog-reason").val();
var direction = $("#wmeac-advanced-closure-dialog-direction").val();
var sc = require("Waze/Modules/Closures/Models/SharedClosure");
direction = direction == "1" ? sc.DIRECTION.A_TO_B : direction == "2" ? sc.DIRECTION.B_TO_A : sc.DIRECTION.TWO_WAY;
var directionStr = direction == 1 ? "(A → B)" : direction == 2 ? "(B → A)" : "(⇆)";
var isIT = $("#wmeac-advanced-closure-dialog-ignoretraffic").is(":checked");
var mte = Waze.model.majorTrafficEvents.get($("#wmeac-advanced-closure-dialog-mteid").val());
closureList = rc.list.map(function(e) {
var details = {reason:reason, direction:direction, startDate:e.start, endDate:e.end, location:"", permanent:isIT};
if (mte) {
details.eventId = mte.id;
}
return details;
});
var selection = _.pluck(Waze.selectionManager.selectedItems, "model");
Waze.selectionManager.events.unregister("selectionchanged", null, WMEAC.refreshClosureList);
WMEAC.addClosureListFromSelection(closureList, function(i, e) {
$("#wmeac-advanced-closure-dialog-preview-" + i).html(e).css({color:"#44D544"});
}, function(i, e) {
$("#wmeac-advanced-closure-dialog-preview-" + i).html(e).css({color:"#D5444F"});
}, function() {
Waze.selectionManager.select(selection);
var tmp = function selectionReady() {
if (Waze.selectionManager.selectedItems.isEmpty()) {
window.setTimeout(selectionReady, 500);
} else {
Waze.selectionManager.events.register("selectionchanged", null, WMEAC.refreshClosureList);
$('a[href="#segment-edit-closures"]').click();
}
};
window.setTimeout(tmp, 500);
}, 0);
});
}
if (typeof $.fn.datepicker !== "undefined") {
$("#wmeac-advanced-closure-dialog-rangestartdate,#wmeac-advanced-closure-dialog-rangeenddate").datepicker({format:"yyyy-mm-dd", todayHighlight:!0, autoclose:!0});
} else {
if (typeof $.fn.daterangepicker !== "undefined") {
$("#wmeac-advanced-closure-dialog-rangestartdate,#wmeac-advanced-closure-dialog-rangeenddate").daterangepicker({singleDatePicker:!0, locale:{format:"YYYY-MM-DD"}});
}
}
$("#wmeac-advanced-closure-dialog-rangestartdate,#wmeac-advanced-closure-dialog-rangeenddate").on("change", function() {
WMEAC.refreshMTEList();
});
$("#wmeac-advanced-closure-dialog-starttime,#wmeac-advanced-closure-dialog-durationtime").timepicker({defaultTime:"00:00", showMeridian:!1, template:!1});
$("#wmeac-add-advanced-closure-dialog").find(".input-group").find(".input-group-addon").on("click", function(e) {
$(e.target).parent().find("input").focus();
}).find("i").on("click", function(e) {
$(e.target).parent().parent().find("input").focus();
});
$("#wmeac-advanced-closure-dialog-each-dayall").on("click", function() {
var atLeastOneChecked = false;
for (var i = 0;i < 7;i++) {
atLeastOneChecked = atLeastOneChecked || $("#wmeac-advanced-closure-dialog-each-" + i).is(":checked");
}
for (var i = 0;i < 7;i++) {
$("#wmeac-advanced-closure-dialog-each-" + i).prop("checked", !atLeastOneChecked);
}
$("#wmeac-advanced-closure-dialog-each-dayall").prop("checked", !atLeastOneChecked);
});
if (typeof $.fn.spinner !== "undefined") {
$("#wmeac-advanced-closure-dialog-repeat-every-day").spinner({min:0, spin:function(event, ui) {
$(this).trigger("change");
}});
$("#wmeac-advanced-closure-dialog-repeat-every-hour").spinner({min:0, spin:function(event, ui) {
if (ui.value >= 24) {
$(this).spinner("value", ui.value - 24);
$("#wmeac-advanced-closure-dialog-repeat-every-day").spinner("stepUp");
return false;
} else {
if (ui.value < 0) {
$(this).spinner("value", ui.value + 24);
$("#wmeac-advanced-closure-dialog-repeat-every-day").spinner("stepDown");
return false;
}
}
$(this).trigger("change");
}});
$("#wmeac-advanced-closure-dialog-repeat-every-minute").spinner({spin:function(event, ui) {
if (ui.value >= 60) {
$(this).spinner("value", ui.value - 60);
$("#wmeac-advanced-closure-dialog-repeat-every-hour").spinner("stepUp");
return false;
} else {
if (ui.value < 0) {
$(this).spinner("value", ui.value + 60);
$("#wmeac-advanced-closure-dialog-repeat-every-hour").spinner("stepDown");
return false;
}
}
$(this).trigger("change");
}, change:function(event) {
if (event.target.value < 0 || event.target.value > 59) {
$(this).spinner("value", 0);
}
}});
$("#wmeac-advanced-closure-dialog-duration-day").spinner({min:0, spin:function(event, ui) {
$(this).trigger("change");
}});
}
$("#wmeac-advanced-closure-dialog-repeat,#wmeac-advanced-closure-dialog-each,#wmeac-advanced-closure-dialog-holiday").on("click", function(e) {
window.setTimeout(WMEAC.refreshClosureList);
});
$("#wmeac-advanced-closure-dialog-holiday-refresh").on("click", function(e) {
var hDiv = $("#wmeac-advanced-closure-dialog-holiday-list");
$("#wmeac-advanced-closure-dialog-holiday-refresh-spinner").css({display:"block"});
WMEAC.removeChildElements(hDiv[0]);
window.setTimeout(function() {
WMEAC.getHolidays({rangeStart:$("#wmeac-advanced-closure-dialog-rangestartdate").val(), rangeEnd:$("#wmeac-advanced-closure-dialog-rangeenddate").val(), countries:_.pluck(WMEAC.getCountriesFromSegmentSet(_.pluck(Waze.selectionManager.selectedItems, "model")), "abbr"), handlerFinished:function(holidays) {
WMEAC.lastGeneratedHolidays = holidays;
if (holidays.isEmpty()) {
hDiv.html("No holiday found.");
} else {
holidays.forEach(function(h, i) {
var chkBx = WMEAC.createElement({type:"div", className:"checkbox"});
chkBx.innerHTML = '<label class="control-label" style="font-weight: bold;"> <input id="wmeac-advanced-closure-dialog-holidays-' + i + '" type="checkbox"> ' + h.date + ": " + h.name + " (" + h.country + ") </label> ";
$(chkBx).on("click", function(e) {
window.setTimeout(WMEAC.refreshClosureList);
});
hDiv.append(chkBx);
});
}
$("#wmeac-advanced-closure-dialog-holiday-refresh-spinner").css({display:"none"});
}});
});
});
$("#wmeac-add-advanced-closure-dialog").on("change", function(e) {
window.setTimeout(WMEAC.refreshClosureList);
});
WMEAC.reloadPresets();
$("#wmeac-advanced-closure-dialog-presets-load").on("click", function(e) {
var presetIndex = parseInt($("#wmeac-advanced-closure-dialog-presets-list").val());
$("#wmeac-advanced-closure-dialog-starttime").val(WMEAC.presets[presetIndex].values.starttime);
if (WMEAC.presets[presetIndex].values.duration.hasOwnProperty("day")) {
$("#wmeac-advanced-closure-dialog-duration-day").val(WMEAC.presets[presetIndex].values.duration.day);
} else {
$("#wmeac-advanced-closure-dialog-duration-day").val(Math.floor(WMEAC.presets[presetIndex].values.duration.hour / 24));
}
$("#wmeac-advanced-closure-dialog-durationtime").val("" + WMEAC.presets[presetIndex].values.duration.hour % 24 + ":" + WMEAC.presets[presetIndex].values.duration.minute);
$("#wmeac-advanced-closure-dialog-reason").val(WMEAC.presets[presetIndex].values.description);
$("#wmeac-advanced-closure-dialog-direction").val(WMEAC.presets[presetIndex].values.direction);
$("#wmeac-advanced-closure-dialog-ignoretraffic").prop("checked", WMEAC.presets[presetIndex].values.ignoretraffic);
$("#wmeac-advanced-closure-dialog-repeat-ntimes").val(WMEAC.presets[presetIndex].values.repeat.ntimes);
if (WMEAC.presets[presetIndex].values.repeat.hasOwnProperty("day")) {
$("#wmeac-advanced-closure-dialog-repeat-every-day").val(WMEAC.presets[presetIndex].values.repeat.day);
} else {
$("#wmeac-advanced-closure-dialog-repeat-every-day").val(Math.floor(WMEAC.presets[presetIndex].values.repeat.hour / 24));
}
$("#wmeac-advanced-closure-dialog-repeat-every-hour").val(WMEAC.presets[presetIndex].values.repeat.hour % 24);
$("#wmeac-advanced-closure-dialog-repeat-every-minute").val(WMEAC.presets[presetIndex].values.repeat.minute);
for (var i = 0;i < 7;i++) {
$("#wmeac-advanced-closure-dialog-each-" + i).prop("checked", WMEAC.presets[presetIndex].values.each[i]);
}
});
$("#wmeac-advanced-closure-dialog-presets-load-fromseg").on("click", function() {
closureId = $("#wmeac-advanced-closure-dialog-segclosure-list").val();
if (closureId) {
var c = Waze.model.roadClosures.objects[closureId];
if (c) {
$("#wmeac-advanced-closure-dialog-starttime").val(c.startDate.split(" ")[1]);
var duration = new Date(c.endDate) - new Date(c.startDate);
$("#wmeac-advanced-closure-dialog-duration-day").val(Math.floor(duration / 864E5));
$("#wmeac-advanced-closure-dialog-durationtime").val("" + (new Date(duration)).getHours() + ":" + (new Date(duration)).getMinutes());
$("#wmeac-advanced-closure-dialog-reason").val(c.reason.trim());
if (WMEAC.getOppositeClosure(c).isEmpty()) {
$("#wmeac-advanced-closure-dialog-direction").val(c.forward ? 1 : 2);
} else {
$("#wmeac-advanced-closure-dialog-direction").val(3);
}
$("#wmeac-advanced-closure-dialog-ignoretraffic").prop("checked", c.permanent);
}
}
});
$("#wmeac-advanced-closure-dialog-presets-delete").on("click", function(e) {
var presetIndex = parseInt($("#wmeac-advanced-closure-dialog-presets-list").val());
WMEAC.presets.splice(presetIndex, 1);
WMEAC.save();
WMEAC.reloadPresets();
});
$("#wmeac-advanced-closure-dialog-presets-save").on("click", function(e) {
var name = $("#wmeac-advanced-closure-dialog-presets-name").val();
var presetIndex = WMEAC.presets.findIndex(function(e) {
return e.name == name;
});
var preset = {name:name, values:{duration:{}, repeat:{}, each:[]}};
if (presetIndex != -1) {
preset = WMEAC.presets[presetIndex];
}
preset.values.starttime = $("#wmeac-advanced-closure-dialog-starttime").val();
preset.values.duration.day = $("#wmeac-advanced-closure-dialog-duration-day").val();
preset.values.duration.hour = parseInt($("#wmeac-advanced-closure-dialog-durationtime").val().split(":")[0]);
preset.values.duration.minute = parseInt($("#wmeac-advanced-closure-dialog-durationtime").val().split(":")[1]);
preset.values.description = $("#wmeac-advanced-closure-dialog-reason").val();
preset.values.direction = $("#wmeac-advanced-closure-dialog-direction").val();
preset.values.ignoretraffic = $("#wmeac-advanced-closure-dialog-ignoretraffic").is(":checked");
preset.values.repeat.ntimes = $("#wmeac-advanced-closure-dialog-repeat-ntimes").val();
preset.values.repeat.day = $("#wmeac-advanced-closure-dialog-repeat-every-day").val();
preset.values.repeat.hour = $("#wmeac-advanced-closure-dialog-repeat-every-hour").val();
preset.values.repeat.minute = $("#wmeac-advanced-closure-dialog-repeat-every-minute").val();
for (var i = 0;i < 7;i++) {
preset.values.each[i] = $("#wmeac-advanced-closure-dialog-each-" + i).is(":checked");
}
if (presetIndex == -1) {
WMEAC.presets.push(preset);
}
WMEAC.save();
WMEAC.reloadPresets();
});
WMEAC.setDraggable($("#wmeac-add-advanced-closure-dialog"), {controller:$("#wmeac-add-advanced-closure-dialog h1:first-child"), container:$("#WazeMap")});
};
WMEAC.connectAdvancedClosureTabHandlers = function() {
var e = null;
e = WMEAC.getId("wmeac-csv-file");
if (e) {
e.addEventListener("change", WMEAC.CSVFileChanged);
}
e = WMEAC.getId("wmeac-csv-closures-controls-check");
if (e) {
e.addEventListener("change", function(e) {
WMEAC.CSVCheckAll(e.target.checked);
});
}
e = WMEAC.getId("wmeac-csv-closures-controls-apply");
if (e) {
e.addEventListener("click", WMEAC.CSVApplyChecked);
}
e = WMEAC.getId("wmeac-csv-closures-controls-segs");
if (e) {
e.addEventListener("click", WMEAC.CSVCheckSegsChecked);
}
};
WMEAC.reloadPresets = function() {
var optionList = WMEAC.presets.map(function(p, i) {
return '<option value="' + i + '">' + p.name + "</option>";
});
$("#wmeac-advanced-closure-dialog-presets-list").html(optionList.join(""));
};
WMEAC.ProgressBar = function(id) {
this.id = id;
this.divpbi = WMEAC.createElement({type:"div", id:id, className:id});
var elt = WMEAC.createElement({type:"div", id:"wmeac-progressBar"});
elt.style.width = "100%";
elt.style.display = "none";
elt.innerHTML = '<div class="wmeac-progressBarBG"></div><span class="wmeac-progressBarFG">100%</span>';
this.divpbi.appendChild(elt);
elt = WMEAC.createElement({type:"div", id:"wmeac-progressBar-info"});
this.divpbi.appendChild(elt);
this.isShown = function() {
return this.divpbi.style.display != "none";
};
this.show = function(toShow) {
this.divpbi.style.display = toShow ? "block" : "none";
};
this.update = function(value) {
if (value == -1) {
this.divpbi.children[0].style.display = "none";
this.divpbi.children[1].style.display = "none";
return;
}
value = Math.round(value);
this.divpbi.children[0].style.display = "block";
this.divpbi.children[1].style.display = "block";
this.divpbi.children[0].children[0].style.width = value + "%";
this.divpbi.children[0].children[1].innerHTML = value + "%";
};
this.info = function(text) {
this.divpbi.children[1].innerHTML = text;
};
};
WMEAC.ClassClosure = function(options) {
WMEAC.log("options", options);
this.isValid = false;
this.errorMessage = "";
var validProperties = ["reason", "startDate", "endDate", "direction", "segIDs", "lonlat", "permanent", "id", "zoom"];
var goodOptions = 0;
validProperties.forEach(function(p) {
if (options.hasOwnProperty(p)) {
this[p] = options[p];
goodOptions++;
} else {
this.errorMessage += "Missing property " + p + "\n";
}
}, this);
if (goodOptions == validProperties.length) {
this.isValid = true;
} else {
return;
}
this.comment = "";
if (options.hasOwnProperty("comment")) {
this.comment = options.comment;
}
this.eventId = null;
if (options.hasOwnProperty("eventId") && options.eventId != "") {
this.eventId = options.eventId;
}
this.segIDs = this.segIDs.split(";");
var matches = this.lonlat.match(/lon=(-?\d+\.?\d*)&lat=(-?\d+\.?\d*)/);
if (matches && matches.length == 3) {
this.lonlat = {lon:parseFloat(matches[1]), lat:parseFloat(matches[2])};
} else {
matches = this.lonlat.match(/lat=(-?\d+\.?\d*)&lon=(-?\d+\.?\d*)/);
if (matches && matches.length == 3) {
this.lonlat = {lon:parseFloat(matches[2]), lat:parseFloat(matches[1])};
} else {
this.isValid = false;
this.errorMessage = "Can't parse lonlat: " + this.lonlat + "\n";
return;
}
}
if (this.direction != "A to B" && this.direction != "B to A" && this.direction != "TWO WAY") {
this.isValid = false;
this.errorMessage = "Can't determine direction: " + this.direction + "\n";
return;
}
this.zoom = parseInt(this.zoom);
if (this.zoom < 2 || this.zoom > 10) {
this.isValid = false;
this.errorMessage = "Wrong zoom (2 to 10): " + this.zoom + "\n";
return;
}
this.applyInWME = function(successHandler, failureHandler) {
var segs = WMEAC.segmentsIDsToSegments(this.segIDs);
WMEAC.log("Segs: ", segs);
segs = segs.filter(function(seg) {
return seg.isAllowed(seg.PERMISSIONS.EDIT_CLOSURES);
});
if (segs.length == 0) {
failureHandler([{attributes:{details:"No segment. Check permissions or existence."}}]);
} else {
var cityStreets = WMEAC.getCityStreetsFromSegmentSet(segs);
var closureLocation = Object.keys(cityStreets).map(function(c) {
return Object.keys(cityStreets[c]).map(function(s) {
if (s == "noStreet") {
return I18n.translations[I18n.locale].edit.address.no_street;
}
return s;
}).join(", ") + (c == "noCity" ? "" : " (" + c + ")");
}).join(" ; ");
var sc = require("Waze/Modules/Closures/Models/SharedClosure");
var closureDetails = {reason:this.reason, direction:this.direction == "A to B" ? sc.DIRECTION.A_TO_B : this.direction == "B to A" ? sc.DIRECTION.B_TO_A : sc.DIRECTION.TWO_WAY, startDate:this.startDate, endDate:this.endDate, location:closureLocation, permanent:this.permanent == "Yes", segments:segs};
if (this.eventId != null) {
closureDetails.eventId = this.eventId;
}
WMEAC.addClosure(closureDetails, successHandler, failureHandler);
}
};
this.removeInWME = function(successHandler, failureHandler) {
var segs = WMEAC.segmentsIDsToSegments(this.segIDs);
segs = segs.filter(function(seg) {
return seg.isAllowed(seg.PERMISSIONS.EDIT_CLOSURES);
});
var allClosuresToRemove = [];
var countToMatch = this.segIDs.length * (this.direction == "TWO WAY" ? 2 : 1);
segs.forEach(function(s) {
var that = this;
var closures = Waze.model.roadClosures.getObjectArray(function(c) {
return c.startDate == that.startDate && c.endDate == that.endDate && c.reason.trim() == that.reason && c.segID == s.attributes.id && c.permanent == (that.permanent == "Yes");
});
if (this.direction == "TWO WAY" || this.direction == "A to B" && closures.length == 1 && closures[0].forward == true || this.direction == "B to A" && closures.length == 1 && closures[0].forward == false) {
allClosuresToRemove = allClosuresToRemove.concat(closures);
}
}, this);
if (allClosuresToRemove.isEmpty()) {
failureHandler([{attributes:{details:"No segment. Check permissions or existence."}}]);
} else {
WMEAC.removeClosure(allClosuresToRemove, successHandler, failureHandler);
}
};
};
WMEAC.buildClosuresListFromRecurringUI = function() {
var list = [];
var rangeStartDate = new Date($("#wmeac-advanced-closure-dialog-rangestartdate").val());
if (!WMEAC.isValidDate(rangeStartDate)) {
return {list:list, error:"Range start date is not valid"};
}
var rangeEndDate = new Date($("#wmeac-advanced-closure-dialog-rangeenddate").val());
if (!WMEAC.isValidDate(rangeEndDate)) {
return {list:list, error:"Range end date is not valid"};
}
if (rangeEndDate < rangeStartDate) {
return {list:list, error:"Range end date is before range start date"};
}
var dD = parseInt($("#wmeac-advanced-closure-dialog-duration-day").val());
if (isNaN(dD) || dH < 0) {
return {list:list, error:"Duration days is invalid"};
}
var dH = parseInt($("#wmeac-advanced-closure-dialog-durationtime").val().split(":")[0]);
var dM = parseInt($("#wmeac-advanced-closure-dialog-durationtime").val().split(":")[1]);
if (dD == 0 && dH == 0 && dM == 0) {
return {list:list, error:"Duration is null"};
}
var rangeStartTimeM = 0;
var rangeEndTimeM = 1440;
var rangeEndDateTime = rangeEndDate.clone();
rangeEndDateTime.addMinutes(rangeEndTimeM);
var startTimeM = $("#wmeac-advanced-closure-dialog-starttime").val().split(":").map(function(e) {
return parseInt(e);
}).reduce(function(p, c, i) {
return p * 60 + c;
});
if ($("#wmeac-advanced-closure-dialog-tabrepeat").attr("class").indexOf("active") != -1) {
var ntimes = parseInt($("#wmeac-advanced-closure-dialog-repeat-ntimes").val());
if (isNaN(ntimes) || ntimes < 1) {
return {list:list, error:"Repeat count is invalid"};
}
var evD = parseInt($("#wmeac-advanced-closure-dialog-repeat-every-day").val());
if (isNaN(evD) || evD < 0) {
return {list:list, error:"Repeat every day is invalid"};
}
var evH = parseInt($("#wmeac-advanced-closure-dialog-repeat-every-hour").val());
if (isNaN(evH) || evH < 0) {
return {list:list, error:"Repeat every hour is invalid"};
}
var evM = parseInt($("#wmeac-advanced-closure-dialog-repeat-every-minute").val());
if (isNaN(evM) || evM < 0 || evM >= 60) {
return {list:list, error:"Repeat every minute is invalid"};
}
if (evD * 1440 + evH * 60 + evM < dD * 1440 + dH * 60 + dM) {
return {list:list, error:"Repeat must be greater than duration"};
}
var firstDateTimeStart = rangeStartDate.clone();
if (startTimeM < rangeStartTimeM) {
firstDateTimeStart.addDays(1);
}
firstDateTimeStart.setMinutes(startTimeM);
var firstDateTimeEnd = firstDateTimeStart.clone();
firstDateTimeEnd.addMinutes(dD * 1440 + dH * 60 + dM);
for (var i = 0;i < ntimes;i++) {
var start = firstDateTimeStart.clone();
start.addMinutes((evD * 1440 + evH * 60 + evM) * i);
var end = start.clone();
end.addMinutes(dD * 1440 + dH * 60 + dM);
if (end > rangeEndDateTime) {
break;
}
list.push({start:WMEAC.dateToClosureStr(start), end:WMEAC.dateToClosureStr(end)});
}
return {list:list, error:""};
} else {
if ($("#wmeac-advanced-closure-dialog-tabeach").attr("class").indexOf("active") != -1) {
var dow = WMEAC.daysOfWeek.map(function(e, i) {
return $("#wmeac-advanced-closure-dialog-each-" + i).is(":checked");
});
var dayCount = Math.ceil((rangeEndDate - rangeStartDate + 1) / 864E5);
var day0 = rangeStartDate.clone();
day0.addMinutes(startTimeM);
if (startTimeM < rangeStartTimeM) {
day0.addDays(1);
}
for (var d = 0;d < dayCount;d++) {
var start = day0.clone();
start.addMinutes(d * 1440);
if (dow[start.getUTCDay()]) {
var end = start.clone();
end.addMinutes(dD * 1440 + dH * 60 + dM);
if (end > rangeEndDateTime) {
break;
}
list.push({start:WMEAC.dateToClosureStr(start), end:WMEAC.dateToClosureStr(end)});
}
}
return {list:list, error:""};
} else {
if ($("#wmeac-advanced-closure-dialog-tabholiday").attr("class").indexOf("active") != -1) {
WMEAC.lastGeneratedHolidays.forEach(function(e, i) {
if ($("#wmeac-advanced-closure-dialog-holidays-" + i).is(":checked")) {
var start = (new Date(e.date)).addMinutes(startTimeM);
var end = start.clone();
end.addMinutes(dD * 1440 + dH * 60 + dM);
list.push({start:WMEAC.dateToClosureStr(start), end:WMEAC.dateToClosureStr(end)});
}
});
return {list:list, error:""};
} else {
return {list:list, error:"Wrong tab active"};
}
}
}
};
WMEAC.refreshClosureList = function() {
try {
var rc = WMEAC.buildClosuresListFromRecurringUI();
if (rc.error != "") {
$("#wmeac-csv-closures-preview-content").html(rc.error);
} else {
var reason = $("#wmeac-advanced-closure-dialog-reason").val();
var direction = $("#wmeac-advanced-closure-dialog-direction").val();
var directionStr = direction == 1 ? "(A → B)" : direction == 2 ? "(B → A)" : "(⇆)";
var isIT = $("#wmeac-advanced-closure-dialog-ignoretraffic").is(":checked");
var existingClosures = Waze.selectionManager.selectedItems.reduce(function(p, c, i) {
var revSegs = Waze.selectionManager.getReversedSegments();
var isReversed = revSegs.hasOwnProperty(c.model.attributes.id) && revSegs[c.model.attributes.id];
var realWay = isReversed ? direction == 1 ? 2 : 1 : direction;
return p.concat(Waze.model.roadClosures.getObjectArray(function(e) {
return e.segID == c.model.attributes.id && (direction == 3 || e.forward && realWay == 1 || !e.forward && realWay == 2);
}));
}, []);
var mte = Waze.model.majorTrafficEvents.get($("#wmeac-advanced-closure-dialog-mteid").val());
$("#wmeac-csv-closures-preview-content").html("" + rc.list.length + " closure(s) to apply: <br>" + rc.list.map(function(e, i) {
var overlap = existingClosures.filter(function(c) {
return WMEAC.dateTimeOverlaps({startDate:e.start, endDate:e.end}, c);
}).map(function(c) {
var msg = (c.reason ? c.reason + " " : "") + "(" + c.segID + ")";
if (Waze.model.segments.objects.hasOwnProperty(c.segID) == false) {
return msg;
}
if (Waze.model.segments.objects[c.segID].attributes.primaryStreetID == null) {
return msg;
}
if (Waze.model.streets.objects.hasOwnProperty(Waze.model.segments.objects[c.segID].attributes.primaryStreetID) == false) {
return msg;
}
var street = Waze.model.streets.objects[Waze.model.segments.objects[c.segID].attributes.primaryStreetID];
if (!street.isEmpty) {
msg = street.name + ": " + msg;
}
return msg;
});
var mteOK = !(mte && (new Date(e.start) < new Date(mte.attributes.startDate) || new Date(e.end) > new Date(mte.attributes.endDate)));
return reason + ": " + e.start + " → " + e.end + " " + directionStr + ' <i class="fa fa-car' + (isIT ? " slashed" : "") + '"></i>' + (overlap.length != 0 ? ' <i title="Warning: overlap on existing closure!\n' + overlap.join("\n") + '" class="fa fa-exclamation-circle" style="color: orange"></i>' : "") + (mteOK ? "" : ' <i title="Warning: closure dates not inside MTE date!" class="fa fa-exclamation-circle" style="color: orange"></i>') + ' <span id="wmeac-advanced-closure-dialog-preview-' +
i + '"></span>';
}).join("<br>"));
}
} catch (e) {
WMEAC.logError("Error while refreshing closure list: ", e);
}
};
WMEAC.refreshMTEList = function() {
var currentMTEid = $("#wmeac-advanced-closure-dialog-mteid").val();
var rangeStart = new Date($("#wmeac-advanced-closure-dialog-rangestartdate").val());
var rangeEnd = new Date($("#wmeac-advanced-closure-dialog-rangeenddate").val());
var options = [{name:"none", value:""}];
$("#wmeac-advanced-closure-dialog-mteid").empty();
if (WMEAC.isValidDate(rangeStart) && WMEAC.isValidDate(rangeEnd)) {
rangeEnd.addDays(1);
Waze.model.majorTrafficEvents.getObjectArray(function(mte) {
return WMEAC.dateTimeOverlaps({startDate:rangeStart, endDate:rangeEnd}, {startDate:new Date(mte.attributes.startDate), endDate:new Date(mte.attributes.endDate)});
}).forEach(function(mte) {
options.push({name:mte.attributes.names[0].value, value:mte.attributes.id});
});
}
options.forEach(function(o) {
var el = WMEAC.createElement({type:"option"});
el.setAttribute("value", o.value);
if (currentMTEid == o.value) {
el.setAttribute("selected", "");
}
el.innerHTML = o.name;
$("#wmeac-advanced-closure-dialog-mteid").append(el);
});
if (options.length > 1) {
$("#wmeac-advanced-closure-dialog-mteid").removeAttr("disabled");
} else {
$("#wmeac-advanced-closure-dialog-mteid").attr("disabled", "");
}
};
WMEAC.refreshClosureListFromSelection = function() {
try {
var currentSegClosure = $("#wmeac-advanced-closure-dialog-segclosure-list").val();
$("#wmeac-advanced-closure-dialog-segclosure-list").empty();
if (Waze.selectionManager.selectedItems.length != 0) {
var blackList = [];
Waze.model.roadClosures.getObjectArray(function(c) {
return c.segID == Waze.selectionManager.selectedItems[0].model.attributes.id;
}).sort(function(a, b) {
return new Date(a.startDate) - new Date(b.startDate);
}).forEach(function(c) {
if (blackList.indexOf(c.id) != -1) {
return;
}
var direction = c.forward ? "A to B" : "B to A";
var oppositeClosure = WMEAC.getOppositeClosure(c);
if (!oppositeClosure.isEmpty()) {
direction = "Two way";
blackList.push(oppositeClosure[0].id);
}
var el = WMEAC.createElement({type:"option"});
el.setAttribute("value", c.id);
if (currentSegClosure == c.id) {
el.setAttribute("selected", "");
}
el.innerHTML = c.reason.trim() + " " + direction + " " + c.startDate + "→" + c.endDate;
$("#wmeac-advanced-closure-dialog-segclosure-list").append(el);
});
}
} catch (e) {
WMEAC.logError("Error while refreshing closure list from selection: ", e);
}
};
WMEAC.abbrToISO3166_1alpha2 = function(abbr) {
switch(abbr) {
case "GM":
return "DE";
break;
case "BU":
return "BG";
break;
case "EZ":
return "CZ";
break;
case "SP":
return "ES";
break;
case "UK":
return "GB";
break;
case "LO":
return "SK";
break;
default:
return abbr;
break;
}
return abbr;
};
WMEAC.getHolidays = function(options) {
var holidays = [];
var currentCountryIndex = 0;
var rangeStart = new Date(options.rangeStart);
var rangeEnd = (new Date(options.rangeEnd)).addDays(1);
var years = [];
for (y = parseInt(options.rangeStart.substring(0, 4));y <= parseInt(options.rangeEnd.substring(0, 4));y++) {
years.push(y);
}
var currentYearIndex = 0;
var tmp1 = function downloadNext() {
if (currentCountryIndex >= options.countries.length) {
if (options.handlerFinished) {
holidays.sort(function(a, b) {
return new Date(a.date) - new Date(b.date);
});
holidays = holidays.filter(function(h, i) {
if (i != 0) {
if (holidays[i - 1].date == h.date) {
holidays[i - 1].name += " / " + h.name;
holidays[i - 1].country += " / " + h.country;
return false;
}
}
return true;
});
options.handlerFinished(holidays);
return;
}
}
window.WMEAC_downloadHelper.add("https://holidayapi.com/v1/holidays?key=ca1a7925-0045-47da-bcf6-ee0de583e384&country=" + WMEAC.abbrToISO3166_1alpha2(options.countries[currentCountryIndex]) + "&year=" + years[currentYearIndex], function(data) {
if (data.status == "success") {
try {
nextHoliday = JSON.parse(data.data);
if (nextHoliday.hasOwnProperty("holidays")) {
for (var hd in nextHoliday.holidays) {
if (!nextHoliday.holidays.hasOwnProperty(hd)) {
continue;
}
if (nextHoliday.holidays[hd].length == 0) {
continue;
}
var name = _.pluck(nextHoliday.holidays[hd], "name").join(" / ");
var h = nextHoliday.holidays[hd][0];
var d = new Date(h.date);
if (d >= rangeStart && d < rangeEnd) {
holidays.push({date:h.date, name:name, country:h.country});
}
}
currentYearIndex++;
if (currentYearIndex < years.length) {
window.setTimeout(downloadNext);
return;
}
}
} catch (e) {
WMEAC.log("Error while getting holiday from server!", e);
WMEAC.log("data", data.data);
}
currentCountryIndex++;
currentYearIndex = 0;
window.setTimeout(downloadNext);
}
});
};
tmp1();
};
WMEAC.addClosure = function(options, successHandler, failureHandler) {
if (options && options.hasOwnProperty("segments") && options.hasOwnProperty("reason") && options.hasOwnProperty("direction") && options.hasOwnProperty("startDate") && options.hasOwnProperty("endDate") && options.hasOwnProperty("location") && options.hasOwnProperty("permanent")) {
WMEAC.log("Addinf closure: ", options);
var fail = function(e) {
return function(f) {
if (failureHandler) {
failureHandler(f);
} else {
WMEAC.log("Failed to create closure:", f);
}
};
};
var done = function(e) {
return function(f) {
if (successHandler) {
successHandler(f);
} else {
WMEAC.log("Closure successful:", f);
}
};
};
var cab = require("Waze/Modules/Closures/Models/ClosureActionBuilder");
var sc = require("Waze/Modules/Closures/Models/SharedClosure");
var t = {};
var closureDetails = {reason:options.reason + String.fromCharCode(160), direction:options.direction, startDate:options.startDate, endDate:options.endDate, location:options.location, permanent:options.permanent, segments:options.segments, reverseSegments:{}};
if (options.hasOwnProperty("eventId") && options.eventId != null) {
closureDetails.eventId = options.eventId;
}
var c = new sc(closureDetails);
t.actions = [cab.add(c)];
W.controller.save(t).done(done()).fail(fail());
return true;
}
return false;
};
WMEAC.addClosureListFromSelection = function(closureList, successHandler, failureHandler, endHandler, i) {
if (i >= closureList.length) {
WMEAC.reloadClosuresLayer(function() {
if (endHandler) {
endHandler();
}
});
return;
}
var c = closureList[i];
var fail = function(e) {
return function(f) {
if (failureHandler) {
var details = [];
f.forEach(function(err) {
if (err.hasOwnProperty("attributes") && err.attributes.hasOwnProperty("details")) {
details.push(err.attributes.details);
}
});
failureHandler(i, details.join(" | "));
} else {
WMEAC.log("Failed to create closure:", f);
}
WMEAC.addClosureListFromSelection(closureList, successHandler, failureHandler, endHandler, i + 1);
};
};
var done = function(e) {
return function(f) {
if (successHandler) {
successHandler(i, "OK");
} else {
WMEAC.log("Closure successful:", f);
}
WMEAC.addClosureListFromSelection(closureList, successHandler, failureHandler, endHandler, i + 1);
};
};
var cab = require("Waze/Modules/Closures/Models/ClosureActionBuilder");
var sc = require("Waze/Modules/Closures/Models/SharedClosure");
var t = {};
var segs = _.pluck(Waze.selectionManager.selectedItems, "model");
var cityStreets = WMEAC.getCityStreetsFromSegmentSet(segs);
var closureLocation = Object.keys(cityStreets).map(function(c) {
return Object.keys(cityStreets[c]).map(function(s) {
if (s == "noStreet") {
return I18n.translations[I18n.locale].edit.address.no_street;
}
return s;
}).join(", ") + (c == "noCity" ? "" : " (" + c + ")");
}).join(" ; ");
var closureDetails = {reason:closureList[i].reason + String.fromCharCode(160), direction:closureList[i].direction, startDate:closureList[i].startDate, endDate:closureList[i].endDate, location:closureLocation, permanent:closureList[i].permanent, segments:segs, reverseSegments:Waze.selectionManager.getReversedSegments()};
if (closureList[i].hasOwnProperty("eventId") && closureList[i].eventId != null) {
closureDetails.eventId = closureList[i].eventId;
}
var c = new sc(closureDetails);
t.actions = [cab.add(c)];
W.controller.save(t).done(done()).fail(fail());
};
WMEAC.addClosureFromSelection = function(options, successHandler, failureHandler) {
if (options && options.hasOwnProperty("reason") && options.hasOwnProperty("direction") && options.hasOwnProperty("startDate") && options.hasOwnProperty("endDate") && options.hasOwnProperty("location") && options.hasOwnProperty("permanent")) {
WMEAC.log("Addinf closure: ", options);
var fail = function(e) {
return function(f) {
if (failureHandler) {
failureHandler(f);
} else {
WMEAC.log("Failed to create closure:", f);
}
};
};
var done = function(e) {
return function(f) {
if (successHandler) {
successHandler(f);
} else {
WMEAC.log("Closure successful:", f);
}
};
};
var cab = require("Waze/Modules/Closures/Models/ClosureActionBuilder");
var sc = require("Waze/Modules/Closures/Models/SharedClosure");
var t = {};
var segs = _.pluck(Waze.selectionManager.selectedItems, "model");
var closureDetails = {reason:options.reason + String.fromCharCode(160), direction:options.direction, startDate:options.startDate, endDate:options.endDate, location:options.location, permanent:options.permanent, segments:segs, reverseSegments:Waze.selectionManager.getReversedSegments()};
if (options.hasOwnProperty("eventId") && options.eventId != null) {
closureDetails.eventId = options.eventId;
}
var c = new sc(closureDetails);
t.actions = [cab.add(c)];
W.controller.save(t).done(done()).fail(fail());
return true;
}
return false;
};
WMEAC.removeClosure = function(closures, successHandler, failureHandler) {
var fail = function(e) {
return function(f) {
if (failureHandler) {
failureHandler(f);
} else {
WMEAC.log("Failed to delete closure:", f);
}
};
};
var done = function(e) {
return function(f) {
if (successHandler) {
successHandler(f);
} else {
WMEAC.log("Closure deletion successful:", f);
}
};
};
var cab = require("Waze/Modules/Closures/Models/ClosureActionBuilder");
var sc = require("Waze/Modules/Closures/Models/SharedClosure");
var t = {};
var c = new sc({closures:[].concat(closures)});
t.actions = [cab["delete"](c)];
W.controller.save(t).done(done()).fail(fail());
return true;
};
WMEAC.save = function() {
WMEAC.log("save data...");
localStorage.WMEAC = JSON.stringify({presets:WMEAC.presets});
};
WMEAC.load = function() {
try {
var saved = JSON.parse(localStorage.WMEAC);
WMEAC.presets = saved.presets;
WMEAC.log("presets", WMEAC.presets);
} catch (err) {
WMEAC.log("Error while loading data from storage: ", err);
}
};
WMEAC.parseCSV = function(csvString) {
if (csvString != null) {
var csvArray = WMEAC.CSVtoArray(csvString);
WMEAC.log("CSV as array:", csvArray);
var isValid = WMEAC.csv[0].validate(csvArray);
if (isValid.isValid) {
WMEAC.log("CSV is valid!");
var closures = WMEAC.csv[0].filter(csvArray).map(function(e, i) {
return {action:e[0], closure:new WMEAC.ClassClosure({reason:e[1], startDate:e[2], endDate:e[3], direction:e[4], segIDs:e[6], lonlat:e[7], permanent:e[5], zoom:e[8], id:i, eventId:e[9], comment:e.length == 11 ? e[10] : ""}), UI:null};
});
WMEAC.log("Closure list:", closures);
WMEAC.csvCurrentClosureList = closures;
var listUI = WMEAC.getId("wmeac-csv-closures-list-elts");
WMEAC.removeChildElements(listUI);
closures.forEach(function(c) {
c.UI = WMEAC.buildInlineClosureUI(c.closure, c.action);
listUI.appendChild(c.UI);
});
WMEAC.csvShowList(true);
WMEAC.csvAddLog("CSV parse successful\n");
return true;
} else {
WMEAC.log("CSV is NOT valid!:" + isValid.feedBack + "\n");
WMEAC.csvAddLog(isValid.feedBack + "\n");
WMEAC.csvShowList(false);
WMEAC.csvCurrentClosureList = null;
return false;
}
return false;
}
return false;
};
WMEAC.CSVFileChanged = function(evt) {
var files = evt.target.files;
for (var i = 0, f;f = files[i];i++) {
var reader = new FileReader;
reader.onload = function(theFile) {
return function(e) {
WMEAC.log("import CSV file read");
WMEAC.csvClearLog();
if (WMEAC.parseCSV(e.target.result)) {
WMEAC.csvCurrentBatchClosureList = WMEAC.csvCurrentClosureList.slice();
}
};
}(f);
reader.readAsText(f);
}
this.value = null;
WMEAC.getId("wmeac-csv-closures-controls-check").checked = false;
};
WMEAC.ClassCSV = function(options) {
this.isValid = false;
if (options.hasOwnProperty("version")) {
this.version = options.version;
} else {
return;
}
if (options.hasOwnProperty("regexpValidation")) {
this.regexpValidation = options.regexpValidation;
} else {
return;
}
this.isValid = true;
this.validate = function(data) {
var regexps = this.regexpValidation;
var feedBack = "";
this.filter(data).forEach(function(line, l) {
var isLineValid = line.reduce(function(stillValid, cell, i) {
var isCellValid = cell.match(regexps[i]) != null;
if (!isCellValid) {
feedBack += "Error while parsing line " + l + " cell " + i + ': "' + cell + '" in line ' + line.join(",");
}
return stillValid && isCellValid;
}, true);
}, this);
return {isValid:feedBack == "", feedBack:feedBack};
};
this.filter = function(data) {
return data.filter(function(line) {
return line.length >= 1 && ["add", "remove"].indexOf(line[0]) != -1;
});
};
};
WMEAC.csv.push(new WMEAC.ClassCSV({version:1, regexpValidation:[/.*/, /.*/, /^\d{4}-\d{2}-\d{2} \d{2}:\d{2}$/, /^\d{4}-\d{2}-\d{2} \d{2}:\d{2}$/, /(^A to B$)|(^B to A$)|(^TWO WAY$)/, /(Yes)|(No)/, /^(\d+(;|$))+/, /(lon=(-?\d+\.?\d*)&lat=(-?\d+\.?\d*))|(lat=(-?\d+\.?\d*)&lon=(-?\d+\.?\d*))/, /^\d$/, /(^$)|(^-?\d+\.-?\d+\.-?\d+$)/]}));
WMEAC.buildInlineClosureUI = function(closure, action) {
var liElt = WMEAC.createElement({type:"li", className:"wmeac-csv-closures-list-" + action});
liElt.setAttribute("closureID", closure.id);
liElt.innerHTML = '<div class="wmeac-csv-closures-list-col-action"><input type="checkbox" /></div> <div class="wmeac-csv-closures-list-col-lr"><div title="' + closure.reason + '">' + closure.reason + '</div><div title="' + closure.comment + '">' + closure.comment + '</div></div> <div class="wmeac-csv-closures-list-col-dates"><div title="' + closure.startDate + '">' + closure.startDate + '</div><div title="' + closure.endDate + '">' + closure.endDate + '</div></div> <div class="wmeac-csv-closures-list-col-dir">' +
(closure.direction == "A to B" ? "A→B" : closure.direction == "B to A" ? "B→A" : "A↔B") + '</div> <div class="wmeac-csv-closures-list-col-it"><input type="checkbox" ' + (closure.permanent == "Yes" ? "checked" : "") + ' disabled/></div> <div class="wmeac-csv-closures-list-col-target"><a href="#" title="Go there!"><i class="fa fa-crosshairs"></i></a></div> <div class="wmeac-csv-closures-list-col-apply"><a href="#" title="Apply action of this closure"><i class="fa fa-arrow-circle-right"></i></a></div> <div class="wmeac-csv-closures-minilog" style="display: block;">' +
(action == "add" ? "Ready to apply" : action == "remove" ? "Ready to remove" : "") + "</div>";
liElt.children[5].children[0].addEventListener("click", function(e) {
WMEAC.csvClearLog();
var cid = parseInt(e.target.parentNode.parentNode.parentNode.getAttribute("closureID"));
var closure = WMEAC.csvCurrentClosureList.find(function(c) {
return c.closure.id == cid;
});
WMEAC.log("Closure to target:", closure);
var xy = OpenLayers.Layer.SphericalMercator.forwardMercator(closure.closure.lonlat.lon, closure.closure.lonlat.lat);
Waze.map.setCenter(xy, closure.closure.zoom);
var tmp3 = function selectSegments() {
WMEAC.log("Now select segments...");
var segs = WMEAC.segmentsIDsToSegments(closure.closure.segIDs);
if (segs.length != closure.closure.segIDs.length) {
if (segs.length == 0) {
WMEAC.csvAddLog("No segment found: " + closure.closure.comment + "(" + closure.closure.reason + ")\n");
WMEAC.setCSVMiniLog(closure, "Selection failed: no segment found", 3);
} else {
WMEAC.csvAddLog("Partial selection (" + segs.length + "/" + closure.closure.segIDs.length + "): " + closure.closure.comment + "(" + closure.closure.reason + ")\n");
WMEAC.setCSVMiniLog(closure, "Partial selection: " + segs.length + "/" + closure.closure.segIDs.length, 2);
}
alert("Warning: missing segments.\nFound " + segs.length + "/" + closure.closure.segIDs.length + " segment(s)");
} else {
WMEAC.csvAddLog("Selection ok (" + segs.length + "): " + closure.closure.comment + "(" + closure.closure.reason + ")\n");
WMEAC.setCSVMiniLog(closure, "Selection OK: " + segs.length, 1);
}
if (segs.length != 0) {
Waze.selectionManager.select(segs);
var tmp = function selectionReady() {
if (Waze.selectionManager.selectedItems.isEmpty()) {
window.setTimeout(selectionReady, 500);
} else {
$('a[href="#segment-edit-closures"]').click();
}
};
window.setTimeout(tmp, 500);
}
};
var tmp2 = function readyToSelect() {
WMEAC.log("Test if ready to select...");
if (WMEAC.pendingOps == true) {
WMEAC.log("Not yet. Waiting for WME...");
window.setTimeout(readyToSelect, 500);
} else {
tmp3();
}
};
var tmp1 = function mapMovedEnd() {
WMEAC.log("Test if roads are reloaded...");
if (WMEAC.pendingOps == true) {
WMEAC.log("Not yet. Waiting for WME...");
window.setTimeout(mapMovedEnd, 500);
} else {
WMEAC.reloadRoadLayer();
tmp2();
}
};
window.setTimeout(tmp1, 500);
});
liElt.children[6].children[0].addEventListener("click", function(e) {
WMEAC.csvClearLog();
var liElt = e.target.parentNode.parentNode.parentNode;
var cid = parseInt(liElt.getAttribute("closureID"));
var closure = WMEAC.csvCurrentClosureList.find(function(c) {
return c.closure.id == cid;
});
WMEAC.log("Closure to apply:", closure);
WMEAC.csvApplyClosure(closure, null);
});
return liElt;
};
WMEAC.csvApplyClosure = function(closure, handler) {
var xy = OpenLayers.Layer.SphericalMercator.forwardMercator(closure.closure.lonlat.lon, closure.closure.lonlat.lat);
Waze.map.setCenter(xy, closure.closure.zoom);
function applySuccess(evt) {
WMEAC.csvAddLog("Closure OK: " + closure.closure.comment + "(" + closure.closure.reason + ")\n");
closure.UI.className = "wmeac-csv-closures-list-done";
WMEAC.setCSVMiniLog(closure, "OK", 1);
handler && handler(true);
}
function applyFailure(evt) {
var details = "";
evt.forEach(function(err) {
if (err.hasOwnProperty("attributes") && err.attributes.hasOwnProperty("details")) {
details += err.attributes.details + "\n";
}
});
WMEAC.csvAddLog("Closure KO: " + closure.closure.comment + " (" + closure.closure.reason + ")\n" + details + "\n");
WMEAC.setCSVMiniLog(closure, "KO: " + details, 3);
closure.UI.className = "wmeac-csv-closures-list-failed";
handler && handler(false);
}
var tmp3 = function applyClosure() {
WMEAC.log("Now apply closure...");
if (closure.action == "add") {
closure.closure.applyInWME(applySuccess, applyFailure);
} else {
if (closure.action == "remove") {
closure.closure.removeInWME(applySuccess, applyFailure);
}
}
};
var tmp2 = function readyToApply() {
WMEAC.log("Test if ready to apply...");
if (WMEAC.pendingOps == true) {
WMEAC.log("Not yet. Waiting for WME...");
window.setTimeout(readyToApply, 500);
} else {
tmp3();
}
};
var tmp1 = function mapMovedEnd() {
WMEAC.log("Test if roads are reloaded...");
if (WMEAC.pendingOps == true) {
WMEAC.log("Not yet. Waiting for WME...");
window.setTimeout(mapMovedEnd, 500);
} else {
WMEAC.reloadRoadLayer();
tmp2();
}
};
window.setTimeout(tmp1, 1500);
};
WMEAC.csvAddLog = function(text) {
var divLog = WMEAC.getId("wmeac-csv-closures-log");
divLog.innerHTML += text.replace(/\n/g, "<br>");
};
WMEAC.csvClearLog = function() {
var divLog = WMEAC.getId("wmeac-csv-closures-log");
divLog.innerHTML = "";
};
WMEAC.csvShowList = function(show) {
var divList = WMEAC.getId("wmeac-csv-closures");
divList.style.display = show ? "block" : "none";
};
WMEAC.csvCheckAllSegments = function(i) {
if (i == -1) {
WMEAC.pb.update(0);
WMEAC.pb.show(true);
window.setTimeout(function() {
WMEAC.csvCheckAllSegments(0);
});
return;
}
var continueSegmentCheck = function() {
window.setTimeout(function() {
WMEAC.csvCheckAllSegments(i + 1);
});
};
if (i < WMEAC.csvCurrentBatchClosureList.length) {
var currentClosure = WMEAC.csvCurrentBatchClosureList[i];
WMEAC.pb.update(i * 100 / WMEAC.csvCurrentBatchClosureList.length);
WMEAC.pb.info("Scanning segments. please wait...");
var c = OpenLayers.Layer.SphericalMercator.forwardMercator(currentClosure.closure.lonlat.lon, currentClosure.closure.lonlat.lat);
var b = Waze.map.calculateBounds();
var zoomRatio = Math.pow(2, Waze.map.zoom - currentClosure.closure.zoom);
var w = b.getWidth() * Waze.controller.ratio * zoomRatio;
var h = b.getHeight() * Waze.controller.ratio * zoomRatio;
var tileBounds = new OpenLayers.Bounds(c.lon - w / 2, c.lat - h / 2, c.lon + w / 2, c.lat + h / 2);
tileBounds = tileBounds.transform(Waze.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")).toBBOX();
var roadTypes = Waze.model.repos.segments.zoomToRoadType[currentClosure.closure.zoom] == -1 ? _.range(1, 22) : Waze.model.repos.segments.zoomToRoadType[currentClosure.closure.zoom];
var WFVS = require("Waze/Feature/Vector/Segment");
var aseg = new WFVS;
var req = new XMLHttpRequest;
req.open("GET", document.location.protocol + "//" + document.location.host + Waze.Config.api_base + "/Features?roadTypes=" + roadTypes.join("%2C") + "&problemFilter=0&mapUpdateRequestFilter=0&roadClosures=true&userAreas=false&managedAreas=false&majorTrafficEvents=false&bbox=" + encodeURIComponent(tileBounds) + "&language=en", true);
req.onreadystatechange = function(e) {
if (req.readyState == 4) {
if (req.status == 200) {
try {
var data = JSON.parse(req.responseText);
WMEAC.log("data", data);
var existingSegs = currentClosure.closure.segIDs.filter(function(sid) {
return data.segments.objects.find(function(seg) {
return sid == seg.id;
}) != null;
});
var editableClosuresSegs = currentClosure.closure.segIDs.filter(function(sid) {
return data.segments.objects.find(function(seg) {
return sid == seg.id && seg.permissions & aseg.PERMISSIONS.EDIT_CLOSURES;
}) != null;
});
var overlaps = [];
var existingClosures = existingSegs.forEach(function(sid) {
var cl = data.roadClosures.objects.filter(function(c) {
return c.segID == sid;
});
console.log("cl", cl);
cl.forEach(function(c) {
var forwardMustBe = currentClosure.closure.direction == "A to B" ? true : currentClosure.closure.direction == "B to A" ? false : null;
console.log("forwardMustBe", forwardMustBe);
console.log("dateTimeOverlaps", currentClosure.closure);
console.log("dateTimeOverlaps", c);
if (WMEAC.dateTimeOverlaps(currentClosure.closure, c)) {
if (forwardMustBe == null || forwardMustBe == c.forward) {
var segment = data.segments.objects.find(function(seg) {
return seg.id == sid;
});
var streetName = null;
if (segment && segment.primaryStreetID != null) {
var street = data.streets.objects.find(function(st) {
return st.id == segment.primaryStreetID;
});
if (street && street.name != null) {
streetName = street.name;
}
}
overlaps.push("Overlap with " + c.reason + (streetName != null ? " :" + streetName : "") + " (" + sid + ")");
}
}
});
});
if (existingSegs.length == currentClosure.closure.segIDs.length && editableClosuresSegs.length == currentClosure.closure.segIDs.length && overlaps.length == 0) {
WMEAC.csvAddLog("Seg check OK: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\n" + existingSegs.length + " editable seg(s) found\n");
WMEAC.setCSVMiniLog(currentClosure, "segs OK: " + existingSegs.length + " editable seg(s) found", 1);
} else {
if (existingSegs.length == currentClosure.closure.segIDs.length && editableClosuresSegs.length == currentClosure.closure.segIDs.length && overlaps.length != 0) {
WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\nOverlap detected on existing closures:\n" + overlaps.join("\n") + "\n");
WMEAC.setCSVMiniLog(currentClosure, "segs KO: " + overlaps.length + " overlap(s) detected", 2);
} else {
if (existingSegs.length == currentClosure.closure.segIDs.length && editableClosuresSegs.length != currentClosure.closure.segIDs.length) {
WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\n" + existingSegs.length + "/" + currentClosure.closure.segIDs.length + " seg(s) found but " + (currentClosure.closure.segIDs.length - editableClosuresSegs.length) + " are not editable\n");
WMEAC.setCSVMiniLog(currentClosure, "segs KO: " + existingSegs.length + "/" + currentClosure.closure.segIDs.length + " seg(s) found and " + (currentClosure.closure.segIDs.length - editableClosuresSegs.length) + " are not editable", 2);
} else {
WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\n" + existingSegs.length + "/" + currentClosure.closure.segIDs.length + " seg(s) found\n");
WMEAC.setCSVMiniLog(currentClosure, "segs KO: " + existingSegs.length + "/" + currentClosure.closure.segIDs.length + " seg(s) found", 3);
}
}
}
} catch (err) {
WMEAC.log("Failed to parse Waze's server response: " + req.responseText);
WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\nFailed to parse response from Waze\n");
WMEAC.setCSVMiniLog(currentClosure, "segs KO: Failed to parse response from Waze", 3);
}
} else {
WMEAC.log("Error on road tile: " + e.target.status);
WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\nCommunication failed with Waze\n");
WMEAC.setCSVMiniLog(currentClosure, "segs KO: Communication failed with Waze", 3);
}
continueSegmentCheck();
}
};
req.onError = function(e) {
WMEAC.log("Error on road tile: " + e.target.status);
WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\nCommunication failed with Waze's server\n");
WMEAC.setCSVMiniLog(currentClosure, "segs KO: Communication failed with Waze", 3);
continueSegmentCheck();
};
req.send(null);
} else {
WMEAC.pb.show(false);
}
};
WMEAC.setCSVMiniLog = function(closure, text, level) {
var c = null;
if (closure.hasOwnProperty("UI")) {
c = closure;
} else {
c = WMEAC.csvCurrentClosureList.find(function(e) {
return e.closure.id == closure.id;
});
}
if (c != null) {
c.UI.children[7].innerHTML = text;
var colors = ["#000000", "#54C600", "#FFA000", "#FF0000"];
if (arguments.length == 3) {
c.UI.children[7].style.color = colors[level];
} else {
c.UI.children[7].style.color = colors[0];
}
}
};
WMEAC.CSVCheckAll = function(check) {
WMEAC.csvCurrentClosureList.forEach(function(e) {
e.UI.children[0].children[0].checked = check;
});
};
WMEAC.CSVApplyChecked = function() {
WMEAC.csvCurrentBatchClosureList = WMEAC.csvCurrentClosureList.filter(function(e) {
return e.UI.children[0].children[0].checked;
});
WMEAC.csvClearLog();
if (WMEAC.csvCurrentBatchClosureList.isEmpty()) {
WMEAC.csvAddLog("No closure checked!\n");
} else {
WMEAC.showClosuresLayer(true);
WMEAC.pb.update(0);
WMEAC.pb.info("Applying closures. please wait...");
WMEAC.pb.show(true);
WMEAC.csvAddLog("Start to apply selected closures\n");
window.setTimeout(function() {
WMEAC.CSVBatchApply(0);
});
}
};
WMEAC.CSVBatchApply = function(i) {
WMEAC.pb.update(i * 100 / WMEAC.csvCurrentBatchClosureList.length);
if (i < WMEAC.csvCurrentBatchClosureList.length) {
if (WMEAC.csvCurrentBatchClosureList[i].action != "add" && WMEAC.csvCurrentBatchClosureList[i].action != "remove") {
WMEAC.csvAddLog("Closure KO: " + WMEAC.csvCurrentBatchClosureList[i].closure.comment + " (" + WMEAC.csvCurrentBatchClosureList[i].closure.reason + "): action " + WMEAC.csvCurrentBatchClosureList[i].action + " not supported yet\n");
WMEAC.setCSVMiniLog(WMEAC.csvCurrentBatchClosureList[i], "KO: action " + WMEAC.csvCurrentBatchClosureList[i].action + " not supported yet", 2);
WMEAC.CSVBatchApply(i + 1);
} else {
WMEAC.csvApplyClosure(WMEAC.csvCurrentBatchClosureList[i], function(success) {
if (success) {
WMEAC.csvAddLog("Closure OK: " + WMEAC.csvCurrentBatchClosureList[i].closure.comment + " (" + WMEAC.csvCurrentBatchClosureList[i].closure.reason + ")\n");
} else {
WMEAC.csvAddLog("Closure KO: " + WMEAC.csvCurrentBatchClosureList[i].closure.comment + " (" + WMEAC.csvCurrentBatchClosureList[i].closure.reason + ")\n");
}
WMEAC.CSVBatchApply(i + 1);
});
}
} else {
WMEAC.csvAddLog("Apply selected closures ended\n");
WMEAC.reloadClosuresLayer();
WMEAC.pb.show(false);
}
};
WMEAC.CSVCheckSegsChecked = function() {
WMEAC.csvClearLog();
WMEAC.csvCurrentBatchClosureList = WMEAC.csvCurrentClosureList.filter(function(e) {
return e.UI.children[0].children[0].checked;
});
if (WMEAC.csvCurrentBatchClosureList.isEmpty()) {
WMEAC.csvAddLog("No closure checked!\n");
} else {
WMEAC.csvCheckAllSegments(-1);
}
};
WMEAC.refreshHighlight = function() {
try {
var l = Waze.map.getLayersBy("uniqueName", "closures");
if (l.length == 1) {
l = l[0];
}
for (var m in l.markers) {
if (!l.markers.hasOwnProperty(m)) {
continue;
}
var marker = l.markers[m];
if (marker.model.reason && marker.model.reason.length >= 1 && marker.model.reason.charCodeAt(marker.model.reason.length - 1) == 160) {
marker.icon.$div.addClass("wmeac-hl");
}
}
} catch (e) {
WMEAC.log("Highlight error: ", e);
}
};
WMEAC.log("Ready");
WMEAC.bootstrapAC();
}
var WMEAC_Injected_script = document.createElement("script");
WMEAC_Injected_script.textContent = "" + WMEAC_Injected.toString() + " \n" + "WMEAC_Injected();";
WMEAC_Injected_script.setAttribute("type", "application/javascript");
document.body.appendChild(WMEAC_Injected_script);