WME Advanced Closures

Recurrent and imported closures in the Waze Map Editor

  1. // ==UserScript==
  2. // @name WME Advanced Closures
  3. // @version 1.8.2
  4. // @description Recurrent and imported closures in the Waze Map Editor
  5. // @namespace WMEAC
  6. // @include https://www.waze.com/editor*
  7. // @include https://www.waze.com/*/editor*
  8. // @include https://beta.waze.com/*
  9. // @exclude https://www.waze.com/user/*
  10. // @exclude https://www.waze.com/*/user/*
  11. // @icon data:image/png;base64,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
  12. // @grant unsafeWindow
  13. // @grant GM_xmlhttpRequest
  14. // @connect holidayapi.com
  15. // @copyright 2018, dummyd2, seb-d59
  16. // @author dummyd2, seb-d59
  17. // ==/UserScript==
  18.  
  19.  
  20. /*******
  21. *
  22. * You are free to:
  23. * Share, copy, and redistribute the script in any medium or format
  24. * under the following terms:
  25. * Attribution - You must give appropriate credit. You may do so in any
  26. * reasonable manner, but not in any way that suggests the licensor
  27. * endorses you or your use.
  28. *
  29. * NonCommercial - You may not use the script for commercial purposes.
  30. *
  31. * NoModifications - You may NOT MODIFY the script.
  32. *
  33. * You are invited to contact authors on waze forum for more details.
  34. *
  35. ********/
  36.  
  37. /*jshint multistr: true */
  38.  
  39. function WMEAC_downloadHelperInjected() {
  40. window.WMEAC_downloadHelper = {jobs:[], _waitForData:function(id) {
  41. if (this.jobs.length <= id) {
  42. this.jobs[id].callback({url:null, data:null, callback:this.jobs[id].callback, status:"error", error:"Request not found"});
  43. } else {
  44. if (this.jobs[id].status == "success" || this.jobs[id].status == "error") {
  45. this.jobs[id].callback(this.jobs[id]);
  46. } else {
  47. if (this.jobs[id].status == "downloading" && this.jobs[id].progressCallback) {
  48. this.jobs[id].progressCallback(this.jobs[id]);
  49. }
  50. var _this = this;
  51. window.setTimeout(function() {
  52. _this._waitForData(id);
  53. }, 500);
  54. }
  55. }
  56. }, add:function(url, callback, progressCallback) {
  57. this.jobs.push({url:url, data:null, callback:callback, progressCallback:progressCallback, status:"added", progression:0, error:""});
  58. var _this = this;
  59. window.setTimeout(function() {
  60. _this._waitForData(_this.jobs.length - 1);
  61. }, 500);
  62. }};
  63. }
  64. var WMEAC_downloadHelperInjectedScript = document.createElement("script");
  65. WMEAC_downloadHelperInjectedScript.textContent = "" + WMEAC_downloadHelperInjected.toString() + " \n" + "WMEAC_downloadHelperInjected();";
  66. WMEAC_downloadHelperInjectedScript.setAttribute("type", "application/javascript");
  67. document.body.appendChild(WMEAC_downloadHelperInjectedScript);
  68. function lookFordownloadHelperJob() {
  69. for (var i = 0; i < unsafeWindow.WMEAC_downloadHelper.jobs.length; i++) {
  70. if (unsafeWindow.WMEAC_downloadHelper.jobs[i].status == "added") {
  71. unsafeWindow.WMEAC_downloadHelper.jobs[i].status = cloneInto("downloading", unsafeWindow.WMEAC_downloadHelper.jobs[i]);
  72. var f = function() {
  73. var job = i;
  74. GM_xmlhttpRequest({method:"GET", headers:{"User-Agent":"Mozilla/5.0", "Accept":"text/plain"}, synchronous:false, timeout:10000, url:unsafeWindow.WMEAC_downloadHelper.jobs[job].url, onerror:function(r) {
  75. unsafeWindow.WMEAC_downloadHelper.jobs[job].status = cloneInto("error", unsafeWindow.WMEAC_downloadHelper.jobs[job]);
  76. }, ontimeout:function(r) {
  77. console.debug("TOTO Timeout while getting area from server: ", r);
  78. unsafeWindow.WMEAC_downloadHelper.jobs[job].status = cloneInto("error", unsafeWindow.WMEAC_downloadHelper.jobs[job]);
  79. }, onload:function(r) {
  80. unsafeWindow.WMEAC_downloadHelper.jobs[job].status = cloneInto("success", unsafeWindow.WMEAC_downloadHelper.jobs[job]);
  81. unsafeWindow.WMEAC_downloadHelper.jobs[job].data = cloneInto(r.responseText, unsafeWindow.WMEAC_downloadHelper.jobs[job]);
  82. }, onprogress:function(r) {
  83. unsafeWindow.WMEAC_downloadHelper.jobs[job].progression = cloneInto(r.total == 0 ? 0 : r.loaded / r.total, unsafeWindow.WMEAC_downloadHelper.jobs[job]);
  84. }});
  85. }();
  86. }
  87. }
  88. window.setTimeout(lookFordownloadHelperJob, 2000);
  89. }
  90. window.setTimeout(lookFordownloadHelperJob);
  91. function WMEAC_Injected() {
  92. var WMEAC = {};
  93. WMEAC.isDebug = false;
  94. WMEAC.ac_version = "1.8.2";
  95. WMEAC.closureTabTimeout = null;
  96. WMEAC.csv = [];
  97. WMEAC.csvCurrentClosureList = null;
  98. WMEAC.csvCurrentBatchClosureList = null;
  99. WMEAC.pendingOps = false;
  100. WMEAC.pb = null;
  101. WMEAC.daysOfWeek = ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"];
  102. WMEAC.lastGeneratedHolidays = [];
  103. WMEAC.presets = [];
  104. WMEAC.getElementsByClassName = function(classname, node) {
  105. if (!node) {
  106. node = document.getElementsByTagName("body")[0];
  107. }
  108. var a = [];
  109. var re = new RegExp("\\b" + classname + "\\b");
  110. var els = node.getElementsByTagName("*");
  111. for (var i = 0, j = els.length; i < j; i++) {
  112. if (re.test(els[i].className)) {
  113. a.push(els[i]);
  114. }
  115. }
  116. return a;
  117. };
  118. WMEAC.removeChildElements = function(node) {
  119. while (node.firstChild) {
  120. WMEAC.removeChildElements(node.firstChild);
  121. node.removeChild(node.firstChild);
  122. }
  123. };
  124. WMEAC.createElement = function(options) {
  125. if (options.hasOwnProperty("type") == false) {
  126. return null;
  127. }
  128. var el = document.createElement(options.type);
  129. if (options.hasOwnProperty("id") == true) {
  130. el.id = options.id;
  131. }
  132. if (options.hasOwnProperty("className") == true) {
  133. el.className = options.className;
  134. }
  135. return el;
  136. };
  137. WMEAC.getId = function(node) {
  138. var el = document.getElementById(node);
  139. return el;
  140. };
  141. WMEAC.logBeta = function(msg, obj) {
  142. };
  143. WMEAC.logDebug = function(msg, obj) {
  144. if (WMEAC.isDebug) {
  145. WMEAC.log("DEBUG - " + msg, obj);
  146. }
  147. };
  148. WMEAC.logError = function(msg, obj) {
  149. console.error("Advanced closures v" + WMEAC.ac_version + " - " + msg, obj);
  150. };
  151. WMEAC.log = function(msg, obj) {
  152. if (obj == null) {
  153. console.log("Advanced closures v" + WMEAC.ac_version + " - " + msg);
  154. } else {
  155. console.debug("Advanced closures v" + WMEAC.ac_version + " - " + msg + " ", obj);
  156. }
  157. };
  158. WMEAC.isValidDate = function(d) {
  159. if (Object.prototype.toString.call(d) === "[object Date]") {
  160. if (isNaN(d.getTime())) {
  161. return false;
  162. } else {
  163. return true;
  164. }
  165. } else {
  166. return false;
  167. }
  168. };
  169. WMEAC.dateToClosureStr = function(d) {
  170. var yyyy = d.getUTCFullYear().toString();
  171. var MM = (d.getUTCMonth() + 1).toString();
  172. var dd = d.getUTCDate().toString();
  173. var hh = d.getUTCHours().toString();
  174. var mm = d.getUTCMinutes().toString();
  175. 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]);
  176. };
  177. WMEAC.CSVtoArray = function(text) {
  178. var b = [];
  179. var re_valid = /^\s*(?:'[^'\\]*(?:\\[\S\s][^'\\]*)*'|"[^"\\]*(?:\\[\S\s][^"\\]*)*"|[^,'"\s\\]*(?:\s+[^,'"\s\\]+)*)\s*(?:,\s*(?:'[^'\\]*(?:\\[\S\s][^'\\]*)*'|"[^"\\]*(?:\\[\S\s][^"\\]*)*"|[^,'"\s\\]*(?:\s+[^,'"\s\\]+)*)\s*)*$/;
  180. var re_value = /(?!\s*$)\s*(?:'([^'\\]*(?:\\[\S\s][^'\\]*)*)'|"([^"\\]*(?:\\[\S\s][^"\\]*)*)"|([^,'"\s\\]*(?:\s+[^,'"\s\\]+)*))\s*(?:,|$)/g;
  181. var lines = text.split("\n");
  182. lines.forEach(function(line) {
  183. if (!re_valid.test(line)) {
  184. return;
  185. }
  186. var a = [];
  187. line.replace(re_value, function(m0, m1, m2, m3) {
  188. if (m1 !== undefined) {
  189. a.push(m1.replace(/\\'/g, "'"));
  190. } else {
  191. if (m2 !== undefined) {
  192. a.push(m2.replace(/\\"/g, '"'));
  193. } else {
  194. if (m3 !== undefined) {
  195. a.push(m3);
  196. }
  197. }
  198. }
  199. return "";
  200. });
  201. if (/,\s*$/.test(line)) {
  202. a.push("");
  203. }
  204. b.push(a);
  205. });
  206. return b;
  207. };
  208. WMEAC.segmentsIDsToSegments = function(ids) {
  209. return ids.filter(function(e) {
  210. return W.model.segments.objects.hasOwnProperty(e);
  211. }).map(function(e) {
  212. return W.model.segments.objects[e];
  213. });
  214. };
  215. WMEAC.reloadRoadLayer = function() {
  216. var l = W.map.getLayersBy("uniqueName", "roads")[0];
  217. l.redraw({force:!0});
  218. l.removeBackBuffer();
  219. W.controller.reload();
  220. };
  221. WMEAC.reloadClosuresLayer = function(endHandler) {
  222. var l = W.map.getLayersBy("uniqueName", "closures")[0];
  223. l.redraw({force:!0});
  224. W.controller.reload();
  225. if (endHandler) {
  226. var tmp = function reloaded() {
  227. WMEAC.log("Test if reloaded...");
  228. if (WMEAC.pendingOps == true) {
  229. WMEAC.log("Not yet. Waiting for WME...");
  230. window.setTimeout(reloaded, 500);
  231. } else {
  232. endHandler();
  233. }
  234. };
  235. window.setTimeout(tmp, 500);
  236. }
  237. };
  238. WMEAC.showClosuresLayer = function(show) {
  239. var l = W.map.getLayersBy("uniqueName", "closures");
  240. if (l.length == 1) {
  241. l[0].setVisibility(show);
  242. }
  243. };
  244. WMEAC.setDraggable = function(element, options) {
  245. if (!options.hasOwnProperty("controller")) {
  246. options.controller = element;
  247. }
  248. if (!options.hasOwnProperty("container")) {
  249. options.container = [$("body")];
  250. }
  251. options.controller.css({cursor:"move"});
  252. options.controller.on("mousedown", function(e) {
  253. var x = e.pageX - element.offset().left;
  254. var y = e.pageY - element.offset().top;
  255. $("body").on("mouseup", function(e) {
  256. options.container.forEach(function(c) {
  257. c.off("mousemove", elemmousemove);
  258. });
  259. });
  260. function elemmousemove(e) {
  261. e.preventDefault();
  262. element.offset({top:e.pageY - y, left:e.pageX - x});
  263. }
  264. options.container.forEach(function(c) {
  265. c.on("mousemove", elemmousemove);
  266. });
  267. });
  268. };
  269. WMEAC.dateTimeOverlaps = function(dt1, dt2) {
  270. return dt1.startDate < dt2.endDate && dt1.endDate > dt2.startDate;
  271. };
  272. WMEAC.solveOverlaps = function(closureToAdd, existingClosureList, mode) {
  273. var ecs = existingClosureList.map(function(e) {
  274. return {isNew:false, ref:e, startDate:e.startDate, endDate:e.endDate};
  275. });
  276. closureToAdd.isNew = true;
  277. ecs.push(closureToAdd);
  278. var changes = true;
  279. while (changes) {
  280. changes = false;
  281. ecs.sort(function(a, b) {
  282. return new Date(a.startDate) - new Date(b.startDate);
  283. });
  284. for (var i = 1; i < ecs.length; i++) {
  285. if (WMEAC.dateTimeOverlaps(ecs[i - 1], ecs[i])) {
  286. var indexOfNew = i - 1;
  287. var indexOfExisting = i;
  288. if (ecs[i].isNew) {
  289. indexOfNew = i;
  290. indexOfExisting = i - 1;
  291. }
  292. var r1 = ecs[indexOfNew];
  293. var r2 = ecs[indexOfExisting];
  294. var range1 = {};
  295. var range2 = {};
  296. switch(mode) {
  297. case 0:
  298. return [];
  299. break;
  300. case 1:
  301. ecs.splice(indexOfExisting, 1);
  302. changes = true;
  303. break;
  304. case 2:
  305. range1.start = new Date(r1.startDate);
  306. range1.end = new Date(r1.endDate);
  307. range2.start = new Date(r2.startDate);
  308. range2.end = new Date(r2.endDate);
  309. changes = true;
  310. if (range1.start >= range2.start && range1.end <= range2.end) {
  311. ecs.splice(indexOfNew, 1);
  312. } else {
  313. if (range1.start < range2.start && range1.end > range2.end) {
  314. ecs.push({isNew:true, startDate:r2.endDate, endDate:r1.endDate});
  315. r1.endDate = r2.startDate;
  316. } else {
  317. if (range1.start < range2.start) {
  318. r1.endDate = r2.startDate;
  319. } else {
  320. r1.startDate = r2.endDate;
  321. }
  322. }
  323. }
  324. break;
  325. case 3:
  326. range1.start = new Date(r1.startDate);
  327. range1.end = new Date(r1.endDate);
  328. range2.start = new Date(r2.startDate);
  329. range2.end = new Date(r2.endDate);
  330. changes = true;
  331. if (range1.start > range2.start && range1.end < range2.end) {
  332. ecs.push({isNew:false, startDate:r1.endDate, endDate:r2.endDate, ref:r2.ref});
  333. r2.endDate = r1.startDate;
  334. } else {
  335. if (range1.start <= range2.start && range1.end >= range2.end) {
  336. ecs.splice(indexOfExisting, 1);
  337. } else {
  338. if (range1.start < range2.start) {
  339. r2.startDate = r1.endDate;
  340. } else {
  341. r2.endDate = r1.startDate;
  342. }
  343. }
  344. }
  345. break;
  346. }
  347. }
  348. }
  349. }
  350. return ecs;
  351. };
  352. WMEAC.getCountriesFromSegmentSet = function(segs) {
  353. var cids = segs.map(function(s) {
  354. if (s.attributes.hasOwnProperty("primaryStreetID") && s.attributes.primaryStreetID != null) {
  355. var stid = s.attributes.primaryStreetID;
  356. if (W.model.streets.objects.hasOwnProperty(stid)) {
  357. var st = W.model.streets.objects[stid];
  358. if (st.hasOwnProperty("cityID") && st.cityID != null && typeof st.cityID != "undefined") {
  359. var ctid = st.cityID;
  360. if (W.model.cities.objects.hasOwnProperty(ctid)) {
  361. return W.model.cities.objects[ctid].countryID;
  362. }
  363. }
  364. }
  365. }
  366. return null;
  367. }).filter(function(cid) {
  368. return cid != null;
  369. });
  370. return W.model.countries.getObjectArray(function(c) {
  371. return cids.indexOf(c.id) != -1;
  372. });
  373. };
  374. WMEAC.getOppositeClosure = function(closure) {
  375. return W.model.roadClosures.getObjectArray(function(c) {
  376. return closure.reason == c.reason && closure.startDate == c.startDate && closure.endDate == c.endDate && closure.segID == c.segID && closure.forward != c.forward;
  377. });
  378. };
  379. WMEAC.getCityStreetsFromSegmentSet = function(segs) {
  380. var r = {};
  381. function add(city, street) {
  382. if (!r.hasOwnProperty(city)) {
  383. r[city] = {};
  384. }
  385. if (!r[city].hasOwnProperty(street)) {
  386. r[city][street] = 0;
  387. }
  388. r[city][street]++;
  389. }
  390. segs.forEach(function(s) {
  391. var city = "noCity";
  392. if (s.attributes.primaryStreetID != null && W.model.streets.objects.hasOwnProperty(s.attributes.primaryStreetID)) {
  393. var st = W.model.streets.objects[s.attributes.primaryStreetID];
  394. if (st.hasOwnProperty("cityID") && st.cityID != null && typeof st.cityID != "undefined") {
  395. var ctid = st.cityID;
  396. if (W.model.cities.objects.hasOwnProperty(ctid)) {
  397. if (!W.model.cities.objects[ctid].isEmpty) {
  398. city = W.model.cities.objects[ctid].name;
  399. }
  400. }
  401. }
  402. if (W.model.streets.objects[s.attributes.primaryStreetID].isEmpty) {
  403. add(city, "noStreet");
  404. } else {
  405. add(city, W.model.streets.objects[s.attributes.primaryStreetID].name);
  406. }
  407. }
  408. });
  409. return r;
  410. };
  411. WMEAC.download = function(data, filename) {
  412. var element = document.createElement("a");
  413. element.style.display = "none";
  414. element.setAttribute("href", encodeURI("data:text/plain," + data));
  415. element.setAttribute("download", filename);
  416. document.body.appendChild(element);
  417. element.click();
  418. document.body.removeChild(element);
  419. };
  420. WMEAC.buildPermalink = function(data) {
  421. var getvars = [];
  422. for (var m in data) {
  423. if (data.hasOwnProperty(m)) {
  424. getvars.push("" + m + "=" + data[m]);
  425. }
  426. }
  427. return document.location.protocol + "//" + document.location.host + document.location.pathname + "?" + getvars.join("&amp;");
  428. };
  429. var cssElt = WMEAC.createElement({type:"style"});
  430. cssElt.type = "text/css";
  431. var css = "";
  432. css += ".slashed:after { content: ''; position: relative; width: 140%; height: 1px; display: block; background: red; transform: rotate(-30deg); margin-top: -50%; margin-left: -20%; }";
  433. 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 }";
  434. 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; }";
  435. 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;}";
  436. css += ".wmeac-progressBarFG { float: left; position: relative; bottom: 22px; height: 0px; text-align: center; width: 100% }";
  437. css += ".wmeac-button { border: none; border-bottom-left-radius: 5px; border-bottom-right-radius: 5px; border-top-left-radius: 5px; border-top-right-radius: 5px; background-color: #F1DDDB; display: inline-block; padding: 6px 12px; cursor: pointer; text-align: center; font-weight: bold; }";
  438. 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;}";
  439. 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; }";
  440. css += ".wmeac-closuredialog h1 { background-color: #F1DDDB; text-align: center; font-size: medium; margin-top: 0px; padding: 10px;}";
  441. css += ".wmeac-closuredialog .content { padding: 10px;}";
  442. css += ".wmeac-closuredialog .content table { witdh: 100%; border: none; font-size: 10px; text-transform: uppercase;}";
  443. css += ".wmeac-closuredialog .content table tbody tr { vertical-align: top;}";
  444. css += ".wmeac-closuredialog .content table tbody tr td { padding-right: 2px; padding-left: 2px;}";
  445. css += ".wmeac-closuredialog-fromgroup { display: inline-block; }";
  446. css += ".wmeac-nav-tabs>ul { border-bottom: 1px solid #F6C3BE }";
  447. css += ".wmeac-nav-tabs>li { float: left; margin-bottom: -1px; }";
  448. css += ".wmeac-nav-tabs>li>a { border: 1px solid #F6C3BE; border-top-left-radius: 5px; border-top-right-radius: 5px; margin-right: 2px;}";
  449. css += ".wmeac-nav-tabs>li.active>a { background-color: rgba(0, 0, 0, 0); border-bottom: 1px solid #FDEDEB}";
  450. css += ".wmeac-nav-tabs>li:not(.active)>a { background-color: #DADBDC}";
  451. 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;}";
  452. css += ".wmeac-closuredialog .footer { height: 40px; padding: 0 10px;}";
  453. css += "#wmeac-csv-closures-list ul { list-style-type: none; padding: 0px;}";
  454. css += "#wmeac-csv-closures-list ul li { width: 100%; height: 42px; border-radius: 4px; margin-top: 1px; }";
  455. css += "#wmeac-csv-closures-list ul li > * { display: table-cell; vertical-align: middle;}";
  456. css += ".wmeac-csv-closures-list-add { background-color: #C6DFFF; }";
  457. css += ".wmeac-csv-closures-list-remove { background-color: #FFC65F; }";
  458. css += ".wmeac-csv-closures-list-failed { background-color: #FF8585; }";
  459. css += ".wmeac-csv-closures-list-done { background-color: #B9FAB1; }";
  460. css += ".wmeac-csv-closures-list-col-action { width: 14px; min-width: 14px; }";
  461. css += ".wmeac-csv-closures-list-col-lr { font-size: xx-small; width: 100%; }";
  462. css += ".wmeac-csv-closures-list-col-lr > * { height: 14px; overflow-y: hidden; vertical-align: middle; }";
  463. css += ".wmeac-csv-closures-list-col-dates { width: 75px; min-width: 75px; font-size: xx-small; text-align: center; }";
  464. css += ".wmeac-csv-closures-list-col-dates > * { height: 14px; overflow-y: hidden; vertical-align: center; }";
  465. css += ".wmeac-csv-closures-list-col-dir { width: 35px; min-width: 35px; text-align: center; }";
  466. css += ".wmeac-csv-closures-list-col-it { width: 15px; min-width: 15px; }";
  467. css += ".wmeac-csv-closures-list-col-target { width: 15px; min-width: 15px; }";
  468. css += ".wmeac-csv-closures-list-col-apply { width: 15px; min-width: 15px; }";
  469. 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; }";
  470. 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; }";
  471. 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; }';
  472. 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; }";
  473. 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; }';
  474. css += '.wmeac-hl:after { content: "\\f018"; position: relative; display: block; margin-top: -100%; margin-left: 50%; font-family: FontAwesome; }';
  475. cssElt.innerHTML = css;
  476. document.body.appendChild(cssElt);
  477. WMEAC.bootstrapAC = function() {
  478. window.setTimeout(WMEAC.initialize, 500);
  479. };
  480. WMEAC.initialize = function() {
  481. WMEAC.log("init");
  482. WMEAC.waitForWaze(function() {
  483. WMEAC.load();
  484. WMEAC.log("presets", WMEAC.presets);
  485. WMEAC.initUI();
  486. });
  487. WMEAC.log("init done");
  488. };
  489. WMEAC.waitForWaze = function(handler) {
  490. var APIRequired = [{o:"W"}, {o:"W.model"}, {o:"W.map"}, {o:"W.loginManager"}, {o:"W.vent"}, {o:"W.Config"}, {o:"W.controller"}];
  491. for (var i = 0; i < APIRequired.length; i++) {
  492. var path = APIRequired[i].o.split(".");
  493. var object = window;
  494. for (var j = 0; j < path.length; j++) {
  495. object = object[path[j]];
  496. if (typeof object == "undefined" || object == null) {
  497. window.setTimeout(function() {
  498. WMEAC.waitForWaze(handler);
  499. }, 500);
  500. return;
  501. }
  502. }
  503. }
  504. var userInfo = WMEAC.getId("user-info");
  505. if (userInfo == null) {
  506. window.setTimeout(function() {
  507. WMEAC.waitForWaze(handler);
  508. }, 500);
  509. return;
  510. }
  511. var navTabs = userInfo.getElementsByTagName("ul");
  512. if (navTabs.length == 0) {
  513. window.setTimeout(function() {
  514. WMEAC.waitForWaze(handler);
  515. }, 500);
  516. return;
  517. }
  518. if (typeof navTabs[0] == "undefined") {
  519. window.setTimeout(function() {
  520. WMEAC.waitForWaze(handler);
  521. }, 500);
  522. return;
  523. }
  524. var tabContents = userInfo.getElementsByTagName("div");
  525. if (tabContents.length == 0) {
  526. window.setTimeout(function() {
  527. WMEAC.waitForWaze(handler);
  528. }, 500);
  529. return;
  530. }
  531. if (typeof tabContents[0] == "undefined") {
  532. window.setTimeout(function() {
  533. WMEAC.waitForWaze(handler);
  534. }, 500);
  535. return;
  536. }
  537. handler();
  538. };
  539. WMEAC.initUI = function() {
  540. var addon = WMEAC.createElement({type:"section", id:"wmeac-addon"});
  541. WMEAC.pb = new WMEAC.ProgressBar("wmeac-progressBarInfo");
  542. addon.appendChild(WMEAC.pb.divpbi);
  543. var section = WMEAC.createElement({type:"p", id:"wmeac-main-title"});
  544. section.style.paddingTop = "0px";
  545. section.style.marginTop = "-15px";
  546. section.style.textIndent = "8px";
  547. var title = '<b><a target="_blank" href="https://greasyfork.org/scripts/20705-wme-advanced-closures"><u>Advanced Closures</u></a> <a target="_blank" href="https://www.waze.com/forum/viewtopic.php?f=1316&t=193462">Fr</a> <a target="_blank" href="https://www.waze.com/forum/viewtopic.php?f=819&t=193465">En</a> </b> v' + WMEAC.ac_version;
  548. section.innerHTML = title;
  549. addon.appendChild(section);
  550. var divAdvCl = WMEAC.createElement({type:"div", className:"wmeac-sidepanel", id:"wmeac-ac"});
  551. var addACBtn = WMEAC.createElement({type:"div", id:"wmeac-add-advanced-closure-button", className:"wmeac-button"});
  552. addACBtn.style.width = "100%";
  553. addACBtn.innerHTML = '<i class="fa fa-clock-o"></i> Add advanced closure';
  554. addACBtn.addEventListener("click", WMEAC.showAddAdvancedClosure);
  555. divAdvCl.appendChild(addACBtn);
  556. var divCSV = WMEAC.createElement({type:"div", className:"wmeac-sidepanel", id:"wmeac-csv"});
  557. var csvHTML = '<label for="wmeac-csv-file" class="wmeac-button">Parse CSV</label> <input id="wmeac-csv-file" type="file" name="files[]" style="display: none;" />';
  558. 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>';
  559. divCSV.innerHTML = csvHTML;
  560. addon.appendChild(divAdvCl);
  561. addon.appendChild(WMEAC.createElement({type:"hr"}));
  562. addon.appendChild(divCSV);
  563. var userTabs = WMEAC.getId("user-tabs");
  564. var userInfo = WMEAC.getId("user-info");
  565. var sidePanelPrefs = WMEAC.getId("sidepanel-prefs");
  566. var navTabs = WMEAC.getElementsByClassName("nav-tabs", userTabs)[0];
  567. var tabContent = sidePanelPrefs.parentNode;
  568. newtab = WMEAC.createElement({type:"li"});
  569. newtab.innerHTML = '<a title="Advanced closures" href="#sidepanel-wmeac" data-toggle="tab"><span class="fa fa-road slashed"></span></a>';
  570. navTabs.appendChild(newtab);
  571. addon.id = "sidepanel-wmeac";
  572. addon.className = "tab-pane";
  573. addon.style.marginLeft = "-10px";
  574. tabContent.appendChild(addon);
  575. var observer = new MutationObserver(function(mutations) {
  576. mutations.forEach(function(mutation) {
  577. function rescurse(node) {
  578. if (node.id == "segment-edit-closures") {
  579. WMEAC.installButtonInClosureTab(node);
  580. } else {
  581. if (node.className == "closures-list") {
  582. var target = WMEAC.getId("segment-edit-closures");
  583. if (target) {
  584. WMEAC.installButtonInClosureTab(target);
  585. }
  586. } else {
  587. for (var j = 0; j < node.childNodes.length; j++) {
  588. rescurse(node.childNodes[j]);
  589. }
  590. }
  591. }
  592. }
  593. for (var i = 0; i < mutation.addedNodes.length; i++) {
  594. rescurse(mutation.addedNodes[i]);
  595. }
  596. });
  597. });
  598. observer.observe(WMEAC.getId("edit-panel"), {childList:true, subtree:true});
  599. WMEAC.installButtonInClosureTab();
  600. W.vent.on("operationPending", function(e) {
  601. if (e.operation.id != "pending.road_data") {
  602. return;
  603. }
  604. WMEAC.pendingOps = true;
  605. });
  606. W.vent.on("operationDone", function(e) {
  607. if (e.operation.id != "pending.road_data") {
  608. return;
  609. }
  610. WMEAC.pendingOps = false;
  611. });
  612. W.model.events.register("mergeend", null, WMEAC.refreshHighlight);
  613. WMEAC.refreshHighlight();
  614. window.setTimeout(WMEAC.connectAdvancedClosureTabHandlers);
  615. };
  616. WMEAC.installButtonInClosureTab = function(node) {
  617. if (!node) {
  618. node = WMEAC.getId("segment-edit-closures");
  619. }
  620. if (!node) {
  621. return;
  622. }
  623. if ($(node).find("#wmeac-closuretab-add-advanced-closure-button").length == 0) {
  624. var addACBtn = WMEAC.createElement({type:"div", id:"wmeac-closuretab-add-advanced-closure-button", className:"wmeac-button"});
  625. addACBtn.style.width = "100%";
  626. addACBtn.style.marginBottom = "10px";
  627. addACBtn.innerHTML = '<i class="fa fa-clock-o"></i> Add advanced closure';
  628. addACBtn.addEventListener("click", WMEAC.showAddAdvancedClosure);
  629. $(node).find(".closures-list").prepend(addACBtn);
  630. }
  631. };
  632. WMEAC.showAddAdvancedClosure = function() {
  633. var ACDiv = WMEAC.getId("wmeac-add-advanced-closure-dialog");
  634. if (ACDiv == null) {
  635. ACDiv = WMEAC.createElement({type:"div", id:"wmeac-add-advanced-closure-dialog", className:"wmeac-closuredialog"});
  636. ACDiv.innerHTML = WMEAC.HTMLTemplates.advancedClosureDialog;
  637. W.map.div.appendChild(ACDiv);
  638. window.setTimeout(WMEAC.connectAdvancedClosureDialogHandlers);
  639. ACDiv.style.display = "none";
  640. }
  641. if (ACDiv.style.display == "block") {
  642. $(ACDiv).css({left:"80px", top:"10px"});
  643. } else {
  644. ACDiv.style.display = "block";
  645. W.selectionManager.events.register("selectionchanged", null, WMEAC.refreshClosureList);
  646. W.selectionManager.events.register("selectionchanged", null, WMEAC.refreshClosureListFromSelection);
  647. WMEAC.refreshClosureListFromSelection();
  648. }
  649. $(ACDiv).find(".input-group-addon").css({display:"table-cell"});
  650. WMEAC.showClosuresLayer(true);
  651. };
  652. WMEAC.HTMLTemplates = {};
  653. 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>';
  654. 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>';
  655. 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>';
  656. 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>';
  657. 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 (&#xf0ec;)</option><option value="1">One way (A &#8594; B)</option><option value="2">One way (B &#8594; A)</option> </select> </div> </div>';
  658. 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>';
  659. 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>';
  660. 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>';
  661. 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>';
  662. var daysOfWeekUI = _(WMEAC.daysOfWeek).clone();
  663. daysOfWeekUI.push(daysOfWeekUI.shift());
  664. var tabEachUI = '<div class="checkbox"> <label class="control-label" style="font-weight: bold;"> <input id="wmeac-advanced-closure-dialog-each-dayall" type="checkbox" name="closure_each_dayall"> All </label> </div> ' + daysOfWeekUI.map(function(d, i) {
  665. return '<div class="checkbox"> <label class="control-label" style="font-weight: bold;"> <input id="wmeac-advanced-closure-dialog-each-' + (i + 1) % 7 + '" type="checkbox" name="closure_each_' + d + '"> ' + d + " </label> </div>";
  666. }).join("");
  667. 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>';
  668. 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>';
  669. 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"> ' +
  670. 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>";
  671. 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>';
  672. WMEAC.HTMLTemplates.advancedClosureDialog = '<h1>Advanced closures</h1><div class="content"> <table> <tr> <td style="width: 50%;">' + rangeStartEndUI + startTimeAndDurationUI + " </td> <td>" + descriptionUI + directionUI + ignoreTrafficUI + MTEUI + " </td> </tr> </table>" + tabsUI + "</div>" + footerUI;
  673. WMEAC.connectAdvancedClosureDialogHandlers = function() {
  674. var e = null;
  675. e = WMEAC.getId("wmeac-advanced-closure-dialog-exportCSV-button");
  676. if (e) {
  677. e.addEventListener("click", function() {
  678. var rc = WMEAC.buildClosuresListFromRecurringUI();
  679. if (rc.error != "") {
  680. alert("Can't apply closures.\nPlease, check all parameters.");
  681. return;
  682. }
  683. if (W.selectionManager.getSelectedFeatures().length == 0 || W.selectionManager.getSelectedFeatures()[0].model.type != "segment") {
  684. alert("Please, select segment(s) before.");
  685. return;
  686. }
  687. var reason = $("#wmeac-advanced-closure-dialog-reason").val();
  688. var direction = $("#wmeac-advanced-closure-dialog-direction").val();
  689. var isIT = $("#wmeac-advanced-closure-dialog-ignoretraffic").is(":checked");
  690. var mteId = $("#wmeac-advanced-closure-dialog-mteid").val();
  691. var mte = W.model.majorTrafficEvents.get(mteId);
  692. closureList = rc.list.map(function(e) {
  693. var details = {reason:reason, direction:direction, startDate:e.start, endDate:e.end, location:"", permanent:isIT};
  694. if (mte) {
  695. details.eventId = mte.id;
  696. }
  697. return details;
  698. });
  699. var selection = _.pluck(W.selectionManager.getSelectedFeatures(), "model");
  700. var selectionReversed = [];
  701. if (direction != "3") {
  702. var rev = W.selectionManager.getReversedSegments();
  703. selection = selection.filter(function(e) {
  704. if (rev[e.attributes.id]) {
  705. selectionReversed.push(e);
  706. return false;
  707. }
  708. return true;
  709. });
  710. }
  711. var lonlat = W.map.center.transform(W.map.projection.projCode, "EPSG:4326");
  712. var csv = "header,reason,start date (yyyy-mm-dd hh:mm),end date (yyyy-mm-dd hh:mm),direction (A to B|B to A|TWO WAY),ignore trafic (Yes|No),segment IDs (id1;id2;...),lon/lat (like in a permalink: lon=xxx&lat=yyy),zoom (2 to 10),MTE id (empty cell if not),comment (optional)\n";
  713. closureList.forEach(function(e) {
  714. 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) {
  715. return s.attributes.id;
  716. }).join(";") + '","lon=' + lonlat.lon + "&lat=" + lonlat.lat + '",' + W.map.zoom + "," + mteId + ',"Generated by WMEAC"\n';
  717. });
  718. if (!selectionReversed.length == 0) {
  719. closureList.forEach(function(e) {
  720. 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) {
  721. return s.attributes.id;
  722. }).join(";") + '","lon=' + lonlat.lon + "&lat=" + lonlat.lat + '",' + W.map.zoom + "," + mteId + ',"Generated by WMEAC"\n';
  723. });
  724. }
  725. WMEAC.download(csv, "closures.csv");
  726. });
  727. }
  728. e = WMEAC.getId("wmeac-advanced-closure-dialog-close-button");
  729. if (e) {
  730. e.addEventListener("click", function() {
  731. var d = WMEAC.getId("wmeac-add-advanced-closure-dialog");
  732. if (d) {
  733. W.selectionManager.events.unregister("selectionchanged", null, WMEAC.refreshClosureList);
  734. W.selectionManager.events.unregister("selectionchanged", null, WMEAC.refreshClosureListFromSelection);
  735. d.style.display = "none";
  736. }
  737. });
  738. }
  739. e = WMEAC.getId("wmeac-advanced-closure-dialog-apply-button");
  740. if (e) {
  741. e.addEventListener("click", function() {
  742. var rc = WMEAC.buildClosuresListFromRecurringUI();
  743. if (rc.error != "") {
  744. alert("Can't apply closures.\nPlease, check all parameters.");
  745. return;
  746. }
  747. if (W.selectionManager.getSelectedFeatures().length == 0 || W.selectionManager.getSelectedFeatures()[0].model.type != "segment") {
  748. alert("Please, select segment(s) before.");
  749. return;
  750. }
  751. if (W.selectionManager.getSelectedFeatures().every(function(e) {
  752. return e.model.isAllowed(e.model.PERMISSIONS.EDIT_CLOSURES);
  753. }) == false) {
  754. alert("You don't have permission to edit closures on all those segments.");
  755. return;
  756. }
  757. var reason = $("#wmeac-advanced-closure-dialog-reason").val();
  758. var direction = $("#wmeac-advanced-closure-dialog-direction").val();
  759. var sc = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/SharedClosure");
  760. direction = direction == "1" ? sc.DIRECTION.A_TO_B : direction == "2" ? sc.DIRECTION.B_TO_A : sc.DIRECTION.TWO_WAY;
  761. var directionStr = direction == 1 ? "(A &#8594; B)" : direction == 2 ? "(B &#8594; A)" : "(&#8646;)";
  762. var isIT = $("#wmeac-advanced-closure-dialog-ignoretraffic").is(":checked");
  763. var mte = W.model.majorTrafficEvents.get($("#wmeac-advanced-closure-dialog-mteid").val());
  764. closureList = rc.list.map(function(e) {
  765. var details = {reason:reason, direction:direction, startDate:e.start, endDate:e.end, location:"", permanent:isIT};
  766. if (mte) {
  767. details.eventId = mte.id;
  768. }
  769. return details;
  770. });
  771. var selection = _.pluck(W.selectionManager.getSelectedFeatures(), "model");
  772. W.selectionManager.events.unregister("selectionchanged", null, WMEAC.refreshClosureList);
  773. WMEAC.addClosureListFromSelection(closureList, function(i, e) {
  774. $("#wmeac-advanced-closure-dialog-preview-" + i).html(e).css({color:"#44D544"});
  775. }, function(i, e) {
  776. $("#wmeac-advanced-closure-dialog-preview-" + i).html(e).css({color:"#D5444F"});
  777. }, function() {
  778. W.selectionManager.setSelectedModels(selection);
  779. var tmp = function selectionReady() {
  780. if (W.selectionManager.getSelectedFeatures().length == 0) {
  781. window.setTimeout(selectionReady, 500);
  782. } else {
  783. W.selectionManager.events.register("selectionchanged", null, WMEAC.refreshClosureList);
  784. $('a[href="#segment-edit-closures"]').click();
  785. }
  786. };
  787. window.setTimeout(tmp, 500);
  788. }, 0);
  789. });
  790. }
  791. if (typeof $.fn.datepicker !== "undefined") {
  792. $("#wmeac-advanced-closure-dialog-rangestartdate,#wmeac-advanced-closure-dialog-rangeenddate").datepicker({format:"yyyy-mm-dd", todayHighlight:!0, autoclose:!0});
  793. } else {
  794. if (typeof $.fn.daterangepicker !== "undefined") {
  795. $("#wmeac-advanced-closure-dialog-rangestartdate,#wmeac-advanced-closure-dialog-rangeenddate").daterangepicker({singleDatePicker:!0, locale:{format:"YYYY-MM-DD"}});
  796. }
  797. }
  798. $("#wmeac-advanced-closure-dialog-rangestartdate,#wmeac-advanced-closure-dialog-rangeenddate").on("change", function() {
  799. WMEAC.refreshMTEList();
  800. });
  801. $("#wmeac-advanced-closure-dialog-starttime,#wmeac-advanced-closure-dialog-durationtime").timepicker({defaultTime:"00:00", showMeridian:!1, template:!1});
  802. $("#wmeac-add-advanced-closure-dialog").find(".input-group").find(".input-group-addon").on("click", function(e) {
  803. $(e.target).parent().find("input").focus();
  804. }).find("i").on("click", function(e) {
  805. $(e.target).parent().parent().find("input").focus();
  806. });
  807. $("#wmeac-advanced-closure-dialog-each-dayall").on("click", function() {
  808. var atLeastOneChecked = false;
  809. for (var i = 0; i < 7; i++) {
  810. atLeastOneChecked = atLeastOneChecked || $("#wmeac-advanced-closure-dialog-each-" + i).is(":checked");
  811. }
  812. for (var i = 0; i < 7; i++) {
  813. $("#wmeac-advanced-closure-dialog-each-" + i).prop("checked", !atLeastOneChecked);
  814. }
  815. $("#wmeac-advanced-closure-dialog-each-dayall").prop("checked", !atLeastOneChecked);
  816. });
  817. if (typeof $.fn.spinner !== "undefined") {
  818. $("#wmeac-advanced-closure-dialog-repeat-every-day").spinner({min:0, spin:function(event, ui) {
  819. $(this).trigger("change");
  820. }});
  821. $("#wmeac-advanced-closure-dialog-repeat-every-hour").spinner({min:0, spin:function(event, ui) {
  822. if (ui.value >= 24) {
  823. $(this).spinner("value", ui.value - 24);
  824. $("#wmeac-advanced-closure-dialog-repeat-every-day").spinner("stepUp");
  825. return false;
  826. } else {
  827. if (ui.value < 0) {
  828. $(this).spinner("value", ui.value + 24);
  829. $("#wmeac-advanced-closure-dialog-repeat-every-day").spinner("stepDown");
  830. return false;
  831. }
  832. }
  833. $(this).trigger("change");
  834. }});
  835. $("#wmeac-advanced-closure-dialog-repeat-every-minute").spinner({spin:function(event, ui) {
  836. if (ui.value >= 60) {
  837. $(this).spinner("value", ui.value - 60);
  838. $("#wmeac-advanced-closure-dialog-repeat-every-hour").spinner("stepUp");
  839. return false;
  840. } else {
  841. if (ui.value < 0) {
  842. $(this).spinner("value", ui.value + 60);
  843. $("#wmeac-advanced-closure-dialog-repeat-every-hour").spinner("stepDown");
  844. return false;
  845. }
  846. }
  847. $(this).trigger("change");
  848. }, change:function(event) {
  849. if (event.target.value < 0 || event.target.value > 59) {
  850. $(this).spinner("value", 0);
  851. }
  852. }});
  853. $("#wmeac-advanced-closure-dialog-duration-day").spinner({min:0, spin:function(event, ui) {
  854. $(this).trigger("change");
  855. }});
  856. }
  857. $("#wmeac-advanced-closure-dialog-repeat,#wmeac-advanced-closure-dialog-each,#wmeac-advanced-closure-dialog-holiday").on("click", function(e) {
  858. window.setTimeout(WMEAC.refreshClosureList);
  859. });
  860. $("#wmeac-advanced-closure-dialog-holiday-refresh").on("click", function(e) {
  861. var hDiv = $("#wmeac-advanced-closure-dialog-holiday-list");
  862. $("#wmeac-advanced-closure-dialog-holiday-refresh-spinner").css({display:"block"});
  863. WMEAC.removeChildElements(hDiv[0]);
  864. window.setTimeout(function() {
  865. WMEAC.getHolidays({rangeStart:$("#wmeac-advanced-closure-dialog-rangestartdate").val(), rangeEnd:$("#wmeac-advanced-closure-dialog-rangeenddate").val(), countries:_.pluck(WMEAC.getCountriesFromSegmentSet(_.pluck(W.selectionManager.getSelectedFeatures(), "model")), "abbr"), handlerFinished:function(holidays) {
  866. WMEAC.lastGeneratedHolidays = holidays;
  867. if (holidays.length == 0) {
  868. hDiv.html("No holiday found.");
  869. } else {
  870. holidays.forEach(function(h, i) {
  871. var chkBx = WMEAC.createElement({type:"div", className:"checkbox"});
  872. 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> ";
  873. $(chkBx).on("click", function(e) {
  874. window.setTimeout(WMEAC.refreshClosureList);
  875. });
  876. hDiv.append(chkBx);
  877. });
  878. }
  879. $("#wmeac-advanced-closure-dialog-holiday-refresh-spinner").css({display:"none"});
  880. }});
  881. });
  882. });
  883. $("#wmeac-add-advanced-closure-dialog").on("change", function(e) {
  884. window.setTimeout(WMEAC.refreshClosureList);
  885. });
  886. WMEAC.reloadPresets();
  887. $("#wmeac-advanced-closure-dialog-presets-load").on("click", function(e) {
  888. var presetIndex = parseInt($("#wmeac-advanced-closure-dialog-presets-list").val());
  889. $("#wmeac-advanced-closure-dialog-starttime").val(WMEAC.presets[presetIndex].values.starttime);
  890. if (WMEAC.presets[presetIndex].values.duration.hasOwnProperty("day")) {
  891. $("#wmeac-advanced-closure-dialog-duration-day").val(WMEAC.presets[presetIndex].values.duration.day);
  892. } else {
  893. $("#wmeac-advanced-closure-dialog-duration-day").val(Math.floor(WMEAC.presets[presetIndex].values.duration.hour / 24));
  894. }
  895. $("#wmeac-advanced-closure-dialog-durationtime").val("" + WMEAC.presets[presetIndex].values.duration.hour % 24 + ":" + WMEAC.presets[presetIndex].values.duration.minute);
  896. $("#wmeac-advanced-closure-dialog-reason").val(WMEAC.presets[presetIndex].values.description);
  897. $("#wmeac-advanced-closure-dialog-direction").val(WMEAC.presets[presetIndex].values.direction);
  898. $("#wmeac-advanced-closure-dialog-ignoretraffic").prop("checked", WMEAC.presets[presetIndex].values.ignoretraffic);
  899. $("#wmeac-advanced-closure-dialog-repeat-ntimes").val(WMEAC.presets[presetIndex].values.repeat.ntimes);
  900. if (WMEAC.presets[presetIndex].values.repeat.hasOwnProperty("day")) {
  901. $("#wmeac-advanced-closure-dialog-repeat-every-day").val(WMEAC.presets[presetIndex].values.repeat.day);
  902. } else {
  903. $("#wmeac-advanced-closure-dialog-repeat-every-day").val(Math.floor(WMEAC.presets[presetIndex].values.repeat.hour / 24));
  904. }
  905. $("#wmeac-advanced-closure-dialog-repeat-every-hour").val(WMEAC.presets[presetIndex].values.repeat.hour % 24);
  906. $("#wmeac-advanced-closure-dialog-repeat-every-minute").val(WMEAC.presets[presetIndex].values.repeat.minute);
  907. for (var i = 0; i < 7; i++) {
  908. $("#wmeac-advanced-closure-dialog-each-" + i).prop("checked", WMEAC.presets[presetIndex].values.each[i]);
  909. }
  910. });
  911. $("#wmeac-advanced-closure-dialog-presets-load-fromseg").on("click", function() {
  912. closureId = $("#wmeac-advanced-closure-dialog-segclosure-list").val();
  913. if (closureId) {
  914. var c = W.model.roadClosures.objects[closureId];
  915. if (c) {
  916. $("#wmeac-advanced-closure-dialog-starttime").val(c.startDate.split(" ")[1]);
  917. var duration = new Date(c.endDate) - new Date(c.startDate);
  918. var days = Math.floor(duration / 86400000);
  919. $("#wmeac-advanced-closure-dialog-duration-day").val(days);
  920. var hours = Math.floor((duration - days * 86400000) / 3600000);
  921. var minutes = Math.floor((duration - days * 86400000 - hours * 3600000) / 60000);
  922. $("#wmeac-advanced-closure-dialog-durationtime").val("" + hours + ":" + minutes);
  923. $("#wmeac-advanced-closure-dialog-reason").val(c.reason.trim());
  924. if (WMEAC.getOppositeClosure(c).length == 0) {
  925. $("#wmeac-advanced-closure-dialog-direction").val(c.forward ? 1 : 2);
  926. } else {
  927. $("#wmeac-advanced-closure-dialog-direction").val(3);
  928. }
  929. $("#wmeac-advanced-closure-dialog-ignoretraffic").prop("checked", c.permanent);
  930. if (c.eventId != null) {
  931. var options = [];
  932. $("#wmeac-advanced-closure-dialog-mteid option").each(function() {
  933. options.push($(this).val());
  934. });
  935. if (options.indexOf(c.eventId) != -1) {
  936. $("#wmeac-advanced-closure-dialog-mteid").val(c.eventId);
  937. } else {
  938. $("#wmeac-advanced-closure-dialog-mteid").val("");
  939. }
  940. }
  941. }
  942. }
  943. });
  944. $("#wmeac-advanced-closure-dialog-presets-delete").on("click", function(e) {
  945. var presetIndex = parseInt($("#wmeac-advanced-closure-dialog-presets-list").val());
  946. WMEAC.presets.splice(presetIndex, 1);
  947. WMEAC.save();
  948. WMEAC.reloadPresets();
  949. });
  950. $("#wmeac-advanced-closure-dialog-presets-save").on("click", function(e) {
  951. var name = $("#wmeac-advanced-closure-dialog-presets-name").val();
  952. var presetIndex = WMEAC.presets.findIndex(function(e) {
  953. return e.name == name;
  954. });
  955. var preset = {name:name, values:{duration:{}, repeat:{}, each:[]}};
  956. if (presetIndex != -1) {
  957. preset = WMEAC.presets[presetIndex];
  958. }
  959. preset.values.starttime = $("#wmeac-advanced-closure-dialog-starttime").val();
  960. preset.values.duration.day = $("#wmeac-advanced-closure-dialog-duration-day").val();
  961. preset.values.duration.hour = parseInt($("#wmeac-advanced-closure-dialog-durationtime").val().split(":")[0]);
  962. preset.values.duration.minute = parseInt($("#wmeac-advanced-closure-dialog-durationtime").val().split(":")[1]);
  963. preset.values.description = $("#wmeac-advanced-closure-dialog-reason").val();
  964. preset.values.direction = $("#wmeac-advanced-closure-dialog-direction").val();
  965. preset.values.ignoretraffic = $("#wmeac-advanced-closure-dialog-ignoretraffic").is(":checked");
  966. preset.values.repeat.ntimes = $("#wmeac-advanced-closure-dialog-repeat-ntimes").val();
  967. preset.values.repeat.day = $("#wmeac-advanced-closure-dialog-repeat-every-day").val();
  968. preset.values.repeat.hour = $("#wmeac-advanced-closure-dialog-repeat-every-hour").val();
  969. preset.values.repeat.minute = $("#wmeac-advanced-closure-dialog-repeat-every-minute").val();
  970. for (var i = 0; i < 7; i++) {
  971. preset.values.each[i] = $("#wmeac-advanced-closure-dialog-each-" + i).is(":checked");
  972. }
  973. if (presetIndex == -1) {
  974. WMEAC.presets.push(preset);
  975. }
  976. WMEAC.save();
  977. WMEAC.reloadPresets();
  978. });
  979. WMEAC.setDraggable($("#wmeac-add-advanced-closure-dialog"), {controller:$("#wmeac-add-advanced-closure-dialog h1:first-child"), container:[$("#OpenLayers_Map_200_OpenLayers_ViewPort"), $("#WazeMap")]});
  980. };
  981. WMEAC.connectAdvancedClosureTabHandlers = function() {
  982. var e = null;
  983. e = WMEAC.getId("wmeac-csv-file");
  984. if (e) {
  985. e.addEventListener("change", WMEAC.CSVFileChanged);
  986. }
  987. e = WMEAC.getId("wmeac-csv-closures-controls-check");
  988. if (e) {
  989. e.addEventListener("change", function(e) {
  990. WMEAC.CSVCheckAll(e.target.checked);
  991. });
  992. }
  993. e = WMEAC.getId("wmeac-csv-closures-controls-apply");
  994. if (e) {
  995. e.addEventListener("click", WMEAC.CSVApplyChecked);
  996. }
  997. e = WMEAC.getId("wmeac-csv-closures-controls-segs");
  998. if (e) {
  999. e.addEventListener("click", WMEAC.CSVCheckSegsChecked);
  1000. }
  1001. };
  1002. WMEAC.reloadPresets = function() {
  1003. var optionList = WMEAC.presets.map(function(p, i) {
  1004. return '<option value="' + i + '">' + p.name + "</option>";
  1005. });
  1006. $("#wmeac-advanced-closure-dialog-presets-list").html(optionList.join(""));
  1007. };
  1008. WMEAC.ProgressBar = function(id) {
  1009. this.id = id;
  1010. this.divpbi = WMEAC.createElement({type:"div", id:id, className:id});
  1011. var elt = WMEAC.createElement({type:"div", id:"wmeac-progressBar"});
  1012. elt.style.width = "100%";
  1013. elt.style.display = "none";
  1014. elt.innerHTML = '<div class="wmeac-progressBarBG"></div><span class="wmeac-progressBarFG">100%</span>';
  1015. this.divpbi.appendChild(elt);
  1016. elt = WMEAC.createElement({type:"div", id:"wmeac-progressBar-info"});
  1017. this.divpbi.appendChild(elt);
  1018. this.isShown = function() {
  1019. return this.divpbi.style.display != "none";
  1020. };
  1021. this.show = function(toShow) {
  1022. this.divpbi.style.display = toShow ? "block" : "none";
  1023. };
  1024. this.update = function(value) {
  1025. if (value == -1) {
  1026. this.divpbi.children[0].style.display = "none";
  1027. this.divpbi.children[1].style.display = "none";
  1028. return;
  1029. }
  1030. value = Math.round(value);
  1031. this.divpbi.children[0].style.display = "block";
  1032. this.divpbi.children[1].style.display = "block";
  1033. this.divpbi.children[0].children[0].style.width = value + "%";
  1034. this.divpbi.children[0].children[1].innerHTML = value + "%";
  1035. };
  1036. this.info = function(text) {
  1037. this.divpbi.children[1].innerHTML = text;
  1038. };
  1039. };
  1040. WMEAC.ClassClosure = function(options) {
  1041. WMEAC.log("options", options);
  1042. this.isValid = false;
  1043. this.errorMessage = "";
  1044. var validProperties = ["reason", "startDate", "endDate", "direction", "segIDs", "lonlat", "permanent", "id", "zoom"];
  1045. var goodOptions = 0;
  1046. validProperties.forEach(function(p) {
  1047. if (options.hasOwnProperty(p)) {
  1048. this[p] = options[p];
  1049. goodOptions++;
  1050. } else {
  1051. this.errorMessage += "Missing property " + p + "\n";
  1052. }
  1053. }, this);
  1054. if (goodOptions == validProperties.length) {
  1055. this.isValid = true;
  1056. } else {
  1057. return;
  1058. }
  1059. this.comment = "";
  1060. if (options.hasOwnProperty("comment")) {
  1061. this.comment = options.comment;
  1062. }
  1063. this.eventId = null;
  1064. if (options.hasOwnProperty("eventId") && options.eventId != "") {
  1065. this.eventId = options.eventId;
  1066. }
  1067. this.segIDs = this.segIDs.split(";");
  1068. var matches = this.lonlat.match(/lon=(-?\d+\.?\d*)&lat=(-?\d+\.?\d*)/);
  1069. if (matches && matches.length == 3) {
  1070. this.lonlat = {lon:parseFloat(matches[1]), lat:parseFloat(matches[2])};
  1071. } else {
  1072. matches = this.lonlat.match(/lat=(-?\d+\.?\d*)&lon=(-?\d+\.?\d*)/);
  1073. if (matches && matches.length == 3) {
  1074. this.lonlat = {lon:parseFloat(matches[2]), lat:parseFloat(matches[1])};
  1075. } else {
  1076. this.isValid = false;
  1077. this.errorMessage = "Can't parse lonlat: " + this.lonlat + "\n";
  1078. return;
  1079. }
  1080. }
  1081. if (this.direction != "A to B" && this.direction != "B to A" && this.direction != "TWO WAY") {
  1082. this.isValid = false;
  1083. this.errorMessage = "Can't determine direction: " + this.direction + "\n";
  1084. return;
  1085. }
  1086. this.zoom = parseInt(this.zoom);
  1087. if (this.zoom < 2 || this.zoom > 10) {
  1088. this.isValid = false;
  1089. this.errorMessage = "Wrong zoom (2 to 10): " + this.zoom + "\n";
  1090. return;
  1091. }
  1092. this.applyInWME = function(successHandler, failureHandler) {
  1093. var segs = WMEAC.segmentsIDsToSegments(this.segIDs);
  1094. WMEAC.log("Segs: ", segs);
  1095. segs = segs.filter(function(seg) {
  1096. return seg.isAllowed(seg.PERMISSIONS.EDIT_CLOSURES);
  1097. });
  1098. if (segs.length == 0) {
  1099. failureHandler([{attributes:{details:"No segment. Check permissions or existence."}}]);
  1100. } else {
  1101. var cityStreets = WMEAC.getCityStreetsFromSegmentSet(segs);
  1102. var closureLocation = Object.keys(cityStreets).map(function(c) {
  1103. return Object.keys(cityStreets[c]).map(function(s) {
  1104. if (s == "noStreet") {
  1105. return I18n.translations[I18n.locale].edit.address.no_street;
  1106. }
  1107. return s;
  1108. }).join(", ") + (c == "noCity" ? "" : " (" + c + ")");
  1109. }).join(" ; ");
  1110. var sc = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/SharedClosure");
  1111. var closureDetails = {reason:this.reason, direction:this.direction == "A to B" ? sc.DIRECTION.A_TO_B : this.direction == "B to A" ? sc.DIRECTION.B_TO_A : sc.DIRECTION.TWO_WAY, startDate:this.startDate, endDate:this.endDate, location:closureLocation, permanent:this.permanent == "Yes", segments:segs};
  1112. if (this.eventId != null) {
  1113. closureDetails.eventId = this.eventId;
  1114. }
  1115. WMEAC.addClosure(closureDetails, successHandler, failureHandler);
  1116. }
  1117. };
  1118. this.removeInWME = function(successHandler, failureHandler) {
  1119. var segs = WMEAC.segmentsIDsToSegments(this.segIDs);
  1120. segs = segs.filter(function(seg) {
  1121. return seg.isAllowed(seg.PERMISSIONS.EDIT_CLOSURES);
  1122. });
  1123. var allClosuresToRemove = [];
  1124. var countToMatch = this.segIDs.length * (this.direction == "TWO WAY" ? 2 : 1);
  1125. segs.forEach(function(s) {
  1126. var that = this;
  1127. var closures = W.model.roadClosures.getObjectArray(function(c) {
  1128. return c.startDate == that.startDate && c.endDate == that.endDate && c.reason.trim() == that.reason && c.segID == s.attributes.id && c.permanent == (that.permanent == "Yes");
  1129. });
  1130. if (this.direction == "TWO WAY" || this.direction == "A to B" && closures.length == 1 && closures[0].forward == true || this.direction == "B to A" && closures.length == 1 && closures[0].forward == false) {
  1131. allClosuresToRemove = allClosuresToRemove.concat(closures);
  1132. }
  1133. }, this);
  1134. if (allClosuresToRemove.length == 0) {
  1135. failureHandler([{attributes:{details:"No segment. Check permissions or existence."}}]);
  1136. } else {
  1137. WMEAC.removeClosure(allClosuresToRemove, successHandler, failureHandler);
  1138. }
  1139. };
  1140. };
  1141. WMEAC.buildClosuresListFromRecurringUI = function() {
  1142. var list = [];
  1143. var rangeStartDate = new Date($("#wmeac-advanced-closure-dialog-rangestartdate").val());
  1144. if (!WMEAC.isValidDate(rangeStartDate)) {
  1145. return {list:list, error:"Range start date is not valid"};
  1146. }
  1147. var rangeEndDate = new Date($("#wmeac-advanced-closure-dialog-rangeenddate").val());
  1148. if (!WMEAC.isValidDate(rangeEndDate)) {
  1149. return {list:list, error:"Range end date is not valid"};
  1150. }
  1151. if (rangeEndDate < rangeStartDate) {
  1152. return {list:list, error:"Range end date is before range start date"};
  1153. }
  1154. var dD = parseInt($("#wmeac-advanced-closure-dialog-duration-day").val());
  1155. if (isNaN(dD) || dH < 0) {
  1156. return {list:list, error:"Duration days is invalid"};
  1157. }
  1158. var dH = parseInt($("#wmeac-advanced-closure-dialog-durationtime").val().split(":")[0]);
  1159. var dM = parseInt($("#wmeac-advanced-closure-dialog-durationtime").val().split(":")[1]);
  1160. if (dD == 0 && dH == 0 && dM == 0) {
  1161. return {list:list, error:"Duration is null"};
  1162. }
  1163. var rangeStartTimeM = 0;
  1164. var rangeEndTimeM = 1440;
  1165. var rangeEndDateTime = rangeEndDate.clone();
  1166. rangeEndDateTime.addMinutes(rangeEndTimeM);
  1167. var startTimeM = $("#wmeac-advanced-closure-dialog-starttime").val().split(":").map(function(e) {
  1168. return parseInt(e);
  1169. }).reduce(function(p, c, i) {
  1170. return p * 60 + c;
  1171. });
  1172. if ($("#wmeac-advanced-closure-dialog-tabrepeat").attr("class").indexOf("active") != -1) {
  1173. var ntimes = parseInt($("#wmeac-advanced-closure-dialog-repeat-ntimes").val());
  1174. if (isNaN(ntimes) || ntimes < 1) {
  1175. return {list:list, error:"Repeat count is invalid"};
  1176. }
  1177. var evD = parseInt($("#wmeac-advanced-closure-dialog-repeat-every-day").val());
  1178. if (isNaN(evD) || evD < 0) {
  1179. return {list:list, error:"Repeat every day is invalid"};
  1180. }
  1181. var evH = parseInt($("#wmeac-advanced-closure-dialog-repeat-every-hour").val());
  1182. if (isNaN(evH) || evH < 0) {
  1183. return {list:list, error:"Repeat every hour is invalid"};
  1184. }
  1185. var evM = parseInt($("#wmeac-advanced-closure-dialog-repeat-every-minute").val());
  1186. if (isNaN(evM) || evM < 0 || evM >= 60) {
  1187. return {list:list, error:"Repeat every minute is invalid"};
  1188. }
  1189. if (evD * 1440 + evH * 60 + evM < dD * 1440 + dH * 60 + dM) {
  1190. return {list:list, error:"Repeat must be greater than duration"};
  1191. }
  1192. var firstDateTimeStart = rangeStartDate.clone();
  1193. if (startTimeM < rangeStartTimeM) {
  1194. firstDateTimeStart.addDays(1);
  1195. }
  1196. firstDateTimeStart.setMinutes(startTimeM);
  1197. var firstDateTimeEnd = firstDateTimeStart.clone();
  1198. firstDateTimeEnd.addMinutes(dD * 1440 + dH * 60 + dM);
  1199. for (var i = 0; i < ntimes; i++) {
  1200. var start = firstDateTimeStart.clone();
  1201. start.addMinutes((evD * 1440 + evH * 60 + evM) * i);
  1202. var end = start.clone();
  1203. end.addMinutes(dD * 1440 + dH * 60 + dM);
  1204. if (end > rangeEndDateTime) {
  1205. break;
  1206. }
  1207. list.push({start:WMEAC.dateToClosureStr(start), end:WMEAC.dateToClosureStr(end)});
  1208. }
  1209. return {list:list, error:""};
  1210. } else {
  1211. if ($("#wmeac-advanced-closure-dialog-tabeach").attr("class").indexOf("active") != -1) {
  1212. var dow = WMEAC.daysOfWeek.map(function(e, i) {
  1213. return $("#wmeac-advanced-closure-dialog-each-" + i).is(":checked");
  1214. });
  1215. var dayCount = Math.ceil((rangeEndDate - rangeStartDate + 1) / 86400000);
  1216. var day0 = rangeStartDate.clone();
  1217. day0.addMinutes(startTimeM);
  1218. if (startTimeM < rangeStartTimeM) {
  1219. day0.addDays(1);
  1220. }
  1221. for (var d = 0; d < dayCount; d++) {
  1222. var start = day0.clone();
  1223. start.addMinutes(d * 1440);
  1224. if (dow[start.getUTCDay()]) {
  1225. var end = start.clone();
  1226. end.addMinutes(dD * 1440 + dH * 60 + dM);
  1227. if (end > rangeEndDateTime) {
  1228. break;
  1229. }
  1230. list.push({start:WMEAC.dateToClosureStr(start), end:WMEAC.dateToClosureStr(end)});
  1231. }
  1232. }
  1233. return {list:list, error:""};
  1234. } else {
  1235. if ($("#wmeac-advanced-closure-dialog-tabholiday").attr("class").indexOf("active") != -1) {
  1236. WMEAC.lastGeneratedHolidays.forEach(function(e, i) {
  1237. if ($("#wmeac-advanced-closure-dialog-holidays-" + i).is(":checked")) {
  1238. var start = (new Date(e.date)).addMinutes(startTimeM);
  1239. var end = start.clone();
  1240. end.addMinutes(dD * 1440 + dH * 60 + dM);
  1241. list.push({start:WMEAC.dateToClosureStr(start), end:WMEAC.dateToClosureStr(end)});
  1242. }
  1243. });
  1244. return {list:list, error:""};
  1245. } else {
  1246. return {list:list, error:"Wrong tab active"};
  1247. }
  1248. }
  1249. }
  1250. };
  1251. WMEAC.refreshClosureList = function() {
  1252. try {
  1253. var rc = WMEAC.buildClosuresListFromRecurringUI();
  1254. if (rc.error != "") {
  1255. $("#wmeac-csv-closures-preview-content").html(rc.error);
  1256. } else {
  1257. var reason = $("#wmeac-advanced-closure-dialog-reason").val();
  1258. var direction = $("#wmeac-advanced-closure-dialog-direction").val();
  1259. var directionStr = direction == 1 ? "(A &#8594; B)" : direction == 2 ? "(B &#8594; A)" : "(&#8646;)";
  1260. var isIT = $("#wmeac-advanced-closure-dialog-ignoretraffic").is(":checked");
  1261. var existingClosures = W.selectionManager.getSelectedFeatures().reduce(function(p, c, i) {
  1262. var revSegs = W.selectionManager.getReversedSegments();
  1263. var isReversed = revSegs.hasOwnProperty(c.model.attributes.id) && revSegs[c.model.attributes.id];
  1264. var realWay = isReversed ? direction == 1 ? 2 : 1 : direction;
  1265. return p.concat(W.model.roadClosures.getObjectArray(function(e) {
  1266. return e.segID == c.model.attributes.id && (direction == 3 || e.forward && realWay == 1 || !e.forward && realWay == 2);
  1267. }));
  1268. }, []);
  1269. var mte = W.model.majorTrafficEvents.get($("#wmeac-advanced-closure-dialog-mteid").val());
  1270. $("#wmeac-csv-closures-preview-content").html("" + rc.list.length + " closure(s) to apply: <br>" + rc.list.map(function(e, i) {
  1271. var overlap = existingClosures.filter(function(c) {
  1272. return WMEAC.dateTimeOverlaps({startDate:e.start, endDate:e.end}, c);
  1273. }).map(function(c) {
  1274. var msg = (c.reason ? c.reason + " " : "") + "(" + c.segID + ")";
  1275. if (W.model.segments.objects.hasOwnProperty(c.segID) == false) {
  1276. return msg;
  1277. }
  1278. if (W.model.segments.objects[c.segID].attributes.primaryStreetID == null) {
  1279. return msg;
  1280. }
  1281. if (W.model.streets.objects.hasOwnProperty(W.model.segments.objects[c.segID].attributes.primaryStreetID) == false) {
  1282. return msg;
  1283. }
  1284. var street = W.model.streets.objects[W.model.segments.objects[c.segID].attributes.primaryStreetID];
  1285. if (!street.isEmpty) {
  1286. msg = street.name + ": " + msg;
  1287. }
  1288. return msg;
  1289. });
  1290. var mteOK = !(mte && (new Date(e.start) < new Date(mte.attributes.startDate) || new Date(e.end) > new Date(mte.attributes.endDate)));
  1291. return reason + ": " + e.start + " &#8594; " + 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-' +
  1292. i + '"></span>';
  1293. }).join("<br>"));
  1294. }
  1295. } catch (e) {
  1296. WMEAC.logError("Error while refreshing closure list: ", e);
  1297. }
  1298. };
  1299. WMEAC.refreshMTEList = function() {
  1300. var currentMTEid = $("#wmeac-advanced-closure-dialog-mteid").val();
  1301. var rangeStart = new Date($("#wmeac-advanced-closure-dialog-rangestartdate").val());
  1302. var rangeEnd = new Date($("#wmeac-advanced-closure-dialog-rangeenddate").val());
  1303. var options = [{name:"none", value:""}];
  1304. $("#wmeac-advanced-closure-dialog-mteid").empty();
  1305. if (WMEAC.isValidDate(rangeStart) && WMEAC.isValidDate(rangeEnd)) {
  1306. rangeEnd.addDays(1);
  1307. W.model.majorTrafficEvents.getObjectArray(function(mte) {
  1308. return WMEAC.dateTimeOverlaps({startDate:rangeStart, endDate:rangeEnd}, {startDate:new Date(mte.attributes.startDate), endDate:new Date(mte.attributes.endDate)});
  1309. }).forEach(function(mte) {
  1310. options.push({name:mte.attributes.names[0].value, value:mte.attributes.id});
  1311. });
  1312. }
  1313. options.forEach(function(o) {
  1314. var el = WMEAC.createElement({type:"option"});
  1315. el.setAttribute("value", o.value);
  1316. if (currentMTEid == o.value) {
  1317. el.setAttribute("selected", "");
  1318. }
  1319. el.innerHTML = o.name;
  1320. $("#wmeac-advanced-closure-dialog-mteid").append(el);
  1321. });
  1322. if (options.length > 1) {
  1323. $("#wmeac-advanced-closure-dialog-mteid").removeAttr("disabled");
  1324. } else {
  1325. $("#wmeac-advanced-closure-dialog-mteid").attr("disabled", "");
  1326. }
  1327. };
  1328. WMEAC.refreshClosureListFromSelection = function() {
  1329. try {
  1330. var currentSegClosure = $("#wmeac-advanced-closure-dialog-segclosure-list").val();
  1331. $("#wmeac-advanced-closure-dialog-segclosure-list").empty();
  1332. if (W.selectionManager.getSelectedFeatures().length != 0) {
  1333. var blackList = [];
  1334. W.model.roadClosures.getObjectArray(function(c) {
  1335. return c.segID == W.selectionManager.getSelectedFeatures()[0].model.attributes.id;
  1336. }).sort(function(a, b) {
  1337. return new Date(a.startDate) - new Date(b.startDate);
  1338. }).forEach(function(c) {
  1339. if (blackList.indexOf(c.id) != -1) {
  1340. return;
  1341. }
  1342. var direction = c.forward ? "A to B" : "B to A";
  1343. var oppositeClosure = WMEAC.getOppositeClosure(c);
  1344. if (!oppositeClosure.length == 0) {
  1345. direction = "Two way";
  1346. blackList.push(oppositeClosure[0].id);
  1347. }
  1348. var el = WMEAC.createElement({type:"option"});
  1349. el.setAttribute("value", c.id);
  1350. if (currentSegClosure == c.id) {
  1351. el.setAttribute("selected", "");
  1352. }
  1353. el.innerHTML = c.reason.trim() + " " + direction + " " + c.startDate + "&#8594;" + c.endDate;
  1354. $("#wmeac-advanced-closure-dialog-segclosure-list").append(el);
  1355. });
  1356. }
  1357. } catch (e) {
  1358. WMEAC.logError("Error while refreshing closure list from selection: ", e);
  1359. }
  1360. };
  1361. WMEAC.abbrToISO3166_1alpha2 = function(abbr) {
  1362. switch(abbr) {
  1363. case "GM":
  1364. return "DE";
  1365. break;
  1366. case "BU":
  1367. return "BG";
  1368. break;
  1369. case "EZ":
  1370. return "CZ";
  1371. break;
  1372. case "SP":
  1373. return "ES";
  1374. break;
  1375. case "UK":
  1376. return "GB";
  1377. break;
  1378. case "LO":
  1379. return "SK";
  1380. break;
  1381. default:
  1382. return abbr;
  1383. break;
  1384. }
  1385. return abbr;
  1386. };
  1387. WMEAC.getHolidays = function(options) {
  1388. var holidays = [];
  1389. var currentCountryIndex = 0;
  1390. var rangeStart = new Date(options.rangeStart);
  1391. var rangeEnd = (new Date(options.rangeEnd)).addDays(1);
  1392. var years = [];
  1393. for (y = parseInt(options.rangeStart.substring(0, 4)); y <= parseInt(options.rangeEnd.substring(0, 4)); y++) {
  1394. years.push(y);
  1395. }
  1396. var currentYearIndex = 0;
  1397. var tmp1 = function downloadNext() {
  1398. if (currentCountryIndex >= options.countries.length) {
  1399. if (options.handlerFinished) {
  1400. holidays.sort(function(a, b) {
  1401. return new Date(a.date) - new Date(b.date);
  1402. });
  1403. holidays = holidays.filter(function(h, i) {
  1404. if (i != 0) {
  1405. if (holidays[i - 1].date == h.date) {
  1406. holidays[i - 1].name += " / " + h.name;
  1407. holidays[i - 1].country += " / " + h.country;
  1408. return false;
  1409. }
  1410. }
  1411. return true;
  1412. });
  1413. options.handlerFinished(holidays);
  1414. return;
  1415. }
  1416. }
  1417. window.WMEAC_downloadHelper.add("https://holidayapi.com/v1/holidays?key=ca1a7925-0045-47da-bcf6-ee0de583e384&country=" + WMEAC.abbrToISO3166_1alpha2(options.countries[currentCountryIndex]) + "&year=" + years[currentYearIndex], function(data) {
  1418. if (data.status == "success") {
  1419. try {
  1420. nextHoliday = JSON.parse(data.data);
  1421. if (nextHoliday.hasOwnProperty("holidays")) {
  1422. for (var hd in nextHoliday.holidays) {
  1423. if (!nextHoliday.holidays.hasOwnProperty(hd)) {
  1424. continue;
  1425. }
  1426. if (nextHoliday.holidays[hd].length == 0) {
  1427. continue;
  1428. }
  1429. var name = _.pluck(nextHoliday.holidays[hd], "name").join(" / ");
  1430. var h = nextHoliday.holidays[hd][0];
  1431. var d = new Date(h.date);
  1432. if (d >= rangeStart && d < rangeEnd) {
  1433. holidays.push({date:h.date, name:name, country:h.country});
  1434. }
  1435. }
  1436. currentYearIndex++;
  1437. if (currentYearIndex < years.length) {
  1438. window.setTimeout(downloadNext);
  1439. return;
  1440. }
  1441. }
  1442. } catch (e) {
  1443. WMEAC.log("Error while getting holiday from server!", e);
  1444. WMEAC.log("data", data.data);
  1445. }
  1446. currentCountryIndex++;
  1447. currentYearIndex = 0;
  1448. window.setTimeout(downloadNext);
  1449. }
  1450. });
  1451. };
  1452. tmp1();
  1453. };
  1454. WMEAC.addClosure = function(options, successHandler, failureHandler) {
  1455. if (options && options.hasOwnProperty("segments") && options.hasOwnProperty("reason") && options.hasOwnProperty("direction") && options.hasOwnProperty("startDate") && options.hasOwnProperty("endDate") && options.hasOwnProperty("location") && options.hasOwnProperty("permanent")) {
  1456. WMEAC.log("Addinf closure: ", options);
  1457. var fail = function(e) {
  1458. return function(f) {
  1459. if (failureHandler) {
  1460. failureHandler(f);
  1461. } else {
  1462. WMEAC.log("Failed to create closure:", f);
  1463. }
  1464. };
  1465. };
  1466. var done = function(e) {
  1467. return function(f) {
  1468. if (successHandler) {
  1469. successHandler(f);
  1470. } else {
  1471. WMEAC.log("Closure successful:", f);
  1472. }
  1473. };
  1474. };
  1475. var cab = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/ClosureActionBuilder");
  1476. var sc = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/SharedClosure");
  1477. var t = {};
  1478. 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:{}};
  1479. if (options.hasOwnProperty("eventId") && options.eventId != null) {
  1480. closureDetails.eventId = options.eventId;
  1481. }
  1482. var c = new sc["default"](closureDetails);
  1483. t.actions = [cab.add(c)];
  1484. W.controller.save(t).then(done())["catch"](fail());
  1485. return true;
  1486. }
  1487. return false;
  1488. };
  1489. WMEAC.addClosureListFromSelection = function(closureList, successHandler, failureHandler, endHandler, i) {
  1490. if (i >= closureList.length) {
  1491. WMEAC.reloadClosuresLayer(function() {
  1492. if (endHandler) {
  1493. endHandler();
  1494. }
  1495. });
  1496. return;
  1497. }
  1498. var c = closureList[i];
  1499. var fail = function(e) {
  1500. return function(f) {
  1501. if (failureHandler) {
  1502. var details = [];
  1503. f.errors.forEach(function(err) {
  1504. if (err.hasOwnProperty("attributes") && err.attributes.hasOwnProperty("details")) {
  1505. details.push(err.attributes.details);
  1506. }
  1507. });
  1508. failureHandler(i, details.join(" | "));
  1509. } else {
  1510. WMEAC.log("Failed to create closure:", f);
  1511. }
  1512. WMEAC.addClosureListFromSelection(closureList, successHandler, failureHandler, endHandler, i + 1);
  1513. };
  1514. };
  1515. var done = function(e) {
  1516. return function(f) {
  1517. if (successHandler) {
  1518. successHandler(i, "OK");
  1519. } else {
  1520. WMEAC.log("Closure successful:", f);
  1521. }
  1522. WMEAC.addClosureListFromSelection(closureList, successHandler, failureHandler, endHandler, i + 1);
  1523. };
  1524. };
  1525. var cab = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/ClosureActionBuilder");
  1526. var sc = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/SharedClosure");
  1527. var t = {};
  1528. var segs = _.pluck(W.selectionManager.getSelectedFeatures(), "model");
  1529. var cityStreets = WMEAC.getCityStreetsFromSegmentSet(segs);
  1530. var closureLocation = Object.keys(cityStreets).map(function(c) {
  1531. return Object.keys(cityStreets[c]).map(function(s) {
  1532. if (s == "noStreet") {
  1533. return I18n.translations[I18n.locale].edit.address.no_street;
  1534. }
  1535. return s;
  1536. }).join(", ") + (c == "noCity" ? "" : " (" + c + ")");
  1537. }).join(" ; ");
  1538. 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()};
  1539. if (closureList[i].hasOwnProperty("eventId") && closureList[i].eventId != null) {
  1540. closureDetails.eventId = closureList[i].eventId;
  1541. }
  1542. var c = new sc["default"](closureDetails);
  1543. t.actions = [cab.add(c)];
  1544. W.controller.save(t).then(done())["catch"](fail());
  1545. };
  1546. WMEAC.addClosureFromSelection = function(options, successHandler, failureHandler) {
  1547. if (options && options.hasOwnProperty("reason") && options.hasOwnProperty("direction") && options.hasOwnProperty("startDate") && options.hasOwnProperty("endDate") && options.hasOwnProperty("location") && options.hasOwnProperty("permanent")) {
  1548. WMEAC.log("Addinf closure: ", options);
  1549. var fail = function(e) {
  1550. return function(f) {
  1551. if (failureHandler) {
  1552. failureHandler(f);
  1553. } else {
  1554. WMEAC.log("Failed to create closure:", f);
  1555. }
  1556. };
  1557. };
  1558. var done = function(e) {
  1559. return function(f) {
  1560. if (successHandler) {
  1561. successHandler(f);
  1562. } else {
  1563. WMEAC.log("Closure successful:", f);
  1564. }
  1565. };
  1566. };
  1567. var cab = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/ClosureActionBuilder");
  1568. var sc = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/SharedClosure");
  1569. var t = {};
  1570. var segs = _.pluck(W.selectionManager.getSelectedFeatures(), "model");
  1571. 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()};
  1572. if (options.hasOwnProperty("eventId") && options.eventId != null) {
  1573. closureDetails.eventId = options.eventId;
  1574. }
  1575. var c = new sc["default"](closureDetails);
  1576. t.actions = [cab.add(c)];
  1577. W.controller.save(t).then(done())["catch"](fail());
  1578. return true;
  1579. }
  1580. return false;
  1581. };
  1582. WMEAC.removeClosure = function(closures, successHandler, failureHandler) {
  1583. var fail = function(e) {
  1584. return function(f) {
  1585. if (failureHandler) {
  1586. failureHandler(f);
  1587. } else {
  1588. WMEAC.log("Failed to delete closure:", f);
  1589. }
  1590. };
  1591. };
  1592. var done = function(e) {
  1593. return function(f) {
  1594. if (successHandler) {
  1595. successHandler(f);
  1596. } else {
  1597. WMEAC.log("Closure deletion successful:", f);
  1598. }
  1599. };
  1600. };
  1601. var cab = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/ClosureActionBuilder");
  1602. var sc = WMEAC.WMEAPI.require("Waze/Modules/Closures/Models/SharedClosure");
  1603. var t = {};
  1604. var c = new sc["default"]({closures:[].concat(closures)});
  1605. t.actions = [cab["delete"](c)];
  1606. W.controller.save(t).then(done())["catch"](fail());
  1607. return true;
  1608. };
  1609. WMEAC.save = function() {
  1610. WMEAC.log("save data...");
  1611. localStorage.WMEAC = JSON.stringify({presets:WMEAC.presets});
  1612. };
  1613. WMEAC.load = function() {
  1614. try {
  1615. var saved = JSON.parse(localStorage.WMEAC);
  1616. WMEAC.presets = saved.presets;
  1617. WMEAC.log("presets", WMEAC.presets);
  1618. } catch (err) {
  1619. WMEAC.log("Error while loading data from storage: ", err);
  1620. }
  1621. };
  1622. WMEAC.parseCSV = function(csvString) {
  1623. if (csvString != null) {
  1624. var csvArray = WMEAC.CSVtoArray(csvString);
  1625. WMEAC.log("CSV as array:", csvArray);
  1626. var isValid = WMEAC.csv[0].validate(csvArray);
  1627. if (isValid.isValid) {
  1628. WMEAC.log("CSV is valid!");
  1629. var closures = WMEAC.csv[0].filter(csvArray).map(function(e, i) {
  1630. 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};
  1631. });
  1632. WMEAC.log("Closure list:", closures);
  1633. WMEAC.csvCurrentClosureList = closures;
  1634. var listUI = WMEAC.getId("wmeac-csv-closures-list-elts");
  1635. WMEAC.removeChildElements(listUI);
  1636. closures.forEach(function(c) {
  1637. c.UI = WMEAC.buildInlineClosureUI(c.closure, c.action);
  1638. listUI.appendChild(c.UI);
  1639. });
  1640. WMEAC.csvShowList(true);
  1641. WMEAC.csvAddLog("CSV parse successful\n");
  1642. return true;
  1643. } else {
  1644. WMEAC.log("CSV is NOT valid!:" + isValid.feedBack + "\n");
  1645. WMEAC.csvAddLog(isValid.feedBack + "\n");
  1646. WMEAC.csvShowList(false);
  1647. WMEAC.csvCurrentClosureList = null;
  1648. return false;
  1649. }
  1650. return false;
  1651. }
  1652. return false;
  1653. };
  1654. WMEAC.CSVFileChanged = function(evt) {
  1655. var files = evt.target.files;
  1656. for (var i = 0, f; f = files[i]; i++) {
  1657. var reader = new FileReader;
  1658. reader.onload = function(theFile) {
  1659. return function(e) {
  1660. WMEAC.log("import CSV file read");
  1661. WMEAC.csvClearLog();
  1662. if (WMEAC.parseCSV(e.target.result)) {
  1663. WMEAC.csvCurrentBatchClosureList = WMEAC.csvCurrentClosureList.slice();
  1664. }
  1665. };
  1666. }(f);
  1667. reader.readAsText(f);
  1668. }
  1669. this.value = null;
  1670. WMEAC.getId("wmeac-csv-closures-controls-check").checked = false;
  1671. };
  1672. WMEAC.ClassCSV = function(options) {
  1673. this.isValid = false;
  1674. if (options.hasOwnProperty("version")) {
  1675. this.version = options.version;
  1676. } else {
  1677. return;
  1678. }
  1679. if (options.hasOwnProperty("regexpValidation")) {
  1680. this.regexpValidation = options.regexpValidation;
  1681. } else {
  1682. return;
  1683. }
  1684. this.isValid = true;
  1685. this.validate = function(data) {
  1686. var regexps = this.regexpValidation;
  1687. var feedBack = "";
  1688. this.filter(data).forEach(function(line, l) {
  1689. var isLineValid = line.reduce(function(stillValid, cell, i) {
  1690. var isCellValid = cell.match(regexps[i]) != null;
  1691. if (!isCellValid) {
  1692. feedBack += "Error while parsing line " + l + " cell " + i + ': "' + cell + '" in line ' + line.join(",");
  1693. }
  1694. return stillValid && isCellValid;
  1695. }, true);
  1696. }, this);
  1697. return {isValid:feedBack == "", feedBack:feedBack};
  1698. };
  1699. this.filter = function(data) {
  1700. return data.filter(function(line) {
  1701. return line.length >= 1 && ["add", "remove"].indexOf(line[0]) != -1;
  1702. });
  1703. };
  1704. };
  1705. WMEAC.csv.push(new WMEAC.ClassCSV({version:1, regexpValidation:[/.*/, /.*/, /^\d{4}-\d{2}-\d{2} \d{2}:\d{2}$/, /^\d{4}-\d{2}-\d{2} \d{2}:\d{2}$/, /(^A to B$)|(^B to A$)|(^TWO WAY$)/, /(Yes)|(No)/, /^(\d+(;|$))+/, /(lon=(-?\d+\.?\d*)&lat=(-?\d+\.?\d*))|(lat=(-?\d+\.?\d*)&lon=(-?\d+\.?\d*))/, /^\d$/, /(^$)|(^-?\d+\.-?\d+\.-?\d+$)/]}));
  1706. WMEAC.buildInlineClosureUI = function(closure, action) {
  1707. var liElt = WMEAC.createElement({type:"li", className:"wmeac-csv-closures-list-" + action});
  1708. liElt.setAttribute("closureID", closure.id);
  1709. 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">' +
  1710. (closure.direction == "A to B" ? "A&#8594;B" : closure.direction == "B to A" ? "B&#8594;A" : "A&#8596;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;">' +
  1711. (action == "add" ? "Ready to apply" : action == "remove" ? "Ready to remove" : "") + "</div>";
  1712. liElt.children[5].children[0].addEventListener("click", function(e) {
  1713. WMEAC.csvClearLog();
  1714. var cid = parseInt(e.target.parentNode.parentNode.parentNode.getAttribute("closureID"));
  1715. var closure = WMEAC.csvCurrentClosureList.find(function(c) {
  1716. return c.closure.id == cid;
  1717. });
  1718. WMEAC.log("Closure to target:", closure);
  1719. var xy = OpenLayers.Layer.SphericalMercator.forwardMercator(closure.closure.lonlat.lon, closure.closure.lonlat.lat);
  1720. W.map.setCenter(xy, closure.closure.zoom);
  1721. var tmp3 = function selectSegments() {
  1722. WMEAC.log("Now select segments...");
  1723. var segs = WMEAC.segmentsIDsToSegments(closure.closure.segIDs);
  1724. if (segs.length != closure.closure.segIDs.length) {
  1725. if (segs.length == 0) {
  1726. WMEAC.csvAddLog("No segment found: " + closure.closure.comment + "(" + closure.closure.reason + ")\n");
  1727. WMEAC.setCSVMiniLog(closure, "Selection failed: no segment found", 3);
  1728. } else {
  1729. WMEAC.csvAddLog("Partial selection (" + segs.length + "/" + closure.closure.segIDs.length + "): " + closure.closure.comment + "(" + closure.closure.reason + ")\n");
  1730. WMEAC.setCSVMiniLog(closure, "Partial selection: " + segs.length + "/" + closure.closure.segIDs.length, 2);
  1731. }
  1732. alert("Warning: missing segments.\nFound " + segs.length + "/" + closure.closure.segIDs.length + " segment(s)");
  1733. } else {
  1734. WMEAC.csvAddLog("Selection ok (" + segs.length + "): " + closure.closure.comment + "(" + closure.closure.reason + ")\n");
  1735. WMEAC.setCSVMiniLog(closure, "Selection OK: " + segs.length, 1);
  1736. }
  1737. if (segs.length != 0) {
  1738. W.selectionManager.setSelectedModels(segs);
  1739. var tmp = function selectionReady() {
  1740. if (W.selectionManager.getSelectedFeatures().length == 0) {
  1741. window.setTimeout(selectionReady, 500);
  1742. } else {
  1743. $('a[href="#segment-edit-closures"]').click();
  1744. }
  1745. };
  1746. window.setTimeout(tmp, 500);
  1747. }
  1748. };
  1749. var tmp2 = function readyToSelect() {
  1750. WMEAC.log("Test if ready to select...");
  1751. if (WMEAC.pendingOps == true) {
  1752. WMEAC.log("Not yet. Waiting for WME...");
  1753. window.setTimeout(readyToSelect, 500);
  1754. } else {
  1755. tmp3();
  1756. }
  1757. };
  1758. var tmp1 = function mapMovedEnd() {
  1759. WMEAC.log("Test if roads are reloaded...");
  1760. if (WMEAC.pendingOps == true) {
  1761. WMEAC.log("Not yet. Waiting for WME...");
  1762. window.setTimeout(mapMovedEnd, 500);
  1763. } else {
  1764. WMEAC.reloadRoadLayer();
  1765. tmp2();
  1766. }
  1767. };
  1768. window.setTimeout(tmp1, 500);
  1769. e.preventDefault();
  1770. });
  1771. liElt.children[6].children[0].addEventListener("click", function(e) {
  1772. WMEAC.csvClearLog();
  1773. var liElt = e.target.parentNode.parentNode.parentNode;
  1774. var cid = parseInt(liElt.getAttribute("closureID"));
  1775. var closure = WMEAC.csvCurrentClosureList.find(function(c) {
  1776. return c.closure.id == cid;
  1777. });
  1778. WMEAC.log("Closure to apply:", closure);
  1779. WMEAC.csvApplyClosure(closure, null);
  1780. });
  1781. return liElt;
  1782. };
  1783. WMEAC.csvApplyClosure = function(closure, handler) {
  1784. var xy = OpenLayers.Layer.SphericalMercator.forwardMercator(closure.closure.lonlat.lon, closure.closure.lonlat.lat);
  1785. W.map.setCenter(xy, closure.closure.zoom);
  1786. function applySuccess(evt) {
  1787. WMEAC.csvAddLog("Closure OK: " + closure.closure.comment + "(" + closure.closure.reason + ")\n");
  1788. closure.UI.className = "wmeac-csv-closures-list-done";
  1789. WMEAC.setCSVMiniLog(closure, "OK", 1);
  1790. handler && handler(true);
  1791. }
  1792. function applyFailure(evt) {
  1793. var details = "";
  1794. evt.errors.forEach(function(err) {
  1795. if (err.hasOwnProperty("attributes") && err.attributes.hasOwnProperty("details")) {
  1796. details += err.attributes.details + "\n";
  1797. }
  1798. });
  1799. WMEAC.csvAddLog("Closure KO: " + closure.closure.comment + " (" + closure.closure.reason + ")\n" + details + "\n");
  1800. WMEAC.setCSVMiniLog(closure, "KO: " + details, 3);
  1801. closure.UI.className = "wmeac-csv-closures-list-failed";
  1802. handler && handler(false);
  1803. }
  1804. var tmp3 = function applyClosure() {
  1805. WMEAC.log("Now apply closure...");
  1806. if (closure.action == "add") {
  1807. closure.closure.applyInWME(applySuccess, applyFailure);
  1808. } else {
  1809. if (closure.action == "remove") {
  1810. closure.closure.removeInWME(applySuccess, applyFailure);
  1811. }
  1812. }
  1813. };
  1814. var tmp2 = function readyToApply() {
  1815. WMEAC.log("Test if ready to apply...");
  1816. if (WMEAC.pendingOps == true) {
  1817. WMEAC.log("Not yet. Waiting for WME...");
  1818. window.setTimeout(readyToApply, 500);
  1819. } else {
  1820. tmp3();
  1821. }
  1822. };
  1823. var tmp1 = function mapMovedEnd() {
  1824. WMEAC.log("Test if roads are reloaded...");
  1825. if (WMEAC.pendingOps == true) {
  1826. WMEAC.log("Not yet. Waiting for WME...");
  1827. window.setTimeout(mapMovedEnd, 500);
  1828. } else {
  1829. WMEAC.reloadRoadLayer();
  1830. tmp2();
  1831. }
  1832. };
  1833. window.setTimeout(tmp1, 1500);
  1834. };
  1835. WMEAC.csvAddLog = function(text) {
  1836. var divLog = WMEAC.getId("wmeac-csv-closures-log");
  1837. divLog.innerHTML += text.replace(/\n/g, "<br>");
  1838. };
  1839. WMEAC.csvClearLog = function() {
  1840. var divLog = WMEAC.getId("wmeac-csv-closures-log");
  1841. divLog.innerHTML = "";
  1842. };
  1843. WMEAC.csvShowList = function(show) {
  1844. var divList = WMEAC.getId("wmeac-csv-closures");
  1845. divList.style.display = show ? "block" : "none";
  1846. };
  1847. WMEAC.csvCheckAllSegments = function(i) {
  1848. if (i == -1) {
  1849. WMEAC.pb.update(0);
  1850. WMEAC.pb.show(true);
  1851. window.setTimeout(function() {
  1852. WMEAC.csvCheckAllSegments(0);
  1853. });
  1854. return;
  1855. }
  1856. var continueSegmentCheck = function() {
  1857. window.setTimeout(function() {
  1858. WMEAC.csvCheckAllSegments(i + 1);
  1859. });
  1860. };
  1861. if (i < WMEAC.csvCurrentBatchClosureList.length) {
  1862. var currentClosure = WMEAC.csvCurrentBatchClosureList[i];
  1863. WMEAC.pb.update(i * 100 / WMEAC.csvCurrentBatchClosureList.length);
  1864. WMEAC.pb.info("Scanning segments. please wait...");
  1865. var c = OpenLayers.Layer.SphericalMercator.forwardMercator(currentClosure.closure.lonlat.lon, currentClosure.closure.lonlat.lat);
  1866. var b = W.map.calculateBounds();
  1867. var zoomRatio = Math.pow(2, W.map.zoom - currentClosure.closure.zoom);
  1868. var w = b.getWidth() * 1.7 * zoomRatio;
  1869. var h = b.getHeight() * 1.7 * zoomRatio;
  1870. var tileBounds = new OpenLayers.Bounds(c.lon - w / 2, c.lat - h / 2, c.lon + w / 2, c.lat + h / 2);
  1871. tileBounds = tileBounds.transform(W.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")).toBBOX();
  1872. var roadTypes = W.model.repos.segments.zoomToRoadType[currentClosure.closure.zoom] == -1 ? _.range(1, 22) : W.model.repos.segments.zoomToRoadType[currentClosure.closure.zoom];
  1873. var WFVS = WMEAC.WMEAPI.require("Waze/Feature/Vector/Segment");
  1874. var aseg = new WFVS["default"];
  1875. var req = new XMLHttpRequest;
  1876. 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);
  1877. req.onreadystatechange = function(e) {
  1878. if (req.readyState == 4) {
  1879. if (req.status == 200) {
  1880. try {
  1881. var data = JSON.parse(req.responseText);
  1882. WMEAC.log("data", data);
  1883. var existingSegs = currentClosure.closure.segIDs.filter(function(sid) {
  1884. return data.segments.objects.find(function(seg) {
  1885. return sid == seg.id;
  1886. }) != null;
  1887. });
  1888. var editableClosuresSegs = currentClosure.closure.segIDs.filter(function(sid) {
  1889. return data.segments.objects.find(function(seg) {
  1890. return sid == seg.id && seg.permissions & aseg.PERMISSIONS.EDIT_CLOSURES;
  1891. }) != null;
  1892. });
  1893. var overlaps = [];
  1894. var existingClosures = existingSegs.forEach(function(sid) {
  1895. var cl = data.roadClosures.objects.filter(function(c) {
  1896. return c.segID == sid;
  1897. });
  1898. console.log("cl", cl);
  1899. cl.forEach(function(c) {
  1900. var forwardMustBe = currentClosure.closure.direction == "A to B" ? true : currentClosure.closure.direction == "B to A" ? false : null;
  1901. console.log("forwardMustBe", forwardMustBe);
  1902. console.log("dateTimeOverlaps", currentClosure.closure);
  1903. console.log("dateTimeOverlaps", c);
  1904. if (WMEAC.dateTimeOverlaps(currentClosure.closure, c)) {
  1905. if (forwardMustBe == null || forwardMustBe == c.forward) {
  1906. var segment = data.segments.objects.find(function(seg) {
  1907. return seg.id == sid;
  1908. });
  1909. var streetName = null;
  1910. if (segment && segment.primaryStreetID != null) {
  1911. var street = data.streets.objects.find(function(st) {
  1912. return st.id == segment.primaryStreetID;
  1913. });
  1914. if (street && street.name != null) {
  1915. streetName = street.name;
  1916. }
  1917. }
  1918. overlaps.push("Overlap with " + c.reason + (streetName != null ? " :" + streetName : "") + " (" + sid + ")");
  1919. }
  1920. }
  1921. });
  1922. });
  1923. if (existingSegs.length == currentClosure.closure.segIDs.length && editableClosuresSegs.length == currentClosure.closure.segIDs.length && overlaps.length == 0) {
  1924. WMEAC.csvAddLog("Seg check OK: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\n" + existingSegs.length + " editable seg(s) found\n");
  1925. WMEAC.setCSVMiniLog(currentClosure, "segs OK: " + existingSegs.length + " editable seg(s) found", 1);
  1926. } else {
  1927. if (existingSegs.length == currentClosure.closure.segIDs.length && editableClosuresSegs.length == currentClosure.closure.segIDs.length && overlaps.length != 0) {
  1928. WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\nOverlap detected on existing closures:\n" + overlaps.join("\n") + "\n");
  1929. WMEAC.setCSVMiniLog(currentClosure, "segs KO: " + overlaps.length + " overlap(s) detected", 2);
  1930. } else {
  1931. if (existingSegs.length == currentClosure.closure.segIDs.length && editableClosuresSegs.length != currentClosure.closure.segIDs.length) {
  1932. 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");
  1933. 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);
  1934. } else {
  1935. WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\n" + existingSegs.length + "/" + currentClosure.closure.segIDs.length + " seg(s) found\n");
  1936. WMEAC.setCSVMiniLog(currentClosure, "segs KO: " + existingSegs.length + "/" + currentClosure.closure.segIDs.length + " seg(s) found", 3);
  1937. }
  1938. }
  1939. }
  1940. } catch (err) {
  1941. WMEAC.log("Failed to parse Waze's server response: " + req.responseText);
  1942. WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\nFailed to parse response from Waze\n");
  1943. WMEAC.setCSVMiniLog(currentClosure, "segs KO: Failed to parse response from Waze", 3);
  1944. }
  1945. } else {
  1946. WMEAC.log("Error on road tile: " + e.target.status);
  1947. WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\nCommunication failed with Waze\n");
  1948. WMEAC.setCSVMiniLog(currentClosure, "segs KO: Communication failed with Waze", 3);
  1949. }
  1950. continueSegmentCheck();
  1951. }
  1952. };
  1953. req.onError = function(e) {
  1954. WMEAC.log("Error on road tile: " + e.target.status);
  1955. WMEAC.csvAddLog("Seg check KO: " + currentClosure.closure.comment + " (" + currentClosure.closure.reason + "):\nCommunication failed with Waze's server\n");
  1956. WMEAC.setCSVMiniLog(currentClosure, "segs KO: Communication failed with Waze", 3);
  1957. continueSegmentCheck();
  1958. };
  1959. req.send(null);
  1960. } else {
  1961. WMEAC.pb.show(false);
  1962. }
  1963. };
  1964. WMEAC.setCSVMiniLog = function(closure, text, level) {
  1965. var c = null;
  1966. if (closure.hasOwnProperty("UI")) {
  1967. c = closure;
  1968. } else {
  1969. c = WMEAC.csvCurrentClosureList.find(function(e) {
  1970. return e.closure.id == closure.id;
  1971. });
  1972. }
  1973. if (c != null) {
  1974. c.UI.children[7].innerHTML = text;
  1975. var colors = ["#000000", "#54C600", "#FFA000", "#FF0000"];
  1976. if (arguments.length == 3) {
  1977. c.UI.children[7].style.color = colors[level];
  1978. } else {
  1979. c.UI.children[7].style.color = colors[0];
  1980. }
  1981. }
  1982. };
  1983. WMEAC.CSVCheckAll = function(check) {
  1984. WMEAC.csvCurrentClosureList.forEach(function(e) {
  1985. e.UI.children[0].children[0].checked = check;
  1986. });
  1987. };
  1988. WMEAC.CSVApplyChecked = function() {
  1989. WMEAC.csvCurrentBatchClosureList = WMEAC.csvCurrentClosureList.filter(function(e) {
  1990. return e.UI.children[0].children[0].checked;
  1991. });
  1992. WMEAC.csvClearLog();
  1993. if (WMEAC.csvCurrentBatchClosureList.length == 0) {
  1994. WMEAC.csvAddLog("No closure checked!\n");
  1995. } else {
  1996. WMEAC.showClosuresLayer(true);
  1997. WMEAC.pb.update(0);
  1998. WMEAC.pb.info("Applying closures. please wait...");
  1999. WMEAC.pb.show(true);
  2000. WMEAC.csvAddLog("Start to apply selected closures\n");
  2001. window.setTimeout(function() {
  2002. WMEAC.CSVBatchApply(0);
  2003. });
  2004. }
  2005. };
  2006. WMEAC.CSVBatchApply = function(i) {
  2007. WMEAC.pb.update(i * 100 / WMEAC.csvCurrentBatchClosureList.length);
  2008. if (i < WMEAC.csvCurrentBatchClosureList.length) {
  2009. if (WMEAC.csvCurrentBatchClosureList[i].action != "add" && WMEAC.csvCurrentBatchClosureList[i].action != "remove") {
  2010. WMEAC.csvAddLog("Closure KO: " + WMEAC.csvCurrentBatchClosureList[i].closure.comment + " (" + WMEAC.csvCurrentBatchClosureList[i].closure.reason + "): action " + WMEAC.csvCurrentBatchClosureList[i].action + " not supported yet\n");
  2011. WMEAC.setCSVMiniLog(WMEAC.csvCurrentBatchClosureList[i], "KO: action " + WMEAC.csvCurrentBatchClosureList[i].action + " not supported yet", 2);
  2012. WMEAC.CSVBatchApply(i + 1);
  2013. } else {
  2014. WMEAC.csvApplyClosure(WMEAC.csvCurrentBatchClosureList[i], function(success) {
  2015. if (success) {
  2016. WMEAC.csvAddLog("Closure OK: " + WMEAC.csvCurrentBatchClosureList[i].closure.comment + " (" + WMEAC.csvCurrentBatchClosureList[i].closure.reason + ")\n");
  2017. } else {
  2018. WMEAC.csvAddLog("Closure KO: " + WMEAC.csvCurrentBatchClosureList[i].closure.comment + " (" + WMEAC.csvCurrentBatchClosureList[i].closure.reason + ")\n");
  2019. }
  2020. WMEAC.CSVBatchApply(i + 1);
  2021. });
  2022. }
  2023. } else {
  2024. WMEAC.csvAddLog("Apply selected closures ended\n");
  2025. WMEAC.reloadClosuresLayer();
  2026. WMEAC.pb.show(false);
  2027. }
  2028. };
  2029. WMEAC.CSVCheckSegsChecked = function() {
  2030. WMEAC.csvClearLog();
  2031. WMEAC.csvCurrentBatchClosureList = WMEAC.csvCurrentClosureList.filter(function(e) {
  2032. return e.UI.children[0].children[0].checked;
  2033. });
  2034. if (WMEAC.csvCurrentBatchClosureList.length == 0) {
  2035. WMEAC.csvAddLog("No closure checked!\n");
  2036. } else {
  2037. WMEAC.csvCheckAllSegments(-1);
  2038. }
  2039. };
  2040. WMEAC.refreshHighlight = function() {
  2041. try {
  2042. var l = W.map.getLayersBy("uniqueName", "closures");
  2043. if (l.length == 1) {
  2044. l = l[0];
  2045. }
  2046. for (var m in l.markers) {
  2047. if (!l.markers.hasOwnProperty(m)) {
  2048. continue;
  2049. }
  2050. var marker = l.markers[m];
  2051. if (marker.model.reason && marker.model.reason.length >= 1 && marker.model.reason.charCodeAt(marker.model.reason.length - 1) == 160) {
  2052. marker.icon.$div.addClass("wmeac-hl");
  2053. }
  2054. }
  2055. } catch (e) {
  2056. WMEAC.log("Highlight error: ", e);
  2057. }
  2058. };
  2059. if (typeof window.require != "undefined" && typeof window.define == "undefined") {
  2060. WMEAC.WMEAPI = {require:window.require};
  2061. WMEAC.bootstrapAC();
  2062. } else {
  2063. var WMEAPI = {};
  2064. WMEAPI.scripts = document.getElementsByTagName("script");
  2065. WMEAPI.url = null;
  2066. for (i = 0; i < WMEAPI.scripts.length; i++) {
  2067. if (WMEAPI.scripts[i].src.indexOf("/assets-editor/js/app") != -1) {
  2068. WMEAPI.url = WMEAPI.scripts[i].src;
  2069. break;
  2070. }
  2071. }
  2072. if (WMEAPI.url == null) {
  2073. throw new Error("WME require patch: can't detect WME main JS");
  2074. }
  2075. WMEAPI.require = function(e) {
  2076. if (WMEAPI.require.define.modules.hasOwnProperty(e)) {
  2077. return WMEAPI.require.define.modules[e];
  2078. } else {
  2079. console.error("Require failed on " + e, WMEAPI.require.define.modules);
  2080. }
  2081. return null;
  2082. };
  2083. WMEAPI.require.define = function(m) {
  2084. if (WMEAPI.require.define.hasOwnProperty("modules") == false) {
  2085. WMEAPI.require.define.modules = {};
  2086. }
  2087. for (var p in m) {
  2088. WMEAPI.require.define.modules[p] = m[p];
  2089. }
  2090. };
  2091. WMEAPI.tmp = window.webpackJsonp;
  2092. WMEAPI.t = function(n) {
  2093. if (WMEAPI.s[n]) {
  2094. return WMEAPI.s[n].exports;
  2095. }
  2096. var r = WMEAPI.s[n] = {exports:{}, id:n, loaded:!1};
  2097. return WMEAPI.e[n].call(r.exports, r, r.exports, WMEAPI.t), r.loaded = !0, r.exports;
  2098. };
  2099. WMEAPI.e = [];
  2100. window.webpackJsonp = function(a, i) {
  2101. var api = {};
  2102. for (var o, d, u = 0, l = []; u < a.length; u++) {
  2103. d = a[u], WMEAPI.r[d] && l.push.apply(l, WMEAPI.r[d]), WMEAPI.r[d] = 0;
  2104. }
  2105. var unknownCount = 0;
  2106. var classname, funcStr;
  2107. for (o in i) {
  2108. WMEAPI.e[o] = i[o];
  2109. funcStr = i[o].toString();
  2110. classname = funcStr.match(/CLASS_NAME:"([^"]*)"/);
  2111. if (classname) {
  2112. api[classname[1].replace(/\./g, "/").replace(/^W\//, "Waze/")] = {index:o, func:WMEAPI.e[o]};
  2113. } else {
  2114. api["Waze/Unknown/" + unknownCount] = {index:o, func:WMEAPI.e[o]};
  2115. unknownCount++;
  2116. }
  2117. }
  2118. for (; l.length;) {
  2119. l.shift().call(null, WMEAPI.t);
  2120. }
  2121. WMEAPI.s[0] = 0;
  2122. var module = {};
  2123. var apiFuncName;
  2124. unknownCount = 0;
  2125. for (o in i) {
  2126. funcStr = i[o].toString();
  2127. classname = funcStr.match(/CLASS_NAME:"([^"]*)"/);
  2128. if (classname) {
  2129. module = {};
  2130. apiFuncName = classname[1].replace(/\./g, "/").replace(/^W\//, "Waze/");
  2131. module[apiFuncName] = WMEAPI.t(api[apiFuncName].index);
  2132. WMEAPI.require.define(module);
  2133. } else {
  2134. var matches = funcStr.match(/SEGMENT:"segment",/);
  2135. if (matches) {
  2136. module = {};
  2137. apiFuncName = "Waze/Model/ObjectType";
  2138. module[apiFuncName] = WMEAPI.t(api["Waze/Unknown/" + unknownCount].index);
  2139. WMEAPI.require.define(module);
  2140. } else {
  2141. if (matches = funcStr.match(/TWO_WAY:/)) {
  2142. module = {};
  2143. apiFuncName = "Waze/Modules/Closures/Models/SharedClosure";
  2144. module[apiFuncName] = WMEAPI.t(api["Waze/Unknown/" + unknownCount].index);
  2145. WMEAPI.require.define(module);
  2146. } else {
  2147. if (matches = funcStr.match(/nextClosureID:/)) {
  2148. module = {};
  2149. apiFuncName = "Waze/Modules/Closures/Models/ClosureActionBuilder";
  2150. module[apiFuncName] = WMEAPI.t(api["Waze/Unknown/" + unknownCount].index);
  2151. WMEAPI.require.define(module);
  2152. }
  2153. }
  2154. }
  2155. unknownCount++;
  2156. }
  2157. }
  2158. window.webpackJsonp = WMEAPI.tmp;
  2159. WMEAC.WMEAPI = WMEAPI;
  2160. setTimeout(WMEAC.bootstrapAC);
  2161. };
  2162. WMEAPI.s = {};
  2163. WMEAPI.r = {0:0};
  2164. WMEAPI.WMEHACK_Injected_script = document.createElement("script");
  2165. WMEAPI.WMEHACK_Injected_script.setAttribute("type", "application/javascript");
  2166. WMEAPI.WMEHACK_Injected_script.src = WMEAPI.url;
  2167. document.body.appendChild(WMEAPI.WMEHACK_Injected_script);
  2168. }
  2169. WMEAC.log("Ready");
  2170. }
  2171. var WMEAC_Injected_script = document.createElement("script");
  2172. WMEAC_Injected_script.textContent = "" + WMEAC_Injected.toString() + " \n" + "WMEAC_Injected();";
  2173. WMEAC_Injected_script.setAttribute("type", "application/javascript");
  2174. document.body.appendChild(WMEAC_Injected_script);