WME DOT Advisories

Overlay DOT Advisories on the WME Map Object

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

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         WME DOT Advisories
// @namespace    https://greasyfork.org/en/users/668704-phuz
// @require      https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @version      1.54
// @description  Overlay DOT Advisories on the WME Map Object
// @author       phuz
// @include      /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @require      http://cdnjs.cloudflare.com/ajax/libs/jquery/1.9.1/jquery.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/tablesort/5.2.1/tablesort.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/tablesort/5.2.1/sorts/tablesort.number.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/tablesort/5.2.1/sorts/tablesort.date.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.1/moment.min.js
// @grant        GM_xmlhttpRequest
// @grant        GM_info
// @grant        GM_fetch
// @grant        GM_addStyle
// @require      https://cdn.jsdelivr.net/npm/hls.js@latest
// @connect      511nj.org
// @connect      511ny.org
// @connect      511pa.com
// @connect      511wi.gov
// @connect      arcgis.com
// @connect      deldot.gov
// @connect      essentialintegrations.com
// @connect      fl511.com
// @connect      md.gov
// @connect      mi.us
// @connect      ohgo.com
// @connect      rehostjson.phuz.repl.co
// @connect      tripcheck.com
// @connect      iteris-atis.com
// @connect      carsprogram.org
// @connect      phoenix.gov
// @connect      511virginia.org
/* global OpenLayers */
/* global W */
/* global WazeWrap */
/* global $ */
/* global I18n */
/* global _ */
// ==/UserScript==

let AKDOTLayer, AZDOTLayer, CTDOTLayer, DEDOTLayer, FLDOTLayer, GADOTLayer, ILDOTLayer, INDOTLayer, LADOTLayer, MDDOTLayer, MIDOTLayer, MNDOTLayer, NCDOTLayer, NJDOTLayer, NVDOTLayer, NYDOTLayer, OHDOTLayer, ORDOTLayer, PADOTLayer, TXDOTLayer, VADOTLayer, WADOTLayer, WIDOTLayer, WVDOTLayer;
let promisesAK, promisesAZ, promisesCT, promisesDE, promisesFL, promisesGA, promisesIL, promisesIN, promisesLA, promisesMD, promisesMI, promisesMN, promisesNC, promisesNJ, promisesNV, promisesNY, promisesOH, promisesOR, promisesPA, promisesTX, promisesVA, promisesWA, promisesWI, promisesWV;
let advisoriesAK, advisoriesAZ, advisoriesCT, advisoriesDE, advisoriesFL, advisoriesGA, advisoriesIL, advisoriesIN, advisoriesLA, advisoriesMD, advisoriesMI, advisoriesMN, advisoriesNC, advisoriesNJ, advisoriesNV, advisoriesNY, advisoriesOH, advisoriesOR, advisoriesPA, advisoriesTX, advisoriesVA, advisoriesWA, advisoriesWI, advisoriesWV;
let ALFeed = [], AKFeed = [], AZFeed = [], ARFeed = [], CAFeed = [], COFeed = [], CTFeed = [], DEFeed = [], DCFeed = [], FLFeed = [], GAFeed = [], HIFeed = [], IDFeed = [], ILFeed = [], INFeed = [], IAFeed = [], KSFeed = [], KYFeed = [], LAFeed = [], MEFeed = [], MDFeed = [], MAFeed = [], MIFeed = [], MNFeed = [], MSFeed = [], MOFeed = [], MTFeed = [], NEFeed = [], NVFeed = [], NHFeed = [], NJFeed = [], NMFeed = [], NYFeed = [], NWFeed = [], NCFeed = [], NDFeed = [], OHFeed = [], OKFeed = [], ORFeed = [], PAFeed = [], RIFeed = [], SCFeed = [], SDFeed = [], TNFeed = [], TXFeed = [], UTFeed = [], VTFeed = [], VAFeed = [], WAFeed = [], WIFeed = [], WVFeed = [], WYFeed;
let settings;
let state, stateLength, advisories;
const updateMessage = "► Added MN";
const DEIconC = '';
const DEIconSchRestriction = '';
const DEIconSchClosure = '';
const Incident = '';
const Roadwork = '';
const PLIcon = '';
const reportIcon = '';
const NJURLDetail = 'https://511nj.org/API/client/Map/getEventPopupData?EventId=';
const NotNY = ['Pennsylvania Statewide', 'New Jersey Statewide', 'Connecticut Statewide'];
const NJConstruction = ['Construction', 'ScheduledConstruction'];

