WME DOT Advisories

Overlay DOT Advisories on the WME Map Object

当前为 2020-10-29 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name WME DOT Advisories
  3. // @namespace https://greasyfork.org/en/users/668704-phuz
  4. // @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
  5. // @version 1.09
  6. // @description Overlay DOT Advisories on the WME Map Object
  7. // @author phuz
  8. // @include /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
  9. // @require http://cdnjs.cloudflare.com/ajax/libs/jquery/1.9.1/jquery.min.js
  10. // @require https://cdnjs.cloudflare.com/ajax/libs/tablesort/5.2.1/tablesort.min.js
  11. // @require https://cdnjs.cloudflare.com/ajax/libs/tablesort/5.2.1/sorts/tablesort.number.min.js
  12. // @require https://cdnjs.cloudflare.com/ajax/libs/tablesort/5.2.1/sorts/tablesort.date.min.js
  13. // @require https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.1/moment.min.js
  14. // @grant GM_xmlhttpRequest
  15. // @grant GM_info
  16. // @grant GM_fetch
  17. // @grant GM_addStyle
  18. // @require https://cdn.jsdelivr.net/npm/hls.js@latest
  19. // @connect 511pa.com
  20. // @connect deldot.gov
  21. // @connect 511ny.org
  22. // @connect 511nj.org
  23. // @connect rehostjson.phuz.repl.co
  24. /* global OpenLayers */
  25. /* global W */
  26. /* global WazeWrap */
  27. /* global $ */
  28. /* global I18n */
  29. /* global _ */
  30. // ==/UserScript==
  31.  
  32. let DEDOTLayer;
  33. let NJDOTLayer;
  34. let NYDOTLayer;
  35. let PADOTLayer;
  36. let WADOTLayer;
  37. var settings;
  38. var newZIndex;
  39. const DEIconC = '';
  40. const DEIconSchRestriction = '';
  41. const DEIconSchClosure = '';
  42. const Incident = '';
  43. const Roadwork = '';
  44. const PLIcon = '';
  45. const reportIcon = '';
  46. const DEAdvURL = 'https://tmc.deldot.gov/json/advisory.json';
  47. const DESchURL = 'https://deldot.gov/json/str.json';
  48. const NJURLList = 'https://511nj.org/API/client/Map/getEventData';
  49. const NJURLDetail = 'https://511nj.org/API/client/Map/getEventPopupData?EventId=';
  50. const NYURL = 'https://rehostjson.phuz.repl.co/NY';
  51. const PAURL = 'https://rehostjson.phuz.repl.co/PA';
  52. const WAURL = 'https://rehostjson.phuz.repl.co/WA';
  53. const NotNY = ['Pennsylvania Statewide', 'New Jersey Statewide', 'Connecticut Statewide'];
  54. const NJConstruction = ['Construction', 'ScheduledConstruction'];
  55.  
  56. (function() {
  57. 'use strict';
  58. //Bootstrap
  59. function bootstrap(tries = 1) {
  60. if (W && W.loginManager && W.map && W.loginManager.user && W.model
  61. && W.model.states && W.model.states.getObjectArray().length && WazeWrap && WazeWrap.Ready) {
  62. console.log("WME DOT Advisories Loaded!");
  63. init();
  64. if (!OpenLayers.Icon) {
  65. installIcon();
  66. }
  67.  
  68. } else if (tries < 1000) {
  69. setTimeout(function () {bootstrap(++tries);}, 200);
  70. }
  71. }
  72. //Build the Tab and Settings Division
  73. function init()
  74. {
  75. var $section = $("<div>");
  76. $section.html([
  77. '<div id="chkEnables">',
  78. '<a href="https://www.waze.com/forum/viewtopic.php?f=819&t=308141" target="_blank">WME DOT Advisories</a> v' + GM_info.script.version + '<br>',
  79. '* The WME Refresh Button will update reports.',
  80. '<table border=1 style="text-align:center;width:100%;padding:10px;">',
  81. '<tr><td colspan=2 style="text-align:center"><b>Enable</b></td><td style="text-align"><b>State</b></td><td width=30><b>Rpt</b></td></tr>',
  82. '<tr><td colspan=2 align=center><input type="checkbox" id="chkDEDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>DE</td><td><div class=DOTreport data-report="report" id="DOTDEPopup"><img src=' + reportIcon + '></div></td></tr>',
  83. '<tr><td colspan=2 align=center><input type="checkbox" id="chkNJDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>NJ</td><td><div class=DOTreport data-report="report" id="DOTNJPopup"><img src=' + reportIcon + '></div></td></tr>',
  84. '<tr><td colspan=2 align=center><input type="checkbox" id="chkNYDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>NY</td><td><div class=DOTreport data-report="report" id="DOTNYPopup"><img src=' + reportIcon + '></div></td></tr>',
  85. '<tr><td colspan=2 align=center><input type="checkbox" id="chkPADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>PA</td><td><div class=DOTreport data-report="report" id="DOTPAPopup"><img src=' + reportIcon + '></div></td></tr>',
  86. '<tr><td colspan=2 align=center><input type="checkbox" id="chkWADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>WA</td><td><div class=DOTreport data-report="report" id="DOTWAPopup"><img src=' + reportIcon + '></div></td></tr>',
  87. '</table>',
  88. '</div></div>'
  89. ].join(' '));
  90. new WazeWrap.Interface.Tab('DOT Advisories', $section.html(), initializeSettings);
  91. }
  92. getFeed("https://rehostjson.phuz.repl.co/public/CSS", function(result) {
  93. GM_addStyle(result.responseText);
  94. })
  95. //Build the State Layers
  96. function buildDOTAdvLayers(state) {
  97. eval(state.substring(0,2) + 'DOTLayer = new OpenLayers.Layer.Markers(' + state.substring(0,2) + 'DOTLayer)');
  98. eval('W.map.addLayer(' + state.substring(0,2) + 'DOTLayer)');
  99. eval(state.substring(0,2) + "DOTLayer.setZIndex(" + newZIndex + ")");
  100. }
  101. function getFeed(url,callback) {
  102. GM_xmlhttpRequest({
  103. method: "GET",
  104. url: url,
  105. onload: function(response) {
  106. var result = response;
  107. callback(result);
  108. }
  109. });
  110. }
  111. function getDEDOT() {
  112. getDEDOTAdv();
  113. }
  114. //Get the DE Advisory JSON Feed
  115. function getDEDOTAdv(type) {
  116. getFeed(DEAdvURL, function(result) {
  117. var resultObj = JSON.parse(result.responseText).advisories;
  118. var i=0;
  119. while (i<resultObj.length) {
  120. if (type == "report") {
  121. let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
  122. var row = table.insertRow(-1);
  123. var cell1 = row.insertCell(0);
  124. var cell2 = row.insertCell(1);
  125. var cell3 = row.insertCell(2);
  126. var cell4 = row.insertCell(3);
  127. cell1.innerHTML = '<div class="gotoPL" data-lat="' + resultObj[i].where.lat + '" data-lon="' + resultObj[i].where.lon + '"><img src=' + PLIcon + '></div>';
  128. cell2.innerHTML = resultObj[i].where.location;
  129. cell3.innerHTML = resultObj[i].where.county.name;
  130. cell4.innerHTML = moment(new Date(resultObj[i].timestamp)).format('YYYY/MM/DD HH:mm');
  131. } else {
  132. drawMarkers("DEAdv",resultObj[i].id,resultObj[i].type.name,resultObj[i].where.lon,resultObj[i].where.lat,"DEIconC",resultObj[i].where.location,resultObj[i].timestamp,resultObj[i].published.linkbackUrl);
  133. }
  134. i++;
  135. }
  136. getDEDOTSch(type);
  137. })
  138. }
  139. //Get the DE Schedule JSON Feed
  140. function getDEDOTSch(type) {
  141. getFeed(DESchURL, function(result) {
  142. var resultObj = JSON.parse(result.responseText);
  143. var i=0;
  144. while (i<resultObj.length) {
  145. if (resultObj[i].str.impactType == "Closure") {
  146. if (type == "report") {
  147. let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
  148. var row = table.insertRow();
  149. var cell1 = row.insertCell(0);
  150. var cell2 = row.insertCell(1);
  151. var cell3 = row.insertCell(2);
  152. var cell4 = row.insertCell(3);
  153. cell1.innerHTML = '<div class="gotoPL" data-lat="' + resultObj[i].str.latitude + '" data-lon="' + resultObj[i].str.longitude + '"><img src=' + PLIcon + '></div>';
  154. cell2.innerHTML = resultObj[i].str.title;
  155. cell3.innerHTML = resultObj[i].str.county;
  156. cell4.innerHTML = moment(new Date(resultObj[i].str.actualStartDate)).format('YYYY/MM/DD HH:mm');
  157. } else {
  158. drawMarkers("DESch",resultObj[i].str.strId,resultObj[i].str.title,resultObj[i].str.longitude,resultObj[i].str.latitude,"DEIconSch" + resultObj[i].str.impactType,resultObj[i].str.construction,resultObj[i].str.startDate,resultObj[i].str.releaseId);
  159. }
  160. }
  161. i++;
  162. }
  163. if (type == "report") {
  164. reportWorker();
  165. }
  166. })
  167. }
  168. //Get the NJ Schedule JSON Feed
  169. function getNJDOT(type) {
  170. getFeed(NJURLList, function(result) {
  171. var resultObj = JSON.parse(result.responseText).Data.features;
  172. var length = resultObj.length;
  173. let advisories = [];
  174. var i = 0;
  175. for (let i = 0; i < length; i++) {
  176. if (["Incident", "Construction", "ScheduledConstruction", "Detour"].includes(resultObj[i].properties.CategoryName)) {
  177. advisories.push(resultObj[i].properties.EventID);
  178. }
  179. }
  180. getNJDetails(type,advisories);
  181. })
  182. }
  183. function getNJDetails(type,advisories) {
  184. var promises = [];
  185. for (let i = 0; i < advisories.length; i++) {
  186. promises.push(new Promise((resolve,reject) => {
  187. getFeed(NJURLDetail + advisories[i], function(result2) {
  188. var eventObj = JSON.parse(result2.responseText).Data;
  189. if ((eventObj[0].FullText.toUpperCase()).includes("ALL LANES CLOSE") || (eventObj[0].FullText.toUpperCase()).includes("RAMP CLOSE"))
  190. {
  191. var icon;
  192. switch(eventObj[0].CategoryName) {
  193. case "Incident":
  194. icon = "Incident";
  195. break;
  196. case "Detour":
  197. icon = "Incident";
  198. break;
  199. case "Construction":
  200. icon = "Roadwork";
  201. break;
  202. case "ScheduledConstruction":
  203. icon = "Roadwork";
  204. break;
  205. default:
  206. icon = "Incident";
  207. }
  208. if (type == "report") {
  209. let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
  210. var row = table.insertRow(-1);
  211. var cell1 = row.insertCell(0);
  212. var cell2 = row.insertCell(1);
  213. var cell3 = row.insertCell(2);
  214. var cell4 = row.insertCell(3);
  215. cell1.innerHTML = '<div class="gotoPL" data-lat="' + eventObj[0].Latitude + '" data-lon="' + eventObj[0].Longitude + '"><img src=' + PLIcon + '></div>';
  216. cell2.innerHTML = eventObj[0].FullText
  217. cell3.innerHTML = eventObj[0].County
  218. cell4.innerHTML = moment(new Date(eventObj[0].LastUpdateDate_String)).format('YYYY/MM/DD HH:mm');
  219. } else {
  220. drawMarkers("NJ",eventObj[0].markerId,eventObj[0].County + " - " + eventObj[0].CategoryName,eventObj[0].Longitude,eventObj[0].Latitude,icon,eventObj[0].FullText,eventObj[0].LastUpdateDate_String,"");
  221. }
  222. }
  223. resolve(); //Let the Promise.all() know that we finished this dependency
  224. })
  225. }))
  226. }
  227. Promise.all(promises).then(function () {
  228. if (type=="report") { reportWorker(); } //Run the reportWorker after we've resolved all promises (ie, fetched the details of all pertinent report IDs)
  229. });
  230. }
  231. //Get the NY Event JSON Feed
  232. function getNYDOT(type) {
  233. getFeed(NYURL, function(result) {
  234. var resultObj = JSON.parse(result.responseText);
  235. var i=0;
  236. var icon;
  237. while (i<resultObj.length) {
  238. if (NotNY.includes(resultObj[i].RegionName) == false && resultObj[i].EventType != 'transitMode' && resultObj[i].EventSubType != 'Capacity related') { //skip anything not NY & transit notices & parking notices
  239. if (resultObj[i].EventType == 'closures' || (resultObj[i].EventType != 'closures' && resultObj[i].LanesAffected == 'all lanes' && (resultObj[i].LanesStatus == 'closed' || resultObj[i].LanesStatus == "blocked"))) {
  240. switch(resultObj[i].EventType) {
  241. case "accidentsAndIncidents":
  242. icon = "Incident";
  243. break;
  244. case "roadwork":
  245. icon = "Roadwork";
  246. break;
  247. case "specialEvents":
  248. icon = "Incident";
  249. break;
  250. case "closures":
  251. icon = "Roadwork";
  252. break;
  253. case "transitMode":
  254. icon = "Roadwork";
  255. break;
  256. case "generalInfo":
  257. icon = "Roadwork";
  258. break;
  259. case "winterDrivingIndex":
  260. icon = "Roadwork";
  261. break;
  262. default:
  263. icon = "Incident";
  264. }
  265. if (type == "report") {
  266. let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
  267. var row = table.insertRow(-1);
  268. var cell1 = row.insertCell(0);
  269. var cell2 = row.insertCell(1);
  270. var cell3 = row.insertCell(2);
  271. var cell4 = row.insertCell(3);
  272. cell1.innerHTML = '<div class="gotoPL" data-lat="' + resultObj[i].Latitude + '" data-lon="' + resultObj[i].Longitude + '"><img src=' + PLIcon + '></div>';
  273. cell2.innerHTML = resultObj[i].Description;
  274. cell3.innerHTML = resultObj[i].CountyName;
  275. cell4.innerHTML = moment(moment(resultObj[i].LastUpdated,"DD/MM/YYYY HH:mm:ss")).format('YYYY/MM/DD HH:mm');
  276. } else {
  277. drawMarkers("NY",resultObj[i].ID,"",resultObj[i].Longitude,resultObj[i].Latitude,icon,resultObj[i].Description,resultObj[i].LastUpdated,"");
  278. }
  279. }
  280. }
  281. i++;
  282. }
  283. if (type == "report") {
  284. reportWorker();
  285. }
  286. })
  287. }
  288. //Get the PA Event JSON Feed
  289. function getPADOT(type) {
  290. getFeed(PAURL, function(result) {
  291. var resultObj = JSON.parse(result.responseText);
  292. var i=0;
  293. var icon;
  294. var x,y;
  295. while (i<resultObj.length) {
  296. if ((resultObj[i].LaneStatus == "closed") || (resultObj[i].LaneStatus == "ramp closure")) {
  297. switch(resultObj[i].EventType) {
  298. case "roadwork":
  299. icon = "Roadwork";
  300. break;
  301. default:
  302. icon = "Incident";
  303. }
  304. if (resultObj[i].LaneStatus != "ramp closure") {
  305. x = resultObj[i].FromLocLatLong.split(",")[1];
  306. y = resultObj[i].FromLocLatLong.split(",")[0];
  307. } else {
  308. x = resultObj[i].IncidentLocLatLong.split(",")[1];
  309. y = resultObj[i].IncidentLocLatLong.split(",")[0];
  310. }
  311. if (type == "report") {
  312. let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
  313. var row = table.insertRow(-1);
  314. var cell1 = row.insertCell(0);
  315. var cell2 = row.insertCell(1);
  316. var cell3 = row.insertCell(2);
  317. var cell4 = row.insertCell(3);
  318. cell1.innerHTML = '<div class="gotoPL" data-lat="' + y + '" data-lon="' + x + '"><img src=' + PLIcon + '></div>';
  319. cell2.innerHTML = resultObj[i].Description;
  320. cell3.innerHTML = resultObj[i].CountyName;
  321. cell4.innerHTML = moment(new Date(resultObj[i].DateTimeNotified)).format('YYYY/MM/DD HH:mm');
  322. } else {
  323. drawMarkers("PA",resultObj[i].EventID,resultObj[i].Facility,x,y,icon,resultObj[i].Description,resultObj[i].LastUpdate,"");
  324. }
  325. }
  326. i++;
  327. }
  328. if (type == "report") {
  329. reportWorker();
  330. }
  331. })
  332. }
  333. //Get the WA Event JSON Feed
  334. function getWADOT(type) {
  335. getFeed(WAURL, function(result) {
  336. var resultObj = JSON.parse(result.responseText);
  337. var i=0;
  338. var icon;
  339. var county;
  340. while (i<resultObj.length) {
  341. if (resultObj[i].EventCategory == 'Closure' || resultObj[i].EventCategory == 'Construction' || resultObj[i].EventCategory == 'Bridge') {
  342. switch(resultObj[i].EventCategory) {
  343. case "Construction":
  344. icon = "Roadwork";
  345. break;
  346. case "Closure":
  347. icon = "Roadwork";
  348. break;
  349. case "Collision":
  350. icon = "Incident";
  351. break;
  352. default:
  353. icon = "Incident";
  354. }
  355. var unixtime = parseInt(resultObj[i].LastUpdatedTime.replace("/Date(","").replace(")/","").split("-")[0]);
  356. let datetime = new Date(unixtime).toLocaleString();
  357. if (resultObj[i].County == null) {
  358. county = resultObj[i].Region;
  359. } else {
  360. county = resultObj[i].County;
  361. }
  362. if (type == "report") {
  363. let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
  364. var row = table.insertRow(-1);
  365. var cell1 = row.insertCell(0);
  366. var cell2 = row.insertCell(1);
  367. var cell3 = row.insertCell(2);
  368. var cell4 = row.insertCell(3);
  369. cell1.innerHTML = '<div class="gotoPL" data-lat="' + resultObj[i].StartRoadwayLocation.Latitude + '" data-lon="' + resultObj[i].StartRoadwayLocation.Longitude + '"><img src=' + PLIcon + '></div>';
  370. cell2.innerHTML = resultObj[i].HeadlineDescription;
  371. cell3.innerHTML = county;
  372. cell4.innerHTML = moment(new Date(datetime)).format('YYYY/MM/DD HH:mm'); //moment(new Date(resultObj[i].DateTimeNotified)).format('YYYY/MM/DD hh:mm a');
  373. } else {
  374. drawMarkers("WA",resultObj[i].AlertID,"",resultObj[i].StartRoadwayLocation.Longitude,resultObj[i].StartRoadwayLocation.Latitude,icon,resultObj[i].HeadlineDescription,datetime,"");
  375. }
  376. }
  377. i++;
  378. }
  379. if (type == "report") {
  380. reportWorker();
  381. }
  382. })
  383. }
  384. function reportWorker() {
  385. var elements = document.getElementsByClassName("gotoPL");
  386. for (var i = 0; i < elements.length; i++) {
  387. elements[i].addEventListener('click', moveMap, false);
  388. }
  389. refreshReportTable();
  390. }
  391. function refreshReportTable() {
  392. var sort = new Tablesort(document.getElementById('reportTable'),{descending: true});
  393. sort.refresh();
  394. }
  395. function getReportData(id) {
  396. switch(id) {
  397. case "DOTDEPopup":
  398. popupdetails("Delaware");
  399. getDEDOTAdv("report");
  400. break;
  401. case "DOTNJPopup":
  402. popupdetails("New Jersey");
  403. getNJDOT("report");
  404. break;
  405. case "DOTNYPopup":
  406. popupdetails("New York");
  407. getNYDOT("report");
  408. break;
  409. case "DOTPAPopup":
  410. popupdetails("Pennsylvania");
  411. getPADOT("report");
  412. break;
  413. case "DOTWAPopup":
  414. popupdetails("Washington");
  415. getWADOT("report");
  416. break;
  417. }
  418. }
  419. //Generate the Advisory markers
  420. function drawMarkers(state,id,title,x,y,icontype,desc,timestamp,link) {
  421. var size = new OpenLayers.Size(20,20);
  422. var offset = new OpenLayers.Pixel(-(size.w/2), -size.h);
  423. var icon = new OpenLayers.Icon(eval(icontype),size);
  424. var epsg4326 = new OpenLayers.Projection("EPSG:4326"); //WGS 1984 projection
  425. var projectTo = W.map.getProjectionObject(); //The map projection (Spherical Mercator)
  426. var lonLat = new OpenLayers.LonLat(x,y).transform(epsg4326,projectTo);
  427. var newMarker = new OpenLayers.Marker(lonLat,icon);
  428. newMarker.desc = desc;
  429. newMarker.eventId = id;
  430. newMarker.title = title;
  431. newMarker.state = state;
  432. newMarker.timestamp = timestamp;
  433. newMarker.events.register('click',newMarker,popup);
  434. newMarker.location = lonLat;
  435. if (link != "-1" && state == "DESch") {
  436. newMarker.link = '<a href=https://deldot.gov/About/news/index.shtml?dc=release&id=' + link + ' target="_blank">';
  437. } else if (link == "-1" && state == "DESch") {
  438. newMarker.link = `<a href='javascript:alert("Sorry! No publication available for this closure")'>`;
  439. } else {
  440. newMarker.link = link;
  441. }
  442. eval(state.substring(0,2) + "DOTLayer.addMarker(newMarker)");
  443. }
  444. function popup(evt) {
  445. $("#gmPopupContainer").remove ();
  446. $("#gmPopupContainer").hide ();
  447. var popupHTML;
  448. W.map.moveTo(this.location);
  449. switch(this.state) {
  450. case "DEAdv":
  451. this.timestamp = new Date(this.timestamp);
  452. popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px">' +
  453. '<a href="#close" id="gmCloseDlgBtn" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
  454. '<table border=0><tr><td><div id="mydivheader" style="min-height: 20px;">' + this.title + '</div><hr class="myhrline"/>' +
  455. 'Updated: ' + this.timestamp.toLocaleString() + '<hr class="myhrline"/></td></tr>' +
  456. '<tr><td>' + this.desc + '</td></tr>' +
  457. '<tr><td><a href="' + this.link + '" target="_blank">DelDot Link</a></td></tr>' +
  458. '</table>' +
  459. '</div>'
  460. ]);
  461. break;
  462. case "DESch":
  463. popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px">' +
  464. '<a href="#close" id="gmCloseDlgBtn" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
  465. '<table border=0><tr><td><div id="mydivheader" style="min-height: 20px;">Scheduled Closure</div><hr class="myhrline"/>' +
  466. '<tr><td><h4>' + this.title.toString() + '</h4><hr class="myhrline"/></td></tr>' +
  467. '<tr><td>Period: ' + this.timestamp.replace(/ 12:00 AM/g,"") + '<hr class="myhrline"/></td></tr>' +
  468. '<tr><td>' + this.desc + '</td></tr>' +
  469. '<tr><td>' + this.link + 'DelDot Link</a></td></tr>' +
  470. '</table>' +
  471. '</div>'
  472. ]);
  473. break;
  474. case "NJ":
  475. case "NY":
  476. case "PA":
  477. popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px">' +
  478. '<a href="#close" id="gmCloseDlgBtn" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
  479. '<table border=0><tr><td><div id="mydivheader" style="min-height: 20px;"></div><hr class="myhrline"/>' +
  480. '<tr><td><h4>' + this.title.toString() + '</h4><hr class="myhrline"/></td></tr>' +
  481. '<tr><td>Time: ' + new Date(this.timestamp).toLocaleString() + '<hr class="myhrline"/></td></tr>' +
  482. '<tr><td>' + this.desc + '</td></tr>' +
  483. '</table>' +
  484. '</div>'
  485. ]);
  486. break;
  487. case "WA":
  488. popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px">' +
  489. '<a href="#close" id="gmCloseDlgBtn" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
  490. '<table border=0><tr><td><div id="mydivheader" style="min-height: 20px;"></div><hr class="myhrline"/>' +
  491. 'Updated: ' + this.timestamp.toLocaleString() + '<hr class="myhrline"/></td></tr>' +
  492. '<tr><td>' + this.desc + '</td></tr>' +
  493. '</table>' +
  494. '</div>'
  495. ]);
  496. }
  497. $("body").append(popupHTML);
  498. //Position the modal based on the position of the click event
  499. $("#gmPopupContainer").css({left: document.getElementById("user-tabs").offsetWidth + W.map.getPixelFromLonLat(W.map.getCenter()).x - document.getElementById("gmPopupContainer").clientWidth - 10 });
  500. $("#gmPopupContainer").css({top: document.getElementById("left-app-head").offsetHeight + W.map.getPixelFromLonLat(W.map.getCenter()).y - (document.getElementById("gmPopupContainer").clientHeight / 2) });
  501. $("#gmPopupContainer").show();
  502. //Add listener for popup's "Close" button
  503. $("#gmCloseDlgBtn").click ( function () {
  504. $("#gmPopupContainer").remove ();
  505. $("#gmPopupContainer").hide ();
  506. });
  507. dragElement(document.getElementById("gmPopupContainer"));
  508. }
  509. function popupdetails(state) {
  510. $("#gmPopupContainer").remove ();
  511. $("#gmPopupContainer").hide ();
  512. var popupHTML;
  513. popupHTML = (['<div id="gmPopupContainer" style="max-width:750px;max-height:500px;margin: 1;text-align: center;padding: 5px;">' +
  514. '<a href="#close" id="popupdetailsclose" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
  515. '<table border=0><tr><td><div id="mydivheader"">' + state + ' Reports</div></td></tr>' +
  516. '<tr><td>' +
  517. '<div style="width:720px; height:450px; overflow:auto;"><table id="reportTable" border=1>' +
  518. '<thead><tr><td data-sort-method="none" width=30><b>PL</b></td><th width=394>Description</th><th width=100>Location</th><th data-sort-default width=175>Time</th></tr></thead>' +
  519. '<tbody></tbody></table></div>' +
  520. '</td></tr></table>' +
  521. '</div>'
  522. ]);
  523. $("body").append(popupHTML);
  524. //Position the modal based on the position of the click event
  525. $("#gmPopupContainer").css({left: 400});
  526. $("#gmPopupContainer").css({top: 100});
  527. $("#gmPopupContainer").show();
  528. //Add listener for popup's "Close" button
  529. $("#popupdetailsclose").click ( function () {
  530. $("#gmPopupContainer").remove ();
  531. $("#gmPopupContainer").hide ();
  532. });
  533. dragElement(document.getElementById("gmPopupContainer"));
  534. }
  535. // Make the DIV element draggable
  536. function dragElement(elmnt) {
  537. var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
  538. if (document.getElementById("mydivheader")) {
  539. // if present, the header is where you move the DIV from:
  540. document.getElementById("mydivheader").onmousedown = dragMouseDown;
  541. } else {
  542. // otherwise, move the DIV from anywhere inside the DIV:
  543. elmnt.onmousedown = dragMouseDown;
  544. }
  545. function dragMouseDown(e) {
  546. e = e || window.event;
  547. e.preventDefault();
  548. // get the mouse cursor position at startup:
  549. pos3 = e.clientX;
  550. pos4 = e.clientY;
  551. document.onmouseup = closeDragElement;
  552. // call a function whenever the cursor moves:
  553. document.onmousemove = elementDrag;
  554. }
  555.  
  556. function elementDrag(e) {
  557. e = e || window.event;
  558. e.preventDefault();
  559. // calculate the new cursor position:
  560. pos1 = pos3 - e.clientX;
  561. pos2 = pos4 - e.clientY;
  562. pos3 = e.clientX;
  563. pos4 = e.clientY;
  564. // set the element's new position:
  565. elmnt.style.top = (elmnt.offsetTop - pos2) + "px";
  566. elmnt.style.left = (elmnt.offsetLeft - pos1) + "px";
  567. }
  568. function closeDragElement() {
  569. // stop moving when mouse button is released:
  570. document.onmouseup = null;
  571. document.onmousemove = null;
  572. }
  573. }
  574. //Move map to coordinates specified
  575. function moveMap() {
  576. var epsg4326 = new OpenLayers.Projection("EPSG:4326"); //WGS 1984 projection
  577. var projectTo = W.map.getProjectionObject(); //The map projection (Spherical Mercator)
  578. var lat = this.getAttribute("data-lat");
  579. var lon = this.getAttribute("data-lon");
  580. W.map.moveTo(new OpenLayers.LonLat(lon,lat).transform(epsg4326,projectTo),6);
  581. }
  582. //Initialize Settings
  583. function initializeSettings()
  584. {
  585. for (var i = 0; i < document.getElementsByClassName("DOTreport").length; i++) {
  586. document.getElementsByClassName("DOTreport")[i].addEventListener('click', function(e) {getReportData(this.getAttribute("id"));}, false);
  587. }
  588. loadSettings();
  589. setChecked('chkPADOTEnabled', settings.PADOTEnabled);
  590. setChecked('chkDEDOTEnabled', settings.DEDOTEnabled);
  591. setChecked('chkNYDOTEnabled', settings.NYDOTEnabled);
  592. setChecked('chkNJDOTEnabled', settings.NJDOTEnabled);
  593. setChecked('chkWADOTEnabled', settings.WADOTEnabled);
  594. $(".overlay-button").click(function(){
  595. if (document.getElementById('chkDEDOTEnabled').checked) { eval('W.map.removeLayer(DEDOTLayer)'); }
  596. if (document.getElementById('chkNJDOTEnabled').checked) { eval('W.map.removeLayer(NJDOTLayer)'); }
  597. if (document.getElementById('chkNYDOTEnabled').checked) { eval('W.map.removeLayer(NYDOTLayer)'); }
  598. if (document.getElementById('chkPADOTEnabled').checked) { eval('W.map.removeLayer(PADOTLayer)'); }
  599. if (document.getElementById('chkWADOTEnabled').checked) { eval('W.map.removeLayer(WADOTLayer)'); }
  600. initializeSettings();
  601. });
  602.  
  603. //Add Handler for Checkbox Setting Changes
  604. $('.WMEDOTAdvSettingsCheckbox').change(function() {
  605. var settingName = $(this)[0].id.substr(3);
  606. settings[settingName] = this.checked;
  607. saveSettings();
  608. if(this.checked) {
  609. buildDOTAdvLayers(settingName.substring(0,2));
  610. eval("get" + settingName.substring(0,2) + "DOT()");
  611. }
  612. else
  613. {
  614. //eval(settingName.substring(0,2) + "DOTLayer.destroy()");
  615. eval('W.map.removeLayer(' + settingName.substring(0,2) + 'DOTLayer)');
  616. }
  617. });
  618. if (document.getElementById('WMEFUzoom') != null) {
  619. document.getElementById('WMEFUzoom').style.zIndex = "45000";
  620. }
  621. if (document.getElementById('chkDEDOTEnabled').checked) { buildDOTAdvLayers("DE"); getDEDOT();}
  622. if (document.getElementById('chkNJDOTEnabled').checked) { buildDOTAdvLayers("NJ"); getNJDOT();}
  623. if (document.getElementById('chkNYDOTEnabled').checked) { buildDOTAdvLayers("NY"); getNYDOT();}
  624. if (document.getElementById('chkPADOTEnabled').checked) { buildDOTAdvLayers("PA"); getPADOT();}
  625. if (document.getElementById('chkWADOTEnabled').checked) { buildDOTAdvLayers("WA"); getWADOT();}
  626. }
  627. //Set Checkbox from Settings
  628. function setChecked(checkboxId, checked) {
  629. $('#' + checkboxId).prop('checked', checked);
  630. }
  631. //Load Saved Settings
  632. function loadSettings() {
  633. var loadedSettings = $.parseJSON(localStorage.getItem("WMEDOT_Settings"));
  634. var defaultSettings = {
  635. Enabled: false,
  636. };
  637. settings = loadedSettings ? loadedSettings : defaultSettings;
  638. for (var prop in defaultSettings) {
  639. if (!settings.hasOwnProperty(prop)) {
  640. settings[prop] = defaultSettings[prop];
  641. }
  642. }
  643. }
  644. //Save Tab Settings
  645. function saveSettings() {
  646. if (localStorage) {
  647. var localsettings = {
  648. PADOTEnabled: settings.PADOTEnabled,
  649. DEDOTEnabled: settings.DEDOTEnabled,
  650. NYDOTEnabled: settings.NYDOTEnabled,
  651. NJDOTEnabled: settings.NJDOTEnabled,
  652. WADOTEnabled: settings.WADOTEnabled,
  653. };
  654. localStorage.setItem("WMEDOT_Settings", JSON.stringify(localsettings));
  655. }
  656. }
  657. //Add the Icon Class to OpenLayers
  658. function installIcon() {
  659. console.log('Installing OpenLayers.Icon');
  660. OpenLayers.Icon = OpenLayers.Class({
  661. url: null,
  662. size: null,
  663. offset: null,
  664. calculateOffset: null,
  665. imageDiv: null,
  666. px: null,
  667. initialize: function(a,b,c,d){
  668. this.url=a;
  669. this.size=b||{w: 20,h: 20};
  670. this.offset=c||{x: -(this.size.w/2),y: -(this.size.h/2)};
  671. this.calculateOffset=d;
  672. a=OpenLayers.Util.createUniqueID("OL_Icon_");
  673. let div = this.imageDiv=OpenLayers.Util.createAlphaImageDiv(a);
  674. $(div.firstChild).removeClass('olAlphaImg'); // LEAVE THIS LINE TO PREVENT WME-HARDHATS SCRIPT FROM TURNING ALL ICONS INTO HARDHAT WAZERS --MAPOMATIC
  675. },
  676. destroy: function(){ this.erase();OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild);this.imageDiv.innerHTML="";this.imageDiv=null; },
  677. clone: function(){ return new OpenLayers.Icon(this.url,this.size,this.offset,this.calculateOffset); },
  678. setSize: function(a){ null!==a&&(this.size=a); this.draw(); },
  679. setUrl: function(a){ null!==a&&(this.url=a); this.draw(); },
  680. draw: function(a){
  681. OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv,null,null,this.size,this.url,"absolute");
  682. this.moveTo(a);
  683. return this.imageDiv;
  684. },
  685. erase: function(){ null!==this.imageDiv&&null!==this.imageDiv.parentNode&&OpenLayers.Element.remove(this.imageDiv); },
  686. setOpacity: function(a){ OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv,null,null,null,null,null,null,null,a); },
  687. moveTo: function(a){
  688. null!==a&&(this.px=a);
  689. null!==this.imageDiv&&(null===this.px?this.display(!1): (
  690. this.calculateOffset&&(this.offset=this.calculateOffset(this.size)),
  691. OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv,null,{x: this.px.x+this.offset.x,y: this.px.y+this.offset.y})
  692. ));
  693. },
  694. display: function(a){ this.imageDiv.style.display=a?"": "none"; },
  695. isDrawn: function(){ return this.imageDiv&&this.imageDiv.parentNode&&11!=this.imageDiv.parentNode.nodeType; },
  696. CLASS_NAME: "OpenLayers.Icon"
  697. });
  698. }
  699. bootstrap();
  700. })();