WME DOT Advisories

Overlay DOT Advisories on the WME Map Object

目前為 2020-11-02 提交的版本,檢視 最新版本

// ==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.13
// @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      511pa.com
// @connect      deldot.gov
// @connect      511ny.org
// @connect      511nj.org
// @connect      rehostjson.phuz.repl.co
/* global OpenLayers */
/* global W */
/* global WazeWrap */
/* global $ */
/* global I18n */
/* global _ */
// ==/UserScript==

let AKDOTLayer;
let DEDOTLayer;
let LADOTLayer;
let NJDOTLayer;
let NYDOTLayer;
let PADOTLayer;
let WADOTLayer;
var settings;
const updateMessage = "&#9658; Added Alaska, Louisiana<br>&#9658; Bug Fixes";
const DEIconC = '';
const DEIconSchRestriction = '';
const DEIconSchClosure = '';
const Incident = '';
const Roadwork = '';
const PLIcon = '';
const reportIcon = '';
const AKURL = 'https://rehostjson.phuz.repl.co/AK';
const DEAdvURL = 'https://tmc.deldot.gov/json/advisory.json';
const DESchURL = 'https://deldot.gov/json/str.json';
const LAURL = 'https://rehostjson.phuz.repl.co/LA';
const NJURLList = 'https://511nj.org/API/client/Map/getEventData';
const NJURLDetail = 'https://511nj.org/API/client/Map/getEventPopupData?EventId=';
const NYURL = 'https://rehostjson.phuz.repl.co/NY';
const PAURL = 'https://rehostjson.phuz.repl.co/PA';
const WAURL = 'https://rehostjson.phuz.repl.co/WA';
const NotNY = ['Pennsylvania Statewide', 'New Jersey Statewide', 'Connecticut Statewide'];
const NJConstruction = ['Construction', 'ScheduledConstruction'];

