WME DOT Cameras

Overlay DOT Cameras on the WME Map Object

As of 2021-09-15. See the latest version.

// ==UserScript==
// @name         WME DOT Cameras
// @namespace    https://greasyfork.org/en/users/668704-phuz
// @require      https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @version      1.33
// @description  Overlay DOT Cameras on the WME Map Object
// @author       phuz, doctorblah
// @include      /^https:\/\/(www|beta)\.waze\.com\/(?!user\/)(.{2,6}\/)?editor\/?.*$/
// @require      http://ajax.googleapis.com/ajax/libs/jquery/1.8.3/jquery.min.js
// @grant        GM_xmlhttpRequest
// @grant        GM_info
// @grant        GM_fetch
// @grant        GM_addStyle
// @require      https://cdn.jsdelivr.net/npm/hls.js@latest
// @require      https://unpkg.com/video.js/dist/video.js
// @require      https://unpkg.com/@videojs/http-streaming/dist/videojs-http-streaming.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/x2js/1.2.0/xml2json.min.js
// @connect      essentialintegrations.com
// @connect      repl.co
// @connect      jsdelivr.net
// @connect      511pa.com
// @connect      deldot.gov
// @connect      511ny.org
// @connect      511nj.org
// @connect      maryland.gov
// @connect      511virginia.org
// @connect      newengland511.org
// @connect      algotraffic.com
// @connect      nvroads.com
// @connect      tn.gov
// @connect      511ga.org
// @connect      iteriscdn.com
// @connect      skyvdn.com
// @connect      idrivearkansas.com
// @connect      akamaihd.net
// @connect      goakamai.org
// @connect      trafficwise.org
// @connect      ga.gov
// @connect      google.com
// @connect      fl511.com
// @connect      511ia.org
// @connect      idaho.gov
// @connect      arcgis.com
// @connect      kandrive.org
// @connect      mass511.com
// @connect      mi.us
// @connect      511mn.org
// @connect      modot.org
// @connect      mt.gov
// @connect      azureedge.net
// @connect      nebraska.gov
// @connect      nmroads.com
// @connect      ohgo.com
// @connect      tripcheck.com
// @connect      ri.gov
// @connect      utah.gov
// @connect      ca.gov
// @connect      txdot.gov
// @connect      modttraffic.om
// @connect      cttravelsmart.org
// @connect      vaisala.com
// @connect      skyvdn.com
// @connect      cotrip.org
// @connect      austintexas.gov
// @connect      wyoroad.info
// @connect      ncdot.gov
/* global OpenLayers */
/* global W */
/* global WazeWrap */
/* global $ */
/* global I18n */
/* global _ */
// ==/UserScript==

let ALLayer, AKLayer, AZLayer, ARLayer, CALayer, COLayer, CTLayer, DELayer, DCLayer, FLLayer, GALayer, HILayer, IDLayer, ILLayer, INLayer, IALayer, KSLayer, KYLayer, LALayer, MELayer, MDLayer, MALayer, MILayer, MNLayer, MSLayer, MOLayer, MTLayer, NELayer, NVLayer, NHLayer, NJLayer, NMLayer, NYLayer, NWLayer, NCLayer, NDLayer, OHLayer, OKLayer, ORLayer, PALayer, RILayer, SCLayer, SDLayer, TNLayer, TXLayer, UTLayer, VTLayer, VALayer, WALayer, WILayer, WVLayer, WYLayer;
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;
var settings, video, player, hls, staticUpdateID, newZIndex;
var state, stateLength;
let paWowzaKey = "";
let mapBounds;
const updateMessage = "► Performance improvement";
const x2js = new X2JS();
const camIcon = '';
const camRed = '';
const staticIcon = '';
const warning = '';

