Overlay DOT Cameras on the WME Map Object
目前為
// ==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.05
// @description Overlay DOT Cameras on the WME Map Object
// @author phuz
// @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
// @connect jsdelivr.net
// @connect 511pa.com
// @connect deldot.gov
// @connect 511ny.org
// @connect 511nj.org
// @connect maryland.gov
// @connect 511virginia.org
/* global OpenLayers */
/* global W */
/* global WazeWrap */
/* global $ */
/* global I18n */
/* global _ */
// ==/UserScript==
let PALayer;
let DELayer;
let NYLayer;
let NJLayer;
let MDLayer;
let VALayer;
var settings;
var video;
var player;
var hls;
const camIcon = '';
const warning = '';
const PAURL = 'https://www.511pa.com/wsvc/gmap.asmx/buildCamerasJSONjs';
const DEURL = 'https://tmc.deldot.gov/json/videocamera.json';
const NYAPI = 'ZGE3YzZkNzBmMWY4NGEyZWJhOWFhODBmYTE2NmI2ZTg=';
const NYURL = `https://511ny.org/api/getcameras?key=${atob(NYAPI)}&format=json`;
const NJURL = 'https://511nj.org/api/client/camera/GetCameraDataByTourId?tourid=&rnd=202007201015';
const MDURL = 'https://chartexp1.sha.maryland.gov//CHARTExportClientService/getCameraMapDataJSON.do';
const VAURL = "https://www.511virginia.org/data/geojson/icons.cameras.geojson";
(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 Cameras Loaded!");
init();
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">',
'<table border=1 style="text-align:center;width:100%;padding:10px;">',
'<tr><td width=50 valign=middle><img src="' + warning + '" height=16 width=16></td><td style="text-align:center">Warning: WME Toolbox has caused interference with methods this script uses to play video feeds. Until the Toolbox issues are resolved, it needs to remain disabled in order to run this script.</td><td width=50><img src="' + warning + '" height=16 width=16></td></tr>',
'<tr><td colspan=2 style="text-align:center"><b>Enable</b></td><td style="text-align"><b>State</b></td></tr>',
'<tr><td colspan=2 align=center><input type="checkbox" id="chkDECamEnabled" class="wmedotSettingsCheckbox"></td><td align=center>DE</td></tr>',
'<tr><td colspan=2 align=center><input type="checkbox" id="chkMDCamEnabled" class="wmedotSettingsCheckbox"></td><td align=center>MD</td></tr>',
'<tr><td colspan=2 align=center><input type="checkbox" id="chkNYCamEnabled" class="wmedotSettingsCheckbox"></td><td align=center>NY</td></tr>',
'<tr><td colspan=2 align=center><input type="checkbox" id="chkNJCamEnabled" class="wmedotSettingsCheckbox"></td><td align=center>NJ</td></tr>',
'<tr><td colspan=2 align=center><input type="checkbox" id="chkPACamEnabled" class="wmedotSettingsCheckbox"></td><td align=center>PA</td></tr>',
'<tr><td colspan=2 align=center><input type="checkbox" id="chkVACamEnabled" class="wmedotSettingsCheckbox"></td><td align=center>VA</td></tr>',
'</table>',
'Click <a href="https://www.waze.com/forum/viewtopic.php?f=819&t=145570&start=2270#p2078310">here</a> to see the forum post regarding the conflict with the current version of WME Toolbox',
'</div></div>'
].join(' '));
new WazeWrap.Interface.Tab('DOT Cameras', $section.html(), initializeSettings);
}
//Build the State Layers
function buildDOTCamLayers(state) {
switch(state) {
case "VA":
VALayer = new OpenLayers.Layer.Markers("VALayer");
W.map.addLayer(VALayer);
break;
case "MD":
MDLayer = new OpenLayers.Layer.Markers("MDLayer");
W.map.addLayer(MDLayer);
break;
case "PA":
PALayer = new OpenLayers.Layer.Markers("PALayer");
W.map.addLayer(PALayer);
break;
case "DE":
DELayer = new OpenLayers.Layer.Markers("DELayer");
W.map.addLayer(DELayer);
break;
case "NY":
NYLayer = new OpenLayers.Layer.Markers("NYLayer");
W.map.addLayer(NYLayer);
break;
case "NJ":
NJLayer = new OpenLayers.Layer.Markers("NJLayer");
W.map.addLayer(NJLayer);
}
}
function getCamFeed(url,type,callback) {
GM_xmlhttpRequest({
method: "GET",
url: url,
onload: function(response) {
var result = response.responseText;
callback(result);
}
});
}
//Get the PA Camera JSON Feed (We can't used the canned getCamFeed function because of the extra legwork we have to do with the PA feed)
function getPA() {
$.ajax ({
type: 'GET',
url: PAURL,
dataType: 'text',
success: function (results) {
let result = results.toString().match(/camera_data = ([\s\S]*)/);
var resultObj = JSON.parse(result[1]).cams;
var i=0;
while (i<resultObj.length) {
drawCameras("PA",resultObj[i].md5,resultObj[i].start_lng,resultObj[i].start_lat,resultObj[i].md5,resultObj[i].title);
i++;
}
}
});
}
//Get the DE Camera JSON Feed
function getDE() {
getCamFeed(DEURL,"json", function(result) {
var resultObj = JSON.parse(result).videoCameras;
var i=0;
while (i<resultObj.length) {
drawCameras("DE",resultObj[i].id,resultObj[i].lon,resultObj[i].lat,resultObj[i].urls.m3u8s,resultObj[i].title + " (" + resultObj[i].county + ")",550,300);
i++;
}
});
}
//Get the VA Camera JSON Feed
function getVA() {
getCamFeed(VAURL,"json", function(result) {
var resultObj = JSON.parse(result).features;
var i=0;
while (i<resultObj.length) {
drawCameras("VA",resultObj[i].properties.id,resultObj[i].geometry.coordinates[0],resultObj[i].geometry.coordinates[1],resultObj[i].properties.https_url,resultObj[i].properties.description,550,300);
i++;
}
});
}
//Get the NY Camera JSON Feed
function getNY() {
getCamFeed(NYURL,"json", function(result) {
var resultObj = JSON.parse(result);
var i=0;
while (i<resultObj.length) {
if (resultObj[i].VideoUrl != null) {
drawCameras("NY",resultObj[i].ID,resultObj[i].Longitude,resultObj[i].Latitude,resultObj[i].VideoUrl,resultObj[i].Name,550,300);
}
i++;
}
});
}
//Get the NJ Camera JSON Feed
function getNJ() {
getCamFeed(NJURL,"json", function(result) {
var resultObj = JSON.parse(result).Data.CameraData;
var i=0;
while (i<resultObj.length) {
drawCameras("NJ",resultObj[i].id,resultObj[i].longitude,resultObj[i].latitude,resultObj[i].CameraMainDetail[0].URL,resultObj[i].name,480,360);
i++;
}
});
}
function getMD() {
getCamFeed(MDURL,"json", function(result){
var resultObj = JSON.parse(result).data;
var i=0;
while (i<resultObj.length){
drawCameras("MD",resultObj[i].id,resultObj[i].lon,resultObj[i].lat,'https://' + resultObj[i].cctvIp + '/rtplive/' + resultObj[i].id + '/playlist.m3u8',resultObj[i].description,480,360);
i++;
}
});
}
//Generate the Camera markers
function drawCameras(state,id,x,y,url,title,width,height) {
var size = new OpenLayers.Size(20,20);
var offset = new OpenLayers.Pixel(-(size.w/2), -size.h);
var icon = new OpenLayers.Icon(camIcon,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.title = title;
newMarker.url = url;
newMarker.id = id;
newMarker.width = width;
newMarker.height = height;
newMarker.state = state;
newMarker.events.register('click',newMarker,popupCam);
switch(state) {
case "VA":
VALayer.addMarker(newMarker);
break;
case "MD":
MDLayer.addMarker(newMarker);
break;
case "PA":
PALayer.addMarker(newMarker);
break;
case "DE":
DELayer.addMarker(newMarker);
break;
case "NY":
NYLayer.addMarker(newMarker);
break;
case "NJ":
NJLayer.addMarker(newMarker);
break;
}
}
//Generate the Camera Popup
function popupCam(evt) {
//Check to see if WME Toolbox is running, and if it is, go no further (hopefully temporary!)
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++;
}
$("#gmPopupContainer").remove ();
$("#gmPopupContainer").hide ();
var popupHTMLPA = (['<div id="gmPopupContainer">' +
'<center><h3>' + this.title + '</h3><br>' +
'<iframe class="video" id="fp_embed_player" src="https://www.511pa.com/flowplayeri.aspx?' + this.url + '"&autoplay=1 style="background: #FFFFFF;margin: 5px 20px;" frameborder=0 width=320 height=240 scrolling=no allowfullscreen=allowfullscreen></iframe>' +
'<br><form><button id="gmCloseDlgBtn" type="button">Close</button>' +
'</form></div>'
]);
var popupHTMLDefault = (['<div id="gmPopupContainer">' +
'<center><h3>' + this.title + '</h3>' +
'<div id="videoDiv">' +
'<video id="hlsVideo" width=' + this.width + ' height=' + this.height + ' controls autoplay></video>' +
'</div>' +
'<form><button id="gmCloseDlgBtn" type="button">Close</button></form>' +
'</div>'
]);
switch(this.state) {
case "PA":
$("body").append(popupHTMLPA);
break;
default:
var currentCamURL = this.url;
$("body").append(popupHTMLDefault);
setTimeout(function () {
video = document.getElementById('hlsVideo');
var videoSrc = currentCamURL;
if (Hls.isSupported()) {
hls = new Hls();
hls.loadSource(videoSrc);
hls.attachMedia(video);
hls.on(Hls.Events.MANIFEST_PARSED, function() {
video.play();
});
}
},1000);
}
//Add listener for popup's "Close" button
$("#gmCloseDlgBtn").click ( function () {
if (hls) {
hls.destroy();
}
$("#gmPopupContainer").remove ();
$("#gmPopupContainer").hide ();
});
fetch(this.url)
.then(response => {
if (!response.ok) {
//Good feed
$('#videoDiv').empty();
document.getElementById('videoDiv').innerHTML = "<br>Sorry, this feed is currently offline.";
} else {
//Bad Feed
}
});
}
function initializeSettings()
{
loadSettings();
setChecked('chkPACamEnabled', settings.PACamEnabled);
setChecked('chkDECamEnabled', settings.DECamEnabled);
setChecked('chkNYCamEnabled', settings.NYCamEnabled);
setChecked('chkNJCamEnabled', settings.NJCamEnabled);
setChecked('chkMDCamEnabled', settings.MDCamEnabled);
setChecked('chkVACamEnabled', settings.VACamEnabled);
//Add Handler for Checkbox Setting Changes
$('.wmedotSettingsCheckbox').change(function() {
var settingName = $(this)[0].id.substr(3);
settings[settingName] = this.checked;
saveSettings();
if(this.checked) {
switch(settingName.substring(0,2)) {
case "VA":
buildDOTCamLayers("VA"); getVA();
break;
case "PA":
buildDOTCamLayers("PA"); getPA();
break;
case "DE":
buildDOTCamLayers("DE"); getDE();
break;
case "MD":
buildDOTCamLayers("MD"); getMD();
break;
case "NY":
buildDOTCamLayers("NY"); getNY();
break;
case "NJ":
buildDOTCamLayers("NJ"); getNJ();
}
}
else
{
switch(settingName.substring(0,2)) {
case "VA":
VALayer.destroy();
break;
case "MD":
MDLayer.destroy();
break;
case "PA":
PALayer.destroy();
break;
case "DE":
DELayer.destroy();
break;
case "NY":
NYLayer.destroy();
break;
case "NJ":
NJLayer.destroy();
}
}
});
if (document.getElementById('chkPACamEnabled').checked) { buildDOTCamLayers("PA"); getPA(); }
if (document.getElementById('chkDECamEnabled').checked) { buildDOTCamLayers("DE"); getDE(); }
if (document.getElementById('chkNYCamEnabled').checked) { buildDOTCamLayers("NY"); getNY(); }
if (document.getElementById('chkNJCamEnabled').checked) { buildDOTCamLayers("NJ"); getNJ(); }
if (document.getElementById('chkMDCamEnabled').checked) { buildDOTCamLayers("MD"); getMD(); }
if (document.getElementById('chkVACamEnabled').checked) { buildDOTCamLayers("VA"); getVA(); }
}
//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 = {
PACamEnabled: settings.PACamEnabled,
DECamEnabled: settings.DECamEnabled,
NYCamEnabled: settings.NYCamEnabled,
NJCamEnabled: settings.NJCamEnabled,
MDCamEnabled: settings.MDCamEnabled,
VACamEnabled: settings.VACamEnabled,
};
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"
});
}
//--- CSS styles make it work...
GM_addStyle (" \
#gmPopupContainer { \
position: fixed; \
top: 10%; \
left: 20%; \
padding: 1em; \
background: lightgray; \
border: 3px double black; \
border-radius: 1ex; \
z-index: 777; \
display: flex; \
} \
#gmPopupContainer button{ \
cursor: pointer; \
margin: 1em 1em 0; \
border: 1px outset buttonface; \
} \
");
bootstrap();
})();