// ==UserScript==
// @name WME Advanced Closures
// @version 2024.10.09.01
// @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/editor*
// @include https://beta.waze.com/*/editor*
// @exclude https://www.waze.com/user/*
// @exclude https://www.waze.com/*/user/*
// @exclude https://www.waze.com/discuss/*
// @icon 
// @grant unsafeWindow
// @grant GM_xmlhttpRequest
// @grant GM_addElement
// @connect holidayapi.com
// @copyright 2018, dummyd2, seb-d59, WazeDev
// @author dummyd2, seb-d59, WazeDev
// ==/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.
*
********/
/* global $ */
/* global W */
/* global OpenLayers */
/* global require */
/* global _ */
/* global I18n */
/*jshint multistr: true */
// SKIP_FILE('include/downloadHelper.js');
(function()
{
// WMEAC object and members:
/***********************************************
*** IN INCLUDED FILE : ***
*** include/globalDeclarations.js ***
***********************************************/
// create a custom date class with a few addl functions (originally in Datejs library).
class JDate extends Date {
clone() { return new JDate(this); }
addMinutes(value) {
this.setMinutes(this.getMinutes() + value);
}
addDays(value) {
this.setDate(this.getDate() + value);
}
}
var WMEAC={};
WMEAC.isDebug=false;
WMEAC.ac_version="2024.10.09.01";
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=[];
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/globalDeclarations.js ***
***********************************************/
// WMEAC usefull function member
/***********************************************
*** IN INCLUDED FILE : ***
*** include/util.js ***
***********************************************/
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)
{
//log("Beta - " + 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) // http://stackoverflow.com/questions/1353684/detecting-an-invalid-date-date-instance-in-javascript
{
if ( Object.prototype.toString.call(d) === "[object Date]" ) {
// it is a date
if ( isNaN( d.getTime() ) ) { // d.valueOf() could also work
return false;
}
else {
return true;
}
}
else {
return false;
}
};
WMEAC.dateToClosureStr = function(d) {
var yyyy = d.getUTCFullYear().toString();
var MM = (d.getUTCMonth()+1).toString(); // getMonth() is zero-based
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]); // padding
};
// http://stackoverflow.com/questions/8493195/how-can-i-parse-a-csv-string-with-javascript
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 = []; // Initialize array to receive values.
line.replace(re_value, // "Walk" the string using replace with callback.
function(m0, m1, m2, m3) {
// Remove backslash from \' in single quoted values.
if (m1 !== undefined) a.push(m1.replace(/\\'/g, "'"));
// Remove backslash from \" in double quoted values.
else if (m2 !== undefined) a.push(m2.replace(/\\"/g, '"'));
else if (m3 !== undefined) a.push(m3);
return ''; // Return empty string.
});
// Handle special case of empty last value.
if (/,\s*$/.test(line)) a.push('');
b.push(a);
});
return b;
};
WMEAC.segmentsIDsToSegments = function (ids)
{
return ids.filter(function (e) {
return (W.model.segments.objects.hasOwnProperty(e));
}).map (function (e) {
return (W.model.segments.objects[e]);
});
};
WMEAC.reloadRoadLayer = function ()
{
var l=W.map.getLayerByName("roads");
// 2024-04-09 these seem to be unneeded and causes flashing and delays
// l.redraw({force:!0});
// l.removeBackBuffer();
// W.controller.reloadData();
};
WMEAC.reloadClosuresLayer = function (endHandler)
{
var l=W.map.getLayerByName("closures");
l.redraw({force:!0});
// W.controller.reloadData();
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 = W.map.getLayerByName("closures");
if (l) l.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.forEach(function (c) {
c.off("mousemove", elemmousemove);
});
});
function elemmousemove (e) {
e.preventDefault();
element.offset({
top: e.pageY - y,
left: e.pageX - x
});
}
options.container.forEach(function (c) {
c.on("mousemove", elemmousemove);
});
});
};
WMEAC.dateTimeOverlaps = function ( dt1, dt2 )
{
return (dt1.startDate < dt2.endDate && dt1.endDate > dt2.startDate );
};
WMEAC.solveOverlaps = function (closureToAdd, existingClosureList, mode)
{
// sort existing closures:
var ecs = existingClosureList.map(function (e) {
return { isNew: false, ref: e, startDate: e.startDate, endDate: e.endDate};
});
// append new
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: // keep existing. return empty
return [];
break;
case 1: // delete existing.
ecs.splice(indexOfExisting, 1);
changes=true;
break;
case 2: // fill: keep all existing and cut/split new
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 //if (range1.end>range2.end)
{
r1.startDate = r2.endDate;
}
break;
case 3: // force: cut/split/delete existing and keep new
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 //if (range1.end>range2.end)
{
r2.endDate = r1.startDate;
}
break;
}
}
}
}
return ecs;
};
// tests:
// WMEAC.solveOverlaps({startDate: '2016-01-10 00:00', endDate: '2016-01-20 00:00'},
// [ {startDate: '2016-01-05 00:00', endDate: '2016-01-15 00:00', reason: 'bla bla'}], 0);
// WMEAC.solveOverlaps({startDate: '2016-01-10 00:00', endDate: '2016-01-20 00:00'},
// [ {startDate: '2016-01-15 00:00', endDate: '2016-01-25 00:00', reason: 'bla bla'}], 0);
// WMEAC.solveOverlaps({startDate: '2016-01-10 00:00', endDate: '2016-01-20 00:00'},
// [ {startDate: '2016-01-05 00:00', endDate: '2016-01-25 00:00', reason: 'bla bla'}], 0);
// WMEAC.solveOverlaps({startDate: '2016-01-10 00:00', endDate: '2016-01-20 00:00'},
// [ {startDate: '2016-01-12 00:00', endDate: '2016-01-18 00:00', reason: 'bla bla'}], 0);
// WMEAC.solveOverlaps({startDate: '2016-01-10 00:00', endDate: '2016-01-20 00:00'},
// [ {startDate: '2016-01-10 00:00', endDate: '2016-01-20 00:00', reason: 'bla bla'}], 0);
// WMEAC.solveOverlaps({startDate: '2016-01-10 00:00', endDate: '2016-01-20 00:00'},
// [ {startDate: '2016-01-05 00:00', endDate: '2016-01-10 00:00', reason: 'bla bla'},
// {startDate: '2016-01-20 00:00', endDate: '2016-01-25 00:00', reason: 'bla bla'}], 0);
// WMEAC.solveOverlaps({startDate: '2016-01-10 00:00', endDate: '2016-01-20 00:00'},
// [ {startDate: '2016-01-05 00:00', endDate: '2016-01-15 00:00', reason: 'bla bla'},
// {startDate: '2016-01-16 00:00', endDate: '2016-01-25 00:00', reason: 'bla bla'}], 0);
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 (W.model.streets.objects.hasOwnProperty(stid))
{
var st = W.model.streets.objects[stid];
if (st.hasOwnProperty('cityID') && st.cityID!=null && typeof st.cityID != 'undefined')
{
var ctid = st.cityID;
if (W.model.cities.objects.hasOwnProperty(ctid))
return W.model.cities.objects[ctid].countryID;
}
}
}
return null;
}).filter(function (cid) {
return (cid!=null);
});
return (W.model.countries.getObjectArray(function (c) {
return cids.indexOf(c.id)!=-1;
}));
};
WMEAC.getOppositeClosure = function (closure)
{
return W.model.roadClosures.getObjectArray(function (c) {
return (closure.attributes.reason == c.attributes.reason &&
closure.attributes.startDate == c.attributes.startDate &&
closure.attributes.endDate == c.attributes.endDate &&
closure.attributes.segID == c.attributes.segID &&
closure.attributes.forward != c.attributes.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 &&
W.model.streets.objects.hasOwnProperty(s.attributes.primaryStreetID))
{
var st = W.model.streets.objects[s.attributes.primaryStreetID];
if (st.hasOwnProperty('cityID') && st.cityID!=null && typeof st.cityID != 'undefined')
{
var ctid = st.cityID;
if (W.model.cities.objects.hasOwnProperty(ctid))
{
if (!W.model.cities.objects[ctid].isEmpty)
city=W.model.cities.objects[ctid].name;
}
}
if (W.model.streets.objects[s.attributes.primaryStreetID].isEmpty)
add(city, 'noStreet');
else
add(city, W.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);
};
WMEAC.buildPermalink = function (data)
{
var getvars = [];
for (var m in data)
{
if (data.hasOwnProperty(m))
{
getvars.push('' + m + '=' + data[m]);
}
}
return document.location.protocol + '//' + document.location.host + document.location.pathname + '?' + getvars.join('&');
};
WMEAC.sharedClosureDirection = {
A_TO_B: 1,
B_TO_A: 2,
TWO_WAY: 3
};
WMEAC.zoomToRoadType = function(e) {
let allRoadTypes = [1,2,3,4,5,6,7,8,9,10,15,16,17,18,19,20,22];
if (e < 14) {
return [];
}
switch (e) {
case 14:
return [2, 3, 4, 6, 7, 14];
case 15:
return [2, 3, 4, 6, 7, 8, 9, 10, 14, 16,17, 18, 19, 20, 22];
case 16:
case 17:
case 18:
case 19:
case 20:
case 21:
case 22:
default:
return allRoadTypes;
}
}
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/util.js ***
***********************************************/
/***********************************************
*** IN INCLUDED FILE : ***
*** include/css.js ***
***********************************************/
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-csv-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-nav-tabs>li:not(.active)>a:hover { 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);
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/css.js ***
***********************************************/
// boostrap and init (wait for waze UI and objects)
/***********************************************
*** IN INCLUDED FILE : ***
*** include/init.js ***
***********************************************/
WMEAC.bootstrapAC = function ()
{
if (W?.userscripts?.state?.isReady) {
WMEAC.initialize();
} else {
document.addEventListener("wme-ready", WMEAC.initialize, {
once: true,
});
}
};
WMEAC.initialize = function ()
{
WMEAC.log ("init");
WMEAC.load();
WMEAC.log("presets", WMEAC.presets);
WMEAC.initUI();
WMEAC.log ("init done");
};
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/init.js ***
***********************************************/
// function to setup the UI dom
/***********************************************
*** IN INCLUDED FILE : ***
*** include/setupUI.js ***
***********************************************/
WMEAC.HTMLTemplates={};
WMEAC.initUI = async function ()
{
WMEAC.initUIElements();
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 = "0px";
section.style.textIndent = "8px";
var title='<b><a target="_blank" href="https://greasyfork.org/scripts/370072-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/discuss/t/script-wme-advanced-closures/156277">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: 'wz-button',
id: 'wmeac-add-advanced-closure-button',
className: 'wmeac-button'});
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 = '<wz-file-input upload-button-label="Parse CSV" id="wmeac-file-input" enable-drag-and-drop="1" max-files-batch-size=10 ></wz-file-input>';
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);
const { tabLabel, tabPane } = W.userscripts.registerSidebarTab('advancedclosure');
await W.userscripts.waitForElementConnected(tabPane);
$(tabLabel.parentElement).append(
$('<span>', { class:'fa fa-road slashed', title: 'Advanced Closures' })
);
tabPane.appendChild(addon);
var observer = new MutationObserver(function(mutations) {
mutations.forEach(function(mutation) {
function rescurse(node)
{
if (node.className=='closures-list')
{
var target = WMEAC.getElementsByClassName('add-closure-button', node);
if (target.length > 0)
WMEAC.installButtonInClosureTab(node);
}
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});
// test now if closure tab exists. It happens if WME is opened with a segment id in the url:
WMEAC.installButtonInClosureTab();
//W.selectionManager.addEventListener("selectionchanged", WMEAC.selectionChanged);
W.app.layout.model.on("operationPending", function(e) {
if (e.operation.id!="pending.road_data")
return;
WMEAC.pendingOps = true;
});
W.app.layout.model.on("operationDone", function(e) {
if (e.operation.id!="pending.road_data")
return;
WMEAC.pendingOps = false;
});
W.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) {
var clist = WMEAC.getElementsByClassName('closures-list');
if (clist.length >0) node = clist[0];
}
if (!node) return;
// test if we already there
if ($(node).find('#wmeac-closuretab-add-advanced-closure-button').length==0)
{
var addCL = WMEAC.getElementsByClassName('add-closure-button', node);
var addACBtn = WMEAC.createElement({type: 'wz-button',
id: 'wmeac-closuretab-add-advanced-closure-button',
className: 'wmeac-button'});
addACBtn.innerHTML='<i class="fa fa-clock-o"></i> Add advanced closure';
addACBtn.addEventListener('click', WMEAC.showAddAdvancedClosure);
if (addCL.length > 0) addCL[0].after(addACBtn);
}
};
WMEAC.showAddAdvancedClosure = function()
{
// init if needed and show modal dialog
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;
W.map.getOLMap().div.appendChild(ACDiv);
window.setTimeout(WMEAC.connectAdvancedClosureDialogHandlers);
ACDiv.style.display="none";
//W.selectionManager.addEventListener("selectionchanged", WMEAC.refreshClosureList);
}
if (ACDiv.style.display=="block") // already shown => reset position
{
$(ACDiv).css({left: '80px', top: '20px'});
}
else
{
ACDiv.style.display="block";
W.selectionManager.addEventListener("selectionchanged", WMEAC.refreshClosureList);
W.selectionManager.addEventListener("selectionchanged", WMEAC.refreshClosureListFromSelection);
WMEAC.refreshClosureListFromSelection();
}
//window.setTimeout(function () { $('#wmeac-add-advanced-closure-dialog').find('.input-group-addon').css({display:"table-cell"}); });
$(ACDiv).find('.input-group-addon').css({display:"table-cell"});
WMEAC.refreshMTEList();
WMEAC.showClosuresLayer(true);
};
WMEAC.initUIElements = function()
{
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>\
';
if(!I18n.translations[I18n.locale].date.abbr_day_names){
I18n.translations[I18n.locale].date.abbr_day_names = [];
_.forOwn(I18n.translations[I18n.locale].date, (v,k) => { if(k.indexOf("abbr_day_names_") > -1) { I18n.translations[I18n.locale].date.abbr_day_names.push(v)}});
}
var daysOfWeekUI = _(I18n.translations[I18n.locale].date.abbr_day_names).clone();
daysOfWeekUI.push(daysOfWeekUI.shift());
var tabEachUI = '<div class="box" style="display:flex; flex-wrap:wrap;">\
<div style="width:100%;">\
<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 style="width:14%;">\
<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('') + '</div>';
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 +// overlapModeUI +
'\
</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;
}
const m = W.selectionManager.getSelectedDataModelObjects();
if (m.length==0 || m[0].type!="segment")
{
alert("Please, select segment(s) before.");
return;
}
var reason = $('#wmeac-advanced-closure-dialog-reason').val();
//var cllocation = $('#wmeac-advanced-closure-dialog-location').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 = W.model.majorTrafficEvents.getObjectById(mteId);
closureList = rc.list.map(function (e) {
//return {reason: reason, direction: direction, startDate: e.start, endDate: e.end, location: cllocation, permanent: isIT};
var details = {reason: reason, direction: direction, startDate: e.start, endDate: e.end, location: "", permanent: isIT};
if (mte)
details.eventId = mte.attributes.id;
return details;
});
// save selection list
var selection = W.selectionManager.getSelectedDataModelObjects();
var selectionReversed=[];
if (direction!='3') // not two way
{
var rev = W.selectionManager.getReversedSegments();
selection=selection.filter(function (e) {
if (rev[e.attributes.id])
{
selectionReversed.push(e);
return false;
}
return true;
});
}
const ll = W.map.getCenter();
const lonlat = new OpenLayers.Geometry.Point(ll.lon, ll.lat).transform(W.map.getProjectionObject(), "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 (14 to 22),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.x + '&lat=' + lonlat.y + '",' + W.map.zoom + ',' + mteId + ',"Generated by WMEAC"\n';
});
if (!selectionReversed.length==0)
{
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 + '",' + W.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)
{
W.selectionManager.removeEventListener("selectionchanged", WMEAC.refreshClosureList);
W.selectionManager.removeEventListener("selectionchanged", 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;
}
const m = W.selectionManager.getSelectedDataModelObjects();
if (m.length==0 || m[0].type != "segment")
{
alert("Please, select segment(s) before.");
return;
}
if (m.every(function (e) {
const segid = e.attributes.id;
const seg = W.model.segments.objects[segid];
return seg.isAllowed(seg.permissionFlags.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 cllocation = $('#wmeac-advanced-closure-dialog-location').val();
var direction = $('#wmeac-advanced-closure-dialog-direction').val();
var sc = require("Waze/Modules/Closures/Models/SharedClosure");
direction=(direction=="1"?WMEAC.sharedClosureDirection.A_TO_B:(direction=="2"?WMEAC.sharedClosureDirection.B_TO_A:WMEAC.sharedClosureDirection.TWO_WAY));
var directionStr = direction==1?"(A → B)":(direction==2?"(B → A)":"(⇆)");
var isIT = $('#wmeac-advanced-closure-dialog-ignoretraffic').is(':checked');
var mte = W.model.majorTrafficEvents.getObjectById($("#wmeac-advanced-closure-dialog-mteid").val());
closureList = rc.list.map(function (e) {
//return {reason: reason, direction: direction, startDate: e.start, endDate: e.end, location: cllocation, permanent: isIT};
var details = {reason: reason, direction: direction, startDate: e.start, endDate: e.end, location: "", permanent: isIT};
if (mte)
details.eventId = mte.attributes.id;
return details;
});
// save selection list
var selection = W.selectionManager.getSelectedDataModelObjects();
W.selectionManager.removeEventListener("selectionchanged", WMEAC.refreshClosureList);
WMEAC.addClosureListFromSelection(closureList, function (i, e) {
$('#wmeac-advanced-closure-dialog-preview-' + i).html(e).css({color: "#44D544"}); // green
}, function (i, e) {
$('#wmeac-advanced-closure-dialog-preview-' + i).html(e).css({color: "#D5444F"}); // red
}, function () {
W.selectionManager.setSelectedModels(selection);
//alert ('done');
var tmp = function selectionReady()
{
if (W.selectionManager.getSelectedFeatures().length==0)
window.setTimeout(selectionReady, 500);
else
{
W.selectionManager.addEventListener("selectionchanged", 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') // WME beta
$("#wmeac-advanced-closure-dialog-rangestartdate,#wmeac-advanced-closure-dialog-rangeenddate").daterangepicker({singleDatePicker: !0, autoApply: !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-duration-hour').spinner({
// min: 0,
// spin: function (event, ui) {
// $(this).trigger('change');
// }
// });
// $('#wmeac-advanced-closure-dialog-duration-minute').spinner({
// spin: function (event, ui) {
// if (ui.value >= 60) {
// $(this).spinner('value', ui.value - 60);
// $('#wmeac-advanced-closure-dialog-duration-hour').spinner('stepUp');
// return false;
// } else if (ui.value < 0) {
// $(this).spinner('value', ui.value + 60);
// $('#wmeac-advanced-closure-dialog-duration-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-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: _.map(WMEAC.getCountriesFromSegmentSet(_.map(W.selectionManager.getSelectedFeatures(), 'model')), 'abbr'),
handlerFinished: function (holidays)
{
WMEAC.lastGeneratedHolidays = holidays;
if (holidays.length==0)
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);
// $("#wmeac-advanced-closure-dialog-duration-hour").val(WMEAC.presets[presetIndex].values.duration.hour);
// $("#wmeac-advanced-closure-dialog-duration-minute").val(WMEAC.presets[presetIndex].values.duration.minute);
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-location").val(WMEAC.presets[presetIndex].values.location);
$("#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 = W.model.roadClosures.objects[closureId];
if (c)
{
$("#wmeac-advanced-closure-dialog-starttime").val(c.attributes.startDate.split(' ')[1]);
var duration=new Date(c.attributes.endDate) - new Date(c.attributes.startDate);
// $("#wmeac-advanced-closure-dialog-duration-hour").val(Math.floor(duration/3600000));
// $("#wmeac-advanced-closure-dialog-duration-minute").val(new Date(duration).getMinutes());
var days = Math.floor(duration/86400000);
$("#wmeac-advanced-closure-dialog-duration-day").val(days);
var hours = Math.floor((duration - days * 86400000)/3600000);
var minutes = Math.floor((duration - days * 86400000 - hours * 3600000)/60000);
$("#wmeac-advanced-closure-dialog-durationtime").val('' + hours + ':' + minutes);
$("#wmeac-advanced-closure-dialog-reason").val(c.attributes.reason.trim());
if (WMEAC.getOppositeClosure(c).length==0) // oneway
$("#wmeac-advanced-closure-dialog-direction").val(c.attributes.forward?1:2);
else
$("#wmeac-advanced-closure-dialog-direction").val(3);
$("#wmeac-advanced-closure-dialog-ignoretraffic").prop('checked', c.attributes.permanent);
// MTE
if (c.attributes.eventId!=null)
{
var options = [];
$("#wmeac-advanced-closure-dialog-mteid option").each(function () { options.push($(this).val()); });
if (options.indexOf(c.attributes.eventId)!=-1)
$("#wmeac-advanced-closure-dialog-mteid").val(c.attributes.eventId);
else
$("#wmeac-advanced-closure-dialog-mteid").val('');
}
}
}
});
$('#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) // overwrite existing preset
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.location=$("#wmeac-advanced-closure-dialog-location").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: [$('#OpenLayers_Map_200_OpenLayers_ViewPort'), $('#WazeMap')] });
WMEAC.refreshMTEList();
};
WMEAC.connectAdvancedClosureTabHandlers = function ()
{
$('#wmeac-file-input')[0].addEventListener('filesSelected', (e) => WMEAC.ReadFiles(e.detail) );
var e = null;
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(''));
};
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/setupUI.js ***
***********************************************/
/***********************************************
*** IN INCLUDED FILE : ***
*** include/class.progressBar.js ***
***********************************************/
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'});
//elt.innerHTML=" ";
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;
};
};
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/class.progressBar.js ***
***********************************************/
/***********************************************
*** IN INCLUDED FILE : ***
*** include/class.closure.js ***
***********************************************/
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;
}
// optional options:
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<14||this.zoom>22)
{
this.isValid=false;
this.errorMessage="Wrong zoom (14 to 22): " + this.zoom + "\n";
return;
}
this.applyInWME = function(successHandler, failureHandler)
{
// check if segments are on screen
var segs = WMEAC.segmentsIDsToSegments(this.segIDs);
WMEAC.log("Segs: ", segs);
segs = segs.filter(function (seg) {
return seg.isAllowed(seg.permissionFlags.EDIT_CLOSURES);
});
if (segs.length==0)
{
failureHandler( {errors: [{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"?WMEAC.sharedClosureDirection.A_TO_B:(this.direction=="B to A"?WMEAC.sharedClosureDirection.B_TO_A:WMEAC.sharedClosureDirection.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.permissionFlags.EDIT_CLOSURES);
});
var allClosuresToRemove=[];
var countToMatch=this.segIDs.length*(this.direction=="TWO WAY"?2:1); // two way = 2 closures in WME
segs.forEach(function (s) {
// look for closure(s)
var that = this;
var closures = W.model.roadClosures.getObjectArray(function (c) {
return (c.attributes.startDate==that.startDate &&
c.attributes.endDate==that.endDate &&
c.attributes.reason.trim()==that.reason &&
c.attributes.segID==s.attributes.id &&
c.attributes.permanent == (that.permanent=='Yes'));
});
if ((this.direction=="TWO WAY") || // && closures.length==2 && closures[0].forward!=closures[1].forward) ||
(this.direction=="A to B" && closures.length==1 && closures[0].attributes.forward==true) ||
(this.direction=="B to A" && closures.length==1 && closures[0].attributes.forward==false))
{
allClosuresToRemove=allClosuresToRemove.concat(closures);
}
}, this);
if (allClosuresToRemove.length==0)
{
failureHandler( {errors: [{attributes: {details: "No segment. Check permissions or existence."}}]} );
}
else
WMEAC.removeClosure(allClosuresToRemove, successHandler, failureHandler);
};
};
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/class.closure.js ***
***********************************************/
/***********************************************
*** IN INCLUDED FILE : ***
*** include/recurringClosures.js ***
***********************************************/
WMEAC.buildClosuresListFromRecurringUI = function ()
{
var list = [];
var rangeStartDate = new JDate($('#wmeac-advanced-closure-dialog-rangestartdate').val());
if (!WMEAC.isValidDate(rangeStartDate)) return {list: list, error: "Range start date is not valid"};
var rangeEndDate = new JDate($('#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 dM = parseInt($('#wmeac-advanced-closure-dialog-duration-minute').val());
// if (isNaN(dM) || dM<0 || dM>=60) return {list: list, error: "Duration minute 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 = $('#wmeac-advanced-closure-dialog-rangestarttime').val().split(':').map(function (e) {
// return parseInt(e);
// }).reduce(function (p, c, i) {
// return (p*60+c);
// });
var rangeStartTimeM = 0;
// var rangeEndTimeM = $('#wmeac-advanced-closure-dialog-rangeendtime').val().split(':').map(function (e) {
// return parseInt(e);
// }).reduce(function (p, c, i) {
// return (p*60+c);
// });
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 mode is REPEAT
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 repeat is smaller than duration
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) // starts the day after
firstDateTimeStart.addDays(1);
firstDateTimeStart.setMinutes(startTimeM);
var firstDateTimeEnd = firstDateTimeStart.clone();
firstDateTimeEnd.addMinutes(dD * 1440 + dH * 60 + dM);
// var now = new Date();
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) // stop if after range end
break;
// WMEAC.log('end', end);
// WMEAC.log('now', now);
// if (end < now) // do not add closure that ends before now
// {
// ntimes++;
// continue;
// }
list.push({start: WMEAC.dateToClosureStr(start), end: WMEAC.dateToClosureStr(end)});
}
return {list: list, error: ""};
}
// if mode is EACH
else if ($('#wmeac-advanced-closure-dialog-tabeach').attr('class').indexOf('active')!=-1)
{
// build bits for a week:
var dow = WMEAC.daysOfWeek.map(function (e, i) {
return ($('#wmeac-advanced-closure-dialog-each-' + i)).is(':checked');
});
var dayCount = Math.ceil((rangeEndDate-rangeStartDate+1)/86400000);
var day0 = rangeStartDate.clone();
day0.addMinutes(startTimeM);
if (startTimeM<rangeStartTimeM) // starts the day after
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) // stop if after range end
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 JDate(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 cllocation = $('#wmeac-advanced-closure-dialog-location').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 = W.selectionManager.getSelectedFeatures().reduce(function (p, c, i) {
var revSegs = W.selectionManager.getReversedSegments();
var isReversed = revSegs.hasOwnProperty(c.id) && revSegs[c.id];
var realWay = isReversed?(direction==1?2:1):direction;
return p.concat(W.model.roadClosures.getObjectArray(function (e) {
return (e.segID==c.id &&
(direction==3 || (e.forward && realWay==1) || (!e.forward && realWay==2)));
}));
}, []);
var mte = W.model.majorTrafficEvents.getObjectById($("#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 (W.model.segments.objects.hasOwnProperty(c.segID)==false) return msg;
if (W.model.segments.objects[c.segID].attributes.primaryStreetID==null) return msg;
if (W.model.streets.objects.hasOwnProperty(W.model.segments.objects[c.segID].attributes.primaryStreetID)==false) return msg;
var street = W.model.streets.objects[W.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 +
//' (' + cllocation + '): ' +
': ' +
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 JDate($("#wmeac-advanced-closure-dialog-rangestartdate").val());
var rangeEnd = new JDate($("#wmeac-advanced-closure-dialog-rangeenddate").val());
var mtelist = [];
$("#wmeac-advanced-closure-dialog-mteid").empty();
if (WMEAC.isValidDate(rangeStart) && WMEAC.isValidDate(rangeEnd))
{
rangeEnd.addDays(1);
// filter MTE loaded in WME:
W.model.majorTrafficEvents.getObjectArray(function (mte) {
// check if ranges overlap
return (WMEAC.dateTimeOverlaps({startDate: rangeStart, endDate: rangeEnd}, {startDate: new JDate(mte.attributes.startDate), endDate: new JDate(mte.attributes.endDate)}));
}).forEach(function (mte) {
mtelist.push({name: mte.attributes.names[0].value, value: mte.attributes.id});
});
}
mtelist.sort(function(a,b) {
return a.name.localeCompare(b.name);
});
WMEAC.addMTEitem('None', '', currentMTEid);
mtelist.forEach(function (o) {
WMEAC.addMTEitem(o.name, o.value, currentMTEid);
});
if (mtelist.length>0)
$("#wmeac-advanced-closure-dialog-mteid").removeAttr('disabled');
else
$("#wmeac-advanced-closure-dialog-mteid").attr('disabled', '');
};
WMEAC.addMTEitem = function (n, v, curId)
{
var el = WMEAC.createElement({type: 'option'});
el.setAttribute('value', v);
if (curId==v)
el.setAttribute('selected', '');
el.innerHTML = n;
$("#wmeac-advanced-closure-dialog-mteid").append(el);
};
WMEAC.refreshClosureListFromSelection = function ()
{
try
{
var currentSegClosure = $("#wmeac-advanced-closure-dialog-segclosure-list").val();
$("#wmeac-advanced-closure-dialog-segclosure-list").empty();
if (W.selectionManager.getSelectedFeatures().length!=0)
{
var blackList=[];
W.model.roadClosures.getObjectArray(function (c) {
return c.attributes.segID==W.selectionManager.getSelectedDataModelObjects()[0].attributes.id;
}).sort(function (a,b) {
return (new Date(a.attributes.startDate)-new Date(b.attributes.startDate));
}).forEach(function (c) {
if (blackList.indexOf(c.attributes.id)!=-1) return;
var direction = c.attributes.forward?"A to B":"B to A";
var oppositeClosure = WMEAC.getOppositeClosure(c);
if (!oppositeClosure.length==0)
{
direction = "Two way";
blackList.push(oppositeClosure[0].id);
}
var el = WMEAC.createElement({type: 'option'});
el.setAttribute('value', c.attributes.id);
if (currentSegClosure==c.attributes.id)
el.setAttribute('selected', '');
el.innerHTML = c.attributes.reason.trim() + ' ' + direction + ' ' + c.attributes.startDate + '→' + c.attributes.endDate;
$("#wmeac-advanced-closure-dialog-segclosure-list").append(el);
});
}
}
catch (e)
{
WMEAC.logError("Error while refreshing closure list from selection: ", e);
}
};
// SKIP_FILE('include/holidays.js');
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/recurringClosures.js ***
***********************************************/
/***********************************************
*** IN INCLUDED FILE : ***
*** include/actionClosures.js ***
***********************************************/
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("Adding 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, {dataModel: W.model, segmentSelection: W.selectionManager.getSegmentSelection(), isNew: true});
t.actions=[cab.add(c)];
W.controller.save(t).then(done()).catch(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.errors.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 = W.selectionManager.getSelectedDataModelObjects();
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: W.selectionManager.getReversedSegments()};
if (closureList[i].hasOwnProperty('eventId') && closureList[i].eventId!=null) closureDetails.eventId = closureList[i].eventId;
var c = new sc(closureDetails, {dataModel: W.model, segmentSelection: W.selectionManager.getSegmentSelection(), isNew: true});
t.actions=[cab.add(c)];
W.controller.save(t).then(done()).catch(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("Adding 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 = W.selectionManager.getSelectedDataModelObjects();
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: W.selectionManager.getReversedSegments()};
if (options.hasOwnProperty('eventId') && options.eventId!=null) closureDetails.eventId = options.eventId;
var c = new sc(closureDetails, {dataModel: W.model, segmentSelection: W.selectionManager.getSegmentSelection(), isNew: true});
t.actions=[cab.add(c)];
W.controller.save(t).then(done()).catch(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 = {};
let segs = WMEAC.segmentsIDsToSegments(closures.map(closure => closure.attributes.segID));
t.actions=[cab.delete(W.model, new sc({segments: segs, closures: closures, reverseSegments: W.selectionManager.getReversedSegments()}, {dataModel: W.model, segmentSelection: W.selectionManager.getSegmentSelection(), isNew: true}))];
W.controller.save(t).then(done()).catch(fail());
return true;
};
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/actionClosures.js ***
***********************************************/
// functions to load and save settings
/***********************************************
*** IN INCLUDED FILE : ***
*** include/saveLoad.js ***
***********************************************/
WMEAC.save = function ()
{
WMEAC.log("save data...");
localStorage.WMEAC = JSON.stringify({presets: WMEAC.presets});
};
WMEAC.load = function ()
{
try
{
if (localStorage.WMEAC!==undefined && localStorage.WMEAC.length > 0) {
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);
}
};
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/saveLoad.js ***
***********************************************/
/***********************************************
*** IN INCLUDED FILE : ***
*** include/csv.js ***
***********************************************/
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)
{
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};
});
closures.forEach(function (c) {
if (!c.closure.isValid) {
isValid.isValid = false;
isValid.feedBack += c.closure.errorMessage;
}
});
}
if (isValid.isValid) {
WMEAC.log("CSV is valid!");
WMEAC.log("Closure list:", closures);
WMEAC.csvCurrentClosureList = closures;
var listUI = WMEAC.getId('wmeac-csv-closures-list-elts');
// remove all closures before:
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;
// aply closures: TEST ONLY: this should not be done there!
/*closures.forEach(function (c) {
c.closure.applyInWME(function () { WMEAC.log("Closure success:", c);});
});*/
// END OF aply closures: TEST ONLY: this should not be done there!
}
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.ReadFiles = function (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();
// WMEAC.csvCheckAllSegments(-1);
}
};
})(f);
// Read in the image file as a data URL.
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 && line[0]!="header" && line[0]!="comment");
return (line.length>=1 && ['add','remove'].indexOf(line[0])!=-1);
});
};
};
WMEAC.csv.push(new WMEAC.ClassCSV({version: 1, regexpValidation: [/.*/, // 1st cell: action is free keyword. It will be filtered later
/.*/, // reason is free
// /.*/, // location is free
/^\d{4}-\d{2}-\d{2} \d{2}:\d{2}$/, // start date
/^\d{4}-\d{2}-\d{2} \d{2}:\d{2}$/, // end date
/(^A to B$)|(^B to A$)|(^TWO WAY$)/, // direction
/(Yes)|(No)/, // ignore trafic = permanent
/^(\d+(;|$))+/, // seg ID list
/(lon=(-?\d+\.?\d*)&lat=(-?\d+\.?\d*))|(lat=(-?\d+\.?\d*)&lon=(-?\d+\.?\d*))/, // lonlat
/^\d+$/, // zoom
/(^$)|(^-?\d+\.-?\d+\.-?[0-9a-fA-F\-]{4,36}$)/ // MTE ID is empty or digits.digits.[digits-or-guid]
]}));
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="' + WMEAC.buildPermalink({lon: closure.lonlat.lon, lat: closure.lonlat.lat, segments: closure.segIDs.join(','), zoom: closure.zoom}) + '" 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>';
// attach handlers
liElt.children[5].children[0].addEventListener('click', function (e) {
WMEAC.csvClearLog();
// get closure id:
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);
W.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)
{
W.selectionManager.setSelectedModels(segs);
var tmp = function selectionReady()
{
if (W.selectionManager.getSelectedFeatures().length==0)
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 || W.app.layout.model.attributes.loadingFeatures==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 || W.app.layout.model.attributes.loadingFeatures==true)
{
WMEAC.log("Not yet. Waiting for WME...");
window.setTimeout(mapMovedEnd, 500);
}
else
{
WMEAC.reloadRoadLayer();
tmp2();
}
};
window.setTimeout(tmp1, 500);
e.preventDefault();
});
liElt.children[6].children[0].addEventListener('click', function (e) {
// get closure id:
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);
W.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)
{
//WMEAC.log('evt', evt);
var details="";
evt.errors.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 || W.app.layout.model.attributes.loadingFeatures==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 || W.app.layout.model.attributes.loadingFeatures==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) // firt call: init progressbar
{
WMEAC.pb.update(0);
WMEAC.pb.show(true);
// and call the check on first closure
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...");
// check segments
// catch window tile
var c = OpenLayers.Layer.SphericalMercator.forwardMercator(currentClosure.closure.lonlat.lon, currentClosure.closure.lonlat.lat);
var b = W.map.calculateBounds();
var b1 = new OpenLayers.Bounds(b[0],b[1],b[2],b[3]);
b1 = b1.transform(new OpenLayers.Projection("EPSG:4326"), W.map.getProjectionObject());
var zoomRatio = Math.pow(2, W.map.zoom - currentClosure.closure.zoom);
var w = b1.getWidth()*1.7*zoomRatio;
var h = b1.getHeight()*1.7*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(W.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")).toBBOX();
var roadTypes = (WMEAC.zoomToRoadType(currentClosure.closure.zoom)==-1?_.range(1, 22):WMEAC.zoomToRoadType(currentClosure.closure.zoom));
// var WFVS = require("Waze/Feature/Vector/Segment");
// var aseg = new WFVS;
var req = new XMLHttpRequest();
const EDIT_CLOSURES = 1024;
req.open('GET', document.location.protocol + '//' + document.location.host + W.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)
{
//WMEAC.log(req.responseText);
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.permissionFlags.*/ EDIT_CLOSURES));
})!=null);
});
// look for closures on existing segs and build overlap list
var overlaps=[];
var removeMatches = 0;
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) {
if (currentClosure.action == 'remove') {
if (currentClosure.closure.startDate == c.startDate && currentClosure.closure.endDate == c.endDate) {
removeMatches++;
}
}
else {
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 && (currentClosure.action == 'add' || removeMatches >= existingSegs.length) )
{
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 (currentClosure.action == 'remove') {
WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + ") No matching closures\n");
WMEAC.setCSVMiniLog(currentClosure, "segs KO: no matches", 2);
}
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();
};
/* // Useless since waze server never send content length... :/
req.onprogress = function(e) {
WMEAC.pb.update((i+(e.position / e.totalSize))*100/WMEAC.csvCurrentClosureList.length);
};*/
req.send(null);
}
else // end of check
{
WMEAC.pb.show(false);
}
};
WMEAC.setCSVMiniLog = function(closure, text, level) // level=0: black 1: green, 2:orange, 3: red
{
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.length==0)
{
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.length==0)
{
WMEAC.csvAddLog("No closure checked!\n");
}
else
{
WMEAC.csvCheckAllSegments(-1);
}
};
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/csv.js ***
***********************************************/
/***********************************************
*** IN INCLUDED FILE : ***
*** include/highlight.js ***
***********************************************/
WMEAC.refreshHighlight = function ()
{
try
{
var l = W.map.getLayerByName("closures");
for (var m in l.markers)
{
if (!l.markers.hasOwnProperty(m)) continue;
var marker = l.markers[m];
// 2023-07-15 closure marker doesnt have model anymore, use chaining to avoid error
// TODO - find way to get closure details from marker
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);
}
};
/***********************************************
*** END OF INCLUDED FILE : ***
*** include/highlight.js ***
***********************************************/
//2023-08-20 remove obsolete require patch
WMEAC.WMEAPI={require: window.require};
// start normally
WMEAC.log("starting");
WMEAC.bootstrapAC();
})();