//Begin script function
(function () {
    'use strict';
    //Bootstrap
    window["text123"] = 123;
    function bootstrap(tries = 1) {
        if (W && W.loginManager && W.map && W.loginManager.user && W.model
            && W.model.states && W.model.states.getObjectArray().length && WazeWrap && WazeWrap.Ready) {
            console.log("WME DOT Advisories Loaded!");
            init();
            addListeners();
            if (!OpenLayers.Icon) {
                installIcon();
            }
        } else if (tries < 1000) {
            setTimeout(function () { bootstrap(++tries); }, 200);
        }
    }
    Function.prototype.bind = function (thisObject) {
        var method = this;
        var oldargs = [].slice.call(arguments, 1);
        return function () {
            var newargs = [].slice.call(arguments);
            return method.apply(thisObject, oldargs.concat(newargs));
        };
    }
    //Build the Tab and Settings Division
    function init() {
        var $section = $("<div>");
        $section.html([
            '<div id="chkEnables">',
            '<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>',
            '* The WME Refresh Button will update reports.',
            '<table border=1 style="text-align:center;width:100%;padding:10px;">',
            '<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>',
            '<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>',
            '<tr><td><input type="checkbox" id="chkAZDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>AZ</td><td><div class=DOTreport data-report="report" data-state="Arizona" id="AZ"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td><input type="checkbox" id="chkCTDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>CT</td><td><div class=DOTreport data-report="report" data-state="Connecticut" id="CT"><img src=' + reportIcon + '></div></td></tr>',
            '<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>',
            '<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>',
            '<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>',
            '<tr><td><input type="checkbox" id="chkILDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>IL</td><td><div class=DOTreport data-report="report" data-state="Illinois" id="IL"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td><input type="checkbox" id="chkINDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>IN</td><td><div class=DOTreport data-report="report" data-state="Indiana" id="IN"><img src=' + reportIcon + '></div></td></tr>',
            '<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>',
            '<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>',
            '<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>',
            '<tr><td><input type="checkbox" id="chkMNDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>MN</td><td><div class=DOTreport data-report="report" data-state="Minnesota" id="MN"><img src=' + reportIcon + '></div></td></tr>',
            '<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>',
            '<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>',
            '<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>',
            '<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>',
            '<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>',
            '<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>',
            '<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>',
            '<tr><td><input type="checkbox" id="chkTXDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>TX (Houston)</td><td><div class=DOTreport data-report="report" data-state="Texas" id="TX"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td><input type="checkbox" id="chkVADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>VA</td><td><div class=DOTreport data-report="report" data-state="Virginia" id="VA"><img src=' + reportIcon + '></div></td></tr>',
            '<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>',
            '<tr><td><input type="checkbox" id="chkWIDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>WI</td><td><div class=DOTreport data-report="report" data-state="Wisconsin" id="WI"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td><input type="checkbox" id="chkWVDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td>WV</td><td><div class=DOTreport data-report="report" data-state="West Virginia" id="WV"><img src=' + reportIcon + '></div></td></tr>',
            '</table>',
            '</div></div>'
        ].join(' '));
        new WazeWrap.Interface.Tab('DOT Advisories', $section.html(), initializeSettings);
        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");
    }
    getFeed("http://scripts.essentialintegrations.com/CSS", function (result) {
        GM_addStyle(result.responseText);
    })
    //Build the State Layers
    function buildDOTAdvLayers(state) {
        eval(state.substring(0, 2) + 'DOTLayer = new OpenLayers.Layer.Markers("' + state.substring(0, 2) + 'DOTLayer")');
        W.map.addLayer(eval(state.substring(0, 2) + 'DOTLayer'));
        W.map.getOLMap().setLayerIndex(eval(state.substring(0, 2) + 'DOTLayer'), 0);
    }
    function getFeed(url, callback) {
        GM_xmlhttpRequest({
            method: "GET",
            url: url,
            onload: callback_function1.bind({}, callback)
            //onload: function (response) {
            //    var result = response;
            //    callback(result);
            //}
        });
    }
    function callback_function1(callback, response) {
        var result = response;
        setTimeout(function () { callback(result) }, 150);
    }
    const timer = ms => new Promise(res => setTimeout(res, ms))
    function getAdvisories(state, stateAbv, type) {
        eval('promises' + stateAbv + ' = []');
        eval('advisories' + stateAbv + ' = []');
        let thesepromises = [];
        for (let j = 0; j < state.URL.length; j++) {
            let thispromise = new Promise((resolve, reject) => {
                getFeed(state.URL[j], function (result) {
                    let resultObj = [];
                    resultObj = state.data(JSON.parse(result.responseText), j);
                    async function innerLoop() {
                        for (let i = 0; i < resultObj.length; i++) {
                            if (eval(state.filter(j))) {
                                state.scheme(resultObj[i], j);
                            }
                            if (i == (resultObj.length - 1)) {
                                resolve();
                            }
                            await timer(1);
                        }
                    }
                    innerLoop();
                });
            })
            thesepromises.push(thispromise);
        }
        Promise.all(thesepromises).then(function () {
            setTimeout(function () { promiseWorker(stateAbv, type) }, 1000);
        })
    }
    function promiseWorker(stateAbv, type) {
        let thisadvisory = eval("advisories" + stateAbv);
        Promise.all(eval('promises' + stateAbv))
            .then(function () {
                for (let i = 0; i < thisadvisory.length; i++) {
                    if (type == "report") {
                        let parms = thisadvisory[i];
                        let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
                        var row = table.insertRow(-1);
                        var cell1 = row.insertCell(0);
                        var cell2 = row.insertCell(1);
                        var cell3 = row.insertCell(2);
                        var cell4 = row.insertCell(3);
                        cell1.innerHTML = '<div class="gotoPL" data-lat="' + parms.lat + '" data-lon="' + parms.lon + '"><img src=' + PLIcon + '></div>'; //PL
                        cell2.innerHTML = parms.desc;  //Description
                        cell3.innerHTML = parms.title; //Location
                        cell4.innerHTML = parms.time;  //Time
                    } else {
                        drawMarkers(thisadvisory[i]);
                    }
                }
                if ((type == "report")) { //Wait until we loop through all the advisory URLs before sorting the table
                    reportWorker();
                }
            });
    }
    function reportWorker() {
        var elements = document.getElementsByClassName("gotoPL");
        for (var i = 0; i < elements.length; i++) {
            elements[i].addEventListener('click', moveMap, false);
        }
        refreshReportTable();
    }
    function refreshReportTable() {
        var sort = new Tablesort(document.getElementById('reportTable'), { descending: true });
        sort.refresh();
        document.getElementById("spinner").style.visibility = "hidden";
    }
    function getReportData(stateAbv, stateName) {
        popupdetails(stateName);
        //if (stateAbv != "NJ") {
        eval('getAdvisories(config.' + stateAbv + ',"' + stateAbv + '", "report")');
        //} else { getNJDOT("report"); }
    }
    //Generate the Advisory markers
    function drawMarkers(parms) {
        var icontype;
        var size = new OpenLayers.Size(20, 20);
        var offset = new OpenLayers.Pixel(-(size.w / 2), -size.h);
        for (let i = 0; i < parms.keyword.length; i++) { //Check each of the keywords for roadwork/construction
            if (parms.type == parms.keyword[i]) {
                icontype = Roadwork;
                break;
            } else {
                icontype = Incident;
            }
        }
        var icon = new OpenLayers.Icon(icontype, size);
        var epsg4326 = new OpenLayers.Projection("EPSG:4326"); //WGS 1984 projection
        var projectTo = W.map.getProjectionObject(); //The map projection (Spherical Mercator)
        var lonLat = new OpenLayers.LonLat(parms.lon, parms.lat).transform(epsg4326, projectTo);
        var newMarker = new OpenLayers.Marker(lonLat, icon);
        newMarker.eventId = parms.id;
        newMarker.title = parms.title;
        newMarker.desc = parms.desc;
        newMarker.popupType = parms.popupType;
        newMarker.state = parms.state;
        newMarker.timestamp = parms.time;
        newMarker.startTime = parms.startTime;
        newMarker.plannedEndTime = parms.plannedEndTime;
        newMarker.events.register('click', newMarker, popup);
        newMarker.location = lonLat;
        newMarker.recurrence = parms.recurrence;
        if (parms.link != '') {
            newMarker.link = '<a href="' + parms.link + '" target="_blank">Publication Link</a>';
        } else {
            newMarker.link = '';
        }
        eval(parms.state[0] + "DOTLayer.addMarker(newMarker)");
    }
    function popup(evt) {
        $("#gmPopupContainer").remove();
        $("#gmPopupContainer").hide();
        var popupHTML;
        W.map.moveTo(this.location);
        let htmlString = '<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px;z-index: 1100">' +
            '<a href="#close" id="gmCloseDlgBtn" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
            '<table border=0><tr><td><div id="mydivheader" style="min-height: 20px;">' + this.title + '</div></div>' +
            '<hr class="myhrline"/>Updated: ' + this.timestamp.toLocaleString();
        if (this.startTime != null) {
            htmlString += '<br/>Start: ' + this.startTime.toLocaleString();
        }
        if (this.plannedEndTime != null) {
            htmlString += '<br/>Planned End: ' + this.plannedEndTime.toLocaleString();
        }
        if (this.recurrence != null) {
            htmlString += '<br/>Recurrence: ' + this.recurrence;
        }
        htmlString += '<hr class="myhrline"/></td></tr><tr><td>' + this.desc + '</td></tr>' +
            '<tr><td>' + this.link + '</td></tr>' +
            '</table>' +
            '</div>';
        popupHTML = ([htmlString]);
        $("body").append(popupHTML);
        //Position the modal based on the position of the click event
        $("#gmPopupContainer").css({ left: document.getElementById("user-tabs").offsetWidth + W.map.getPixelFromLonLat(W.map.getCenter()).x - document.getElementById("gmPopupContainer").clientWidth - 10 });
        $("#gmPopupContainer").css({ top: document.getElementById("left-app-head").offsetHeight + W.map.getPixelFromLonLat(W.map.getCenter()).y - (document.getElementById("gmPopupContainer").clientHeight / 2) });
        $("#gmPopupContainer").show();
        //Add listener for popup's "Close" button
        $("#gmCloseDlgBtn").click(function () {
            $("#gmPopupContainer").remove();
            $("#gmPopupContainer").hide();
        });
        dragElement(document.getElementById("gmPopupContainer"));
    }
    function popupdetails(stateName) {
        $("#gmPopupContainer").remove();
        $("#gmPopupContainer").hide();
        var popupHTML;
        popupHTML = (['<div id="gmPopupContainer" style="max-width:750px;max-height:500px;margin:1;text-align:center;padding: 5px;z-index: 1100">' +
            '<a href="#close" id="popupdetailsclose" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
            '<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%">' +
            '<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>' +
            '<tr><td>' +
            '<div style="width:720px; height:450px; overflow:auto;"><table id="reportTable" border=1>' +
            '<thead><tr><td data-sort-method="none" width=30><b>PL</b></td><th width=394>Description</th><th width=100>Misc.</th><th data-sort-default width=210>Time</th></tr></thead>' +
            '<tbody></tbody></table></div>' +
            '</td></tr></table>' +
            '</div>'
        ]);
        $("body").append(popupHTML);
        //Position the modal based on the position of the click event
        $("#gmPopupContainer").css({ left: 350 });
        $("#gmPopupContainer").css({ top: 100 });
        $("#gmPopupContainer").show();
        //Add listener for popup's "Close" button
        $("#popupdetailsclose").click(function () {
            $("#gmPopupContainer").remove();
            $("#gmPopupContainer").hide();
        });
        dragElement(document.getElementById("gmPopupContainer"));
    }
    // Make the DIV element draggable
    function dragElement(elmnt) {
        var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
        if (document.getElementById("mydivheader")) {
            // if present, the header is where you move the DIV from:
            document.getElementById("mydivheader").onmousedown = dragMouseDown;
        } else {
            // otherwise, move the DIV from anywhere inside the DIV:
            elmnt.onmousedown = dragMouseDown;
        }
        function dragMouseDown(e) {
            e = e || window.event;
            e.preventDefault();
            // get the mouse cursor position at startup:
            pos3 = e.clientX;
            pos4 = e.clientY;
            document.onmouseup = closeDragElement;
            // call a function whenever the cursor moves:
            document.onmousemove = elementDrag;
        }

        function elementDrag(e) {
            e = e || window.event;
            e.preventDefault();
            // calculate the new cursor position:
            pos1 = pos3 - e.clientX;
            pos2 = pos4 - e.clientY;
            pos3 = e.clientX;
            pos4 = e.clientY;
            // set the element's new position:
            elmnt.style.top = (elmnt.offsetTop - pos2) + "px";
            elmnt.style.left = (elmnt.offsetLeft - pos1) + "px";
        }
        function closeDragElement() {
            // stop moving when mouse button is released:
            document.onmouseup = null;
            document.onmousemove = null;
        }
    }
    //Move map to coordinates specified
    function moveMap() {
        var epsg4326 = new OpenLayers.Projection("EPSG:4326"); //WGS 1984 projection
        var projectTo = W.map.getProjectionObject(); //The map projection (Spherical Mercator)
        var lat = this.getAttribute("data-lat");
        var lon = this.getAttribute("data-lon");
        W.map.moveTo(new OpenLayers.LonLat(lon, lat).transform(epsg4326, projectTo), 6);
    }
    //Initialize Settings
    function initializeSettings() {
        stateLength = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox").length;
        loadSettings();
        //Set the state checkboxes according to saved settings
        for (var i = 0; i < stateLength; i++) {
            state = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].id.replace("chk", "").replace("DOTEnabled", "");
            setChecked('chk' + state + 'DOTEnabled', eval('settings.' + state + 'DOTEnabled'));
        }
        //Build the layers for the selected states
        for (var i = 0; i < stateLength; i++) {
            state = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].id.replace("chk", "").replace("DOTEnabled", "");
            if (document.getElementById('chk' + state + 'DOTEnabled').checked) { buildDOTAdvLayers(state); eval('getAdvisories(config.' + state + ',"' + state + '")') }
        }
    }
    function addListeners() {
        //Add event listener to report icon
        for (var i = 0; i < document.getElementsByClassName("DOTreport").length; i++) {
            document.getElementsByClassName("DOTreport")[i].addEventListener('click', function (e) { getReportData(this.getAttribute("id"), this.getAttribute("data-state")); }, false);
        }
        //Refresh selected states when WME's refresh button is clicked
        document.getElementsByClassName("reload-button-region")[0].addEventListener('click', function (e) {
            for (var i = 0; i < stateLength; i++) {
                state = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].id.replace("chk", "").replace("DOTEnabled", "");
                if (document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].checked) { eval('W.map.removeLayer(' + state + 'DOTLayer)'); }
            }
            initializeSettings();
        });
        //Add Handler for Checkbox Setting Changes
        $('.WMEDOTAdvSettingsCheckbox').change(function () {
            var settingName = $(this)[0].id.substr(3);
            settings[settingName] = this.checked;
            saveSettings();
            if (this.checked) {
                buildDOTAdvLayers(settingName.substring(0, 2));
                eval('getAdvisories(config.' + settingName.substring(0, 2) + ',' + "settingName.substring(0, 2)" + ')');
            }
            else {
                eval('W.map.removeLayer(' + settingName.substring(0, 2) + 'DOTLayer)');
            }
        });
    }
    //Set Checkbox from Settings
    function setChecked(checkboxId, checked) {
        $('#' + checkboxId).prop('checked', checked);
    }
    //Load Saved Settings
    function loadSettings() {
        var loadedSettings = $.parseJSON(localStorage.getItem("WMEDOT_Settings"));
        var defaultSettings = {
            Enabled: false,
        };
        settings = loadedSettings ? loadedSettings : defaultSettings;
        for (var prop in defaultSettings) {
            if (!settings.hasOwnProperty(prop)) {
                settings[prop] = defaultSettings[prop];
            }
        }
    }
    //Save Tab Settings
    function saveSettings() {
        if (localStorage) {
            var localsettings = {};
            for (var i = 0; i < stateLength; i++) {
                state = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].id.replace("chk", "").replace("DOTEnabled", "");
                eval('localsettings.' + state + 'DOTEnabled = document.getElementsByClassName("WMEDOTAdvSettingsCheckbox")[i].checked');
            }
        }
        localStorage.setItem("WMEDOT_Settings", JSON.stringify(localsettings));
    }
    //Add the Icon Class to OpenLayers
    function installIcon() {
        console.log('Installing OpenLayers.Icon');
        OpenLayers.Icon = OpenLayers.Class({
            url: null,
            size: null,
            offset: null,
            calculateOffset: null,
            imageDiv: null,
            px: null,
            initialize: function (a, b, c, d) {
                this.url = a;
                this.size = b || { w: 20, h: 20 };
                this.offset = c || { x: -(this.size.w / 2), y: -(this.size.h / 2) };
                this.calculateOffset = d;
                a = OpenLayers.Util.createUniqueID("OL_Icon_");
                let div = this.imageDiv = OpenLayers.Util.createAlphaImageDiv(a);
                $(div.firstChild).removeClass('olAlphaImg'); // LEAVE THIS LINE TO PREVENT WME-HARDHATS SCRIPT FROM TURNING ALL ICONS INTO HARDHAT WAZERS --MAPOMATIC
            },
            destroy: function () { this.erase(); OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); this.imageDiv.innerHTML = ""; this.imageDiv = null; },
            clone: function () { return new OpenLayers.Icon(this.url, this.size, this.offset, this.calculateOffset); },
            setSize: function (a) { null !== a && (this.size = a); this.draw(); },
            setUrl: function (a) { null !== a && (this.url = a); this.draw(); },
            draw: function (a) {
                OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, this.size, this.url, "absolute");
                this.moveTo(a);
                return this.imageDiv;
            },
            erase: function () { null !== this.imageDiv && null !== this.imageDiv.parentNode && OpenLayers.Element.remove(this.imageDiv); },
            setOpacity: function (a) { OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, null, null, null, null, a); },
            moveTo: function (a) {
                null !== a && (this.px = a);
                null !== this.imageDiv && (null === this.px ? this.display(!1) : (
                    this.calculateOffset && (this.offset = this.calculateOffset(this.size)),
                    OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { x: this.px.x + this.offset.x, y: this.px.y + this.offset.y })
                ));
            },
            display: function (a) { this.imageDiv.style.display = a ? "" : "none"; },
            isDrawn: function () { return this.imageDiv && this.imageDiv.parentNode && 11 != this.imageDiv.parentNode.nodeType; },
            CLASS_NAME: "OpenLayers.Icon"
        });
    }
    bootstrap();
    const config = { //Configuration data for each state
        AK: {
            data(res, index) {
                let resultText = [res];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['(resultObj[i].LanesAffected).replace(/ +(?= )/g, "") == ("All Lanes Closed")'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesAK.push(new Promise((resolve, reject) => {
                    advisoriesAK.push({
                        state: ['AK', 'Alaska'],
                        id: obj.ID,
                        popupType: 0,
                        title: obj.RoadwayName,
                        lon: obj.Longitude,
                        lat: obj.Latitude,
                        type: obj.EventType,
                        keyword: ['roadwork'], //keywords for roadwork/construction
                        desc: obj.Description,
                        time: moment(new Date(obj.LastUpdated * 1000)).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/AK']
        },
        AZ: {
            data(res, index) {
                let resultText = [res, res.features, res.features, res.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['((resultObj[i].LanesAffected).replace(/ +(?= )/g, "") == ("All Lanes Closed")) || ((resultObj[i].RoadwayName).includes("Road Closed"))', true, true, true];
                return (filtertext[index]);
            },
            scheme(obj, index) {

                promisesAZ.push(new Promise((resolve, reject) => {
                    advisoriesAZ.push({
                        state: ['AZ', 'Arizona'],
                        id: obj.ID,
                        popupType: 0,
                        title: obj.RoadwayName,
                        lon: obj.Longitude,
                        lat: obj.Latitude,
                        type: obj.EventType,
                        keyword: ['roadwork'], //keywords for roadwork/construction
                        desc: obj.Description,
                        time: moment(new Date(obj.LastUpdated * 1000)).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/AZ', 'https://maps.phoenix.gov/pub/rest/services/Public/STR_PubTraffRes/MapServer/0/query?where=Closure_Type+LIKE+%27FULL%27&text=&objectIds=&time=&geometry=&geometryType=esriGeometryPoint&inSR=&spatialRel=esriSpatialRelIntersects&distance=&units=esriSRUnit_Foot&relationParam=&outFields=*&returnGeometry=true&returnTrueCurves=false&maxAllowableOffset=&geometryPrecision=&outSR=4326&havingClause=&returnIdsOnly=false&returnCountOnly=false&orderByFields=&groupByFieldsForStatistics=&outStatistics=&returnZ=false&returnM=false&gdbVersion=&historicMoment=&returnDistinctValues=false&resultOffset=0&resultRecordCount=300&returnExtentOnly=false&datumTransformation=&parameterValues=&rangeValues=&quantizationParameters=&featureEncoding=esriDefault&f=pjson', 'https://maps.phoenix.gov/pub/rest/services/Public/STR_PubTraffRes/MapServer/1/query?where=Closure_Type+LIKE+%27FULL%27&text=&objectIds=&time=&geometry=&geometryType=esriGeometryPoint&inSR=&spatialRel=esriSpatialRelIntersects&distance=&units=esriSRUnit_Foot&relationParam=&outFields=*&returnGeometry=true&returnTrueCurves=false&maxAllowableOffset=&geometryPrecision=&outSR=4326&havingClause=&returnIdsOnly=false&returnCountOnly=false&orderByFields=&groupByFieldsForStatistics=&outStatistics=&returnZ=false&returnM=false&gdbVersion=&historicMoment=&returnDistinctValues=false&resultOffset=0&resultRecordCount=200&returnExtentOnly=false&datumTransformation=&parameterValues=&rangeValues=&quantizationParameters=&featureEncoding=esriDefault&f=pjson', 'https://maps.phoenix.gov/pub/rest/services/Public/STR_PubTraffRes/MapServer/2/query?f=json&where=1%3D1&returnGeometry=true&outSR=4326&spatialRel=esriSpatialRelIntersects&outFields=*&resultOffset=0&resultRecordCount=25']
        },
        CT: {
            data(res, index) {
                let resultText = [res.data];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['resultObj[i].eventSubType != "Queue"'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesCT.push(new Promise((resolve, reject) => {
                    advisoriesCT.push({
                        state: ['CT', 'Connecticut'],
                        id: obj.DT_RowId,
                        popupType: 0,
                        title: obj.roadwayName,
                        lon: obj.location[1],
                        lat: obj.location[0],
                        type: obj.eventSubType,
                        keyword: ['Construction'], //keywords for roadwork/construction
                        desc: obj.description,
                        startTime: obj.startTime,
                        plannedEndTime: obj.endTime,
                        time: obj.lastUpdated,
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/CT']
        },
        DE: {
            data(res, index) {
                let resultText = [res.advisories, res, res.features];
                return (resultText[index]);
            },
            filter(index) {
                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];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                switch (index) {
                    case 0:
                        promisesDE.push(new Promise((resolve, reject) => {
                            advisoriesDE.push({
                                state: ['DE', 'Delaware'],
                                id: obj.id,
                                popupType: 0,
                                title: obj.where.county.name,
                                lon: obj.where.lon,
                                lat: obj.where.lat,
                                type: obj.type.name,
                                keyword: ['Construction'], //keyword for roadwork/construction
                                desc: obj.where.location,
                                time: moment(new Date(obj.timestamp)).format('LLL'),
                                link: obj.published.linkbackUrl
                            });
                            resolve();
                        }))
                        break;
                    case 1:
                        var pubLink;
                        if (obj.str.releaseId) {
                            if (obj.str.releaseId.toString() == "-1") {
                                pubLink = '';
                            } else {
                                pubLink = 'https://deldot.gov/About/news/index.shtml?dc=release&id=' + obj.str.releaseId;
                            }
                        }
                        promisesDE.push(new Promise((resolve, reject) => {
                            advisoriesDE.push({
                                state: ['DE', 'Delaware'],
                                id: obj.str.strId,
                                popupType: 0,
                                title: obj.str.county,
                                lon: obj.str.longitude,
                                lat: obj.str.latitude,
                                type: obj.str.impactType,
                                keyword: ['Closure'], //keywords for roadwork/construction
                                desc: obj.str.title + " - " + obj.str.construction,
                                time: moment(new Date(obj.str.actualStartDate)).format('LLL'),
                                link: pubLink
                            });
                            resolve();
                        }))
                        break;
                    case 2:
                        var direction, WorkType;
                        switch (obj.attributes.direction) {
                            case 'ONE_DIRECTION':
                                direction = "One Way from ";
                                break;
                            case 'BOTH_DIRECTIONS':
                                direction = "Both Ways between ";
                        }
                        switch (obj.attributes.WorkType) {
                            case "1":
                                WorkType = "Gas Work";
                                break;
                            case "2":
                                WorkType = "Street Work (city)";
                                break;
                            case "3":
                                WorkType = "Street Work (DelDOT)";
                                break;
                            case "4":
                                WorkType = "Water Work";
                                break;
                            case "5":
                                WorkType = "Work (other)";
                                break;
                            case "6":
                                WorkType = "Sewer Work";
                        }
                        if (obj.attributes.street != null) {
                            var originShift = 2.0 * Math.PI * 6378137.0 / 2.0;
                            var lon = (obj.geometry.paths[0][0][0] / originShift) * 180.0;
                            var lat = (obj.geometry.paths[0][0][1] / originShift) * 180.0;
                            lat = 180.0 / Math.PI * (2.0 * Math.atan(Math.exp(lat * Math.PI / 180.0)) - Math.PI / 2.0);
                            promisesDE.push(new Promise((resolve, reject) => {
                                advisoriesDE.push({
                                    state: ['DE', 'Delaware'],
                                    id: obj.attributes.OBJECTID,
                                    popupType: 0,
                                    title: 'Wilmington Road Closures',
                                    lon: lon,
                                    lat: lat,
                                    type: 'Closure',
                                    keyword: ['Closure'], //keywords for roadwork/construction
                                    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'),
                                    time: moment(new Date(obj.attributes.EditDate)).format('LLL'),
                                    link: ''
                                });
                                resolve();
                            }))
                        }
                }
            },
            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=']
        },
        FL: {
            data(res, index) {
                let resultText = [res.item2, res.item2, res.item2];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true, true, true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                switch (index) {
                    case 0:
                        promisesFL.push(new Promise((resolve, reject) => {
                            //setTimeout(function () { resolve() }, 4000);
                            getFeed(config.FL.detailURL[0] + obj.itemId.replace("/ /g", "%20"), async function (result) {
                                var eventObj = JSON.parse(result.responseText);
                                if (eventObj.details.detailLang1.eventTypeName == "Closures") {
                                    advisoriesFL.push({
                                        state: ['FL', 'Florida'],
                                        id: eventObj.id.id,
                                        popupType: 0,
                                        title: eventObj.areas.area5.areaLang1,
                                        lon: eventObj.coordinates.locationLongitude / 1000000,
                                        lat: eventObj.coordinates.locationLatitude / 1000000,
                                        type: eventObj.details.detailLang1.eventTypeName,
                                        keyword: ['Construction'], //keywords for roadwork/construction
                                        desc: eventObj.details.detailLang1.eventDescription,
                                        time: moment(new Date(eventObj.dates.lastUpdated)).format('LLL'),
                                        link: ''
                                    });
                                }
                                resolve();
                            });
                        }));
                        break;
                    case 1:
                        promisesFL.push(new Promise((resolve, reject) => {
                            //setTimeout(function () { resolve() }, 4000);
                            getFeed(config.FL.detailURL[1] + obj.itemId.replace("/ /g", "%20"), async function (result) {
                                var eventObj = JSON.parse(result.responseText);
                                if (eventObj.details.detailLang1.eventTypeName == "Closures") {
                                    advisoriesFL.push({
                                        state: ['FL', 'Florida'],
                                        id: eventObj.id.id,
                                        popupType: 0,
                                        title: eventObj.areas.area5.areaLang1,
                                        lon: eventObj.coordinates.locationLongitude / 1000000,
                                        lat: eventObj.coordinates.locationLatitude / 1000000,
                                        type: eventObj.details.detailLang1.eventTypeName,
                                        keyword: ['Closures'], //keywords for roadwork/construction
                                        desc: eventObj.details.detailLang1.eventDescription,
                                        time: moment(new Date(eventObj.dates.lastUpdated)).format('LLL'),
                                        link: ''
                                    });
                                }
                                resolve();
                            });
                        }));
                        break;
                    case 2:
                        promisesFL.push(new Promise((resolve, reject) => {
                            //setTimeout(function () { resolve() }, 4000);
                            getFeed(config.FL.detailURL[1] + obj.itemId.replace("/ /g", "%20"), async function (result) {
                                var eventObj = JSON.parse(result.responseText);
                                if (eventObj.details.detailLang1.eventTypeName == "Closures") {
                                    advisoriesFL.push({
                                        state: ['FL', 'Florida'],
                                        id: eventObj.id.id,
                                        popupType: 0,
                                        title: eventObj.areas.area5.areaLang1,
                                        lon: eventObj.coordinates.locationLongitude / 1000000,
                                        lat: eventObj.coordinates.locationLatitude / 1000000,
                                        type: eventObj.details.detailLang1.eventTypeName,
                                        keyword: ['Constructions'], //keywords for roadwork/construction
                                        desc: eventObj.details.detailLang1.eventDescription,
                                        time: moment(new Date(eventObj.dates.lastUpdated)).format('LLL'),
                                        link: ''
                                    });
                                }
                                resolve();
                            });
                        }));
                }
            },
            URL: ['https://fl511.com/map/mapIcons/Incidents?_=1604955914474', 'https://fl511.com/map/mapIcons/Construction?_=1604965926997', 'https://fl511.com/map/mapIcons/Closures?_=1604965926997'],
            detailURL: ['https://fl511.com/map/data/Incidents/', 'https://fl511.com/map/data/Construction/', 'https://fl511.com/map/data/Closures/']
        },
        GA: {
            data(res, index) {
                let resultText = [res.features, res.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true, true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                switch (index) {
                    case 0:
                        promisesGA.push(new Promise((resolve, reject) => {
                            advisoriesGA.push({
                                state: ['GA', 'Georgia'],
                                id: obj.id,
                                popupType: 0,
                                title: obj.properties.headline + ' - ' + obj.properties.route,
                                lon: obj.geometry.coordinates[0],
                                lat: obj.geometry.coordinates[1],
                                type: obj.properties.headline,
                                keyword: ['Construction', 'Emergency Roadwork'], //keyword for roadwork/construction
                                desc: obj.properties.location_description + '<br>' + obj.properties.lanes,
                                time: moment(new Date(obj.properties.start * 1000)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                        break;
                    case 1:
                        promisesGA.push(new Promise((resolve, reject) => {
                            advisoriesGA.push({
                                state: ['GA', 'Georgia'],
                                id: obj.id,
                                popupType: 0,
                                title: obj.properties.headline + ' - ' + obj.properties.route,
                                lon: obj.geometry.coordinates[0],
                                lat: obj.geometry.coordinates[1],
                                type: obj.properties.headline,
                                keyword: ['Construction', 'Emergency Roadwork'], //keyword for roadwork/construction
                                desc: obj.properties.location_description + '<br>' + obj.properties.lanes,
                                time: moment(new Date(obj.properties.start * 1000)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                }
            },
            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']
        },
        IL: {
            data(res, index) {
                let resultText = [res.features, res.features, res.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true, true, true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                switch (index) {
                    case 0:
                        promisesIL.push(new Promise((resolve, reject) => {
                            advisoriesIL.push({
                                state: ['IL', 'Illinois'],
                                id: obj.attributes.OBJECTID,
                                popupType: 0,
                                title: obj.attributes.NearTown,
                                lon: obj.geometry.x,
                                lat: obj.geometry.y,
                                type: "Construction",
                                keyword: ['Construction'], //keywords for roadwork/construction
                                desc: 'Closed from ' + obj.attributes.Location + '<br> from ' + moment(new Date(obj.attributes.StartDate)).format('LLL') + ' to ' + moment(new Date(obj.attributes.EndDate)).format('LLL'),
                                time: moment(new Date(obj.attributes.StartDate)).format('LLL'),
                                link: obj.attributes.WebAddress
                            });
                            resolve();
                        }))
                        break;
                    case 1:
                        var DateUpdate;
                        if (obj.attributes.DateUpdate == null) {
                            DateUpdate = moment(new Date(obj.attributes.DateEntered)).format('LLL');
                        } else {
                            DateUpdate = moment(new Date(obj.attributes.DateUpdate)).format('LLL');
                        }
                        var originShift = 2.0 * Math.PI * 6378137.0 / 2.0;
                        var lon = (obj.geometry.paths[0][0][0] / originShift) * 180.0;
                        var lat = (obj.geometry.paths[0][0][1] / originShift) * 180.0;
                        lat = 180.0 / Math.PI * (2.0 * Math.atan(Math.exp(lat * Math.PI / 180.0)) - Math.PI / 2.0);
                        if (obj.attributes.SuggestionToMotorist.includes("Closed")) {
                            promisesIL.push(new Promise((resolve, reject) => {
                                advisoriesIL.push({
                                    state: ['IL', 'Illinois'],
                                    id: obj.attributes.OBJECTID,
                                    popupType: 0,
                                    title: obj.attributes.County,
                                    lon: lon,
                                    lat: lat,
                                    type: "Construction",
                                    keyword: ['Construction'], //keywords for roadwork/construction
                                    desc: 'Closed from ' + obj.attributes.Location + '<br> from ' + moment(new Date(obj.attributes.StartDate)).format('LLL') + ' to ' + moment(new Date(obj.attributes.EndDate)).format('LLL'),
                                    time: DateUpdate,
                                    link: obj.attributes.WebAddress
                                });
                                resolve();
                            }))
                        }
                        break;
                    case 2:
                        promisesIL.push(new Promise((resolve, reject) => {
                            advisoriesIL.push({
                                state: ['IL', 'Illinois'],
                                id: obj.attributes.OBJECTID,
                                popupType: 0,
                                title: obj.attributes.NearTown,
                                lon: obj.geometry.paths[0][0][0],
                                lat: obj.geometry.paths[0][0][1],
                                type: "",
                                keyword: ['Flooding'], //keywords for roadwork/construction
                                desc: 'Closed from ' + obj.attributes.Location + '<br> from ' + moment(new Date(obj.attributes.StartDate)).format('LLL') + ' to ' + moment(new Date(obj.attributes.EndDate)).format('LLL'),
                                time: moment(new Date(obj.attributes.StartDate)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                }
            },

            URL: ['https://services2.arcgis.com/aIrBD8yn1TDTEXoz/ArcGIS/rest/services/ClosureIncidents/FeatureServer/0/query?where=1%3D1&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=', 'https://services2.arcgis.com/aIrBD8yn1TDTEXoz/ArcGIS/rest/services/RoadConstruction_View/FeatureServer/0/query?where=1%3D1&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=', 'https://services2.arcgis.com/aIrBD8yn1TDTEXoz/ArcGIS/rest/services/Flooding_Road_Closures/FeatureServer/0/query?where=1%3D1&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=']
        },
        IN: {
            data(res, index) {
                let resultText = [res];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesIN.push(new Promise((resolve, reject) => {
                    advisoriesIN.push({
                        state: ['IN', 'Indiana'],
                        id: obj.id,
                        popupType: 0,
                        title: obj.location.routeDesignator,
                        lon: obj.location.primaryPoint.lon,
                        lat: obj.location.primaryPoint.lat,
                        type: 'Construction',
                        keyword: ['Construction'], //keywords for roadwork/construction
                        desc: obj.eventDescription.tooltip,
                        time: moment(new Date(obj.updateTime.time)).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['https://indot.carsprogram.org/tgevents/api/eventReports?maxPriority=2&maxBeginDateOffset=7200000&minEndDateOffset=0&eventClassifications%5B%5D=roadReports&eventClassifications%5B%5D=winterDriving&eventClassifications%5B%5D=flooding&_=' + Date.now()]
        },
        LA: {
            data(res, index) {
                let resultText = [res];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['(resultObj[i].LanesAffected).replace(/ +(?= )/g, "") == ("All Lanes Closed")'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesLA.push(new Promise((resolve, reject) => {
                    advisoriesLA.push({
                        state: ['LA', 'Louisiana'],
                        id: obj.ID,
                        popupType: 0,
                        title: obj.RoadwayName,
                        lon: obj.Longitude,
                        lat: obj.Latitude,
                        type: obj.EventType,
                        keyword: ['roadwork'], //keywords for roadwork/construction
                        desc: obj.Description,
                        time: moment(new Date(obj.LastUpdated * 1000)).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/LA']
        },
        MD: {
            data(res, index) {
                let resultText = [res.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesMD.push(new Promise((resolve, reject) => {
                    advisoriesMD.push({
                        state: ['MD', 'Maryland'],
                        id: obj.attributes.OBJECTID,
                        popupType: 0,
                        title: obj.attributes.Jurisdiction,
                        lon: obj.attributes.Longitude,
                        lat: obj.attributes.Latitude,
                        type: obj.attributes.typeSummary,
                        keyword: ['Construction'], //keywords for roadwork/construction
                        desc: obj.attributes.ClosureSummary,
                        time: moment(new Date(obj.attributes.EditDate)).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['https://geodata.md.gov/appdata/rest/services/SHA_RoadClosure/RoadClosureActive/MapServer//0/query?where=1%3D1&outFields=*&outSR=4326&f=json']
        },
        MI: {
            data(res, index) {
                let resultText = [res, res];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['(resultObj[i].type == "Total")', true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                switch (index) {
                    case 0:
                        promisesMI.push(new Promise((resolve, reject) => {
                            advisoriesMI.push({
                                state: ['MI', 'Michigan'],
                                id: "00",
                                popupType: 0,
                                title: obj.county,
                                lon: obj.description.match(/(?<=lon=)[\s\S]*(?=&zoom)/)[0],
                                lat: obj.description.match(/(?<=lat=)[\s\S]*(?=&lon)/)[0],
                                type: "Construction",
                                keyword: ['Construction'], //keyword for roadwork/construction
                                desc: obj.description.match(/(?:(?!<).)*/),
                                time: moment(new Date(obj.startDate)).format('LL'),
                                link: ''
                            });
                            resolve();
                        }))
                        break;
                    case 1:
                        promisesMI.push(new Promise((resolve, reject) => {
                            advisoriesMI.push({
                                state: ['MI', 'Michigan'],
                                id: "",
                                popupType: 0,
                                title: obj.county,
                                lon: obj.location.match(/(?<=lon=)[\s\S]*(?=&zoom)/)[0],
                                lat: obj.location.match(/(?<=lat=)[\s\S]*(?=&lon)/)[0],
                                type: "Incident",
                                keyword: ['Construction'], //keyword for roadwork/construction
                                desc: obj.location.match(/(?:(?!<).)*/),
                                time: moment(new Date(obj.reported)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                }
            },
            URL: ['https://mdotjboss.state.mi.us/MiDrive//construction/list/loadConstruction', 'https://mdotjboss.state.mi.us/MiDrive//incident/list/loadIncidents']
        },
        MN: {
            data(res, index) {
                let resultText = [res[0].data.mapFeaturesQuery.mapFeatures];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['resultObj[i].features[0].properties.icon.url == "/images/tg_closure_routine.svg" || resultObj[i].features[0].properties.icon.url == "/images/tg_crash_routine.svg"'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                let advType = "";
                if (obj.tooltip.toUpperCase().includes("CONSTRUCTION")) {
                    advType = "Construction";
                } else { advType = "Incident"; }
                promisesMN.push(new Promise((resolve, reject) => {
                    advisoriesMN.push({
                        state: ['MN', 'Minnesota'],
                        id: obj.features[0].id,
                        popupType: 0,
                        title: "N/A",
                        lon: obj.features[0].geometry.coordinates[0],
                        lat: obj.features[0].geometry.coordinates[1],
                        type: advType,
                        keyword: ['Construction'], //keywords for roadwork/construction
                        desc: obj.tooltip,
                        time: "N/A",
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/MN']
        },
        NC: {
            data(res, index) {
                let resultText = [res.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesNC.push(new Promise((resolve, reject) => {
                    advisoriesNC.push({
                        state: ['NC', 'North Carolina'],
                        id: obj.attributes.OBJECTID,
                        popupType: 0,
                        title: obj.attributes.CountyName,
                        lon: obj.attributes.Longitude,
                        lat: obj.attributes.Latitude,
                        type: obj.attributes.IncidentType,
                        keyword: ['Construction', 'Emergency Road Work', 'Night Time Construction', 'Weekend Construction'], //keywords for roadwork/construction
                        desc: obj.attributes.Reason,
                        time: moment(new Date(obj.attributes.LastUpdateUTC)).format('LLL'),
                        link: obj.attributes.DriveNCLink
                    });
                    resolve();
                }))
            },
            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=']
        },
        NJ: {
            data(res, index) {
                let resultText = [res.Data.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesNJ.push(new Promise((resolve, reject) => {
                    //setTimeout(function () { resolve() }, 4000);
                    getFeed(config.NJ.detailURL[0] + obj.properties.EventID, function (result) {
                        var eventObj = JSON.parse(result.responseText).Data;
                        console.log(result.status);
                        if (((eventObj[0].FullText.toUpperCase()).includes("ALL LANES CLOSE") || (eventObj[0].FullText.toUpperCase()).includes("RAMP CLOSE")) && ((eventObj[0].FullText).includes("NYSDOT") != true)) {
                            advisoriesNJ.push({
                                state: ['NJ', 'New Jersey'],
                                id: eventObj[0].markerId,
                                popupType: 0,
                                title: eventObj[0].County,
                                lon: eventObj[0].Longitude,
                                lat: eventObj[0].Latitude,
                                type: eventObj[0].CategoryName,
                                keyword: ['Construction', 'ScheduledConstruction'], //keywords for roadwork/construction
                                desc: eventObj[0].FullText,
                                time: moment(new Date(eventObj[0].LastUpdateDate_String)).format('LLL'),
                                link: ''
                            });
                        }
                        resolve(true);
                    });
                }));
            },
            URL: ['https://511nj.org/API/client/Map/getEventData'],
            detailURL: ['https://511nj.org/API/client/Map/getEventPopupData?EventId=']
        },
        NV: {
            data(res, index) {
                let resultText = [res.d];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesNV.push(new Promise((resolve, reject) => {
                    let unix = obj.LastUpdate.replace(/\\\//g, "").replace("/Date(", "").replace(")/", "");
                    advisoriesNV.push({
                        state: ['NV', 'Nevada'],
                        id: obj.ID,
                        popupType: 0,
                        title: obj.Facility,
                        lon: obj.Lon,
                        lat: obj.Lat,
                        type: obj.CategoryName,
                        keyword: ['Construction'], //keywords for roadwork/construction
                        desc: obj.Description,
                        time: moment(new Date(parseInt(unix))).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/NV']
        },
        NY: {
            data(res, index) {
                let resultText = [res];
                return (resultText[index]);
            },
            filter(index) {
                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"))))'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesNY.push(new Promise((resolve, reject) => {
                    advisoriesNY.push({
                        state: ['NY', 'New York'],
                        id: obj.ID,
                        popupType: 0,
                        title: obj.CountyName,
                        lon: obj.Longitude,
                        lat: obj.Latitude,
                        type: obj.EventType,
                        keyword: ['roadwork', 'transitMode', 'closures'], //keywords for roadwork/construction
                        desc: obj.Description,
                        time: moment(moment(obj.LastUpdated, "DD/MM/YYYY HH:mm:ss")).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/NY']
        },
        OH: {
            data(res, index) {
                let resultText = [res.ConstructionMarkers];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['(resultObj[i].Status == "Closed")'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesOH.push(new Promise((resolve, reject) => {
                    advisoriesOH.push({
                        state: ['OH', 'Ohio'],
                        id: obj.ID,
                        popupType: 0,
                        title: obj.District,
                        lon: obj.Longitude,
                        lat: obj.Latitude,
                        type: obj.Category,
                        keyword: ['Roadwork - Planned', 'Roadwork - Unplanned'], //keywords for roadwork/construction
                        desc: obj.Description,
                        time: moment(new Date(obj.StartDate)).format('LL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['https://api.ohgo.com/roadmarkers/TrafficSpeedAndAlertMarkers']
        },
        OR: {
            data(res, index) {
                let resultText = [res.features, res.features, res.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true, 'resultObj[i].attributes.comments.includes("clos")', 'resultObj[i].attributes.tmddOther.includes("clos")'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                switch (index) {
                    case 0:
                        let x, y;
                        var lonlat = obj.geometry.paths[0][0];
                        promisesOR.push(new Promise((resolve, reject) => {
                            x = obj.geometry.paths[0][0].toString().split(",")[0];
                            y = obj.geometry.paths[0][0].toString().split(",")[1];
                            advisoriesOR.push({
                                state: ['OR', 'Oregon'],
                                id: obj.attributes.OBJECTID,
                                popupType: 0,
                                title: obj.attributes.FROM_TO,
                                lon: x,
                                lat: y,
                                type: obj.attributes.CLOSURE_EFFECT,
                                keyword: ['Street'], //keywords for roadwork/construction
                                desc: obj.attributes.FROM_TO + " - " + obj.attributes.REMARKS,
                                time: moment(new Date(obj.attributes.LAST_EDITED_DATE)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                        break;
                    case 1:
                        promisesOR.push(new Promise((resolve, reject) => {
                            advisoriesOR.push({
                                state: ['OR', 'Oregon'],
                                id: obj.attributes.incidentId,
                                popupType: 0,
                                title: obj.attributes.locationName,
                                lon: obj.attributes.startLongitude,
                                lat: obj.attributes.startLatitude,
                                type: obj.attributes.type,
                                keyword: ['ROADWORK'], //keywords for roadwork/construction
                                desc: obj.attributes.comments + " <br> " + obj.attributes.beginMarker + " to " + obj.attributes.endMarker,
                                time: moment(new Date(obj.attributes.lastUpdated)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                        break;
                    case 2:
                        promisesOR.push(new Promise((resolve, reject) => {
                            advisoriesOR.push({
                                state: ['OR', 'Oregon'],
                                id: obj.attributes.incidentId,
                                popupType: 0,
                                title: obj.attributes.locationName,
                                lon: obj.attributes.startLongitude,
                                lat: obj.attributes.startLatitude,
                                type: obj.attributes.type,
                                keyword: ['EVENT'], //keywords for roadwork/construction
                                desc: obj.attributes.tmddOther + " <br> " + obj.attributes.beginMarker + " to " + obj.attributes.endMarker,
                                time: moment(new Date(obj.attributes.lastUpdated)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                }
            },
            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']
        },
        PA: {
            data(res, index) {
                let resultText = [res, res.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['(resultObj[i].LaneStatus == "closed") || (resultObj[i].LaneStatus == "ramp closure")', true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                switch (index) {
                    case 0:
                        let status = obj.LaneStatus;
                        let x, y;
                        if (status != "ramp closure") {
                            x = obj.FromLocLatLong.split(",")[1];
                            y = obj.FromLocLatLong.split(",")[0];
                        } else {
                            x = obj.IncidentLocLatLong.split(",")[1];
                            y = obj.IncidentLocLatLong.split(",")[0];
                        }
                        promisesPA.push(new Promise((resolve, reject) => {
                            advisoriesPA.push({
                                state: ['PA', 'Pennsylvania'],
                                id: obj.EventID,
                                popupType: 0,
                                title: obj.CountyName,
                                lon: x,
                                lat: y,
                                type: obj.EventType,
                                keyword: ['roadwork', 'bridge outage'], //keywords for roadwork/construction
                                desc: obj.Description,
                                time: moment(new Date(obj.LastUpdate)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                        break;
                    case 1:
                        var originShift = 2.0 * Math.PI * 6378137.0 / 2.0;
                        var lon = (obj.geometry.x / originShift) * 180.0;
                        var lat = (obj.geometry.y / originShift) * 180.0;
                        lat = 180.0 / Math.PI * (2.0 * Math.atan(Math.exp(lat * Math.PI / 180.0)) - Math.PI / 2.0);
                        var timing;
                        if (obj.attributes.END_DATE == null) {
                            timing = moment(new Date(obj.attributes.START_DATE)).format('LL');
                        } else {
                            timing = moment(new Date(obj.attributes.START_DATE)).format('LL') + ' to ' + moment(new Date(obj.attributes.END_DATE)).format('LL');
                        }
                        if (obj.attributes.START_TIME != null) {
                            timing = timing + '<br>' + obj.attributes.START_TIME + ' to ' + obj.attributes.END_TIME;
                        }
                        promisesPA.push(new Promise((resolve, reject) => {
                            advisoriesPA.push({
                                state: ['PA', 'Pennsylvania'],
                                id: obj.attributes.GlobalID,
                                popupType: 0,
                                title: 'HARRISBURG',
                                lon: lon,
                                lat: lat,
                                type: 'Closure',
                                keyword: ['Closure'], //keywords for roadwork/construction
                                desc: obj.attributes.STREET_NAME + ' between ' + obj.attributes.STREET_FROM + ' and ' + obj.attributes.STREET_TO + '<br>' + obj.attributes.REASON_FOR_CLOSURE + '<br>' + timing,
                                time: moment(new Date(obj.attributes.EditDate)).format('LLL'),
                                link: ''
                            });
                            resolve();
                        }))
                }
            },
            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']
        },
        VA: {
            data(res, index) {
                let resultText = [res.features, res.features, res.features];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true, true, true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                switch (index) {
                    case 0:
                        promisesVA.push(new Promise((resolve, reject) => {
                            let lat = obj.geometry.coordinates[1];
                            let lon = obj.geometry.coordinates[0];
                            let thisID = obj.id;
                            getFeed(config.VA.detailURL + obj.id, async function (result) {
                                var eventObj = JSON.parse(result.responseText);
                                if (eventObj[thisID].display_text.toLowerCase().includes("all west lanes are closed")) {
                                    advisoriesVA.push({
                                        state: ['VA', 'Virginia'],
                                        id: eventObj[thisID].fid,
                                        popupType: 0,
                                        title: "Construction",
                                        lon: lon,
                                        lat: lat,
                                        type: eventObj[thisID].type,
                                        keyword: ['Constructions'], //keywords for roadwork/construction
                                        desc: eventObj[thisID].report,
                                        time: moment(new Date(eventObj[thisID].update * 1000)).format('LLL'),
                                        link: ''
                                    });
                                }
                                resolve();
                            });
                        }));
                        break;
                    case 1:
                        promisesVA.push(new Promise((resolve, reject) => {
                            let lat = obj.geometry.coordinates[1];
                            let lon = obj.geometry.coordinates[0];
                            let thisID = obj.id;
                            getFeed(config.VA.detailURL + obj.id, async function (result) {
                                var eventObj = JSON.parse(result.responseText);
                                advisoriesVA.push({
                                    state: ['VA', 'Virginia'],
                                    id: eventObj[thisID].fid,
                                    popupType: 0,
                                    title: "High Impact Incident",
                                    lon: lon,
                                    lat: lat,
                                    type: eventObj[thisID].type,
                                    keyword: ['Constructions'], //keywords for roadwork/construction
                                    desc: eventObj[thisID].report,
                                    time: moment(new Date(eventObj[thisID].update * 1000)).format('LLL'),
                                    link: ''
                                });
                                resolve();
                            });
                        }));
                        break;
                    case 2:
                        promisesVA.push(new Promise((resolve, reject) => {
                            let lat = obj.geometry.coordinates[1];
                            let lon = obj.geometry.coordinates[0];
                            let thisID = obj.id;
                            getFeed(config.VA.detailURL + obj.id, async function (result) {
                                var eventObj = JSON.parse(result.responseText);
                                advisoriesVA.push({
                                    state: ['VA', 'Virginia'],
                                    id: eventObj[thisID].fid,
                                    popupType: 0,
                                    title: "Weather",
                                    lon: lon,
                                    lat: lat,
                                    type: eventObj[thisID].type,
                                    keyword: ['Constructions'], //keywords for roadwork/construction
                                    desc: eventObj[thisID].report,
                                    time: moment(new Date(eventObj[thisID].update * 1000)).format('LLL'),
                                    link: ''
                                });
                                resolve();
                            });
                        }));
                }
            },
            URL: ['https://www.511virginia.org/data/geojson/icons.construction.geojson', 'https://www.511virginia.org/data/geojson/icons.high_impact_incident.geojson', 'https://www.511virginia.org/data/geojson/icons.weather_closure.geojson'],
            detailURL: ['https://www.511virginia.org/report-json.pl?idents='],
        },
        WA: {
            data(res, index) {
                let resultText = [res];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['(resultObj[i].EventCategory == "Closure" || resultObj[i].EventCategory == "Construction" || resultObj[i].EventCategory == "Bridge")'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                let county;
                if (obj.County == null) {
                    county = obj.Region;
                } else {
                    county = obj.County;
                }
                let unixtime = parseInt(obj.LastUpdatedTime.replace("/Date(", "").replace(")/", "").split("-")[0]);
                promisesWA.push(new Promise((resolve, reject) => {
                    advisoriesWA.push({
                        state: ['WA', 'Washington'],
                        id: obj.AlertID,
                        popupType: 0,
                        title: county,
                        lon: obj.StartRoadwayLocation.Longitude,
                        lat: obj.StartRoadwayLocation.Latitude,
                        type: obj.EventCategory,
                        keyword: ['Construction', 'Closures'], //keywords for roadwork/construction
                        desc: obj.HeadlineDescription,
                        time: moment(new Date(unixtime)).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/WA']
        },
        WI: {
            data(res, index) {
                let resultText = [res];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = ['(resultObj[i].EventType == "roadwork" || resultObj[i].EventType == "closures" || resultObj[i].EventType == "accidentsAndIncidents")'];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                let linkvar = '';
                let eText = '';

                let addObj = function () {
                    if (obj.PlannedEndDate == null || obj.PlannedEndDate > moment().unix()) {
                        promisesWI.push(new Promise((resolve, reject) => {
                            advisoriesWI.push({
                                state: ['WI', 'Wisconsin'],
                                id: obj.ID,
                                popupType: 0,
                                title: obj.County,
                                lon: obj.Longitude,
                                lat: obj.Latitude,
                                type: obj.EventType,
                                keyword: ['roadwork', 'closure'], //keywords for roadwork/construction
                                desc: eText + ': ' + obj.Description,
                                startTime: moment.unix(obj.StartDate).format('LLL'),
                                plannedEndTime: moment.unix(obj.PlannedEndDate).format('LLL'),
                                time: moment.unix(obj.LastUpdated).format('LLL'),
                                link: linkvar,
                                recurrence: obj.Recurrence
                            });
                            resolve();
                        }))
                    }
                };

                if (obj.EventType == 'closures') {
                    linkvar = 'https://511wi.gov/map#ConstructionClosures-' + obj.ID.replace(' ', '%20');
                    new Promise((resolve, reject) => {
                        getFeed('https://511wi.gov/map/data/ConstructionClosures/' + obj.ID.replace(' ', '%20'), function (result) {
                            let resultObj = [];
                            resultObj = JSON.parse(result.responseText);
                            eText = resultObj.details.detailLang1.eventDescription + ' on ' + resultObj.location.linkDesignator + ' at ' + resultObj.location.crossStreetName
                            resolve();
                        });
                    }).then((result) => addObj());
                }
                else if (obj.EventType == 'accidentsAndIncidents') {
                    linkvar = 'https://511wi.gov/map#Incidents-' + obj.ID.replace(' ', '%20');
                    addObj();
                }
                else {
                    addObj();
                }
            },
            URL: ['http://scripts.essentialintegrations.com/WI']
        },
        WV: {
            data(res, index) {
                let resultText = [res.changes["com.orci.opentms.web.public511.components.plannedevent.shared.data.PlannedEventBean"].changes];
                return (resultText[index]);
            },
            filter(index) {
                let filtertext = [true];
                return (filtertext[index]);
            },
            scheme(obj, index) {
                promisesWV.push(new Promise((resolve, reject) => {
                    advisoriesWV.push({
                        state: ['WV', 'West Virginia'],
                        id: obj.entity.dataGatewayId,
                        popupType: 0,
                        title: obj.entity.routeName,
                        lon: obj.entity.x,
                        lat: obj.entity.y,
                        type: 'Construction',
                        keyword: ['Construction'], //keywords for roadwork/construction
                        desc: obj.entity.message,
                        time: moment(new Date(obj.entity.startTime.millis)).format('LLL'),
                        link: ''
                    });
                    resolve();
                }))
            },
            URL: ['http://scripts.essentialintegrations.com/WV']
        }
    };
})();