(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) {
            if (!OpenLayers.Icon) {
                installIcon();
            }
            init();
            console.log("WME DOT Cameras Loaded!");
        } 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=304760" target="_blank">WME DOT Cameras</a> v' + GM_info.script.version + '<br>',
            '<table border=1 style="text-align:center;width:100%;padding:10px;">',
            '<tr><td width=100 style="text-align:center"><b>Enable</b></td><td style="text-align:center"><b>State</b></td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkAKCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>AK</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkALCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>AL</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkARCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>AR</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkAZCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>AZ</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkCACamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>CA</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkCOCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>CO</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkCTCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>CT</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkDCCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>DC</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkDECamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>DE</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkFLCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>FL</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkGACamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>GA</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkHICamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>HI</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkIACamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>IA</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkIDCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>ID</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkILCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>IL</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkINCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>IN</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkKSCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>KS</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkKYCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>KY</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkLACamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>LA</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkMACamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>MA</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkMDCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>MD</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkMICamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>MI</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkMNCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>MN</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkMOCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>MO</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkMSCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>MS</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkMTCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>MT</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkNCCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>NC</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkNDCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>ND</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkNECamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>NE</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkNWCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>New England</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkNJCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>NJ</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkNMCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>NM</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkNVCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>NV</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkNYCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>NY</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkOHCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>OH</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkOKCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>OK</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkORCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>OR</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkPACamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>PA</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkRICamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>RI</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkSCCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>SC</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkSDCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>SD</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkTNCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>TN</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkTXCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>TX (Austin only)</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkUTCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>UT</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkVACamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>VA</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkWACamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>WA</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkWICamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>WI</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkWVCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>WV (waiting for SSL)</td></tr>',
            '<tr><td align=center><input type="checkbox" id="chkWYCamEnabled" class="wmeDOTCamCheckbox"></td><td align=center>WY</td></tr>',
            '</table>',
            '</div></div>'
        ].join(' '));
        new WazeWrap.Interface.Tab('DOT Cameras', $section.html(), initializeSettings);
        WazeWrap.Interface.ShowScriptUpdate("WME DOT Cameras", GM_info.script.version, updateMessage, "https://greasyfork.org/en/scripts/407690-wme-dot-cameras", "https://www.waze.com/forum/viewtopic.php?f=819&t=304760");
        getBounds();
        W.map.events.register("moveend", W.map, function () {
            getBounds();
            redrawCams();
        });
    }
    function getBounds() {
        mapBounds = W.map.getExtent();
        mapBounds.transform(new OpenLayers.Projection("EPSG:900913"), new OpenLayers.Projection("EPSG:4326"));
    }
    getFeed("http://scripts.essentialintegrations.com/CSS", "css", function (result) {
        GM_addStyle(result);
    })
    //Build the State Layers
    function buildDOTCamLayers(state) {
        eval(state.substring(0, 2) + 'Layer = new OpenLayers.Layer.Markers("' + state.substring(0, 2) + 'Layer")');
        eval('W.map.addLayer(' + state.substring(0, 2) + 'Layer)');
        //eval(state + "Layer.setZIndex(" + newZIndex + ")");
        W.map.getOLMap().setLayerIndex(eval(state.substring(0, 2) + 'Layer'), 10);
    }
    function getFeed(url, type, callback) {
        GM_xmlhttpRequest({
            method: "GET",
            url: url,
            onload: function (response) {
                var result = response.responseText;
                callback(result);
            }
        });
    }
    function redrawCams() {
        for (const property in settings) {
            let state = property.replace("chk", "").replace("CamEnabled", "");
            if (state.length == 2) {
                if (document.getElementById('chk' + state + 'CamEnabled').checked) {
                    eval('W.map.removeLayer(' + state + 'Layer)');
                    buildDOTCamLayers(state);
                    eval('testCam(' + state + 'Feed, config.' + state + ')');
                }
            }
        }
    }
    function getCam(state) {
        let j = 0;
        while (j < state.URL.length) {
            console.log(state.URL);
            getFeed(state.URL[j], "json", function (res) {
                let resultObj = [];
                if (state.x) {
                    resultObj = state.x(x2js.xml_str2json(res));
                } else if (state.y) {
                    resultObj = state.y(JSON.parse(res.toString().match(/(?<=camera_data = )[\s\S]*/)));
                } else {
                    resultObj = state.data(JSON.parse(res));
                }
                eval(state.scheme(resultObj[1]).state + 'Feed = resultObj');
                testCam(resultObj, state);
            });
            j++;
        }
    }
    function testCam(resultObj, state) {
        let i = 0;
        while (i < resultObj.length) {
            if ((state.scheme(resultObj[i]).lon > mapBounds.left) && (state.scheme(resultObj[i]).lon < mapBounds.right)) {
                if ((state.scheme(resultObj[i]).lat > mapBounds.bottom) && (state.scheme(resultObj[i]).lat < mapBounds.top)) {
                    drawCam(state.scheme(resultObj[i]));
                }
            }
            i++;
        }
    }
    function drawCam(spec) {
        var icon;
        var size = new OpenLayers.Size(20, 20);
        if (spec.enabled == false) {
            icon = new OpenLayers.Icon(camRed, size);
        } else {
            icon = new OpenLayers.Icon(camIcon, size);
        }
        var offset = new OpenLayers.Pixel(-(size.w / 2), -size.h);
        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(spec.lon, spec.lat).transform(epsg4326, projectTo);
        var newMarker = new OpenLayers.Marker(lonLat, icon);
        newMarker.title = spec.desc;
        newMarker.url = spec.src;
        newMarker.width = spec.width;
        newMarker.height = spec.height;
        newMarker.state = spec.state;
        newMarker.camType = spec.camType;
        newMarker.location = lonLat;
        //newMarker.setOpacity(.8);
        newMarker.events.register('click', newMarker, popupCam);
        eval(spec.state + 'Layer.addMarker(newMarker)');
    }
    //Generate the Camera Popup
    function popupCam(evt) {
        //Code to check if WME Toolbox is running, and if it is, go no further (hopefully temporary!) - Fixed 10-Dec-2020 but I'm leaving the code here because I don't trust TB yet :P
        //var i = 0;
        //while (i < document.getElementsByTagName('script').length) {
        //if (document.getElementsByTagName('script')[i].src == "chrome-extension://ihebciailciabdiknfomleeccodkdejn/scripts/WME_Toolbox.prod.min.js") {
        //alert("WME DOT Cameras cannot run if Toolbox is enabled, due to current issues with the Toolbox extension.  Please disable the Toolbox extension in order to use this script until the issue is resolved.");
        //return;
        //}
        //i++;
        //}
        clearInterval(staticUpdateID);
        $("#gmPopupContainerCam").remove();
        $("#gmPopupContainerCam").hide();
        W.map.moveTo(this.location);
        console.log(this.url)
        var popupHTML = [];
        var titleNC = "";

        popupHTML[0] = (['<div id="gmPopupContainerCam" style="margin: 1;text-align: center;padding: 5px;z-index: 1100">' +
            '<a href="#close" id="gmCloseCamDlgBtn" title="Close" class="modelCloseCam" style="color:#FF0000;">X</a>' +
            '<table border=0><tr><td><div id="mycamdivheader" style="min-height: 20px;"></div></td></tr>' +
            '<tr><td><center><h4>' + this.title + '</h4></td></tr>' +
            '<tr><td><div id="videoDiv">' +
            '<video id="hlsVideo" width=' + this.width + ' height=' + this.height + ' controls autoplay></video>' +
            '</div></td></tr>' +
            '</table></div>'
        ]);
        popupHTML[1] = (['<div id="gmPopupContainerCam" style="margin: 1;text-align: center;padding: 5px;z-index: 1100">' +
            '<a href="#close" id="gmCloseCamDlgBtn" title="Close" class="modelCloseCam" style="color:#FF0000;">X</a>' +
            '<table border=0><tr><td><div id="mycamdivheader" style="min-height: 20px;"></div></td></tr>' +
            '<tr><td><center><h4>' + this.title + '</h4></td></tr>' +
            '<tr><td><img src="' + this.url + '" style="width:400px" id="staticimage"></td></tr>' +
            '</table></div>'
        ]);
        let metaHead = `<meta http-equiv="Content-Security-Policy" content="default-src 'self' data: gap: https://ssl.gstatic.com 'unsafe-eval'; style-src 'self' 'unsafe-inline'; media-src *;**script-src 'self' http://onlineerp.solution.quebec 'unsafe-inline' 'unsafe-eval';** "></meta>`;
        popupHTML[2] = (['<div id="gmPopupContainerCam" style="margin: 1;text-align: center;padding: 5px;z-index: 1100">' +
            '<a href="#close" id="gmCloseCamDlgBtn" title="Close" class="modelCloseCam" style="color:#FF0000;">X</a>' +
            '<table border=0><tr><td><div id="mycamdivheader" style="min-height: 20px;"></div></td></tr>' +
            '<tr><td><center><h4>' + this.title + '</h4></td></tr>' +
            '<tr><td><div id="videoDiv">' +
            '<video id="hlsVideo" width=' + this.width + ' height=' + this.height + ' controls autoplay style="width:400px"></video>' +
            '</div></td></tr>' +
            '</table></div>'
        ]);
        popupHTML[3] = popupHTML[2];
        popupHTML[4] = (['<div id="gmPopupContainerCam" style="margin: 1;text-align: center;padding: 5px;z-index: 1100">' +
            '<a href="#close" id="gmCloseCamDlgBtn" title="Close" class="modelCloseCam" style="color:#FF0000;">X</a>' +
            '<table border=0><tr><td><div id="mycamdivheader" style="min-height: 20px;"></div></td></tr>' +
            '<tr><td><center><h4><div id="titleNC" >' + titleNC + '</div></h4></td></tr>' +
            '<tr><td><img src="' + this.url + '" style="width:400px" id="staticimage"></td></tr>' +
            '</table></div>'
        ]);
        var currentCamURL = this.url;
        switch (this.camType) {
            case 0:
                $("body").append(popupHTML[0]);
                setTimeout(function () {
                    video = document.getElementById('hlsVideo');
                    var videoSrc = currentCamURL;
                    if (hls) { hls.destroy(); }
                    if (Hls.isSupported()) {
                        console.log('Loading video from ' + videoSrc);
                        hls = new Hls();
                        hls.loadSource(videoSrc);
                        hls.attachMedia(video);
                        hls.on(Hls.Events.MANIFEST_PARSED, function () {
                            video.play();
                        });
                    }
                }, 1000);
                break;
            case 1:
                $("body").append(popupHTML[1]);
                staticUpdateID = setInterval(function () {
                    var camImage = document.getElementById('staticimage');
                    if (currentCamURL.includes('?')) { camImage.src = `${currentCamURL}&rand=${Math.random()}` }
                    else { camImage.src = currentCamURL + '?rand=' + Math.random(); }
                }, 5000);
                break;
            case 2:
                $("body").append(popupHTML[2]);
                getFeed('https://www.511pa.com/flowplayeri.aspx?CAMID=' + currentCamURL, 'text', function (res) {
                    let paCamURL = res.match(/(?=pa511).*(?=')/);
                    currentCamURL = "https://" + paCamURL;
                    setTimeout(function () {
                        video = document.getElementById('hlsVideo');
                        var videoSrc = currentCamURL;
                        if (hls) { hls.destroy(); }
                        if (Hls.isSupported()) {
                            console.log('Loading video from ' + videoSrc);
                            hls = new Hls();
                            hls.loadSource(videoSrc);
                            hls.attachMedia(video);
                            hls.on(Hls.Events.MANIFEST_PARSED, function () {
                                video.play();
                            });
                        }
                    }, 1000);
                });
                break;
            case 3:
                $("body").append(popupHTML[3]);
                let id;
                if (currentCamURL.includes("njtpk-wink")) {
                    id = 1;
                } else { id = 2; }
                getFeed('https://511nj.org/api/client/camera/getHlsToken?Id=' + id, 'json', function (res) {
                    currentCamURL = currentCamURL + "?otp=" + JSON.parse(res).Data.Token;
                    setTimeout(function () {
                        video = document.getElementById('hlsVideo');
                        var videoSrc = currentCamURL;
                        if (hls) { hls.destroy(); }
                        if (Hls.isSupported()) {
                            console.log('Loading video from ' + videoSrc);
                            hls = new Hls();
                            hls.loadSource(videoSrc);
                            hls.attachMedia(video);
                            hls.on(Hls.Events.MANIFEST_PARSED, function () {
                                video.play();
                            });
                        }
                    }, 1000);
                });
                break;
            case 4:
                $("body").append(popupHTML[4]);
                getFeed('https://tims.ncdot.gov/tims/api/traffic/cameras/' + currentCamURL, 'json', function (res) {
                    currentCamURL = JSON.parse(res).imageURL;
                    titleNC = JSON.parse(res).locationName;
                    document.getElementById("titleNC").innerHTML = titleNC;
                    staticUpdateID = setInterval(function () {
                        var camImage = document.getElementById('staticimage');
                        if (currentCamURL.includes('?')) { camImage.src = `${currentCamURL}&rand=${Math.random()}` }
                        else { camImage.src = currentCamURL + '?rand=' + Math.random(); }
                    }, 100);
                });
        }
        //Position the modal based on the position of the click event
        $("#gmPopupContainerCam").css({ left: document.getElementById("user-tabs").offsetWidth + W.map.getPixelFromLonLat(W.map.getCenter()).x - document.getElementById("gmPopupContainerCam").clientWidth - 10 });
        $("#gmPopupContainerCam").css({ top: document.getElementById("left-app-head").offsetHeight + W.map.getPixelFromLonLat(W.map.getCenter()).y - (document.getElementById("gmPopupContainerCam").clientHeight / 2) });
        //Add listener for popup's "Close" button
        $("#gmCloseCamDlgBtn").click(function () {
            if (hls) {
                hls.destroy();
            }
            clearInterval(staticUpdateID);
            $("#gmPopupContainerCam").remove();
            $("#gmPopupContainerCam").hide();
        });
        dragElement(document.getElementById("gmPopupContainerCam"));
        setTimeout(function () {
            fetch(currentCamURL)
                .then(response => {
                    if (!response.ok) {
                        //Bad feed
                        $('#videoDiv').empty();
                        document.getElementById('videoDiv').innerHTML = "<br>Sorry, this feed is currently offline.";
                    } else {
                        //Good Feed
                    }
                });
        }, 1500);
    }
    // Make the DIV element draggable:
    function dragElement(elmnt) {
        var pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
        if (document.getElementById("mycamdivheader")) {
            // if present, the header is where you move the DIV from:
            document.getElementById("mycamdivheader").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;
        }
    }
    //Initialize Settings
    function initializeSettings() {
        stateLength = document.getElementsByClassName("wmeDOTCamCheckbox").length;
        loadSettings();
        //Set the state checkboxes according to saved settings
        for (var i = 0; i < stateLength; i++) {
            state = document.getElementsByClassName("wmeDOTCamCheckbox")[i].id.replace("chk", "").replace("CamEnabled", "");
            setChecked('chk' + state + 'CamEnabled', eval('settings.' + state + 'CamEnabled'));
        }
        //Build the layers for the selected states
        for (var i = 0; i < stateLength; i++) {
            state = document.getElementsByClassName("wmeDOTCamCheckbox")[i].id.replace("chk", "").replace("CamEnabled", "");
            if (document.getElementById('chk' + state + 'CamEnabled').checked) { buildDOTCamLayers(state); eval('getCam(config.' + state + ')') }
        }
        document.getElementById('chkKSCamEnabled').disabled = true; // changed their feeds; we're working on it
        document.getElementById('chkMSCamEnabled').disabled = true; // ??
        document.getElementById('chkMTCamEnabled').disabled = true; // parser written but better feed would help
        document.getElementById('chkOKCamEnabled').disabled = true; // parser pending, would prefer better source
        document.getElementById('chkSDCamEnabled').disabled = true; // parser pending, need a good source
        document.getElementById('chkTXCamEnabled').disabled = true; // not working
        document.getElementById('chkWVCamEnabled').disabled = true; // still using http
        document.getElementById('chkHICamEnabled').disabled = true; // still using http
        document.getElementById('chkKYCamEnabled').disabled = true; // still using http

        //Add Handler for Checkbox Setting Changes
        $('.wmeDOTCamCheckbox').change(function () {
            var settingName = $(this)[0].id.substr(3);
            settings[settingName] = this.checked;
            saveSettings();
            if (this.checked) {
                buildDOTCamLayers(settingName.substring(0, 2));
                eval("getCam(config." + settingName.substring(0, 2) + ")");
            } else {
                //eval(settingName.substring(0, 2) + "Layer.destroy()");
                eval('W.map.removeLayer(' + settingName.substring(0, 2) + 'Layer)');
            }
        });

    }
    //Set Checkbox from Settings
    function setChecked(checkboxId, checked) {
        $('#' + checkboxId).prop('checked', checked);
    }
    //Load Saved Settings
    function loadSettings() {
        var loadedSettings = $.parseJSON(localStorage.getItem("Camera_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("wmeDOTCamCheckbox")[i].id.replace("chk", "").replace("CamEnabled", "");
                eval('localsettings.' + state + 'CamEnabled = document.getElementsByClassName("wmeDOTCamCheckbox")[i].checked');
            }
            localStorage.setItem("Camera_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 = {
        AK: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                return {
                    state: "AK",
                    camType: 1,
                    lon: cam[2],
                    lat: cam[3],
                    src: cam[4],
                    desc: cam[5]
                };
            },
            URL: ['https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/4/public/full?alt=json']
        },
        AL: {
            data(res) {
                let cams = [];
                let i = 0;
                while (i < res.length) {
                    cams.push(res[i].entries);
                    i++;
                }
                return cams.flat();
            },
            scheme(obj) {
                return {
                    state: "AL",
                    camType: 0,
                    lon: obj.longitude,
                    lat: obj.latitude,
                    src: obj.streamUrl,
                    desc: `${obj.primaryRoad} @ ${obj.crossStreet}`,
                    width: 480,
                    height: 360
                };
            },
            URL: ['https://algotraffic.com/api/v1/layers/cameras']
        },
        AR: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                return {
                    state: "AR",
                    camType: 0,
                    lon: cam[2],
                    lat: cam[3],
                    src: cam[4],
                    desc: cam[5],
                    width: 480,
                    height: 360
                };
            },
            URL: ['https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/12/public/full?alt=json']
        },
        AZ: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                return {
                    state: "AZ",
                    camType: 1,
                    lon: cam[2],
                    lat: cam[3],
                    src: cam[4],
                    desc: cam[5]
                };
            },
            URL: ['https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/18/public/full?alt=json']
        },
        CA: {
            data(res) {
                return res.data;
            },
            scheme(obj) {
                return {
                    state: "CA",
                    camType: 1,
                    lon: obj.cctv.location.longitude,
                    lat: obj.cctv.location.latitude,
                    src: obj.cctv.imageData.static.currentImageURL,
                    desc: obj.cctv.location.locationName
                };
            },
            URL: ['http://cwwp2.dot.ca.gov/data/d1/cctv/cctvStatusD01.json', 'http://cwwp2.dot.ca.gov/data/d2/cctv/cctvStatusD02.json', 'http://cwwp2.dot.ca.gov/data/d3/cctv/cctvStatusD03.json', 'http://cwwp2.dot.ca.gov/data/d4/cctv/cctvStatusD04.json', 'http://cwwp2.dot.ca.gov/data/d5/cctv/cctvStatusD05.json', 'http://cwwp2.dot.ca.gov/data/d6/cctv/cctvStatusD06.json', 'http://cwwp2.dot.ca.gov/data/d7/cctv/cctvStatusD07.json', 'http://cwwp2.dot.ca.gov/data/d8/cctv/cctvStatusD08.json', 'http://cwwp2.dot.ca.gov/data/d9/cctv/cctvStatusD09.json', 'http://cwwp2.dot.ca.gov/data/d10/cctv/cctvStatusD10.json', 'http://cwwp2.dot.ca.gov/data/d11/cctv/cctvStatusD11.json', 'http://cwwp2.dot.ca.gov/data/d12/cctv/cctvStatusD12.json']
        },
        CO: { // CO has a streaming camera feed as well, but will require a GET request to get the token and undetermined if it will play even after receiving (didn't work with VLC)
            data(res) {
                return res.CameraDetails.Camera;
            },
            scheme(obj) {
                return {
                    state: "CO",
                    camType: 1,
                    lon: obj.Location.Longitude,
                    lat: obj.Location.Latitude,
                    src: `https://i.cotrip.org/${obj.CameraView[0].ImageLocation}`,
                    desc: obj.Description
                };
            },
            URL: ['https://www.cotrip.org/camera/getStillCameras.do']
        },
        CT: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                return {
                    state: "CT",
                    camType: 1,
                    lon: cam[2],
                    lat: cam[3],
                    src: cam[4],
                    desc: cam[5]
                };
            },
            URL: ["https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/14/public/full?alt=json"]
        },
        DC: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                return {
                    state: "DC",
                    camType: 0,
                    lon: cam[2],
                    lat: cam[3],
                    src: cam[4],
                    desc: cam[5],
                    width: 480,
                    height: 360
                };
            },
            URL: ["https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/23/public/full?alt=json"]
        },
        DE: {
            data(res) {
                return res.videoCameras;
            },
            scheme(obj) {
                return {
                    state: "DE",
                    camType: 0,
                    lon: obj.lon,
                    lat: obj.lat,
                    src: obj.urls.m3u8s,
                    desc: `${obj.title} (${obj.county})`,
                    width: 550,
                    height: 300
                };
            },
            URL: ['https://tmc.deldot.gov/json/videocamera.json']
        },
        FL: {
            data(res) {
                return res.data;
            },
            scheme(obj) {
                return {
                    state: "FL",
                    camType: 0,
                    lon: obj.longitude,
                    lat: obj.latitude,
                    src: obj.videoUrl,
                    desc: obj.description2
                };
            },
            URL: ['https://fl511.com/List/GetData/Cameras?query=%7B%22columns%22%3A%5B%7B%22data%22%3Anull%2C%22name%22%3A%22%22%7D%2C%7B%22name%22%3A%22sortId%22%2C%22s%22%3Atrue%7D%2C%7B%22name%22%3A%22region%22%2C%22s%22%3Atrue%7D%2C%7B%22name%22%3A%22county%22%2C%22s%22%3Atrue%7D%2C%7B%22name%22%3A%22roadway%22%2C%22s%22%3Atrue%7D%2C%7B%22data%22%3A5%2C%22name%22%3A%22description2%22%7D%2C%7B%22data%22%3A6%2C%22name%22%3A%22%22%7D%5D%2C%22order%22%3A%5B%7B%22column%22%3A1%2C%22dir%22%3A%22asc%22%7D%2C%7B%22column%22%3A2%2C%22dir%22%3A%22asc%22%7D%5D%2C%22start%22%3A0%2C%22length%22%3A5000%2C%22search%22%3A%7B%22value%22%3A%22%22%7D%7D&lang=en']
        },
        GA: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                /*if (obj.properties.HLS) {return {state: "GA",camType:0,lon:obj.geometry.coordinates[0],lat:obj.geometry.coordinates[1],src:obj.properties.HLS,desc:obj.properties.location_description,width:480,height:360}}
                                   else {}*/
                return {
                    state: "GA",
                    camType: 1,
                    lon: obj.geometry.coordinates[0],
                    lat: obj.geometry.coordinates[1],
                    src: obj.properties.url,
                    desc: obj.properties.location_description
                };
            },
            URL: ['https://ga.cdn.iteris-atis.com/geojson/icons/metadata/icons.cctv.geojson']
        },
        HI: {
            x(res) {
                return res.ArrayOfCamera.Camera;
            },
            scheme(obj) {
                return {
                    state: "HI",
                    camType: 1,
                    lon: obj.Lon,
                    lat: obj.Lat,
                    src: obj.CameraImageURL.__text,
                    desc: obj.Description
                };
            },
            URL: ['http://goakamai.org/services/MapServiceProxy.asmx/GetFullCameraList']
        },
        IA: {
            data(res) {
                return res;
            },
            scheme(obj) {
                if (obj.views[0].type == "FLASH") { // IA only has RTMP streams currently
                    return {
                        state: "IA",
                        camType: 1,
                        lon: obj.location.longitude,
                        lat: obj.location.latitude,
                        src: obj.views[0].videoPreviewUrl,
                        desc: obj.name
                    };
                } else {
                    return {
                        state: "IA",
                        camType: 1,
                        lon: obj.location.longitude,
                        lat: obj.location.latitude,
                        src: obj.views[0].url,
                        desc: obj.name
                    };
                }
            },
            URL: ['https://hb.511ia.org/tgcameras/api/cameras']
        },
        ID: {
            data(res) {
                return res;
            },
            scheme(obj) {
                return {
                    state: "ID",
                    camType: 1,
                    lon: obj.location.longitude,
                    lat: obj.location.latitude,
                    src: obj.views[0].url,
                    desc: `${obj.name} ${obj.location.cityReference}`
                };
            },
            URL: ['https://hb.511.idaho.gov/tgcameras/api/cameras']
        },
        IL: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "IL",
                    camType: 1,
                    lon: obj.attributes.x,
                    lat: obj.attributes.y,
                    src: obj.attributes.SnapShot,
                    desc: obj.attributes.CameraLocation
                };
            },
            URL: ['https://services2.arcgis.com/aIrBD8yn1TDTEXoz/arcgis/rest/services/TrafficCamerasTM_Public/FeatureServer/0//query?where=y+%3E+0&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) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "IN",
                    camType: 1,
                    lon: obj.geometry.coordinates[0],
                    lat: obj.geometry.coordinates[1],
                    src: `http://pws.trafficwise.org/${obj.properties.image}`,
                    desc: obj.properties.description
                };
            },
            URL: ['http://pws.trafficwise.org/aries/cctv.json']
        }, //non-HTTPS will flag mixed media erro},
        KS: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "KS",
                    camType: 1,
                    lon: obj.attributes.lon,
                    lat: obj.attributes.lat,
                    src: `http://www.kandrive.org/cameras/${obj.attributes.url}`,
                    desc: obj.attributes.location
                };
            },
            URL: ['https://wfs.ksdot.org/arcgis_web_adaptor/rest/services/TravelInfo/Devices/MapServer/0/query?where=oid+%3E+0&text=&objectIds=&time=&geometry=&geometryType=esriGeometryPoint&inSR=&spatialRel=esriSpatialRelIntersects&relationParam=&outFields=*&returnGeometry=true&returnTrueCurves=false&maxAllowableOffset=&geometryPrecision=&outSR=&having=&returnIdsOnly=false&returnCountOnly=false&orderByFields=&groupByFieldsForStatistics=&outStatistics=&returnZ=false&returnM=false&gdbVersion=&historicMoment=&returnDistinctValues=false&resultOffset=&resultRecordCount=&queryByDistance=&returnExtentOnly=false&datumTransformation=&parameterValues=&rangeValues=&quantizationParameters=&f=pjson']
        },
        KY: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "KY",
                    camType: 1,
                    lon: obj.attributes.longitude,
                    lat: obj.attributes.latitude,
                    src: obj.attributes.snapshot,
                    desc: obj.attributes.description
                };
            },
            URL: ['https://services2.arcgis.com/CcI36Pduqd0OR4W9/arcgis/rest/services/trafficCamerasCur_Prd/FeatureServer/0/query?where=id+%3E+0&objectIds=&time=&geometry=&geometryType=esriGeometryPoint&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=']
        },
        LA: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                if (cam[6] == "0") {
                    return {
                        state: "LA",
                        camType: 0,
                        lon: cam[2],
                        lat: cam[3],
                        src: cam[4],
                        desc: cam[5],
                        width: 480,
                        height: 360
                    };
                } else {
                    return {
                        state: "LA",
                        camType: 1,
                        lon: cam[2],
                        lat: cam[3],
                        src: cam[4],
                        desc: cam[5]
                    };
                }
            },
            URL: ['https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/20/public/full?alt=json']
        },
        MA: {
            data(res) {
                return res.item2;
            },
            scheme(obj) {
                return {
                    state: "MA",
                    camType: 1,
                    enabled: true,
                    lon: obj.location[1],
                    lat: obj.location[0],
                    src: `https://mass511.com/map/Cctv/${obj.itemId}`,
                    desc: ""
                };
            },
            URL: ['https://mass511.com/map/mapIcons/Cameras']
        },
        MD: {
            data(res) {
                return res.data;
            },
            scheme(obj) {
                return {
                    state: "MD",
                    camType: 0,
                    lon: obj.lon,
                    lat: obj.lat,
                    src: `https://${obj.cctvIp}/rtplive/${obj.id}/playlist.m3u8`,
                    desc: obj.description,
                    width: 480,
                    height: 360
                };
            },
            URL: ['https://chartexp1.sha.maryland.gov//CHARTExportClientService/getCameraMapDataJSON.do']
        },
        MI: {
            data(res) {
                return res;
            },
            scheme(obj) {
                return {
                    state: "MI",
                    camType: 1,
                    lon: obj.county.match(/(?<=lon=)[\s\S]*(?=&zoom)/)[0],
                    lat: obj.county.match(/(?<=lat=)[\s\S]*(?=&lon)/)[0],
                    src: obj.image.match(/(?<=src=")[\s\S]*(?=" height=)/)[0],
                    desc: `${obj.route} ${obj.location}`
                };
            },
            URL: ['https://mdotjboss.state.mi.us/MiDrive//camera/list']
        },
        MN: {
            data(res) {
                return res;
            },
            scheme(obj) {
                if (obj.views[0].type == "WMP") {
                    return {
                        state: "MN",
                        camType: 0,
                        lon: obj.location.longitude,
                        lat: obj.location.latitude,
                        src: obj.views[0].url,
                        desc: `${obj.location.routeId} - ${obj.location.cityReference}`,
                        width: 480,
                        height: 360
                    };
                } else {
                    return {
                        state: "MN",
                        camType: 1,
                        lon: obj.location.longitude,
                        lat: obj.location.latitude,
                        src: obj.views[0].url,
                        desc: `${obj.location.routeId} - ${obj.location.cityReference}`
                    };
                }
            },
            URL: ['https://hb.511mn.org/tgcameras/api/cameras']
        },
        MO: {
            data(res) {
                return res;
            },
            scheme(obj) {
                return {
                    state: "MO",
                    camType: 0,
                    lon: obj.x,
                    lat: obj.y,
                    src: obj.html,
                    desc: obj.location
                };
            },
            URL: ['https://traveler.modot.org/timconfig/feed/desktop/StreamingCams2.json'] // This is disabled until they serve over HTTPS https://traveler.modot.org/timconfig/feed/desktop/StreamingCams2.json
        },
        MS: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                return {
                    state: "MS",
                    camType: 0,
                    lon: cam[2],
                    lat: cam[3],
                    src: cam[4],
                    desc: cam[5],
                    width: 480,
                    height: 360
                };
            },
            URL: ['https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/16/public/full?alt=json']
        },
        MT: {
            URL: ['http://roadreport.mdt.mt.gov/map/getRWISMarkers.php']
        },
        NC: {
            data(res) {
                return res;
            },
            scheme(obj) {
                return {
                    state: "NC",
                    camType: 4,
                    lon: obj.longitude,
                    lat: obj.latitude,
                    src: obj.id,
                    desc: ''
                };
            },
            URL: ['https://tims.ncdot.gov/tims/api/traffic/cameras/']
        },
        ND: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "ND",
                    camType: 1,
                    lon: obj.geometry.coordinates[0],
                    lat: obj.geometry.coordinates[1],
                    src: obj.properties.Cameras[0].LinkPath,
                    desc: obj.properties.Cameras[0].Description
                };
            },
            URL: ['http://essentialintegrations.com/scripts/NDCam']
        },
        NE: {
            data(res) {
                return res;
            },
            scheme(obj) {
                return {
                    state: "NE",
                    camType: 1,
                    lon: obj.location.longitude,
                    lat: obj.location.latitude,
                    src: obj.views[0].url,
                    description: obj.name
                };
            },
            URL: ['https://hb.511.nebraska.gov/tgcameras/api/cameras']
        },
        NJ: {
            data(res) {
                return res.Data.CameraData;
            },
            scheme(obj) {
                let online, type;
                if (obj.StopCameraFlag == true) { online = false; } else { online = true; }
                if ((obj.CameraMainDetail[0].cameratype == "Video") && (online == true)) { type = 3; } else { type = 1; }
                return {
                    state: "NJ",
                    enabled: online,
                    camType: type,
                    lon: obj.longitude,
                    lat: obj.latitude,
                    src: obj.CameraMainDetail[0].URL,
                    desc: obj.name,
                    width: 480,
                    height: 360
                };
            },
            URL: ['https://511nj.org/api/client/camera/GetCameraDataByTourId?tourid=&rnd=202007201015']
        },
        NM: {
            data(res) {
                return res.cameraInfo;
            },
            scheme(obj) {
                return {
                    state: "NM",
                    camType: 1,
                    lon: obj.lon,
                    lat: obj.lat,
                    src: "https://servicev4.nmroads.com/RealMapWAR/GetCameraImage?ts=0&cameraName=" + obj.name + "&" + Date.now(),
                    desc: obj.title
                };
            },
            URL: ['https://servicev4.nmroads.com/RealMapWAR//GetCameraInfo']
        },
        NV: {
            x(res) {
                return res.ArrayOfCamera.Camera;
            },
            scheme(obj) {
                return {
                    state: "NV",
                    camType: 0,
                    lon: obj.Lon,
                    lat: obj.Lat,
                    src: obj.StreamingURL.__text,
                    desc: obj.Description,
                    width: 480,
                    height: 360
                };
            },
            URL: ['https://nvroads.com/services/MapServiceProxy.asmx/GetFullCameraList']
        },
        NW: {
            data(res) {
                return res;
            },
            scheme(obj) {
                return {
                    state: "NW",
                    camType: 1,
                    lon: obj.Longitude,
                    lat: obj.Latitude,
                    src: obj.ImageUrl,
                    desc: obj.Name
                };
            },
            URL: ['http://newengland511.org/Traffic/GetCameras']
        },
        NY: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                if (cam[6] == "0") {
                    return {
                        state: "NY",
                        camType: 0,
                        lon: cam[2],
                        lat: cam[3],
                        src: cam[4],
                        desc: cam[5],
                        width: 480,
                        height: 360
                    };
                } else {
                    return {
                        state: "NY",
                        camType: 1,
                        lon: cam[2],
                        lat: cam[3],
                        src: cam[4],
                        desc: cam[5]
                    };
                }
            },
            URL: ['https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/6/public/full?alt=json']
        },
        OH: {
            data(res) {
                return res;
            },
            scheme(obj) {
                return {
                    state: "OH",
                    camType: 1,
                    lon: obj.Longitude,
                    lat: obj.Latitude,
                    src: obj.Cameras[0].LargeURL,
                    desc: obj.Location
                };
            },
            URL: ['https://api.ohgo.com/roadmarkers/cameras']
        },
        OR: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "OR",
                    camType: 1,
                    lon: obj.attributes.longitude,
                    lat: obj.attributes.latitude,
                    src: `https://tripcheck.com/RoadCams/cams/${obj.attributes.filename}`,
                    desc: obj.attributes.title
                };
            },
            URL: ['https://www.tripcheck.com/Scripts/map/data/cctvinventory.js']
        },
        PA: {
            y(res) {
                return res.cams;
            },
            scheme(obj) {
                let online;
                if (obj.title.includes("UNAVAILABLE")) { online = false; } else { online = true; }
                if (obj.md5.substring(0, 3) == "ptc") {
                    return {
                        state: "PA",
                        enabled: online,
                        camType: 1,
                        lon: obj.start_lng,
                        lat: obj.start_lat,
                        src: "https://www.paturnpike.com/webmap/1_devices/" + obj.md5.replace("ptc_camera_", "cam") + ".jpg",
                        desc: obj.title
                    };
                } else {
                    return {
                        state: "PA",
                        enabled: online,
                        camType: 2,
                        lon: obj.start_lng,
                        lat: obj.start_lat,
                        src: obj.md5,
                        desc: obj.title
                    };
                }
            },
            URL: ['https://www.511pa.com/wsvc/gmap.asmx/buildCamerasJSONjs']
        },
        RI: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "RI",
                    camType: 1,
                    lon: obj.attributes.Longitude,
                    lat: obj.attributes.Latitude,
                    src: obj.attributes.CCVEWebURL,
                    desc: obj.attributes.Description
                };
            },
            URL: ['https://vueworks.dot.ri.gov/arcgis/rest/services/VUEWorks_V10_2/MapServer/24/query?where=OBJECTID+%3E+0&text=&objectIds=&time=&geometry=&geometryType=esriGeometryPoint&inSR=&spatialRel=esriSpatialRelIntersects&relationParam=&outFields=*&returnGeometry=true&returnTrueCurves=false&maxAllowableOffset=&geometryPrecision=&outSR=&returnIdsOnly=false&returnCountOnly=false&orderByFields=&groupByFieldsForStatistics=&outStatistics=&returnZ=false&returnM=false&gdbVersion=&returnDistinctValues=false&resultOffset=&resultRecordCount=&f=pjson']
        },
        SC: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "SC",
                    camType: 0,
                    lon: obj.geometry.coordinates[0],
                    lat: obj.geometry.coordinates[1],
                    src: obj.properties.https_url,
                    desc: obj.properties.title,
                    width: 480,
                    height: 360
                };
            },
            URL: ['https://files0.beta.iteriscdn.com/WebApps/SC/SafeTravel4/data/geojson/icons/metadata/icons.cctv.geojson']
        },
        TN: {
            data(res) {
                return res.actions;
            },
            scheme(obj) {
                return {
                    state: "TN",
                    camType: 0,
                    lon: obj.dataItem.coordinates.lng,
                    lat: obj.dataItem.coordinates.lat,
                    src: obj.dataItem.httpsVideoUrl,
                    desc: obj.dataItem.title,
                    width: 480,
                    height: 360
                };
            },
            URL: ['https://smartway.tn.gov/Traffic/api/Cameras/0']
        },
        TX: {
            data(res) { return res }, scheme(obj) { return { state: "TX", camType: 1, lon: obj.location.longitude, lat: obj.location.latitude, src: obj.screenshot_address, desc: obj.location_name } },
            URL: ['https://data.austintexas.gov/resource/b4k4-adkb.json']
        }, // TX isn't working until we figure out POST/response
        UT: {
            x(res) {
                return res.kml.Document.Placemark;
            },
            scheme(obj) {
                let coordinates = obj.Point.coordinates.split(",");
                return {
                    state: "UT",
                    camType: 1,
                    lon: coordinates[0],
                    lat: coordinates[1],
                    src: obj.ExtendedData.SchemaData.SimpleData[6].__text,
                    desc: obj.name
                };
            },
            URL: ['https://www.udottraffic.utah.gov/KmlFile.aspx?kmlFileType=Camera']
        },
        VA: {
            data(res) {
                return res.features;
            },
            scheme(obj) {
                return {
                    state: "VA",
                    camType: 0,
                    lon: obj.geometry.coordinates[0],
                    lat: obj.geometry.coordinates[1],
                    src: obj.properties.https_url,
                    desc: obj.properties.description
                };
            },
            URL: ["https://www.511virginia.org/data/geojson/icons.cameras.geojson"]
        },
        WA: {
            data(res) {
                return res.feed.entry;
            },
            scheme(obj) {
                let cam = obj.gs$cell.inputValue.split("|");
                return {
                    state: "WA",
                    camType: 1,
                    lon: cam[2],
                    lat: cam[3],
                    src: cam[4],
                    desc: cam[5]
                };
            },
            URL: ['https://spreadsheets.google.com/feeds/cells/1TUXtPnGHtcXsHw8Y3nxwqWGH_Waj9dcMlmwTcb2nW2k/10/public/full?alt=json']
        },
        WI: {
            data(res) {
                return res;
            },
            scheme(obj) {
                return {
                    state: "WI",
                    camType: 1,
                    lon: obj.Longitude,
                    lat: obj.Latitude,
                    src: obj.Url,
                    desc: obj.Name
                };
            },
            URL: ['http://scripts.essentialintegrations.com/WICam']
        },
        WV: {
            data(res) {
                return res.changes["com.orci.opentms.web.public511.components.camera.shared.data.CameraBean"].changes;
            },
            scheme(obj) {
                return {
                    state: "WV",
                    camType: 0,
                    lon: obj.entity.x,
                    lat: obj.entity.y,
                    src: obj.entity.realTimeMessageUrl,
                    desc: obj.entity.description
                };
            },
            URL: ['http://scripts.essentialintegrations.com/WV']
        },
        WY: {
            data(res) {
                return res.features
            },
            scheme(obj) {
                let LonLat = new OpenLayers.LonLat([obj.geometry.x, obj.geometry.y]).transform('EPSG:3857', 'EPSG:4326')
                console.log(obj.attributes.IMAGEMARKUP.match(/(?=https:\/\/webcams)[\s\S]*?(?<=\.jpg)/)[0])
                return { state: "WY", camType: 1, lon: LonLat.lon, lat: LonLat.lat, src: obj.attributes.IMAGEMARKUP.match(/(?=https:\/\/webcams)[\s\S]*?(?<=\.jpg)/)[0], desc: obj.attributes.IMAGEMARKUP.match(/(?<=<p><i>)[\s\S]*?(?=<\/i><br\/><a href)/)[0] }
            },
            URL: ['https://map.wyoroad.info/wtimap/data/wtimap-webcameras.json']
        }
    };
})();