(function () {
    'use strict';
    //Bootstrap
    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();
            if (!OpenLayers.Icon) {
                installIcon();
            }

        } else if (tries < 1000) {
            setTimeout(function () { bootstrap(++tries); }, 200);
        }
    }
    //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 colspan=2 style="text-align:center"><b>Enable</b></td><td style="text-align"><b>State</b></td><td width=30><b>Rpt</b></td></tr>',
            '<tr><td colspan=2 align=center><input type="checkbox" id="chkAKDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>AK</td><td><div class=DOTreport data-report="report" id="DOTAKPopup"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td colspan=2 align=center><input type="checkbox" id="chkDEDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>DE</td><td><div class=DOTreport data-report="report" id="DOTDEPopup"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td colspan=2 align=center><input type="checkbox" id="chkLADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>LA</td><td><div class=DOTreport data-report="report" id="DOTLAPopup"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td colspan=2 align=center><input type="checkbox" id="chkNJDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>NJ</td><td><div class=DOTreport data-report="report" id="DOTNJPopup"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td colspan=2 align=center><input type="checkbox" id="chkNYDOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>NY</td><td><div class=DOTreport data-report="report" id="DOTNYPopup"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td colspan=2 align=center><input type="checkbox" id="chkPADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>PA</td><td><div class=DOTreport data-report="report" id="DOTPAPopup"><img src=' + reportIcon + '></div></td></tr>',
            '<tr><td colspan=2 align=center><input type="checkbox" id="chkWADOTEnabled" class="WMEDOTAdvSettingsCheckbox"></td><td align=center>WA</td><td><div class=DOTreport data-report="report" id="DOTWAPopup"><img src=' + reportIcon + '></div></td></tr>',
            '</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("https://rehostjson.phuz.repl.co/public/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: function (response) {
                var result = response;
                callback(result);
            }
        });
    }
    //Get the AK Event JSON Feed
    function getAKDOT(type) {
        getFeed(AKURL, function (result) {
            var resultObj = JSON.parse(result.responseText);
            var i = 0;
            var icon;
            while (i < resultObj.length) {
                if ((resultObj[i].LanesAffected).replace(/ +(?= )/g, '') == ("All Lanes Closed")) {
                    switch (resultObj[i].EventType) {
                        case "closures":
                            icon = "Incident";
                            break;
                        case "roadwork":
                            icon = "Roadwork";
                            break;
                        default:
                            icon = "Incident";
                    }
                    if (type == "report") {
                        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="' + resultObj[i].Latitude + '" data-lon="' + resultObj[i].Longitude + '"><img src=' + PLIcon + '></div>';
                        cell2.innerHTML = resultObj[i].Description;
                        cell3.innerHTML = resultObj[i].RoadwayName;
                        cell4.innerHTML = moment(new Date(resultObj[i].LastUpdated * 1000)).format('LLL');
                    } else {
                        drawMarkers("AK", resultObj[i].ID, "", resultObj[i].Longitude, resultObj[i].Latitude, icon, resultObj[i].Description, resultObj[i].LastUpdated, "");
                    }
                }
                i++;
            }
            if (type == "report") {
                reportWorker();
            }
        })
    }
    function getDEDOT() {
        getDEDOTAdv();
    }
    //Get the DE Advisory JSON Feed
    function getDEDOTAdv(type) {
        getFeed(DEAdvURL, function (result) {
            var resultObj = JSON.parse(result.responseText).advisories;
            var i = 0;
            while (i < resultObj.length) {
                if (type == "report") {
                    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="' + resultObj[i].where.lat + '" data-lon="' + resultObj[i].where.lon + '"><img src=' + PLIcon + '></div>';
                    cell2.innerHTML = resultObj[i].where.location;
                    cell3.innerHTML = resultObj[i].where.county.name;
                    cell4.innerHTML = moment(new Date(resultObj[i].timestamp)).format('LLL');
                } else {
                    drawMarkers("DEAdv", resultObj[i].id, resultObj[i].type.name, resultObj[i].where.lon, resultObj[i].where.lat, "DEIconC", resultObj[i].where.location, resultObj[i].timestamp, resultObj[i].published.linkbackUrl);
                }
                i++;
            }
            getDEDOTSch(type);
        })
    }
    //Get the DE Schedule JSON Feed
    function getDEDOTSch(type) {
        getFeed(DESchURL, function (result) {
            var resultObj = JSON.parse(result.responseText);
            var i = 0;
            while (i < resultObj.length) {
                if (resultObj[i].str.impactType == "Closure") {
                    if (type == "report") {
                        let table = document.getElementById("reportTable").getElementsByTagName('tbody')[0];
                        var row = table.insertRow();
                        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="' + resultObj[i].str.latitude + '" data-lon="' + resultObj[i].str.longitude + '"><img src=' + PLIcon + '></div>';
                        cell2.innerHTML = resultObj[i].str.title;
                        cell3.innerHTML = resultObj[i].str.county;
                        cell4.innerHTML = moment(new Date(resultObj[i].str.actualStartDate)).format('LLL');
                    } else {
                        drawMarkers("DESch", resultObj[i].str.strId, resultObj[i].str.title, resultObj[i].str.longitude, resultObj[i].str.latitude, "DEIconSch" + resultObj[i].str.impactType, resultObj[i].str.construction, resultObj[i].str.startDate, resultObj[i].str.releaseId);
                    }
                }
                i++;
            }
            if (type == "report") {
                reportWorker();
            }
        })
    }
    //Get the LA Event JSON Feed
    function getLADOT(type) {
        getFeed(LAURL, function (result) {
            var resultObj = JSON.parse(result.responseText);
            var i = 0;
            var icon;
            while (i < resultObj.length) {
                if ((resultObj[i].LanesAffected).replace(/ +(?= )/g, '') == ("All Lanes Closed")) {
                    switch (resultObj[i].EventType) {
                        case "closures":
                            icon = "Incident";
                            break;
                        case "roadwork":
                            icon = "Roadwork";
                            break;
                        default:
                            icon = "Incident";
                    }
                    if (type == "report") {
                        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="' + resultObj[i].Latitude + '" data-lon="' + resultObj[i].Longitude + '"><img src=' + PLIcon + '></div>';
                        cell2.innerHTML = resultObj[i].Description;
                        cell3.innerHTML = resultObj[i].RoadwayName;
                        cell4.innerHTML = moment(new Date(resultObj[i].LastUpdated * 1000)).format('LLL');
                    } else {
                        drawMarkers("LA", resultObj[i].ID, "", resultObj[i].Longitude, resultObj[i].Latitude, icon, resultObj[i].Description, resultObj[i].LastUpdated, "");
                    }
                }
                i++;
            }
            if (type == "report") {
                reportWorker();
            }
        })
    }
    //Get the NJ Schedule JSON Feed
    function getNJDOT(type) {
        getFeed(NJURLList, function (result) {
            var resultObj = JSON.parse(result.responseText).Data.features;
            var length = resultObj.length;
            let advisories = [];
            var i = 0;
            for (let i = 0; i < length; i++) {
                if (["Incident", "Construction", "ScheduledConstruction", "Detour"].includes(resultObj[i].properties.CategoryName)) {
                    advisories.push(resultObj[i].properties.EventID);
                }
            }
            getNJDetails(type, advisories);
        })
    }
    function getNJDetails(type, advisories) {
        var promises = [];
        for (let i = 0; i < advisories.length; i++) {
            promises.push(new Promise((resolve, reject) => {
                getFeed(NJURLDetail + advisories[i], function (result2) {
                    var eventObj = JSON.parse(result2.responseText).Data;
                    if ((eventObj[0].FullText.toUpperCase()).includes("ALL LANES CLOSE") || (eventObj[0].FullText.toUpperCase()).includes("RAMP CLOSE")) {
                        var icon;
                        switch (eventObj[0].CategoryName) {
                            case "Incident":
                                icon = "Incident";
                                break;
                            case "Detour":
                                icon = "Incident";
                                break;
                            case "Construction":
                                icon = "Roadwork";
                                break;
                            case "ScheduledConstruction":
                                icon = "Roadwork";
                                break;
                            default:
                                icon = "Incident";
                        }
                        if (type == "report") {
                            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="' + eventObj[0].Latitude + '" data-lon="' + eventObj[0].Longitude + '"><img src=' + PLIcon + '></div>';
                            cell2.innerHTML = eventObj[0].FullText
                            cell3.innerHTML = eventObj[0].County
                            cell4.innerHTML = moment(new Date(eventObj[0].LastUpdateDate_String)).format('LLL');
                        } else {
                            drawMarkers("NJ", eventObj[0].markerId, eventObj[0].County + " - " + eventObj[0].CategoryName, eventObj[0].Longitude, eventObj[0].Latitude, icon, eventObj[0].FullText, eventObj[0].LastUpdateDate_String, "");
                        }
                    }
                    resolve(); //Let the Promise.all() know that we finished this dependency
                })
            }))
        }
        Promise.all(promises).then(function () {
            if (type == "report") { reportWorker(); } //Run the reportWorker after we've resolved all promises (ie, fetched the details of all pertinent report IDs)
        });
    }
    //Get the NY Event JSON Feed
    function getNYDOT(type) {
        getFeed(NYURL, function (result) {
            var resultObj = JSON.parse(result.responseText);
            var i = 0;
            var icon;
            while (i < resultObj.length) {
                if (NotNY.includes(resultObj[i].RegionName) == false && resultObj[i].EventType != 'transitMode' && resultObj[i].EventSubType != 'Capacity related') { //skip anything not NY & transit notices & parking notices
                    if (resultObj[i].EventType == 'closures' || (resultObj[i].EventType != 'closures' && resultObj[i].LanesAffected == 'all lanes' && (resultObj[i].LanesStatus == 'closed' || resultObj[i].LanesStatus == "blocked"))) {
                        switch (resultObj[i].EventType) {
                            case "accidentsAndIncidents":
                                icon = "Incident";
                                break;
                            case "roadwork":
                                icon = "Roadwork";
                                break;
                            case "specialEvents":
                                icon = "Incident";
                                break;
                            case "closures":
                                icon = "Roadwork";
                                break;
                            case "transitMode":
                                icon = "Roadwork";
                                break;
                            case "generalInfo":
                                icon = "Roadwork";
                                break;
                            case "winterDrivingIndex":
                                icon = "Roadwork";
                                break;
                            default:
                                icon = "Incident";
                        }
                        if (type == "report") {
                            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="' + resultObj[i].Latitude + '" data-lon="' + resultObj[i].Longitude + '"><img src=' + PLIcon + '></div>';
                            cell2.innerHTML = resultObj[i].Description;
                            cell3.innerHTML = resultObj[i].CountyName;
                            cell4.innerHTML = moment(moment(resultObj[i].LastUpdated, "DD/MM/YYYY HH:mm:ss")).format('LLL');
                        } else {
                            drawMarkers("NY", resultObj[i].ID, "", resultObj[i].Longitude, resultObj[i].Latitude, icon, resultObj[i].Description, resultObj[i].LastUpdated, "");
                        }
                    }
                }
                i++;
            }
            if (type == "report") {
                reportWorker();
            }
        })
    }
    //Get the PA Event JSON Feed
    function getPADOT(type) {
        getFeed(PAURL, function (result) {
            var resultObj = JSON.parse(result.responseText);
            var i = 0;
            var icon;
            var x, y;
            while (i < resultObj.length) {
                if ((resultObj[i].LaneStatus == "closed") || (resultObj[i].LaneStatus == "ramp closure")) {
                    switch (resultObj[i].EventType) {
                        case "roadwork":
                            icon = "Roadwork";
                            break;
                        default:
                            icon = "Incident";
                    }
                    if (resultObj[i].LaneStatus != "ramp closure") {
                        x = resultObj[i].FromLocLatLong.split(",")[1];
                        y = resultObj[i].FromLocLatLong.split(",")[0];
                    } else {
                        x = resultObj[i].IncidentLocLatLong.split(",")[1];
                        y = resultObj[i].IncidentLocLatLong.split(",")[0];
                    }
                    if (type == "report") {
                        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="' + y + '" data-lon="' + x + '"><img src=' + PLIcon + '></div>';
                        cell2.innerHTML = resultObj[i].Description;
                        cell3.innerHTML = resultObj[i].CountyName;
                        cell4.innerHTML = moment(new Date(resultObj[i].DateTimeNotified)).format('LLL'); //('MM/DD/YYYY HH:mm');
                    } else {
                        drawMarkers("PA", resultObj[i].EventID, resultObj[i].Facility, x, y, icon, resultObj[i].Description, resultObj[i].LastUpdate, "");
                    }
                }
                i++;
            }
            if (type == "report") {
                reportWorker();
            }
        })
    }
    //Get the WA Event JSON Feed
    function getWADOT(type) {
        getFeed(WAURL, function (result) {
            var resultObj = JSON.parse(result.responseText);
            var i = 0;
            var icon;
            var county;
            while (i < resultObj.length) {
                if (resultObj[i].EventCategory == 'Closure' || resultObj[i].EventCategory == 'Construction' || resultObj[i].EventCategory == 'Bridge') {
                    switch (resultObj[i].EventCategory) {
                        case "Construction":
                            icon = "Roadwork";
                            break;
                        case "Closure":
                            icon = "Roadwork";
                            break;
                        case "Collision":
                            icon = "Incident";
                            break;
                        default:
                            icon = "Incident";
                    }
                    var unixtime = parseInt(resultObj[i].LastUpdatedTime.replace("/Date(", "").replace(")/", "").split("-")[0]);
                    let datetime = new Date(unixtime).toLocaleString();
                    if (resultObj[i].County == null) {
                        county = resultObj[i].Region;
                    } else {
                        county = resultObj[i].County;
                    }
                    if (type == "report") {
                        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="' + resultObj[i].StartRoadwayLocation.Latitude + '" data-lon="' + resultObj[i].StartRoadwayLocation.Longitude + '"><img src=' + PLIcon + '></div>';
                        cell2.innerHTML = resultObj[i].HeadlineDescription;
                        cell3.innerHTML = county;
                        cell4.innerHTML = moment(new Date(datetime)).format('LLL'); //moment(new Date(resultObj[i].DateTimeNotified)).format('YYYY/MM/DD hh:mm a');
                    } else {
                        drawMarkers("WA", resultObj[i].AlertID, "", resultObj[i].StartRoadwayLocation.Longitude, resultObj[i].StartRoadwayLocation.Latitude, icon, resultObj[i].HeadlineDescription, datetime, "");
                    }
                }
                i++;
            }
            if (type == "report") {
                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();
    }
    function getReportData(id) {
        switch (id) {
            case "DOTAKPopup":
                popupdetails("Alaska");
                getAKDOT("report");
                break;
            case "DOTDEPopup":
                popupdetails("Delaware");
                getDEDOTAdv("report");
                break;
            case "DOTLAPopup":
                popupdetails("Louisiana");
                getLADOT("report");
                break;
            case "DOTNJPopup":
                popupdetails("New Jersey");
                getNJDOT("report");
                break;
            case "DOTNYPopup":
                popupdetails("New York");
                getNYDOT("report");
                break;
            case "DOTPAPopup":
                popupdetails("Pennsylvania");
                getPADOT("report");
                break;
            case "DOTWAPopup":
                popupdetails("Washington");
                getWADOT("report");
        }
    }
    //Generate the Advisory markers
    function drawMarkers(state, id, title, x, y, icontype, desc, timestamp, link) {
        var size = new OpenLayers.Size(20, 20);
        var offset = new OpenLayers.Pixel(-(size.w / 2), -size.h);
        var icon = new OpenLayers.Icon(eval(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(x, y).transform(epsg4326, projectTo);
        var newMarker = new OpenLayers.Marker(lonLat, icon);
        newMarker.desc = desc;
        newMarker.eventId = id;
        newMarker.title = title;
        newMarker.state = state;
        newMarker.timestamp = timestamp;
        newMarker.events.register('click', newMarker, popup);
        newMarker.location = lonLat;
        if (link != "-1" && state == "DESch") {
            newMarker.link = '<a href=https://deldot.gov/About/news/index.shtml?dc=release&id=' + link + ' target="_blank">';
        } else if (link == "-1" && state == "DESch") {
            newMarker.link = `<a href='javascript:alert("Sorry!  No publication available for this closure")'>`;
        } else {
            newMarker.link = link;
        }
        eval(state.substring(0, 2) + "DOTLayer.addMarker(newMarker)");
    }
    function popup(evt) {
        $("#gmPopupContainer").remove();
        $("#gmPopupContainer").hide();
        var popupHTML;
        W.map.moveTo(this.location);
        switch (this.state) {
            case "DEAdv":
                this.timestamp = new Date(this.timestamp);
                popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px">' +
                    '<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><hr class="myhrline"/>' +
                    'Updated: ' + this.timestamp.toLocaleString() + '<hr class="myhrline"/></td></tr>' +
                    '<tr><td>' + this.desc + '</td></tr>' +
                    '<tr><td><a href="' + this.link + '" target="_blank">DelDot Link</a></td></tr>' +
                    '</table>' +
                    '</div>'
                ]);
                break;
            case "DESch":
                popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px">' +
                    '<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;">Scheduled Closure</div><hr class="myhrline"/>' +
                    '<tr><td><h4>' + this.title.toString() + '</h4><hr class="myhrline"/></td></tr>' +
                    '<tr><td>Period: ' + this.timestamp.replace(/ 12:00 AM/g, "") + '<hr class="myhrline"/></td></tr>' +
                    '<tr><td>' + this.desc + '</td></tr>' +
                    '<tr><td>' + this.link + 'DelDot Link</a></td></tr>' +
                    '</table>' +
                    '</div>'
                ]);
                break;
            case "NJ":
            case "NY":
            case "PA":
                popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px">' +
                    '<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;"></div><hr class="myhrline"/>' +
                    '<tr><td><h4>' + this.title.toString() + '</h4><hr class="myhrline"/></td></tr>' +
                    '<tr><td>Time: ' + new Date(this.timestamp).toLocaleString() + '<hr class="myhrline"/></td></tr>' +
                    '<tr><td>' + this.desc + '</td></tr>' +
                    '</table>' +
                    '</div>'
                ]);
                break;
            case "AK":
            case "LA":
            case "WA":
                popupHTML = (['<div id="gmPopupContainer" style="max-width:500px;margin: 1;text-align: center;padding: 5px">' +
                    '<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;"></div><hr class="myhrline"/>' +
                    'Updated: ' + this.timestamp.toLocaleString() + '<hr class="myhrline"/></td></tr>' +
                    '<tr><td>' + this.desc + '</td></tr>' +
                    '</table>' +
                    '</div>'
                ]);
        }
        $("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(state) {
        $("#gmPopupContainer").remove();
        $("#gmPopupContainer").hide();
        var popupHTML;
        popupHTML = (['<div id="gmPopupContainer" style="max-width:750px;max-height:500px;margin: 1;text-align: center;padding: 5px;">' +
            '<a href="#close" id="popupdetailsclose" title="Close" class="modalclose" style="color:#FF0000;">X</a>' +
            '<table border=0><tr><td><div id="mydivheader"">' + state + ' Reports</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>Location</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: 400 });
        $("#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() {
        for (var i = 0; i < document.getElementsByClassName("DOTreport").length; i++) {
            document.getElementsByClassName("DOTreport")[i].addEventListener('click', function (e) { getReportData(this.getAttribute("id")); }, false);
        }
        loadSettings();
        setChecked('chkAKDOTEnabled', settings.AKDOTEnabled);
        setChecked('chkDEDOTEnabled', settings.DEDOTEnabled);
        setChecked('chkLADOTEnabled', settings.LADOTEnabled);
        setChecked('chkNJDOTEnabled', settings.NJDOTEnabled);
        setChecked('chkNYDOTEnabled', settings.NYDOTEnabled);
        setChecked('chkPADOTEnabled', settings.PADOTEnabled);
        setChecked('chkWADOTEnabled', settings.WADOTEnabled);
        $(".overlay-button").click(function () {
            if (document.getElementById('chkAKDOTEnabled').checked) { eval('W.map.removeLayer(AKDOTLayer)'); }
            if (document.getElementById('chkDEDOTEnabled').checked) { eval('W.map.removeLayer(DEDOTLayer)'); }
            if (document.getElementById('chkLADOTEnabled').checked) { eval('W.map.removeLayer(LADOTLayer)'); }
            if (document.getElementById('chkNJDOTEnabled').checked) { eval('W.map.removeLayer(NJDOTLayer)'); }
            if (document.getElementById('chkNYDOTEnabled').checked) { eval('W.map.removeLayer(NYDOTLayer)'); }
            if (document.getElementById('chkPADOTEnabled').checked) { eval('W.map.removeLayer(PADOTLayer)'); }
            if (document.getElementById('chkWADOTEnabled').checked) { eval('W.map.removeLayer(WADOTLayer)'); }
            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("get" + settingName.substring(0, 2) + "DOT()");
            }
            else {
                //eval(settingName.substring(0,2) + "DOTLayer.destroy()");
                eval('W.map.removeLayer(' + settingName.substring(0, 2) + 'DOTLayer)');
            }
        });
        if (document.getElementById('WMEFUzoom') != null) {
            document.getElementById('WMEFUzoom').style.zIndex = "4000";
        }
        if (document.getElementById('chkAKDOTEnabled').checked) { buildDOTAdvLayers("AK"); getAKDOT(); }
        if (document.getElementById('chkDEDOTEnabled').checked) { buildDOTAdvLayers("DE"); getDEDOT(); }
        if (document.getElementById('chkLADOTEnabled').checked) { buildDOTAdvLayers("LA"); getLADOT(); }
        if (document.getElementById('chkNJDOTEnabled').checked) { buildDOTAdvLayers("NJ"); getNJDOT(); }
        if (document.getElementById('chkNYDOTEnabled').checked) { buildDOTAdvLayers("NY"); getNYDOT(); }
        if (document.getElementById('chkPADOTEnabled').checked) { buildDOTAdvLayers("PA"); getPADOT(); }
        if (document.getElementById('chkWADOTEnabled').checked) { buildDOTAdvLayers("WA"); getWADOT(); }
    }
    //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 = {
                AKDOTEnabled: settings.AKDOTEnabled,
                DEDOTEnabled: settings.DEDOTEnabled,
                LADOTEnabled: settings.LADOTEnabled,
                NJDOTEnabled: settings.NJDOTEnabled,
                NYDOTEnabled: settings.NYDOTEnabled,
                PADOTEnabled: settings.PADOTEnabled,
                WADOTEnabled: settings.WADOTEnabled,
            };
            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();
})();