WME DOT Advisories

Overlay DOT Advisories on the WME Map Object

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

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴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.09
// @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 DEDOTLayer;
let NJDOTLayer;
let NYDOTLayer;
let PADOTLayer;
let WADOTLayer;
var settings;
var newZIndex;
const DEIconC = '';
const DEIconSchRestriction = '';
const DEIconSchClosure = '';
const Incident = '';
const Roadwork = '';
const PLIcon = '';
const reportIcon = '';
const DEAdvURL = 'https://tmc.deldot.gov/json/advisory.json';
const DESchURL = 'https://deldot.gov/json/str.json';
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="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="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);
    }
    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)');
        eval('W.map.addLayer(' + state.substring(0,2) + 'DOTLayer)');
        eval(state.substring(0,2) + "DOTLayer.setZIndex(" + newZIndex + ")");
    }
    function getFeed(url,callback) {
        GM_xmlhttpRequest({
            method: "GET",
            url: url,
            onload: function(response) {
                var result = response;
                callback(result);
            }
        });
    }
    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('YYYY/MM/DD HH:mm');
                } 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('YYYY/MM/DD HH:mm');
                    } 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 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('YYYY/MM/DD HH:mm');
                        } 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('YYYY/MM/DD HH:mm');
                        } 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('YYYY/MM/DD 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('YYYY/MM/DD HH:mm'); //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 "DOTDEPopup":
                popupdetails("Delaware");
                getDEDOTAdv("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");
                break;
        }
    }
    //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 "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=175>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('chkPADOTEnabled', settings.PADOTEnabled);
        setChecked('chkDEDOTEnabled', settings.DEDOTEnabled);
        setChecked('chkNYDOTEnabled', settings.NYDOTEnabled);
        setChecked('chkNJDOTEnabled', settings.NJDOTEnabled);
        setChecked('chkWADOTEnabled', settings.WADOTEnabled);
        $(".overlay-button").click(function(){
            if (document.getElementById('chkDEDOTEnabled').checked) { eval('W.map.removeLayer(DEDOTLayer)'); }
            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 = "45000";
        }
        if (document.getElementById('chkDEDOTEnabled').checked) { buildDOTAdvLayers("DE"); getDEDOT();}
        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 = {
                PADOTEnabled: settings.PADOTEnabled,
                DEDOTEnabled: settings.DEDOTEnabled,
                NYDOTEnabled: settings.NYDOTEnabled,
                NJDOTEnabled: settings.NJDOTEnabled,
                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();
})();