WME DOT Advisories

Overlay DOT Advisories on the WME Map Object

当前为 2021-03-09 提交的版本,查看 最新版本

  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.29
  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 511nj.org
  20. // @connect 511ny.org
  21. // @connect 511pa.com
  22. // @connect arcgis.com
  23. // @connect deldot.gov
  24. // @connect essentialintegrations.com
  25. // @connect fl511.com
  26. // @connect md.gov
  27. // @connect mi.us
  28. // @connect ohgo.com
  29. // @connect rehostjson.phuz.repl.co
  30. // @connect tripcheck.com
  31. // @connect iteris-atis.com
  32. /* global OpenLayers */
  33. /* global W */
  34. /* global WazeWrap */
  35. /* global $ */
  36. /* global I18n */
  37. /* global _ */
  38. // ==/UserScript==
  39.  
  40. let AKDOTLayer, DEDOTLayer, FLDOTLayer, GADOTLayer, LADOTLayer, MDDOTLayer, MIDOTLayer, NCDOTLayer, NJDOTLayer, NVDOTLayer, NYDOTLayer, OHDOTLayer, ORDOTLayer, PADOTLayer, WADOTLayer;
  41. let promisesAK, promisesDE, promisesFL, promisesGA, promisesLA, promisesMD, promisesMI, promisesNC, promisesNJ, promisesNV, promisesNY, promisesOH, promisesOR, promisesPA, promisesWA;
  42. let advisoriesAK, advisoriesDE, advisoriesFL, advisoriesGA, advisoriesLA, advisoriesMD, advisoriesMI, advisoriesNC, advisoriesNJ, advisoriesNV, advisoriesNY, advisoriesOH, advisoriesOR, advisoriesPA, advisoriesWA;
  43. var settings;
  44. var state, stateLength, advisories;
  45. const updateMessage = "► Added Wilmington closures";
  46. const DEIconC = '';
  47. const DEIconSchRestriction = '';
  48. const DEIconSchClosure = '';
  49. const Incident = '';
  50. const Roadwork = '';
  51. const PLIcon = '';
  52. const reportIcon = '';
  53. const NJURLDetail = 'https://511nj.org/API/client/Map/getEventPopupData?EventId=';
  54. const NotNY = ['Pennsylvania Statewide', 'New Jersey Statewide', 'Connecticut Statewide'];
  55. const NJConstruction = ['Construction', 'ScheduledConstruction'];
  56.  
  57. //Begin script function
  58. (function () {
  59. 'use strict';
  60. //Bootstrap
  61. window["text123"] = 123;
  62. function bootstrap(tries = 1) {
  63. if (W && W.loginManager && W.map && W.loginManager.user && W.model
  64. && W.model.states && W.model.states.getObjectArray().length && WazeWrap && WazeWrap.Ready) {
  65. console.log("WME DOT Advisories Loaded!");
  66. init();
  67. addListeners();
  68. if (!OpenLayers.Icon) {
  69. installIcon();
  70. }
  71. } else if (tries < 1000) {
  72. setTimeout(function () { bootstrap(++tries); }, 200);
  73. }
  74. }
  75. //Build the Tab and Settings Division
  76. function init() {
  77. var $section = $("<div>");
  78. $section.html([
  79. '<div id="chkEnables">',
  80. '<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>',
  81. '* The WME Refresh Button will update reports.',
  82. '<table border=1 style="text-align:center;width:100%;padding:10px;">',
  83. '<tr><td style="text-align:center"><b>Enable</b></td><td style="text-align"><b>State</b></td><td width=30><b>Rpt</b></td></tr>',
  84. '<tr><td><input type="checkbox" id="chkAKDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>AK</td><td><div class=DOTreport data-report="report" data-state="Alaska" id="AK"><img src=' + reportIcon + '></div></td></tr>',
  85. '<tr><td><input type="checkbox" id="chkDEDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>DE</td><td><div class=DOTreport data-report="report" data-state="Delaware" id="DE"><img src=' + reportIcon + '></div></td></tr>',
  86. '<tr><td><input type="checkbox" id="chkFLDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>FL</td><td><div class=DOTreport data-report="report" data-state="Florida" id="FL"><img src=' + reportIcon + '></div></td></tr>',
  87. '<tr><td><input type="checkbox" id="chkGADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>GA</td><td><div class=DOTreport data-report="report" data-state="Georgia" id="GA"><img src=' + reportIcon + '></div></td></tr>',
  88. '<tr><td><input type="checkbox" id="chkLADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>LA</td><td><div class=DOTreport data-report="report" data-state="Louisiana" id="LA"><img src=' + reportIcon + '></div></td></tr>',
  89. '<tr><td><input type="checkbox" id="chkMDDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>MD</td><td><div class=DOTreport data-report="report" data-state="Maryland" id="MD"><img src=' + reportIcon + '></div></td></tr>',
  90. '<tr><td><input type="checkbox" id="chkMIDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>MI</td><td><div class=DOTreport data-report="report" data-state="Michigan" id="MI"><img src=' + reportIcon + '></div></td></tr>',
  91. '<tr><td><input type="checkbox" id="chkNCDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>NC</td><td><div class=DOTreport data-report="report" data-state="North Carolina" id="NC"><img src=' + reportIcon + '></div></td></tr>',
  92. '<tr><td><input type="checkbox" id="chkNJDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>NJ</td><td><div class=DOTreport data-report="report" data-state="New Jersey" id="NJ"><img src=' + reportIcon + '></div></td></tr>',
  93. '<tr><td><input type="checkbox" id="chkNVDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>NV</td><td><div class=DOTreport data-report="report" data-state="Nevada" id="NV"><img src=' + reportIcon + '></div></td></tr>',
  94. '<tr><td><input type="checkbox" id="chkNYDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>NY</td><td><div class=DOTreport data-report="report" data-state="New York" id="NY"><img src=' + reportIcon + '></div></td></tr>',
  95. '<tr><td><input type="checkbox" id="chkOHDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>OH</td><td><div class=DOTreport data-report="report" data-state="Ohio" id="OH"><img src=' + reportIcon + '></div></td></tr>',
  96. '<tr><td><input type="checkbox" id="chkORDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>OR</td><td><div class=DOTreport data-report="report" data-state="Oregon" id="OR"><img src=' + reportIcon + '></div></td></tr>',
  97. '<tr><td><input type="checkbox" id="chkPADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>PA</td><td><div class=DOTreport data-report="report" data-state="Pennsylvania" id="PA"><img src=' + reportIcon + '></div></td></tr>',
  98. '<tr><td><input type="checkbox" id="chkWADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>WA</td><td><div class=DOTreport data-report="report" data-state="Washington" id="WA"><img src=' + reportIcon + '></div></td></tr>',
  99. '</table>',
  100. '</div></div>'
  101. ].join(' '));
  102. new WazeWrap.Interface.Tab('DOT Advisories', $section.html(), initializeSettings);
  103. WazeWrap.Interface.ShowScriptUpdate("WME DOT Advisories", GM_info.script.version, updateMessage, "https://greasyfork.org/en/scripts/412976-wme-dot-advisories", "https://www.waze.com/forum/viewtopic.php?f=819&t=308141");
  104. }
  105. getFeed("http://scripts.essentialintegrations.com/CSS", function (result) {
  106. GM_addStyle(result.responseText);
  107. })
  108. //Build the State Layers
  109. function buildDOTAdvLayers(state) {
  110. eval(state.substring(0, 2) + 'DOTLayer = new OpenLayers.Layer.Markers(' + state.substring(0, 2) + 'DOTLayer)');
  111. W.map.addLayer(eval(state.substring(0, 2) + 'DOTLayer'));
  112. W.map.getOLMap().setLayerIndex(eval(state.substring(0, 2) + 'DOTLayer'), 0);
  113. }
  114. function getFeed(url, callback) {
  115. GM_xmlhttpRequest({
  116. method: "GET",
  117. url: url,
  118. onload: function (response) {
  119. var result = response;
  120. callback(result);
  121. }
  122. });
  123. }
  124.  
  125. function getAdvisories(state, stateAbv, type) {
  126. eval('promises' + stateAbv + ' = []');
  127. eval('advisories' + stateAbv + ' = []');
  128. let thesepromises = [];
  129. for (let j = 0; j < state.URL.length; j++) {
  130. let thispromise = new Promise((resolve, reject) => {
  131. getFeed(state.URL[j], function (result) {
  132. let resultObj = [];
  133. resultObj = state.data(JSON.parse(result.responseText), j);
  134. for (let i = 0; i < resultObj.length; i++) {
  135. if (eval(state.filter(j))) {
  136. state.scheme(resultObj[i], j);
  137. }
  138. }
  139. resolve();
  140. });
  141. })
  142. //eval('promises' + stateAbv + '.push(thispromise)');
  143. thesepromises.push(thispromise);
  144. }
  145. Promise.all(thesepromises).then(function () {
  146. setTimeout(function () { promiseWorker(stateAbv, type) }, 500);
  147. })
  148. }
  149. function promiseWorker(stateAbv, type) {
  150. let thisadvisory = eval("advisories" + stateAbv);
  151. Promise.all(eval('promises' + stateAbv)).then(function () {
  152. for (let i = 0; i < thisadvisory.length; i++) {
  153. if (type == "report") {
  154. let parms = thisadvisory[i];
  155. let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
  156. var row = table.insertRow(-1);
  157. var cell1 = row.insertCell(0);
  158. var cell2 = row.insertCell(1);
  159. var cell3 = row.insertCell(2);
  160. var cell4 = row.insertCell(3);
  161. cell1.innerHTML = '<div class="gotoPL" data-lat="' + parms.lat + '" data-lon="' + parms.lon + '"><img src=' + PLIcon + '></div>'; //PL
  162. cell2.innerHTML = parms.desc; //Description
  163. cell3.innerHTML = parms.title; //Location
  164. cell4.innerHTML = parms.time; //Time
  165. } else {
  166. drawMarkers(thisadvisory[i]);
  167. }
  168. }
  169. if ((type == "report")) { //Wait until we loop through all the advisory URLs before sorting the table
  170. reportWorker();
  171. }
  172. });
  173. }
  174. function reportWorker() {
  175. var elements = document.getElementsByClassName("gotoPL");
  176. for (var i = 0; i < elements.length; i++) {
  177. elements[i].addEventListener('click', moveMap, false);
  178. }
  179. refreshReportTable();
  180. }
  181. function refreshReportTable() {
  182. var sort = new Tablesort(document.getElementById('reportTable'), { descending: true });
  183. sort.refresh();
  184. document.getElementById("spinner").style.visibility = "hidden";
  185. }
  186. function getReportData(stateAbv, stateName) {
  187. popupdetails(stateName);
  188. //if (stateAbv != "NJ") {
  189. eval('getAdvisories(config.' + stateAbv + ',"' + stateAbv + '", "report")');
  190. //} else { getNJDOT("report"); }
  191. }
  192. //Generate the Advisory markers
  193. function drawMarkers(parms) {
  194. var icontype;
  195. var size = new OpenLayers.Size(20, 20);
  196. var offset = new OpenLayers.Pixel(-(size.w / 2), -size.h);
  197. for (let i = 0; i < parms.keyword.length; i++) { //Check each of the keywords for roadwork/construction
  198. if (parms.type == parms.keyword[i]) {
  199. icontype = Roadwork;
  200. break;
  201. } else {
  202. icontype = Incident;
  203. }
  204. }
  205. var icon = new OpenLayers.Icon(icontype, size);
  206. var epsg4326 = new OpenLayers.Projection("EPSG:4326"); //WGS 1984 projection
  207. var projectTo = W.map.getProjectionObject(); //The map projection (Spherical Mercator)
  208. var lonLat = new OpenLayers.LonLat(parms.lon, parms.lat).transform(epsg4326, projectTo);
  209. var newMarker = new OpenLayers.Marker(lonLat, icon);
  210. newMarker.eventId = parms.id;
  211. newMarker.title = parms.title;
  212. newMarker.desc = parms.desc;
  213. newMarker.popupType = parms.popupType;
  214. newMarker.state = parms.state;
  215. newMarker.timestamp = parms.time;
  216. newMarker.events.register('click', newMarker, popup);
  217. newMarker.location = lonLat;
  218. if (parms.link != '') {
  219. newMarker.link = '<a href="' + parms.link + '" target="_blank">Publication Link</a>';
  220. } else {
  221. newMarker.link = '';
  222. }
  223. eval(parms.state[0] + "DOTLayer.addMarker(newMarker)");
  224. }
  225. function popup(evt) {
  226. $("#gmPopupContainer").remove();
  227. $("#gmPopupContainer").hide();
  228. var popupHTML;
  229. W.map.moveTo(this.location);
  230. popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px;z-index: 1100">' +
  231. '<a href="#close" id="gmCloseDlgBtn" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
  232. '<table border=0><tr><td><div id="mydivheader" style="min-height: 20px;">' + this.title + '</div></div><hr class="myhrline"/>' +
  233. 'Updated: ' + this.timestamp.toLocaleString() + '<hr class="myhrline"/></td></tr>' +
  234. '<tr><td>' + this.desc + '</td></tr>' +
  235. '<tr><td>' + this.link + '</td></tr>' +
  236. '</table>' +
  237. '</div>'
  238. ]);
  239. $("body").append(popupHTML);
  240. //Position the modal based on the position of the click event
  241. $("#gmPopupContainer").css({ left: document.getElementById("user-tabs").offsetWidth + W.map.getPixelFromLonLat(W.map.getCenter()).x - document.getElementById("gmPopupContainer").clientWidth - 10 });
  242. $("#gmPopupContainer").css({ top: document.getElementById("left-app-head").offsetHeight + W.map.getPixelFromLonLat(W.map.getCenter()).y - (document.getElementById("gmPopupContainer").clientHeight / 2) });
  243. $("#gmPopupContainer").show();
  244. //Add listener for popup's "Close" button
  245. $("#gmCloseDlgBtn").click(function () {
  246. $("#gmPopupContainer").remove();
  247. $("#gmPopupContainer").hide();
  248. });
  249. dragElement(document.getElementById("gmPopupContainer"));
  250. }
  251. function popupdetails(stateName) {
  252. $("#gmPopupContainer").remove();
  253. $("#gmPopupContainer").hide();
  254. var popupHTML;
  255. popupHTML = (['<div id="gmPopupContainer" style="max-width:750px;max-height:500px;margin:1;text-align:center;padding: 5px;z-index: 1100">' +
  256. '<a href="#close" id="popupdetailsclose" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
  257. '<table border=0><tr><td><div id="mydivheader" style="float:center">' + stateName + ' Reports <div id="spinner" class="spinner" style="float:left;position:relative;left:70%">' +
  258. '<div class="bounce1" style="float:left;position:relative;left:40%"></div><div class="bounce2" style="float:left;position:relative;left:50%"></div><div class="bounce3" style="float:left;position:relative;left:60%"></div></div></td></tr>' +
  259. '<tr><td>' +
  260. '<div style="width:720px; height:450px; overflow:auto;"><table id="reportTable" border=1>' +
  261. '<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=210>Time</th></tr></thead>' +
  262. '<tbody></tbody></table></div>' +
  263. '</td></tr></table>' +
  264. '</div>'
  265. ]);
  266. $("body").append(popupHTML);
  267. //Position the modal based on the position of the click event
  268. $("#gmPopupContainer").css({ left: 350 });
  269. $("#gmPopupContainer").css({ top: 100 });
  270. $("#gmPopupContainer").show();
  271. //Add listener for popup's "Close" button
  272. $("#popupdetailsclose").click(function () {
  273. $("#gmPopupContainer").remove();
  274. $("#gmPopupContainer").hide();
  275. });
  276. dragElement(document.getElementById("gmPopupContainer"));
  277. }
  278. // Make the DIV element draggable
  279. function dragElement(elmnt) {
  280. var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
  281. if (document.getElementById("mydivheader")) {
  282. // if present, the header is where you move the DIV from:
  283. document.getElementById("mydivheader").onmousedown = dragMouseDown;
  284. } else {
  285. // otherwise, move the DIV from anywhere inside the DIV:
  286. elmnt.onmousedown = dragMouseDown;
  287. }
  288. function dragMouseDown(e) {
  289. e = e || window.event;
  290. e.preventDefault();
  291. // get the mouse cursor position at startup:
  292. pos3 = e.clientX;
  293. pos4 = e.clientY;
  294. document.onmouseup = closeDragElement;
  295. // call a function whenever the cursor moves:
  296. document.onmousemove = elementDrag;
  297. }
  298.  
  299. function elementDrag(e) {
  300. e = e || window.event;
  301. e.preventDefault();
  302. // calculate the new cursor position:
  303. pos1 = pos3 - e.clientX;
  304. pos2 = pos4 - e.clientY;
  305. pos3 = e.clientX;
  306. pos4 = e.clientY;
  307. // set the element's new position:
  308. elmnt.style.top = (elmnt.offsetTop - pos2) + "px";
  309. elmnt.style.left = (elmnt.offsetLeft - pos1) + "px";
  310. }
  311. function closeDragElement() {
  312. // stop moving when mouse button is released:
  313. document.onmouseup = null;
  314. document.onmousemove = null;
  315. }
  316. }
  317. //Move map to coordinates specified
  318. function moveMap() {
  319. var epsg4326 = new OpenLayers.Projection("EPSG:4326"); //WGS 1984 projection
  320. var projectTo = W.map.getProjectionObject(); //The map projection (Spherical Mercator)
  321. var lat = this.getAttribute("data-lat");
  322. var lon = this.getAttribute("data-lon");
  323. W.map.moveTo(new OpenLayers.LonLat(lon, lat).transform(epsg4326, projectTo), 6);
  324. }
  325. //Initialize Settings
  326. function initializeSettings() {
  327. stateLength = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox").length;
  328. loadSettings();
  329. //Set the state checkboxes according to saved settings
  330. for (var i = 0; i < stateLength; i++) {
  331. state = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].id.replace("chk", "").replace("DOTEnabled", "");
  332. setChecked('chk' + state + 'DOTEnabled', eval('settings.' + state + 'DOTEnabled'));
  333. }
  334. //Build the layers for the selected states
  335. for (var i = 0; i < stateLength; i++) {
  336. state = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].id.replace("chk", "").replace("DOTEnabled", "");
  337. if (document.getElementById('chk' + state + 'DOTEnabled').checked) { buildDOTAdvLayers(state); eval('getAdvisories(config.' + state + ',"' + state + '")') }
  338. }
  339. }
  340. function addListeners() {
  341. //Add event listener to report icon
  342. for (var i = 0; i < document.getElementsByClassName("DOTreport").length; i++) {
  343. document.getElementsByClassName("DOTreport")[i].addEventListener('click', function (e) { getReportData(this.getAttribute("id"), this.getAttribute("data-state")); }, false);
  344. }
  345. //Refresh selected states when WME's refresh button is clicked
  346. document.getElementsByClassName("reload-button-region")[0].addEventListener('click', function (e) {
  347. for (var i = 0; i < stateLength; i++) {
  348. state = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].id.replace("chk", "").replace("DOTEnabled", "");
  349. if (document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].checked) { eval('W.map.removeLayer(' + state + 'DOTLayer)'); }
  350. }
  351. initializeSettings();
  352. });
  353. //Add Handler for Checkbox Setting Changes
  354. $('.WMEDOTAdvSettingsCheckbox').change(function () {
  355. var settingName = $(this)[0].id.substr(3);
  356. settings[settingName] = this.checked;
  357. saveSettings();
  358. if (this.checked) {
  359. buildDOTAdvLayers(settingName.substring(0, 2));
  360. eval('getAdvisories(config.' + settingName.substring(0, 2) + ',' + "settingName.substring(0, 2)" + ')');
  361. }
  362. else {
  363. //eval(settingName.substring(0,2) + "DOTLayer.destroy()");
  364. eval('W.map.removeLayer(' + settingName.substring(0, 2) + 'DOTLayer)');
  365. }
  366. });
  367. }
  368. //Set Checkbox from Settings
  369. function setChecked(checkboxId, checked) {
  370. $('#' + checkboxId).prop('checked', checked);
  371. }
  372. //Load Saved Settings
  373. function loadSettings() {
  374. var loadedSettings = $.parseJSON(localStorage.getItem("WMEDOT_Settings"));
  375. var defaultSettings = {
  376. Enabled: false,
  377. };
  378. settings = loadedSettings ? loadedSettings : defaultSettings;
  379. for (var prop in defaultSettings) {
  380. if (!settings.hasOwnProperty(prop)) {
  381. settings[prop] = defaultSettings[prop];
  382. }
  383. }
  384. }
  385. //Save Tab Settings
  386. function saveSettings() {
  387. if (localStorage) {
  388. var localsettings = {};
  389. for (var i = 0; i < stateLength; i++) {
  390. state = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].id.replace("chk", "").replace("DOTEnabled", "");
  391. eval('localsettings.' + state + 'DOTEnabled = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].checked');
  392. }
  393. }
  394. localStorage.setItem("WMEDOT_Settings", JSON.stringify(localsettings));
  395. }
  396. //Add the Icon Class to OpenLayers
  397. function installIcon() {
  398. console.log('Installing OpenLayers.Icon');
  399. OpenLayers.Icon = OpenLayers.Class({
  400. url: null,
  401. size: null,
  402. offset: null,
  403. calculateOffset: null,
  404. imageDiv: null,
  405. px: null,
  406. initialize: function (a, b, c, d) {
  407. this.url = a;
  408. this.size = b || { w: 20, h: 20 };
  409. this.offset = c || { x: -(this.size.w / 2), y: -(this.size.h / 2) };
  410. this.calculateOffset = d;
  411. a = OpenLayers.Util.createUniqueID("OL_Icon_");
  412. let div = this.imageDiv = OpenLayers.Util.createAlphaImageDiv(a);
  413. $(div.firstChild).removeClass('olAlphaImg'); // LEAVE THIS LINE TO PREVENT WME-HARDHATS SCRIPT FROM TURNING ALL ICONS INTO HARDHAT WAZERS --MAPOMATIC
  414. },
  415. destroy: function () { this.erase(); OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); this.imageDiv.innerHTML = ""; this.imageDiv = null; },
  416. clone: function () { return new OpenLayers.Icon(this.url, this.size, this.offset, this.calculateOffset); },
  417. setSize: function (a) { null !== a && (this.size = a); this.draw(); },
  418. setUrl: function (a) { null !== a && (this.url = a); this.draw(); },
  419. draw: function (a) {
  420. OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, this.size, this.url, "absolute");
  421. this.moveTo(a);
  422. return this.imageDiv;
  423. },
  424. erase: function () { null !== this.imageDiv && null !== this.imageDiv.parentNode && OpenLayers.Element.remove(this.imageDiv); },
  425. setOpacity: function (a) { OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, null, null, null, null, a); },
  426. moveTo: function (a) {
  427. null !== a && (this.px = a);
  428. null !== this.imageDiv && (null === this.px ? this.display(!1) : (
  429. this.calculateOffset && (this.offset = this.calculateOffset(this.size)),
  430. OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { x: this.px.x + this.offset.x, y: this.px.y + this.offset.y })
  431. ));
  432. },
  433. display: function (a) { this.imageDiv.style.display = a ? "" : "none"; },
  434. isDrawn: function () { return this.imageDiv && this.imageDiv.parentNode && 11 != this.imageDiv.parentNode.nodeType; },
  435. CLASS_NAME: "OpenLayers.Icon"
  436. });
  437. }
  438. bootstrap();
  439. const config = { //Configuration data for each state
  440. AK: {
  441. data(res, index) {
  442. let resultText = [res];
  443. return (resultText[index]);
  444. },
  445. filter(index) {
  446. let filtertext = ['(resultObj[i].LanesAffected).replace(/ +(?= )/g, "") == ("All Lanes Closed")'];
  447. return (filtertext[index]);
  448. },
  449. scheme(obj, index) {
  450. promisesAK.push(new Promise((resolve, reject) => {
  451. advisoriesAK.push({
  452. state: ['AK', 'Alaska'],
  453. id: obj.ID,
  454. popupType: 0,
  455. title: obj.RoadwayName,
  456. lon: obj.Longitude,
  457. lat: obj.Latitude,
  458. type: obj.EventType,
  459. keyword: ['roadwork'], //keywords for roadwork/construction
  460. desc: obj.Description,
  461. time: moment(new Date(obj.LastUpdated * 1000)).format('LLL'),
  462. link: ''
  463. });
  464. resolve();
  465. }))
  466. },
  467. URL: ['http://scripts.essentialintegrations.com/AK']
  468. },
  469. DE: {
  470. data(res, index) {
  471. let resultText = [res.advisories, res, res.features];
  472. return (resultText[index]);
  473. },
  474. filter(index) {
  475. let filtertext = [true, '(resultObj[i].str.impactType == "Closure") || ((resultObj[i].str.impactType == "Restriction") && ((resultObj[i].str.construction.toUpperCase().includes("AMP CLOS") || (resultObj[i].str.construction.toUpperCase().includes("ROAD CLOS")))))', true];
  476. return (filtertext[index]);
  477. },
  478. scheme(obj, index) {
  479. switch (index) {
  480. case 0:
  481. promisesDE.push(new Promise((resolve, reject) => {
  482. advisoriesDE.push({
  483. state: ['DE', 'Delaware'],
  484. id: obj.id,
  485. popupType: 0,
  486. title: obj.where.county.name,
  487. lon: obj.where.lon,
  488. lat: obj.where.lat,
  489. type: obj.type.name,
  490. keyword: ['Construction'], //keyword for roadwork/construction
  491. desc: obj.where.location,
  492. time: moment(new Date(obj.timestamp)).format('LLL'),
  493. link: obj.published.linkbackUrl
  494. });
  495. resolve();
  496. }))
  497. break;
  498. case 1:
  499. var pubLink;
  500. if (obj.str.releaseId) {
  501. if (obj.str.releaseId.toString() == "-1") {
  502. pubLink = '';
  503. } else {
  504. pubLink = 'https://deldot.gov/About/news/index.shtml?dc=release&id=' + obj.str.releaseId;
  505. }
  506. }
  507. promisesDE.push(new Promise((resolve, reject) => {
  508. advisoriesDE.push({
  509. state: ['DE', 'Delaware'],
  510. id: obj.str.strId,
  511. popupType: 0,
  512. title: obj.str.county,
  513. lon: obj.str.longitude,
  514. lat: obj.str.latitude,
  515. type: obj.str.impactType,
  516. keyword: ['Closure'], //keywords for roadwork/construction
  517. desc: obj.str.title + " - " + obj.str.construction,
  518. time: moment(new Date(obj.str.actualStartDate)).format('LLL'),
  519. link: pubLink
  520. });
  521. resolve();
  522. }))
  523. break;
  524. case 2:
  525. var direction, WorkType;
  526. switch (obj.attributes.direction) {
  527. case 'ONE_DIRECTION':
  528. direction = "One Way from ";
  529. break;
  530. case 'BOTH_DIRECTIONS':
  531. direction = "Both Ways between ";
  532. }
  533. switch (obj.attributes.WorkType) {
  534. case "1":
  535. WorkType = "Gas Work";
  536. break;
  537. case "2":
  538. WorkType = "Street Work (city)";
  539. break;
  540. case "3":
  541. WorkType = "Street Work (DelDOT)";
  542. break;
  543. case "4":
  544. WorkType = "Water Work";
  545. break;
  546. case "5":
  547. WorkType = "Work (other)";
  548. break;
  549. case "6":
  550. WorkType = "Sewer Work";
  551. }
  552. if (obj.attributes.street != null) {
  553. var originShift = 2.0 * Math.PI * 6378137.0 / 2.0;
  554. var lon = (obj.geometry.paths[0][0][0] / originShift) * 180.0;
  555. var lat = (obj.geometry.paths[0][0][1] / originShift) * 180.0;
  556. lat = 180.0 / Math.PI * (2.0 * Math.atan(Math.exp(lat * Math.PI / 180.0)) - Math.PI / 2.0);
  557. promisesDE.push(new Promise((resolve, reject) => {
  558. advisoriesDE.push({
  559. state: ['DE', 'Delaware'],
  560. id: obj.attributes.OBJECTID,
  561. popupType: 0,
  562. title: 'Wilmington Road Closures',
  563. lon: lon,
  564. lat: lat,
  565. type: 'Closure',
  566. keyword: ['Closure'], //keywords for roadwork/construction
  567. desc: WorkType + ' - ' + direction + obj.attributes.StartingFrom + ' to ' + obj.attributes.EndingAt + '<br> from ' + moment(new Date(obj.attributes.starttime)).format('LLL') + ' to ' + moment(new Date(obj.attributes.endtime)).format('LLL'),
  568. time: moment(new Date(obj.attributes.EditDate)).format('LLL'),
  569. link: ''
  570. });
  571. resolve();
  572. }))
  573. }
  574. }
  575. },
  576. URL: ['https://tmc.deldot.gov/json/advisory.json', 'https://deldot.gov/json/str.json', 'https://services.arcgis.com/hQ3wdpbjO3fPf612/ArcGIS/rest/services/RoadClosures_5b3e88c5556242dfa6e058198be7eb52_public/FeatureServer/1/query?where=1%3D1&objectIds=&time=&geometry=&geometryType=esriGeometryEnvelope&inSR=&spatialRel=esriSpatialRelIntersects&resultType=standard&distance=0.0&units=esriSRUnit_Meter&returnGeodetic=false&outFields=*&returnGeometry=true&featureEncoding=esriDefault&multipatchOption=xyFootprint&maxAllowableOffset=&geometryPrecision=&outSR=&datumTransformation=&applyVCSProjection=false&returnIdsOnly=false&returnUniqueIdsOnly=false&returnCountOnly=false&returnExtentOnly=false&returnQueryGeometry=false&returnDistinctValues=false&cacheHint=false&orderByFields=&groupByFieldsForStatistics=&outStatistics=&having=&resultOffset=&resultRecordCount=&returnZ=false&returnM=false&returnExceededLimitFeatures=true&quantizationParameters=&sqlFormat=none&f=pjson&token=']
  577. },
  578. FL: {
  579. data(res, index) {
  580. let resultText = [res.item2, res.item2];
  581. return (resultText[index]);
  582. },
  583. filter(index) {
  584. let filtertext = [true, true];
  585. return (filtertext[index]);
  586. },
  587. scheme(obj, index) {
  588. switch (index) {
  589. case 0:
  590. promisesFL.push(new Promise((resolve, reject) => {
  591. getFeed(config.FL.detailURL[0] + obj.itemId.replace("/ /g", "%20"), async function (result) {
  592. var eventObj = JSON.parse(result.responseText);
  593. if (eventObj.lanes) {
  594. if (['Alllanesblocked', 'On-rampclosed', 'Off-rampclosed'].includes(eventObj.lanes.lanesAffected.replace(/ /g, ""))) {
  595. advisoriesFL.push({
  596. state: ['FL', 'Florida'],
  597. id: eventObj.id.id,
  598. popupType: 0,
  599. title: eventObj.areas.area5.areaLang1,
  600. lon: eventObj.coordinates.locationLongitude / 1000000,
  601. lat: eventObj.coordinates.locationLatitude / 1000000,
  602. type: eventObj.details.detailLang1.eventTypeName,
  603. keyword: ['Construction'], //keywords for roadwork/construction
  604. desc: eventObj.details.detailLang1.eventDescription,
  605. time: moment(new Date(eventObj.dates.lastUpdated)).format('LLL'),
  606. link: ''
  607. });
  608. }
  609. }
  610. resolve();
  611. });
  612. }));
  613. break;
  614. case 1:
  615. promisesFL.push(new Promise((resolve, reject) => {
  616. getFeed(config.FL.detailURL[1] + obj.itemId.replace("/ /g", "%20"), async function (result) {
  617. var eventObj = JSON.parse(result.responseText);
  618. if (eventObj.lanes) {
  619. if (['Alllanesblocked', 'On-rampclosed', 'Off-rampclosed'].includes(eventObj.lanes.lanesAffected.replace(/ /g, ""))) {
  620. advisoriesFL.push({
  621. state: ['FL', 'Florida'],
  622. id: eventObj.id.id,
  623. popupType: 0,
  624. title: eventObj.areas.area5.areaLang1,
  625. lon: eventObj.coordinates.locationLongitude / 1000000,
  626. lat: eventObj.coordinates.locationLatitude / 1000000,
  627. type: eventObj.details.detailLang1.eventTypeName,
  628. keyword: ['Constructions'], //keywords for roadwork/construction
  629. desc: eventObj.details.detailLang1.eventDescription,
  630. time: moment(new Date(eventObj.dates.lastUpdated)).format('LLL'),
  631. link: ''
  632. });
  633. }
  634. }
  635. resolve();
  636. });
  637. }));
  638. }
  639. },
  640. URL: ['https://fl511.com/map/mapIcons/Incidents?_=1604955914474', 'https://fl511.com/map/mapIcons/Construction?_=1604965926997'],
  641. detailURL: ['https://fl511.com/map/data/Incidents/', 'https://fl511.com/map/data/Construction/']
  642. },
  643. GA: {
  644. data(res, index) {
  645. let resultText = [res.features, res.features];
  646. return (resultText[index]);
  647. },
  648. filter(index) {
  649. let filtertext = [true, true];
  650. return (filtertext[index]);
  651. },
  652. scheme(obj, index) {
  653. switch (index) {
  654. case 0:
  655. promisesGA.push(new Promise((resolve, reject) => {
  656. advisoriesGA.push({
  657. state: ['GA', 'Georgia'],
  658. id: obj.id,
  659. popupType: 0,
  660. title: obj.properties.headline + ' - ' + obj.properties.route,
  661. lon: obj.geometry.coordinates[0],
  662. lat: obj.geometry.coordinates[1],
  663. type: obj.properties.headline,
  664. keyword: ['Construction', 'Emergency Roadwork'], //keyword for roadwork/construction
  665. desc: obj.properties.location_description + '<br>' + obj.properties.lanes,
  666. time: moment(new Date(obj.properties.start * 1000)).format('LLL'),
  667. link: ''
  668. });
  669. resolve();
  670. }))
  671. break;
  672. case 1:
  673. promisesGA.push(new Promise((resolve, reject) => {
  674. advisoriesGA.push({
  675. state: ['GA', 'Georgia'],
  676. id: obj.id,
  677. popupType: 0,
  678. title: obj.properties.headline + ' - ' + obj.properties.route,
  679. lon: obj.geometry.coordinates[0],
  680. lat: obj.geometry.coordinates[1],
  681. type: obj.properties.headline,
  682. keyword: ['Construction', 'Emergency Roadwork'], //keyword for roadwork/construction
  683. desc: obj.properties.location_description + '<br>' + obj.properties.lanes,
  684. time: moment(new Date(obj.properties.start * 1000)).format('LLL'),
  685. link: ''
  686. });
  687. resolve();
  688. }))
  689. }
  690. },
  691. URL: ['https://ga.cdn.iteris-atis.com/geojson/icons/metadata/icons.construction.geojson', 'https://ga.cdn.iteris-atis.com/geojson/icons/metadata/icons.incident.geojson']
  692. },
  693. LA: {
  694. data(res, index) {
  695. let resultText = [res];
  696. return (resultText[index]);
  697. },
  698. filter(index) {
  699. let filtertext = ['(resultObj[i].LanesAffected).replace(/ +(?= )/g, "") == ("All Lanes Closed")'];
  700. return (filtertext[index]);
  701. },
  702. scheme(obj, index) {
  703. promisesLA.push(new Promise((resolve, reject) => {
  704. advisoriesLA.push({
  705. state: ['LA', 'Louisiana'],
  706. id: obj.ID,
  707. popupType: 0,
  708. title: obj.RoadwayName,
  709. lon: obj.Longitude,
  710. lat: obj.Latitude,
  711. type: obj.EventType,
  712. keyword: ['roadwork'], //keywords for roadwork/construction
  713. desc: obj.Description,
  714. time: moment(new Date(obj.LastUpdated * 1000)).format('LLL'),
  715. link: ''
  716. });
  717. resolve();
  718. }))
  719. },
  720. URL: ['http://scripts.essentialintegrations.com/LA']
  721. },
  722. MD: {
  723. data(res, index) {
  724. let resultText = [res.features];
  725. return (resultText[index]);
  726. },
  727. filter(index) {
  728. let filtertext = [true];
  729. return (filtertext[index]);
  730. },
  731. scheme(obj, index) {
  732. promisesMD.push(new Promise((resolve, reject) => {
  733. advisoriesMD.push({
  734. state: ['MD', 'Maryland'],
  735. id: obj.attributes.OBJECTID,
  736. popupType: 0,
  737. title: obj.attributes.Jurisdiction,
  738. lon: obj.attributes.Longitude,
  739. lat: obj.attributes.Latitude,
  740. type: obj.attributes.typeSummary,
  741. keyword: ['Construction'], //keywords for roadwork/construction
  742. desc: obj.attributes.ClosureSummary,
  743. time: moment(new Date(obj.attributes.EditDate)).format('LLL'),
  744. link: ''
  745. });
  746. resolve();
  747. }))
  748. },
  749. URL: ['https://geodata.md.gov/appdata/rest/services/SHA_RoadClosure/RoadClosureActive/MapServer//0/query?where=1%3D1&outFields=*&outSR=4326&f=json']
  750. },
  751. MI: {
  752. data(res, index) {
  753. let resultText = [res, res];
  754. return (resultText[index]);
  755. },
  756. filter(index) {
  757. let filtertext = ['(resultObj[i].type == "Total")', true];
  758. return (filtertext[index]);
  759. },
  760. scheme(obj, index) {
  761. switch (index) {
  762. case 0:
  763. promisesMI.push(new Promise((resolve, reject) => {
  764. advisoriesMI.push({
  765. state: ['MI', 'Michigan'],
  766. id: "00",
  767. popupType: 0,
  768. title: obj.county,
  769. lon: obj.description.match(/(?<=lon=)[\s\S]*(?=&zoom)/)[0],
  770. lat: obj.description.match(/(?<=lat=)[\s\S]*(?=&lon)/)[0],
  771. type: "Construction",
  772. keyword: ['Construction'], //keyword for roadwork/construction
  773. desc: obj.description.match(/(?:(?!<).)*/),
  774. time: moment(new Date(obj.startDate)).format('LL'),
  775. link: ''
  776. });
  777. resolve();
  778. }))
  779. break;
  780. case 1:
  781. promisesMI.push(new Promise((resolve, reject) => {
  782. advisoriesMI.push({
  783. state: ['MI', 'Michigan'],
  784. id: "",
  785. popupType: 0,
  786. title: obj.county,
  787. lon: obj.location.match(/(?<=lon=)[\s\S]*(?=&zoom)/)[0],
  788. lat: obj.location.match(/(?<=lat=)[\s\S]*(?=&lon)/)[0],
  789. type: "Incident",
  790. keyword: ['Construction'], //keyword for roadwork/construction
  791. desc: obj.location.match(/(?:(?!<).)*/),
  792. time: moment(new Date(obj.reported)).format('LLL'),
  793. link: ''
  794. });
  795. resolve();
  796. }))
  797. }
  798. },
  799. URL: ['https://mdotjboss.state.mi.us/MiDrive//construction/list/loadConstruction', 'https://mdotjboss.state.mi.us/MiDrive//incident/list/loadIncidents']
  800. },
  801. NC: {
  802. data(res, index) {
  803. let resultText = [res.features];
  804. return (resultText[index]);
  805. },
  806. filter(index) {
  807. let filtertext = [true];
  808. return (filtertext[index]);
  809. },
  810. scheme(obj, index) {
  811. promisesNC.push(new Promise((resolve, reject) => {
  812. advisoriesNC.push({
  813. state: ['NC', 'North Carolina'],
  814. id: obj.attributes.OBJECTID,
  815. popupType: 0,
  816. title: obj.attributes.CountyName,
  817. lon: obj.attributes.Longitude,
  818. lat: obj.attributes.Latitude,
  819. type: obj.attributes.IncidentType,
  820. keyword: ['Construction', 'Emergency Road Work', 'Night Time Construction', 'Weekend Construction'], //keywords for roadwork/construction
  821. desc: obj.attributes.Reason,
  822. time: moment(new Date(obj.attributes.LastUpdateUTC)).format('LLL'),
  823. link: obj.attributes.DriveNCLink
  824. });
  825. resolve();
  826. }))
  827. },
  828. URL: ['https://services.arcgis.com/NuWFvHYDMVmmxMeM/ArcGIS/rest/services/NCDOT_TIMSIncidents/FeatureServer/0/query?where=Condition+%3D+%27Road+Closed%27&objectIds=&time=&geometry=&geometryType=esriGeometryEnvelope&inSR=&spatialRel=esriSpatialRelIntersects&resultType=none&distance=0.0&units=esriSRUnit_Meter&returnGeodetic=false&outFields=*&returnGeometry=true&featureEncoding=esriDefault&multipatchOption=xyFootprint&maxAllowableOffset=&geometryPrecision=&outSR=&datumTransformation=&applyVCSProjection=false&returnIdsOnly=false&returnUniqueIdsOnly=false&returnCountOnly=false&returnExtentOnly=false&returnQueryGeometry=false&returnDistinctValues=false&cacheHint=false&orderByFields=&groupByFieldsForStatistics=&outStatistics=&having=&resultOffset=&resultRecordCount=&returnZ=false&returnM=false&returnExceededLimitFeatures=true&quantizationParameters=&sqlFormat=none&f=pjson&token=']
  829. },
  830. NJ: {
  831. data(res, index) {
  832. let resultText = [res.Data.features];
  833. return (resultText[index]);
  834. },
  835. filter(index) {
  836. let filtertext = [true];
  837. return (filtertext[index]);
  838. },
  839. scheme(obj, index) {
  840. promisesNJ.push(new Promise((resolve, reject) => {
  841. getFeed(NJURLDetail + obj.properties.EventID, async function (result) {
  842. var eventObj = JSON.parse(result.responseText).Data;
  843. if (((eventObj[0].FullText.toUpperCase()).includes("ALL LANES CLOSE") || (eventObj[0].FullText.toUpperCase()).includes("RAMP CLOSE")) && ((eventObj[0].FullText).includes("NYSDOT") != true)) {
  844. advisoriesNJ.push({
  845. state: ['NJ', 'New Jersey'],
  846. id: eventObj[0].markerId,
  847. popupType: 0,
  848. title: eventObj[0].County,
  849. lon: eventObj[0].Longitude,
  850. lat: eventObj[0].Latitude,
  851. type: eventObj[0].CategoryName,
  852. keyword: ['Construction', 'ScheduledConstruction'], //keywords for roadwork/construction
  853. desc: eventObj[0].FullText,
  854. time: moment(new Date(eventObj[0].LastUpdateDate_String)).format('LLL'),
  855. link: ''
  856. });
  857. }
  858. resolve();
  859. });
  860. }));
  861. },
  862. URL: ['https://511nj.org/API/client/Map/getEventData']
  863. },
  864. NV: {
  865. data(res, index) {
  866. let resultText = [res.d];
  867. return (resultText[index]);
  868. },
  869. filter(index) {
  870. let filtertext = [true];
  871. return (filtertext[index]);
  872. },
  873. scheme(obj, index) {
  874. promisesNV.push(new Promise((resolve, reject) => {
  875. let unix = obj.LastUpdate.replace(/\\\//g, "").replace("/Date(", "").replace(")/", "");
  876. advisoriesNV.push({
  877. state: ['NV', 'Nevada'],
  878. id: obj.ID,
  879. popupType: 0,
  880. title: obj.Facility,
  881. lon: obj.Lon,
  882. lat: obj.Lat,
  883. type: obj.CategoryName,
  884. keyword: ['Construction'], //keywords for roadwork/construction
  885. desc: obj.Description,
  886. time: moment(new Date(parseInt(unix))).format('LLL'),
  887. link: ''
  888. });
  889. resolve();
  890. }))
  891. },
  892. URL: ['http://scripts.essentialintegrations.com/NV']
  893. },
  894. NY: {
  895. data(res, index) {
  896. let resultText = [res];
  897. return (resultText[index]);
  898. },
  899. filter(index) {
  900. let filtertext = ['((NotNY.includes(resultObj[i].RegionName) == false && resultObj[i].EventType != "transitMode" && resultObj[i].EventSubType != "Capacity related") && (resultObj[i].EventType == "closures" || (resultObj[i].EventType != "closures" && resultObj[i].LanesAffected == "all lanes" && (resultObj[i].LanesStatus == "closed" || resultObj[i].LanesStatus == "blocked"))))'];
  901. return (filtertext[index]);
  902. },
  903. scheme(obj, index) {
  904. promisesNY.push(new Promise((resolve, reject) => {
  905. advisoriesNY.push({
  906. state: ['NY', 'New York'],
  907. id: obj.ID,
  908. popupType: 0,
  909. title: obj.CountyName,
  910. lon: obj.Longitude,
  911. lat: obj.Latitude,
  912. type: obj.EventType,
  913. keyword: ['roadwork', 'transitMode', 'closures'], //keywords for roadwork/construction
  914. desc: obj.Description,
  915. time: moment(moment(obj.LastUpdated, "DD/MM/YYYY HH:mm:ss")).format('LLL'),
  916. link: ''
  917. });
  918. resolve();
  919. }))
  920. },
  921. URL: ['http://scripts.essentialintegrations.com/NY']
  922. },
  923. OH: {
  924. data(res, index) {
  925. let resultText = [res.ConstructionMarkers];
  926. return (resultText[index]);
  927. },
  928. filter(index) {
  929. let filtertext = ['(resultObj[i].Status == "Closed")'];
  930. return (filtertext[index]);
  931. },
  932. scheme(obj, index) {
  933. promisesOH.push(new Promise((resolve, reject) => {
  934. advisoriesOH.push({
  935. state: ['OH', 'Ohio'],
  936. id: obj.ID,
  937. popupType: 0,
  938. title: obj.District,
  939. lon: obj.Longitude,
  940. lat: obj.Latitude,
  941. type: obj.Category,
  942. keyword: ['Roadwork - Planned', 'Roadwork - Unplanned'], //keywords for roadwork/construction
  943. desc: obj.Description,
  944. time: moment(new Date(obj.StartDate)).format('LL'),
  945. link: ''
  946. });
  947. resolve();
  948. }))
  949. },
  950. URL: ['https://api.ohgo.com/roadmarkers/TrafficSpeedAndAlertMarkers']
  951. },
  952. OR: {
  953. data(res, index) {
  954. let resultText = [res.features, res.features, res.features];
  955. return (resultText[index]);
  956. },
  957. filter(index) {
  958. let filtertext = [true, 'resultObj[i].attributes.comments.includes("clos")', 'resultObj[i].attributes.tmddOther.includes("clos")'];
  959. return (filtertext[index]);
  960. },
  961. scheme(obj, index) {
  962. switch (index) {
  963. case 0:
  964. let x, y;
  965. var lonlat = obj.geometry.paths[0][0];
  966. promisesOR.push(new Promise((resolve, reject) => {
  967. x = obj.geometry.paths[0][0].toString().split(",")[0];
  968. y = obj.geometry.paths[0][0].toString().split(",")[1];
  969. advisoriesOR.push({
  970. state: ['OR', 'Oregon'],
  971. id: obj.attributes.OBJECTID,
  972. popupType: 0,
  973. title: obj.attributes.FROM_TO,
  974. lon: x,
  975. lat: y,
  976. type: obj.attributes.CLOSURE_EFFECT,
  977. keyword: ['Street'], //keywords for roadwork/construction
  978. desc: obj.attributes.FROM_TO + " - " + obj.attributes.REMARKS,
  979. time: moment(new Date(obj.attributes.LAST_EDITED_DATE)).format('LLL'),
  980. link: ''
  981. });
  982. resolve();
  983. }))
  984. break;
  985. case 1:
  986. promisesOR.push(new Promise((resolve, reject) => {
  987. advisoriesOR.push({
  988. state: ['OR', 'Oregon'],
  989. id: obj.attributes.incidentId,
  990. popupType: 0,
  991. title: obj.attributes.locationName,
  992. lon: obj.attributes.startLongitude,
  993. lat: obj.attributes.startLatitude,
  994. type: obj.attributes.type,
  995. keyword: ['ROADWORK'], //keywords for roadwork/construction
  996. desc: obj.attributes.comments + " <br> " + obj.attributes.beginMarker + " to " + obj.attributes.endMarker,
  997. time: moment(new Date(obj.attributes.lastUpdated)).format('LLL'),
  998. link: ''
  999. });
  1000. resolve();
  1001. }))
  1002. break;
  1003. case 2:
  1004. promisesOR.push(new Promise((resolve, reject) => {
  1005. advisoriesOR.push({
  1006. state: ['OR', 'Oregon'],
  1007. id: obj.attributes.incidentId,
  1008. popupType: 0,
  1009. title: obj.attributes.locationName,
  1010. lon: obj.attributes.startLongitude,
  1011. lat: obj.attributes.startLatitude,
  1012. type: obj.attributes.type,
  1013. keyword: ['EVENT'], //keywords for roadwork/construction
  1014. desc: obj.attributes.tmddOther + " <br> " + obj.attributes.beginMarker + " to " + obj.attributes.endMarker,
  1015. time: moment(new Date(obj.attributes.lastUpdated)).format('LLL'),
  1016. link: ''
  1017. });
  1018. resolve();
  1019. }))
  1020. }
  1021. },
  1022. URL: ['https://services.arcgis.com/kIA6yS9KDGqZL7U3/ArcGIS/rest/services/RoadWork/FeatureServer/1/query?where=objectid+like+%27%25%27&objectIds=&time=&geometry=&geometryType=esriGeometryEnvelope&inSR=&spatialRel=esriSpatialRelIntersects&resultType=none&distance=0.0&units=esriSRUnit_Meter&returnGeodetic=false&outFields=*&returnGeometry=true&featureEncoding=esriDefault&multipatchOption=xyFootprint&maxAllowableOffset=&geometryPrecision=&outSR=4326&datumTransformation=&applyVCSProjection=true&returnIdsOnly=false&returnUniqueIdsOnly=false&returnCountOnly=false&returnExtentOnly=false&returnQueryGeometry=true&returnDistinctValues=false&cacheHint=false&orderByFields=&groupByFieldsForStatistics=&outStatistics=&having=&resultOffset=&resultRecordCount=&returnZ=false&returnM=false&returnExceededLimitFeatures=true&quantizationParameters=&sqlFormat=none&f=pjson&token=', 'https://tripcheck.com/Scripts/map/data/INCD.js?dt=1607132941398', 'https://tripcheck.com/Scripts/map/data/EVENT.js?dt=1607134261397']
  1023. },
  1024. PA: {
  1025. data(res, index) {
  1026. let resultText = [res, res.features];
  1027. return (resultText[index]);
  1028. },
  1029. filter(index) {
  1030. let filtertext = ['(resultObj[i].LaneStatus == "closed") || (resultObj[i].LaneStatus == "ramp closure")', true];
  1031. return (filtertext[index]);
  1032. },
  1033. scheme(obj, index) {
  1034. switch (index) {
  1035. case 0:
  1036. let status = obj.LaneStatus;
  1037. let x, y;
  1038. if (status != "ramp closure") {
  1039. x = obj.FromLocLatLong.split(",")[1];
  1040. y = obj.FromLocLatLong.split(",")[0];
  1041. } else {
  1042. x = obj.IncidentLocLatLong.split(",")[1];
  1043. y = obj.IncidentLocLatLong.split(",")[0];
  1044. }
  1045. promisesPA.push(new Promise((resolve, reject) => {
  1046. advisoriesPA.push({
  1047. state: ['PA', 'Pennsylvania'],
  1048. id: obj.EventID,
  1049. popupType: 0,
  1050. title: obj.CountyName,
  1051. lon: x,
  1052. lat: y,
  1053. type: obj.EventType,
  1054. keyword: ['roadwork', 'bridge outage'], //keywords for roadwork/construction
  1055. desc: obj.Description,
  1056. time: moment(new Date(obj.LastUpdate)).format('LLL'),
  1057. link: ''
  1058. });
  1059. resolve();
  1060. }))
  1061. break;
  1062. case 1:
  1063. var originShift = 2.0 * Math.PI * 6378137.0 / 2.0;
  1064. var lon = (obj.geometry.x / originShift) * 180.0;
  1065. var lat = (obj.geometry.y / originShift) * 180.0;
  1066. lat = 180.0 / Math.PI * (2.0 * Math.atan(Math.exp(lat * Math.PI / 180.0)) - Math.PI / 2.0);
  1067. var timing;
  1068. if (obj.attributes.END_DATE == null) {
  1069. timing = moment(new Date(obj.attributes.START_DATE)).format('LL');
  1070. } else {
  1071. timing = moment(new Date(obj.attributes.START_DATE)).format('LL') + ' to ' + moment(new Date(obj.attributes.END_DATE)).format('LL');
  1072. }
  1073. if (obj.attributes.START_TIME != null) {
  1074. timing = timing + '<br>' + obj.attributes.START_TIME + ' to ' + obj.attributes.END_TIME;
  1075. }
  1076. promisesPA.push(new Promise((resolve, reject) => {
  1077. advisoriesPA.push({
  1078. state: ['PA', 'Pennsylvania'],
  1079. id: obj.attributes.GlobalID,
  1080. popupType: 0,
  1081. title: 'HARRISBURG',
  1082. lon: lon,
  1083. lat: lat,
  1084. type: 'Closure',
  1085. keyword: ['Closure'], //keywords for roadwork/construction
  1086. desc: obj.attributes.STREET_NAME + ' between ' + obj.attributes.STREET_FROM + ' and ' + obj.attributes.STREET_TO + '<br>' + obj.attributes.REASON_FOR_CLOSURE + '<br>' + timing,
  1087. time: moment(new Date(obj.attributes.EditDate)).format('LLL'),
  1088. link: ''
  1089. });
  1090. resolve();
  1091. }))
  1092. }
  1093. },
  1094. URL: ['http://scripts.essentialintegrations.com/PA', 'https://services5.arcgis.com/9n3LUAMi3B692MBL/arcgis/rest/services/Street_Closures_for_GIS_2017/FeatureServer/0/query?f=json&where=1%3D1&returnGeometry=true&spatialRel=esriSpatialRelIntersects&outFields=*&outSR=102100&resultOffset=0&resultRecordCount=2000']
  1095. },
  1096. WA: {
  1097. data(res, index) {
  1098. let resultText = [res];
  1099. return (resultText[index]);
  1100. },
  1101. filter(index) {
  1102. let filtertext = ['(resultObj[i].EventCategory == "Closure" || resultObj[i].EventCategory == "Construction" || resultObj[i].EventCategory == "Bridge")'];
  1103. return (filtertext[index]);
  1104. },
  1105. scheme(obj, index) {
  1106. let county;
  1107. if (obj.County == null) {
  1108. county = obj.Region;
  1109. } else {
  1110. county = obj.County;
  1111. }
  1112. let unixtime = parseInt(obj.LastUpdatedTime.replace("/Date(", "").replace(")/", "").split("-")[0]);
  1113. promisesWA.push(new Promise((resolve, reject) => {
  1114. advisoriesWA.push({
  1115. state: ['WA', 'Washington'],
  1116. id: obj.AlertID,
  1117. popupType: 0,
  1118. title: county,
  1119. lon: obj.StartRoadwayLocation.Longitude,
  1120. lat: obj.StartRoadwayLocation.Latitude,
  1121. type: obj.EventCategory,
  1122. keyword: ['Construction', 'Closures'], //keywords for roadwork/construction
  1123. desc: obj.HeadlineDescription,
  1124. time: moment(new Date(unixtime)).format('LLL'),
  1125. link: ''
  1126. });
  1127. resolve();
  1128. }))
  1129. },
  1130. URL: ['http://scripts.essentialintegrations.com/WA']
  1131. }
  1132. };
  1133. })();