// ==UserScript==
// @name UROverview Plus (URO+)
// @namespace http://greasemonkey.chizzum.com
// @description Adds filtering and pop-up infobox for UR, MP and camera markers
// @include https://www.waze.com/*/editor/*
// @include https://www.waze.com/editor/*
// @include https://editor-beta.waze.com/*
// @include https://beta.waze.com/*
// @exclude https://www.waze.com/user/*editor/*
// @exclude https://www.waze.com/*/user/*editor/*
// @grant none
// @version 3.90
// ==/UserScript==
/*
TO-DO ITEMS
=======================================================================================================================
Bug fixes - MUST BE CLEARED BEFORE RELEASE
=======================================================================================================================
=======================================================================================================================
Things to be checked
=======================================================================================================================
=======================================================================================================================
Proposed functionality
=======================================================================================================================
***HIGH PRIORITY***
Convert camera XHR code to async operation
***OTHER STUFF***
"Hide All" button for feed - automatically clicks the trashcan on all visible feed entries
Allow custom alert box to be used as a confirm() replacement as well
Improve reliability of yellow/green comment marker choice
Implement some sort of UR "hotspot" marking to highlight areas of the map with significant clustering of URs
Flush settings to localStorage whenever a change is made, or at least before opening a new tab via a popup
User-defined setting presets
Extend unstacking to cameras
Place filtering
- by last user to edit
More localisation
First-run information
- show quickstart guide to URO features if no existing settings are present (i.e. new installation)
=======================================================================================================================
New functionality in progress
=======================================================================================================================
Addition of segment and place watchlist functionality
*/
/* JSHint Directives */
/* globals $: */
/* globals W: true */
/* globals I18n: */
/* globals OL: true */
/* globals OpenLayers: true */
/* globals Waze: true */
/* globals require: */
/* jshint bitwise: false */
/* jshint eqnull: true */
var uroVersion = "3.90";
var uroReleaseDate = "20160912";
// list of changes affecting all users
var uroChanges =
[
'Adding a comment to a UR no longer throws an error',
'Places locked by staff can now be filtered',
'Fix for latest fiddling Waze have done to the layer definitions',
'MP filtering by closing/not-closing editor, which I managed to break back in 3.54 without anyone noticing, now works again...'
];
// list of changes affecting only WME Beta users (at least until the next production release including these parts of the beta code...)
var uroBetaChanges =
[
];
// true enables debug output during script startup
var uroShowDebugOutput = true;
// true keeps debug output enabled after script startup
var uroPersistentDebugOutput = false;
/*
var uroRecentDebug = [];
*/
var uroCtrlsHidden = false;
var uroCurrentTab = 1;
var uroFID = -1;
var uroShownFID = -1;
var uroShownPopupType = null;
var uroInhibitSave = true;
var uroPopupTimer = -2;
var uroPopupDwellTimer = -1;
var uroPopupShown = false;
var uroPopupSuppressed = false;
var uroSetupListeners = true;
var uroRootContainer = null;
var uroPlacesRoot = null;
var uroMaskLayer = null;
var uroConfirmIntercepted = false;
var uroCustomMarkerList = [];
var uroPendingURSessionIDs = [];
var uroRequestedURSessionIDs = [];
var uroPlacesGroupsCollapsed = [];
var uroMouseInPopup = false;
var uroURControlsIdx = null;
var uroProblemControlsIdx = null;
var uroTurnsLayerIdx = null;
var uroNullCamLayer = false;
var uroNullOpenLayers = false;
var uroNullURLayer = false;
var uroNullProblemLayer = false;
var uroNullMapViewport = false;
var uroURDialogIsOpen = false;
var uroHoveredURID = null;
var uroSelectedURID = null;
var uroURReclickAttempts = 0;
var uroPendingCommentDataRefresh = false;
var uroWaitingCommentDataRefresh = false;
var uroExpectedCommentCount = null;
var uroCachedLastCommentID = null;
var uroPlaceSelected = false;
var uroAutoCentreDisabledOn = [];
var uroMouseIsDown = false;
var uroBackfilling = false;
var uroHidePopupOnPanelOpen = false;
var uroUserID = -1;
var uroURIDInURL = null;
var uroDOMHasTurnProblems = false;
var uroBetaEditor = false;
var uroWazeBitsPresent = 0;
var uroMTEMode = false;
var uroFinalisingListenerSetup = false;
var uroInitialised = false;
var uroOWL = null;
var uroDiv = null;
var uroAlerts = null;
var uroControls = null;
var uroCtrlURs = null;
var uroCtrlMPs = null;
var uroCtrlPlaces = null;
var uroCtrlCameras = null;
var uroCtrlMisc = null;
var uroCtrlHides = null;
var uroAMList = [];
var uroCWLGroups = [];
var uroCamWatchObjects = [];
var uroSegWatchObjects = [];
var uroPlaceWatchObjects = [];
var uroFriendlyAreaNames = [];
var uroAreaNameHoverTime = -1;
var uroAreaNameHoverObj = null;
var uroAreaNameOverlayShown = false;
var uroANEditHovered = false;
var uroANEditBox = null;
var uroPrevMouseX = -1;
var uroPrevMouseY = -1;
var dteControlsIdx = -1;
var dteOldestFullDrive = new Date(0);
var dteEpoch = new Date(0);
var dteTopID = '';
var dteClearHighlightsOnPanelClose = false;
var dteArmClearHighlightsOnPanelClose = false;
var dteOffset = 0;
var uroUserTabId = '';
var uroShowFeedFilter = false;
var uroTBRObj = null;
var uroBackfillQueue = [];
var uroUnstackedMasterID = null;
var uroStackList = [];
var uroStackType = null;
var uroAlertBoxStack = [];
var uroAlertBoxTickAction = null;
var uroAlertBoxCrossAction = null;
var uroAlertBoxInUse = false;
var uroNativeMarkerImage = 'problems-sc703e224cf.png';
var uroCustomURTags = ['[ROADWORKS]','[CONSTRUCTION]','[CLOSURE]','[EVENT]','[NOTE]','[WSLM]','[BOG]','[DIFFICULT]'];
var uroAltMarkers =
[
// 0 = closure
"url('')",
// 1 = cone
"url('')",
// 2 = custom text
"url('')",
// 3 = editor note
"url('')",
// 4 = event
"url('')",
// 5 = wslm & native SL URs
"url('')",
// 6 = Elgin
"url('')",
// 7 = TrafficCast
"url('')",
// 8 = TrafficMaster
"url('')",
// 9 = CalTrans
"url('')",
// 10 = TfL
"url('')",
// 11 = boots on ground
"url('')",
// 12 = difficult turn
"url('')",
];
var uroMarkers =
[
// 0 = comment count circle
[""],
// 1 = green comment marker
[""],
// 2 = yellow (own) comment marker
[""]
];
/*
function uroAddDebug(debugtext)
{
var ts = Math.round(performance.now());
if(uroRecentDebug.length == 100)
{
uroRecentDebug.shift();
}
uroRecentDebug.push(ts+': '+debugtext);
console.debug('URO+DBG '+ts+':'+debugtext);
}
function uroDumpDebug()
{
if(uroRecentDebug.length > 0)
{
document.getElementById('WazeMap').innerHTML = '<textarea id="uroDbgOutput" style="width:100%;height:100%">';
var dbgOutput = '';
for(var i=0; i<uroRecentDebug.length; i++)
{
dbgOutput += uroRecentDebug[i]+'\n';
}
document.getElementById('uroDbgOutput').textContent = dbgOutput;
}
}
*/
function uroAddLog(logtext)
{
if(uroShowDebugOutput) console.log('URO+: '+logtext);
}
function uroGetCBChecked(cbID)
{
try
{
return(document.getElementById(cbID).checked);
}
catch(err)
{
return null;
}
}
function uroSetCBChecked(cbID, state)
{
try
{
document.getElementById(cbID).checked = state;
}
catch(err)
{
}
}
function uroGetElmValue(elmID)
{
try
{
return(document.getElementById(elmID).value);
}
catch(err)
{
return null;
}
}
function uroSetStyleDisplay(elm,style)
{
try
{
document.getElementById(elm).style.display = style;
}
catch(err)
{
}
}
function uroSetOnClick(elm,fn)
{
try
{
document.getElementById(elm).onclick = fn;
}
catch(err)
{
}
}
function uroAddEventListener(elm,eventType,eventFn,eventBool)
{
try
{
document.getElementById(elm).addEventListener(eventType, eventFn, eventBool);
}
catch(err)
{
}
}
function uroGetVisibilityBitmask()
{
if(W.map.mapState.getLayerVisibilityBitmask === undefined)
{
return W.map.mapState._getLayerVisibilityBitmask();
}
else return W.map.mapState.getLayerVisibilityBitmask();
}
function uroAlertBoxObj(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction)
{
this.headericon = headericon;
this.title = title;
this.content = content;
this.hasCross = hasCross;
this.tickText = tickText;
this.crossText = crossText;
this.tickAction = tickAction;
this.crossAction = crossAction;
}
function uroCloseAlertBox()
{
document.getElementById('uroAlerts').childNodes[0].innerHTML = '';
document.getElementById('uroAlerts').childNodes[1].innerHTML = '';
document.getElementById('uroAlertTickBtnCaption').innerHTML = '';
document.getElementById('uroAlertCrossBtnCaption').innerHTML = '';
uroAlertBoxTickAction = null;
uroAlertBoxCrossAction = null;
document.getElementById('uroAlerts').style.visibility = "hidden";
document.getElementById('uroAlertCrossBtn').style.visibility = "hidden";
uroAlertBoxInUse = false;
if(uroAlertBoxStack.length > 0)
{
uroBuildAlertBoxFromStack();
}
}
function uroCloseAlertBoxWithTick()
{
if(typeof uroAlertBoxTickAction === 'function')
{
uroAlertBoxTickAction();
}
uroCloseAlertBox();
}
function uroCloseAlertBoxWithCross()
{
if(typeof uroAlertBoxCrossAction === 'function')
{
uroAlertBoxCrossAction();
}
uroCloseAlertBox();
}
function uroShowAlertBox(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction)
{
uroAlertBoxStack.push(new uroAlertBoxObj(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction));
if(uroAlertBoxInUse === false)
{
uroBuildAlertBoxFromStack();
}
}
function uroBuildAlertBoxFromStack()
{
uroAlertBoxInUse = true;
uroAlertBoxTickAction = null;
uroAlertBoxCrossAction = null;
var titleContent = '<span style="font-size:14px;padding:2px;">';
titleContent += '<i class="fa '+uroAlertBoxStack[0].headericon+'"> </i> ';
titleContent += uroAlertBoxStack[0].title;
titleContent += '</span>';
document.getElementById('uroAlerts').childNodes[0].innerHTML = titleContent;
document.getElementById('uroAlerts').childNodes[1].innerHTML = uroAlertBoxStack[0].content;
document.getElementById('uroAlertTickBtnCaption').innerHTML = uroAlertBoxStack[0].tickText;
if(uroAlertBoxStack[0].hasCross)
{
document.getElementById('uroAlertCrossBtnCaption').innerHTML = uroAlertBoxStack[0].crossText;
document.getElementById('uroAlertCrossBtn').style.visibility = "visible";
if(typeof uroAlertBoxStack[0].crossAction === "function")
{
uroAlertBoxCrossAction = uroAlertBoxStack[0].crossAction;
}
}
else
{
document.getElementById('uroAlertCrossBtn').style.visibility = "hidden";
}
if(typeof uroAlertBoxStack[0].tickAction === "function")
{
uroAlertBoxTickAction = uroAlertBoxStack[0].tickAction;
}
document.getElementById('uroAlerts').style.visibility = "";
uroAlertBoxStack.shift();
}
function uroFirstTimerWelcomePack()
{
uroAddLog('welcome new users to Club URO...');
// for now, just show the update notes...
uroShowUpdateNotes();
}
function uroShowUpdateNotes()
{
uroAddLog('let existing users know what\'s new in this release');
var releaseNotes = '';
releaseNotes += '<p>Thanks for upgrading to URO+ '+uroVersion+' ('+uroReleaseDate+'). What\'s changed?</p>';
var loop;
if(uroChanges.length > 0)
{
releaseNotes += '<ul>';
for(loop=0; loop < uroChanges.length; loop++)
{
releaseNotes += '<li>'+uroChanges[loop];
}
releaseNotes += '</ul>';
}
if((uroBetaEditor) && (uroBetaChanges.length > 0))
{
releaseNotes += '<p>For WME Beta:<p>';
releaseNotes += '<ul>';
for(loop=0; loop < uroBetaChanges.length; loop++)
{
releaseNotes += '<li>'+uroBetaChanges[loop];
}
releaseNotes += '</ul>';
}
uroShowAlertBox('fa-info-circle', 'URO+ Release Notes', releaseNotes, false, "OK", "", null, null);
}
function uroAdvertiseCustomIcons()
{
uroAddLog('advertise the benefits of custom UR icons...');
var confirmMsg = '';
confirmMsg += '<p>Hi there. One of the features of URO+ that a lot of users find useful is the ability to use a custom marker for URs and MPs which have been tagged with a specific keyword in their description text.</p>';
confirmMsg += '<p>Markers are defined for <b>[ROADWORKS]</b>, <b>[CONSTRUCTION]</b>, <b>[CLOSURE]</b>, <b>[EVENT]</b>, <b>[NOTE]</b>, <b>[WSLM]</b>, <b>[BOG]</b> and <b>[DIFFICULT]</b> tags in URs, and <b>[TfL Open Data]</b>, <b>[Elgin]</b>, <b>[TM]</b>, <b>[TrafficCast]</b> and <b>[Caltrans]</b> in MPs.</p>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[1]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[0]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[4]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[3]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[5]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[11]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[12]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[10]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[6]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[8]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[7]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<div style="background-image:'+uroAltMarkers[9]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
confirmMsg += '<p style="clear:left;">Would you like me to automatically enable these custom markers?</p>';
confirmMsg += '<p>If you change your mind later on, they can be enabled/disabled via the Misc tab within the URO+ settings</p>';
uroShowAlertBox('fa-info-circle', 'URO+ Message to Users', confirmMsg, true, 'Yes please', 'No thanks', uroSetMarkerCBs, null);
}
function uroSetMarkerCBs()
{
uroSetCBChecked('_cbCustomRoadworksMarkers', true);
uroSetCBChecked('_cbCustomConstructionMarkers', true);
uroSetCBChecked('_cbCustomClosuresMarkers', true);
uroSetCBChecked('_cbCustomEventsMarkers', true);
uroSetCBChecked('_cbCustomNotesMarkers', true);
uroSetCBChecked('_cbCustomBOGMarkers', true);
uroSetCBChecked('_cbCustomDifficultMarkers', true);
uroSetCBChecked('_cbCustomWSLMMarkers', true);
uroSetCBChecked('_cbCustomNativeSLMarkers', true);
uroSetCBChecked('_cbCustomElginMarkers', true);
uroSetCBChecked('_cbCustomTrafficMasterMarkers', true);
uroSetCBChecked('_cbCustomTrafficCastMarkers', true);
uroSetCBChecked('_cbCustomCaltransMarkers', true);
uroSetCBChecked('_cbCustomTFLMarkers', true);
}
function uroGatherSettings(container)
{
var options = '';
var urOptions = document.getElementById(container).getElementsByTagName('input');
for (var optIdx=0;optIdx<urOptions.length;optIdx++)
{
var id = urOptions[optIdx].id;
if((id.indexOf('_cb') === 0)||(id.indexOf('_text') === 0)||(id.indexOf('_input') === 0))
{
options += ':' + id;
if(urOptions[optIdx].type == 'checkbox') options += ',' + urOptions[optIdx].checked.toString();
else if((urOptions[optIdx].type == 'text')||(urOptions[optIdx].type == 'number')) options += ',' + urOptions[optIdx].value.toString();
}
}
return options;
}
function uroGatherCamWatchList()
{
var liststr = '';
for(var loop=0;loop<uroCamWatchObjects.length;loop++)
{
var camObj = uroCamWatchObjects[loop];
if((camObj.fid != null) && (camObj.persistent === true))
{
if(loop > 0) liststr += ':';
liststr += camObj.fid+',';
liststr += camObj.watch.lon+',';
liststr += camObj.watch.lat+',';
liststr += camObj.watch.type+',';
liststr += camObj.watch.azymuth+',';
liststr += camObj.watch.speed+',';
liststr += camObj.watch.validated+',';
liststr += camObj.groupID+',';
liststr += camObj.server;
}
}
return liststr;
}
function uroGatherSegWatchList()
{
var liststr = '';
for(var loop=0;loop<uroSegWatchObjects.length;loop++)
{
var segObj = uroSegWatchObjects[loop];
if((segObj.fid != null) && (segObj.persistent === true))
{
if(loop > 0) liststr += ':';
liststr += segObj.fid+',';
liststr += segObj.watch.left+',';
liststr += segObj.watch.right+',';
liststr += segObj.watch.bottom+',';
liststr += segObj.watch.top+',';
liststr += segObj.watch.fromNode+',';
liststr += segObj.watch.toNode+',';
liststr += segObj.watch.fwdDir+',';
liststr += segObj.watch.revDir+',';
liststr += segObj.watch.length+',';
liststr += segObj.watch.level+',';
liststr += segObj.watch.rank+',';
liststr += segObj.watch.roadType+',';
liststr += segObj.watch.updatedOn+',';
liststr += segObj.groupID+',';
liststr += segObj.server;
}
}
return liststr;
}
function uroGatherPlaceWatchList()
{
var liststr = '';
for(var loop=0;loop<uroPlaceWatchObjects.length;loop++)
{
var placeObj = uroPlaceWatchObjects[loop];
if((placeObj.fid != null) && (placeObj.persistent === true))
{
if(loop > 0) liststr += ':';
liststr += placeObj.fid+',';
liststr += placeObj.watch.left+',';
liststr += placeObj.watch.right+',';
liststr += placeObj.watch.bottom+',';
liststr += placeObj.watch.top+',';
liststr += placeObj.watch.name+',';
liststr += placeObj.watch.imageCount+',';
liststr += placeObj.watch.residential+',';
liststr += placeObj.watch.updatedOn+',';
liststr += placeObj.groupID+',';
liststr += placeObj.server;
}
}
return liststr;
}
function uroGatherCWLGroups()
{
var liststr = '';
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
var groupObj = uroCWLGroups[loop];
if(groupObj.groupID != -1)
{
if(loop > 0) liststr += ':';
liststr += groupObj.groupID+',';
liststr += groupObj.groupName+',';
liststr += groupObj.groupCollapsed;
}
}
return liststr;
}
function uroGatherPlacesGroups()
{
var liststr = '';
for(var loop=0;loop<uroPlacesGroupsCollapsed.length;loop++)
{
if(loop > 0) liststr += ':';
liststr += uroPlacesGroupsCollapsed[loop];
}
return liststr;
}
function uroGatherFriendlyAreaNames()
{
var liststr = '';
for(var loop=0;loop<uroFriendlyAreaNames.length;loop++)
{
var fnObj = uroFriendlyAreaNames[loop];
if(loop > 0) liststr += ':';
liststr += fnObj.fName+',';
liststr += fnObj.area+',';
liststr += fnObj.server;
}
return liststr;
}
function uroSaveSettings()
{
if(uroInhibitSave)
{
uroAddLog('save inhibited');
return;
}
if (localStorage)
{
localStorage.UROverviewUROptions = uroGatherSettings('uroCtrlURs');
localStorage.UROverviewMPOptions = uroGatherSettings('uroCtrlMPs');
localStorage.UROverviewCameraOptions = uroGatherSettings('uroCtrlCameras');
localStorage.UROverviewMiscOptions = uroGatherSettings('uroCtrlMisc');
localStorage.UROverviewPlacesOptions = uroGatherSettings('uroCtrlPlaces');
localStorage.UROverviewCamWatchList = uroGatherCamWatchList();
localStorage.UROverviewSegWatchList = uroGatherSegWatchList();
localStorage.UROverviewPlaceWatchList = uroGatherPlaceWatchList();
localStorage.UROverviewCWLGroups = uroGatherCWLGroups();
localStorage.UROverviewFriendlyAreaNames = uroGatherFriendlyAreaNames();
localStorage.UROverviewPlacesGroups = uroGatherPlacesGroups();
localStorage.UROverviewMasterEnable = uroGetCBChecked('_cbMasterEnable');
localStorage.UROverviewCurrentVersion = uroVersion;
uroAddLog('save complete');
}
else
{
uroAddLog('no localStorage, save blocked');
}
}
function uroApplySettings(settings)
{
var options = settings.split(':');
for(var optIdx=0;optIdx<options.length;optIdx++)
{
var fields = options[optIdx].split(',');
if(fields[0].indexOf('_cb') === 0)
{
if(document.getElementById(fields[0]) !== null)
{
uroSetCBChecked(fields[0], (fields[1] == 'true'));
}
}
else if((fields[0].indexOf('_input') === 0)||(fields[0].indexOf('_text') === 0))
{
if(document.getElementById(fields[0]) !== null) document.getElementById(fields[0]).value = fields[1];
}
}
}
function uroApplyCamWatchList()
{
var objects = localStorage.UROverviewCamWatchList.split(':');
uroCamWatchObjects = [];
if(objects.length > 0)
{
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
if(fields.length >= 7)
{
// following two bits of code add in blank fields if the user has updated their copy of URO+ from an
// older version which didn't include support for either of these field types
// add default groupID field
if(fields.length == 7)
{
fields.push(0);
}
// set default groupID value to 0 (no group)
if(fields[7] == -1)
{
fields[7] = 0;
}
// add default server field
if(fields.length == 8)
{
fields.push('??');
}
// set default server value to unknown
if(fields[8] === 0)
{
fields[8] = '??';
}
uroCamWatchObjects.push(new uroCamWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7],fields[8]));
}
}
}
}
/*
function uroApplySegWatchList()
{
var objects = localStorage.UROverviewSegWatchList.split(':');
uroSegWatchObjects = [];
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
uroSegWatchObjects.push(new uroSegWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7],fields[8],fields[9],fields[10],fields[11],fields[12],fields[13],fields[14],fields[15]));
}
}
function uroApplyPlaceWatchList()
{
var objects = localStorage.UROverviewPlaceWatchList.split(':');
uroPlaceWatchObjects = [];
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
uroPlaceWatchObjects.push(new uroPlaceWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7],fields[8],fields[9],fields[10]));
}
}
*/
function uroApplyCWLGroups()
{
var objects = localStorage.UROverviewCWLGroups.split(':');
uroCWLGroups = [];
if(objects.length === 0)
{
uroCWLGroups.push(new uroOWLGroupObj(0,'No group',false));
}
else
{
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
if(fields.length < 2)
{
fields.push(false);
}
uroCWLGroups.push(new uroOWLGroupObj(fields[0],fields[1],(fields[2] == 'true')));
}
}
}
/*
function uroApplyPlacesGroups()
{
var t = localStorage.UROverviewPlacesGroups.split(':');
for(var i=0;i<t.length;i++)
{
uroPlacesGroupsCollapsed[i] = (t[i] == "true");
}
}
*/
function uroApplyFriendlyAreaNames()
{
var objects = localStorage.UROverviewFriendlyAreaNames.split(':');
uroFriendlyAreaNames = [];
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
uroFriendlyAreaNames.push(new uroAFNObj(fields[0],parseFloat(fields[1]),fields[2]));
}
uroReplaceAreaNames(true);
}
function uroLoadSettings()
{
var isNewInstall = true;
var isUpgradeInstall = true;
var notifyAboutCustomIcons = true;
uroAddLog('loadSettings()');
if (localStorage.UROverviewUROptions != null)
{
uroAddLog('recover UR tab settings');
uroApplySettings(localStorage.UROverviewUROptions);
isNewInstall = false;
}
if (localStorage.UROverviewCameraOptions != null)
{
uroAddLog('recover camera tab settings');
uroApplySettings(localStorage.UROverviewCameraOptions);
isNewInstall = false;
}
if (localStorage.UROverviewMPOptions != null)
{
uroAddLog('recover MP tab settings');
uroApplySettings(localStorage.UROverviewMPOptions);
isNewInstall = false;
}
if (localStorage.UROverviewPlacesOptions != null)
{
uroAddLog('recover Places tab settings');
uroApplySettings(localStorage.UROverviewPlacesOptions);
isNewInstall = false;
}
if (localStorage.UROverviewMiscOptions != null)
{
uroAddLog('recover misc tab settings');
uroApplySettings(localStorage.UROverviewMiscOptions);
isNewInstall = false;
if(localStorage.UROverviewCurrentVersion != null)
{
notifyAboutCustomIcons = false;
}
else
{
if(uroGetCBChecked('_cbCustomRoadworksMarkers') === true) notifyAboutCustomIcons = false;
if(uroGetCBChecked('_cbCustomConstructionMarkers')=== true) notifyAboutCustomIcons = false;
if(uroGetCBChecked('_cbCustomClosuresMarkers') === true) notifyAboutCustomIcons = false;
if(uroGetCBChecked('_cbCustomEventsMarkers') === true) notifyAboutCustomIcons = false;
if(uroGetCBChecked('_cbCustomNotesMarkers') === true) notifyAboutCustomIcons = false;
if(uroGetCBChecked('_cbCustomWSLMMarkers') === true) notifyAboutCustomIcons = false;
if(uroGetCBChecked('_cbCustomBOGMarkers') === true) notifyAboutCustomIcons = false;
if(uroGetCBChecked('_cbCustomDifficultMarkers') === true) notifyAboutCustomIcons = false;
if(uroGetCBChecked('_cbCustomNativeSLMarkers') === true) notifyAboutCustomIcons = false;
}
}
if(localStorage.UROverviewCWLGroups != null)
{
uroAddLog('recover CWL groups');
uroApplyCWLGroups();
isNewInstall = false;
}
else
{
uroAddLog('set default CWL group');
uroCWLGroups.push(new uroOWLGroupObj(0,'No group',false));
}
if(localStorage.UROverviewCamWatchList != null)
{
uroAddLog('recover camera watchlist');
uroApplyCamWatchList();
uroGetCurrentCamWatchListObjects();
isNewInstall = false;
}
/*
if(localStorage.UROverviewSegWatchList != null)
{
uroAddLog('recover segment watchlist');
uroApplySegWatchList();
uroGetCurrentSegWatchListObjects();
isNewInstall = false;
}
if(localStorage.UROverviewPlaceWatchList != null)
{
uroAddLog('recover places watchlist');
uroApplyPlaceWatchList();
//uroGetCurrentPlaceWatchListObjects();
isNewInstall = false;
}
if(localStorage.UROverviewPlacesGroups != null)
{
uroAddLog('recover places groups');
uroApplyPlacesGroups();
isNewInstall = false;
}
*/
if(localStorage.UROverviewCurrentVersion != null)
{
uroAddLog('comparing install versions');
if(localStorage.UROverviewCurrentVersion == uroVersion)
{
isUpgradeInstall = false;
}
}
if(localStorage.UROverviewFriendlyAreaNames != null)
{
uroAddLog('recover friendly area names');
uroApplyFriendlyAreaNames();
isNewInstall = false;
}
if(localStorage.UROverviewMasterEnable != null)
{
uroAddLog('recover master enable state');
document.getElementById('_cbMasterEnable').checked = (localStorage.UROverviewMasterEnable == "true");
}
if(isNewInstall)
{
uroFirstTimerWelcomePack();
}
else if(isUpgradeInstall)
{
uroShowUpdateNotes();
}
if(notifyAboutCustomIcons)
{
uroAdvertiseCustomIcons();
}
uroInhibitSave = false;
}
function uroDefaultSettings()
{
uroShowAlertBox("fa-warning", "URO+ Warning", "Resetting URO+ settings <b>cannot</b> be undone.<br>Are you <i>sure</i> you want to do this?", true, "Reset settings", "Keep settings", uroDefaultSettingsAction, null);
}
function uroDefaultSettingsAction()
{
var defaultSettings = '';
defaultSettings += '[UROverviewMPOptions][len=628]:_cbMPFilterMissingJunction,false:_cbMPFilterMissingRoad,false:_cbMPFilterCrossroadsJunctionMissing,false:_cbMPFilterDrivingDirectionMismatch,false:_cbMPFilterRoadTypeMismatch,false:_cbMPFilterRestrictedTurn,false:_cbMPFilterRoadClosureProblem,false:_cbMPFilterUnknownProblem,false:_cbMPFilterTurnProblem,false:_cbMPFilterReopenedProblem,false:_cbInvertMPFilter,false:_cbMPFilterOutsideArea,false:_cbMPFilterClosed,false:_cbMPFilterSolved,false:_cbMPFilterUnidentified,false:_cbMPClosedUserIDFilter,false:_cbMPNotClosedUserIDFilter,false:_cbMPFilterLowSeverity,false:_cbMPFilterMediumSeverity,false:_cbMPFilterHighSeverity,false[END]';
defaultSettings += '[UROverviewPlaceWatchList][len=0][END]';
defaultSettings += '[UROverviewSegWatchList][len=0][END]';
defaultSettings += '[UROverviewPlacesGroups][len=59]false:false:false:false:false:false:false:false:false:false[END]';
defaultSettings += '[UROverviewMasterEnable][len=4]true[END]';
defaultSettings += '[UROverviewFriendlyAreaNames][len=0][END]';
defaultSettings += '[UROverviewMiscOptions][len=1222]:_cbNativeConvoMarkers,true:_cbNativeBetaConvoMarkers,true:_cbCommentCount,false:_cbURBackfill,false:_inputUnstackSensitivity,15:_inputUnstackZoomLevel,3:_cbCustomRoadworksMarkers,false:_cbCustomConstructionMarkers,false:_cbCustomClosuresMarkers,false:_cbCustomEventsMarkers,false:_cbCustomNotesMarkers,false:_cbCustomWSLMMarkers,false:_cbCustomKeywordMarkers,false:_textCustomKeyword,:_cbCustomElginMarkers,false:_cbCustomTrafficMasterMarkers,false:_cbCustomTrafficCastMarkers,false:_cbCustomCaltransMarkers,false:_inputPopupDwellTimeout,2:_inputPopupEntryTimeout,2:_inputMaxJitter,2:_cbInhibitURPopup,false:_cbInhibitMPPopup,false:_cbInhibitCamPopup,false:_cbInhibitSegPopup,false:_cbInhibitSegGenericPopup,false:_cbInhibitTurnsPopup,false:_cbInhibitLandmarkPopup,false:_cbInhibitPUPopup,false:_cbDateFmtDDMMYY,true:_cbDateFmtMMDDYY,false:_cbDateFmtYYMMDD,false:_cbTimeFmt24H,true:_cbTimeFmt12H,false:_cbWhiteBackground,false:_inputCustomBackgroundRed,255:_inputCustomBackgroundGreen,255:_inputCustomBackgroundBlue,255:_cbInhibitNURButton,false:_cbInhibitNMPButton,false:_cbInhibitNPURButton,false:_cbHideAMLayer,false:_cbDisablePlacesFiltering,false:_cbDisableTabStyling,false:_cbHideEditorInfo,false:_cbEnableDTE,false[END]';
defaultSettings += '[UROverviewUROptions][len=1685]:_cbURFilterOutsideArea,false:_cbNoFilterForURInURL,false:_cbFilterWazeAuto,false:_cbFilterIncorrectTurn,false:_cbFilterIncorrectAddress,false:_cbFilterIncorrectRoute,false:_cbFilterMissingRoundabout,false:_cbFilterGeneralError,false:_cbFilterTurnNotAllowed,false:_cbFilterIncorrectJunction,false:_cbFilterMissingBridgeOverpass,false:_cbFilterWrongDrivingDirection,false:_cbFilterMissingExit,false:_cbFilterMissingRoad,false:_cbFilterBlockedRoad,false:_cbFilterMissingLandmark,false:_cbFilterUndefined,false:_cbFilterRoadworks,false:_cbFilterConstruction,false:_cbFilterClosure,false:_cbFilterEvent,false:_cbFilterNote,false:_cbFilterWSLM,false:_cbInvertURFilter,false:_cbFilterOpenUR,false:_cbFilterClosedUR,false:_cbFilterSolved,false:_cbFilterUnidentified,false:_cbEnableMinAgeFilter,false:_inputFilterMinDays,:_cbEnableMaxAgeFilter,false:_inputFilterMaxDays,:_cbHideMyFollowed,false:_cbHideMyUnfollowed,false:_cbURDescriptionMustBePresent,false:_cbURDescriptionMustBeAbsent,false:_cbEnableKeywordMustBePresent,false:_textKeywordPresent,:_cbEnableKeywordMustBeAbsent,false:_textKeywordAbsent,:_cbCaseInsensitive,false:_cbHideMyComments,false:_cbHideAnyComments,false:_cbHideIfLastCommenter,false:_cbHideIfNotLastCommenter,false:_cbHideIfReporterLastCommenter,false:_cbHideIfReporterNotLastCommenter,false:_cbEnableMinCommentsFilter,false:_inputFilterMinComments,:_cbEnableMaxCommentsFilter,false:_inputFilterMaxComments,:_cbEnableCommentAgeFilter2,false:_inputFilterCommentDays2,:_cbEnableCommentAgeFilter,false:_inputFilterCommentDays,:_cbIgnoreOtherEditorComments,false:_cbURUserIDFilter,false:_cbURResolverIDFilter,false:_cbInvertURStateFilter,false:_cbNoFilterForTaggedURs,false[END]';
defaultSettings += '[UROverviewCameraOptions][len=878]:_cbShowWorldCams,true:_cbShowUSACams,true:_cbShowNonWorldCams,true:_cbShowOnlyCamsCreatedBy,false:_cbShowOnlyCamsEditedBy,false:_textCameraEditor,:_cbShowOnlyMyCams,false:_cbShowApprovedCams,true:_cbShowNonApprovedCams,true:_cbShowOlderCreatedNonApproved,false:_inputCameraMinCreatedDays,:_cbShowOlderUpdatedNonApproved,false:_inputCameraMinUpdatedDays,:_cbShowSpeedCams,true:_cbShowIfSpeedSet,true:_cbShowIfNoSpeedSet,true:_cbShowRedLightCams,true:_cbShowDummyCams,true:_cbHideCreatedByMe,false:_cbHideCreatedByRank0,false:_cbHideCreatedByRank1,false:_cbHideCreatedByRank2,false:_cbHideCreatedByRank3,false:_cbHideCreatedByRank4,false:_cbHideCreatedByRank5,false:_cbHideUpdatedByMe,false:_cbHideUpdatedByRank0,false:_cbHideUpdatedByRank1,false:_cbHideUpdatedByRank2,false:_cbHideUpdatedByRank3,false:_cbHideUpdatedByRank4,false:_cbHideUpdatedByRank5,false:_cbHideCWLCams,false[END]';
defaultSettings += '[UROverviewCamWatchList][len=0][END]';
defaultSettings += '[UROverviewPlacesOptions][len=5312]:_cbFilterUneditablePlaceUpdates,false:_cbFilterLockRankedPlaceUpdates,false:_cbFilterNewPlacePUR,false:_cbFilterUpdatedDetailsPUR,false:_cbFilterNewPhotoPUR,false:_cbFilterFlaggedPUR,false:_cbLeavePURGeos,false:_cbInvertPURFilters,false:_cbPURFilterLowSeverity,false:_cbPURFilterMediumSeverity,false:_cbPURFilterHighSeverity,false:_cbEnablePURMinAgeFilter,false:_inputPURFilterMinDays,:_cbEnablePURMaxAgeFilter,false:_inputPURFilterMaxDays,:_cbPlaceFilterEditedLessThan,false:_inputFilterPlaceEditMinDays,:_cbPlaceFilterEditedMoreThan,false:_inputFilterPlaceEditMaxDays,:_cbHidePlacesL0,false:_cbHidePlacesL1,false:_cbHidePlacesL2,false:_cbHidePlacesL3,false:_cbHidePlacesL4,false:_cbHidePlacesL5,false:_cbHidePlacesAdLocked,false:_cbHideAreaPlaces,false:_cbHidePointPlaces,false:_cbHidePhotoPlaces,false:_cbHideNoPhotoPlaces,false:_cbHideLinkedPlaces,false:_cbHideNoLinkedPlaces,false:_cbHideKeywordPlaces,false:_cbHideNoKeywordPlaces,false:_textKeywordPlace,:_cbPlacesFilter-CAR_SERVICES,false:_cbPlacesFilter-GAS_STATION,false:_cbPlacesFilter-PARKING_LOT,false:_cbPlacesFilter-GARAGE_AUTOMOTIVE_SHOP,false:_cbPlacesFilter-CAR_WASH,false:_cbPlacesFilter-CHARGING_STATION,false:_cbPlacesFilter-TRANSPORTATION,false:_cbPlacesFilter-AIRPORT,false:_cbPlacesFilter-BUS_STATION,false:_cbPlacesFilter-FERRY_PIER,false:_cbPlacesFilter-SEAPORT_MARINA_HARBOR,false:_cbPlacesFilter-SUBWAY_STATION,false:_cbPlacesFilter-TRAIN_STATION,false:_cbPlacesFilter-BRIDGE,false:_cbPlacesFilter-TUNNEL,false:_cbPlacesFilter-TAXI_STATION,false:_cbPlacesFilter-JUNCTION_INTERCHANGE,false:_cbPlacesFilter-PROFESSIONAL_AND_PUBLIC,false:_cbPlacesFilter-COLLEGE_UNIVERSITY,false:_cbPlacesFilter-SCHOOL,false:_cbPlacesFilter-CONVENTIONS_EVENT_CENTER,false:_cbPlacesFilter-GOVERNMENT,false:_cbPlacesFilter-LIBRARY,false:_cbPlacesFilter-CITY_HALL,false:_cbPlacesFilter-ORGANIZATION_OR_ASSOCIATION,false:_cbPlacesFilter-PRISON_CORRECTIONAL_FACILITY,false:_cbPlacesFilter-COURTHOUSE,false:_cbPlacesFilter-CEMETERY,false:_cbPlacesFilter-FIRE_DEPARTMENT,false:_cbPlacesFilter-POLICE_STATION,false:_cbPlacesFilter-MILITARY,false:_cbPlacesFilter-HOSPITAL_MEDICAL_CARE,false:_cbPlacesFilter-OFFICES,false:_cbPlacesFilter-POST_OFFICE,false:_cbPlacesFilter-RELIGIOUS_CENTER,false:_cbPlacesFilter-KINDERGARDEN,false:_cbPlacesFilter-FACTORY_INDUSTRIAL,false:_cbPlacesFilter-EMBASSY_CONSULATE,false:_cbPlacesFilter-INFORMATION_POINT,false:_cbPlacesFilter-SHOPPING_AND_SERVICES,false:_cbPlacesFilter-ARTS_AND_CRAFTS,false:_cbPlacesFilter-BANK_FINANCIAL,false:_cbPlacesFilter-SPORTING_GOODS,false:_cbPlacesFilter-BOOKSTORE,false:_cbPlacesFilter-PHOTOGRAPHY,false:_cbPlacesFilter-CAR_DEALERSHIP,false:_cbPlacesFilter-FASHION_AND_CLOTHING,false:_cbPlacesFilter-CONVENIENCE_STORE,false:_cbPlacesFilter-PERSONAL_CARE,false:_cbPlacesFilter-DEPARTMENT_STORE,false:_cbPlacesFilter-PHARMACY,false:_cbPlacesFilter-ELECTRONICS,false:_cbPlacesFilter-FLOWERS,false:_cbPlacesFilter-FURNITURE_HOME_STORE,false:_cbPlacesFilter-GIFTS,false:_cbPlacesFilter-GYM_FITNESS,false:_cbPlacesFilter-SWIMMING_POOL,false:_cbPlacesFilter-HARDWARE_STORE,false:_cbPlacesFilter-MARKET,false:_cbPlacesFilter-SUPERMARKET_GROCERY,false:_cbPlacesFilter-JEWELRY,false:_cbPlacesFilter-LAUNDRY_DRY_CLEAN,false:_cbPlacesFilter-SHOPPING_CENTER,false:_cbPlacesFilter-MUSIC_STORE,false:_cbPlacesFilter-PET_STORE_VETERINARIAN_SERVICES,false:_cbPlacesFilter-TOY_STORE,false:_cbPlacesFilter-TRAVEL_AGENCY,false:_cbPlacesFilter-ATM,false:_cbPlacesFilter-CURRENCY_EXCHANGE,false:_cbPlacesFilter-CAR_RENTAL,false:_cbPlacesFilter-FOOD_AND_DRINK,false:_cbPlacesFilter-RESTAURANT,false:_cbPlacesFilter-BAKERY,false:_cbPlacesFilter-DESSERT,false:_cbPlacesFilter-CAFE,false:_cbPlacesFilter-FAST_FOOD,false:_cbPlacesFilter-FOOD_COURT,false:_cbPlacesFilter-BAR,false:_cbPlacesFilter-ICE_CREAM,false:_cbPlacesFilter-CULTURE_AND_ENTERTAINEMENT,false:_cbPlacesFilter-ART_GALLERY,false:_cbPlacesFilter-CASINO,false:_cbPlacesFilter-CLUB,false:_cbPlacesFilter-TOURIST_ATTRACTION_HISTORIC_SITE,false:_cbPlacesFilter-MOVIE_THEATER,false:_cbPlacesFilter-MUSEUM,false:_cbPlacesFilter-MUSIC_VENUE,false:_cbPlacesFilter-PERFORMING_ARTS_VENUE,false:_cbPlacesFilter-GAME_CLUB,false:_cbPlacesFilter-STADIUM_ARENA,false:_cbPlacesFilter-THEME_PARK,false:_cbPlacesFilter-ZOO_AQUARIUM,false:_cbPlacesFilter-RACING_TRACK,false:_cbPlacesFilter-THEATER,false:_cbPlacesFilter-OTHER,false:_cbPlacesFilter-CONSTRUCTION_SITE,false:_cbPlacesFilter-LODGING,false:_cbPlacesFilter-HOTEL,false:_cbPlacesFilter-HOSTEL,false:_cbPlacesFilter-CAMPING_TRAILER_PARK,false:_cbPlacesFilter-COTTAGE_CABIN,false:_cbPlacesFilter-BED_AND_BREAKFAST,false:_cbPlacesFilter-OUTDOORS,false:_cbPlacesFilter-PARK,false:_cbPlacesFilter-PLAYGROUND,false:_cbPlacesFilter-BEACH,false:_cbPlacesFilter-SPORTS_COURT,false:_cbPlacesFilter-GOLF_COURSE,false:_cbPlacesFilter-PLAZA,false:_cbPlacesFilter-PROMENADE,false:_cbPlacesFilter-POOL,false:_cbPlacesFilter-SCENIC_LOOKOUT_VIEWPOINT,false:_cbPlacesFilter-SKI_AREA,false:_cbPlacesFilter-NATURAL_FEATURES,false:_cbPlacesFilter-ISLAND,false:_cbPlacesFilter-SEA_LAKE_POOL,false:_cbPlacesFilter-RIVER_STREAM,false:_cbPlacesFilter-FOREST_GROVE,false:_cbPlacesFilter-FARM,false:_cbPlacesFilter-CANAL,false:_cbPlacesFilter-SWAMP_MARSH,false:_cbPlacesFilter-DAM,false:_cbFilterPrivatePlaces,false:_cbInvertPlacesFilter,false[END]';
defaultSettings += '[UROverviewCurrentVersion][len=0][END]';
defaultSettings += '[UROverviewCWLGroups][len=16]0,No group,false[END]';
document.getElementById('_txtSettings').value = defaultSettings;
uroTextToSettings();
document.getElementById('_txtSettings').value = '';
}
function uroSettingsToText()
{
var txtSettings = '';
uroSaveSettings();
for(var lsEntry in localStorage)
{
if(lsEntry.indexOf('UROverview') === 0)
{
txtSettings += '['+lsEntry+'][len=' + localStorage[lsEntry].length + ']' + localStorage[lsEntry] + '[END]\n';
}
}
document.getElementById('_txtSettings').value = txtSettings;
document.getElementById('_txtSettings').focus();
document.getElementById('_txtSettings').select();
}
function uroTextToSettings()
{
var txtSettings = '';
txtSettings = uroGetElmValue('_txtSettings');
if(txtSettings.indexOf('[END]') == -1) return;
var subText = txtSettings.split('[END]');
for(var i=0;i<subText.length;i++)
{
var aPos = subText[i].indexOf('[');
var bPos = subText[i].indexOf(']');
if((aPos != -1) && (bPos != -1))
{
var settingID = subText[i].substr(aPos+1,bPos-1-aPos);
subText[i] = subText[i].substr(bPos+1);
bPos = subText[i].indexOf(']');
if(bPos != -1)
{
var settingLength = subText[i].substr(5,bPos-5);
subText[i] = subText[i].substr(bPos+1);
if(subText[i].length == settingLength)
{
localStorage[settingID] = subText[i];
}
}
}
}
uroLoadSettings();
}
function uroClearSettingsText()
{
document.getElementById('_txtSettings').value = '';
}
function uroDateToDays(dateToConvert)
{
var dateNow = new Date();
var elapsedSinceEpoch = dateNow.getTime();
var elapsedSinceEvent = elapsedSinceEpoch - dateToConvert;
dateNow.setHours(0);
dateNow.setMinutes(0);
dateNow.setSeconds(0);
dateNow.setMilliseconds(0);
var elapsedSinceMidnight = elapsedSinceEpoch - dateNow.getTime();
if(elapsedSinceEvent < elapsedSinceMidnight)
{
// event occurred today...
return 0;
}
else
{
// event occurred at some point prior to midnight this morning, so return a minimum value of 1...
return 1 + Math.floor((elapsedSinceEvent - elapsedSinceMidnight) / 86400000);
}
}
function uroGetURAge(urObj,ageType,getRaw)
{
if(ageType === 0)
{
if((urObj.attributes.driveDate === null)||(urObj.attributes.driveDate === 0)) return -1;
if(getRaw) return urObj.attributes.driveDate;
else return uroDateToDays(urObj.attributes.driveDate);
}
else if(ageType === 1)
{
if((urObj.attributes.resolvedOn === null)||(urObj.attributes.resolvedOn === 0)) return -1;
if(getRaw) return urObj.attributes.resolvedOn;
else return uroDateToDays(urObj.attributes.resolvedOn);
}
else
{
return -1;
}
}
function uroGetPURAge(purObj)
{
if(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded !== null)
{
return uroDateToDays(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded);
}
else
{
return -1;
}
}
function uroGetCameraAge(camObj, mode)
{
if(mode === 0)
{
if(camObj.attributes.updatedOn === null) return -1;
return uroDateToDays(camObj.attributes.updatedOn);
}
if(mode === 1)
{
if(camObj.attributes.createdOn === null) return -1;
return uroDateToDays(camObj.attributes.createdOn);
}
}
function uroGetCommentAge(commentObj)
{
if(commentObj.createdOn === null) return -1;
return uroDateToDays(commentObj.createdOn);
}
function uroParseDaysAgo(days)
{
if(days === 0) return 'today';
else if(days === 1) return '1 day ago';
else return days+' days ago';
}
function uroGetLocalisedSpeedString(camSpeed)
{
if(camSpeed !== null)
{
var conversionFactor = 1; // default to metric
var multipleFactor = 10; // default to limits being set in multiples of 10
var country;
if(W.model.countries.top === undefined)
{
country = W.model.countries.additionalInfo[0].name;
}
else
{
country = W.model.countries.top.name;
}
if(country !== null)
{
// country-specific deviations from the above...
if
(
(country == "United Kingdom") ||
(country == "Jersey") ||
(country == "Guernsey") ||
(country == "United States")
)
{
// countries using MPH
conversionFactor = 1.609;
}
if
(
(country == "United States") ||
(country == "Guernsey")
)
{
// countries with speed limits set in multiples of 5
multipleFactor = 5;
}
}
var speed = Math.round(camSpeed / conversionFactor);
var retval = speed;
if(conversionFactor == 1) retval += "KM/H";
else retval += "MPH";
if(speed % multipleFactor !== 0) retval += " (not valid?)";
return retval;
}
else return "not set";
}
// --------------------------------------------------------------------------------------------------------------------
// AREA FRIENDLYNAME STUFF
// --------------------------------------------------------------------------------------------------------------------
function uroAFNObj(fName, area, server)
{
this.fName = fName;
this.area = area;
this.server = server;
}
function uroUpdateAreaName(name, server, area)
{
var foundExisting = false;
for(var i=0; i<uroFriendlyAreaNames.length; i++)
{
if((uroFriendlyAreaNames[i].server == server) && (uroFriendlyAreaNames[i].area == area))
{
if(name === "")
{
uroFriendlyAreaNames.splice(i,1);
foundExisting = true;
}
else
{
uroFriendlyAreaNames[i].fName = name;
foundExisting = true;
}
}
}
if((foundExisting === false) && (name !== ""))
{
uroFriendlyAreaNames.push(new uroAFNObj(name,area,server));
}
uroReplaceAreaNames(true);
}
function uroAreaNameHover()
{
if((uroAreaNameHoverObj === null) || (uroAreaNameHoverObj != this))
{
uroAreaNameHoverTime = 0;
}
uroAreaNameHoverObj = this;
}
function uroAreaNameUnHover()
{
if(uroANEditHovered === true)
{
return false;
}
if(uroAreaNameOverlayShown)
{
uroAreaNameHoverObj.removeChild(uroANEditBox);
}
uroAreaNameHoverObj = null;
uroAreaNameHoverTime = -1;
uroAreaNameOverlayShown = false;
}
function uroANEditHover()
{
uroANEditHovered = true;
uroAddEventListener('uroANEditBox','mouseout',uroANEditUnHover,false);
uroAddEventListener('uroANEditBox','click',uroANEditClick,false);
}
function uroANEditUnHover()
{
var newName = document.getElementById('_textAreaName').value;
// sanitise name to avoid conflicts with config storage delimiters...
newName = newName.replace(',','');
newName = newName.replace(':','');
var server = W.location.code;
var area = uroGetAreaArea(uroAreaNameHoverObj.parentNode.children[1]);
uroAreaNameHoverObj.removeChild(uroANEditBox);
uroAreaNameOverlayShown = false;
uroANEditHovered = false;
uroUpdateAreaName(newName, server, area);
}
function uroANEditClick(e)
{
// this traps the click to prevent it falling through to the underlying area name element and
// potentially causing the map view to be relocated to that area...
e.stopPropagation();
}
function uroGetAreaArea(listObj)
{
var area = listObj.getElementsByTagName('span')[0].innerHTML;
area = parseFloat(area.split(' ')[0]);
return area;
}
function uroAreaNameOverlaySetup()
{
uroAreaNameOverlayShown = true;
uroANEditBox = document.createElement('div');
uroANEditBox.id = "uroANEditBox";
uroANEditBox.style.position = "absolute";
uroANEditBox.style.top = '7px';
uroANEditBox.style.left = '2px';
uroANEditBox.style.width = "99%";
uroAreaNameHoverObj.appendChild(uroANEditBox);
uroANEditBox.onmouseover = uroANEditHover();
var existingName = uroAreaNameHoverObj.innerHTML;
var italicTagPos = existingName.indexOf(' <i>');
if(italicTagPos == -1)
{
existingName = "";
}
else
{
existingName = existingName.substr(0,italicTagPos);
}
uroANEditBox.innerHTML = '<input type="text" style="font-size:14px; line-height:16px; height:22px; width:100%" id="_textAreaName" value="'+existingName+'">';
}
function uroReplaceAreaNames(replaceAfterNameChange)
{
if(document.getElementById('sidepanel-areas') === undefined)
{
return;
}
if(replaceAfterNameChange === false)
{
if(document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id == "friendlyNamed")
{
return;
}
}
var panelRootObj = document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0];
if(panelRootObj === undefined)
{
// we get here if the user doesn't have any areas defined...
return;
}
var areaCount = panelRootObj.children.length;
if(areaCount === 0)
{
return;
}
var localisedManagedArea = I18n.lookup("user.areas.managed_area");
for(var loop=0; loop < areaCount; loop++)
{
var childObjPElems = panelRootObj.children[loop].getElementsByTagName('p');
var title = childObjPElems[0].innerHTML;
if(title.indexOf(localisedManagedArea) > -1)
{
var area = uroGetAreaArea(childObjPElems[1]);
childObjPElems[0].innerHTML = localisedManagedArea;
for(var fnIdx=0; fnIdx < uroFriendlyAreaNames.length; fnIdx++)
{
var fnObj = uroFriendlyAreaNames[fnIdx];
if((fnObj.area == area) && (fnObj.server == W.location.code))
{
childObjPElems[0].innerHTML = fnObj.fName +' <i>('+localisedManagedArea+')</i>';
break;
}
}
var titleObj = panelRootObj.getElementsByClassName('title')[loop];
titleObj.addEventListener("mouseover", uroAreaNameHover, false);
titleObj.addEventListener("mouseout", uroAreaNameUnHover, false);
titleObj.style.cursor = "text";
}
}
document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id = "friendlyNamed";
}
// --------------------------------------------------------------------------------------------------------------------
// WATCHLIST STUFF
// --------------------------------------------------------------------------------------------------------------------
// Generic Functions
function uroTypeCast(varin)
{
if(varin == "null") return null;
if(typeof varin == "string") return parseInt(varin);
return varin;
}
function uroTruncate(val)
{
if(val === null) return val;
if(val < 0) return Math.ceil(val);
return Math.floor(val);
}
function uroOWLGroupObj(groupID, groupName, groupCollapsed)
{
groupID = uroTypeCast(groupID);
this.groupID = groupID;
this.groupName = groupName;
this.groupCount = 0;
this.groupCollapsed = groupCollapsed;
}
// Camera Functions
function uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon)
{
if(type !== null) type = uroTypeCast(type);
if(azymuth !== null) azymuth = uroTruncate(uroTypeCast(azymuth)%360);
if(speed !== null) speed = uroTruncate(uroTypeCast(speed));
if(typeof validated == "string") validated = (validated == "true");
if(lat !== null) lat = uroTruncate(uroTypeCast(lat));
if(lon !== null) lon = uroTruncate(uroTypeCast(lon));
this.type = type;
this.azymuth = azymuth;
this.speed = speed;
this.validated = validated;
this.lat = lat;
this.lon = lon;
}
function uroCamWatchObj(persistent, fid, lon, lat, type, azymuth, speed, validated, groupID, server)
{
fid = uroTypeCast(fid);
groupID = uroTypeCast(groupID);
if(typeof persistent == "string") persistent = (persistent == "true");
this.fid = fid;
this.persistent = persistent;
this.loaded = false;
this.server = server;
this.groupID = groupID;
this.watch = new uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon);
this.current = new uroCamWatchObjCheckProps(null, null, null, null, null, null);
}
function uroCamDataChanged(idx)
{
var camObj = uroCamWatchObjects[idx];
if(camObj.loaded === false) return false;
if(camObj.current.type != camObj.watch.type) return true;
if(camObj.current.azymuth != camObj.watch.azymuth) return true;
if(camObj.current.speed != camObj.watch.speed) return true;
if(camObj.current.validated != camObj.watch.validated) return true;
if(camObj.current.lat != camObj.watch.lat) return true;
if(camObj.current.lon != camObj.watch.lon) return true;
return false;
}
function uroFindCWLGroupByIdx(groupIdx)
{
var groupName = '';
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
if(uroCWLGroups[loop].groupID == groupIdx)
{
groupName = uroCWLGroups[loop].groupName;
break;
}
}
return groupName;
}
function uroIsCamOnWatchList(fid)
{
for(var loop=0;loop<uroCamWatchObjects.length;loop++)
{
if(uroCamWatchObjects[loop].fid == fid) return loop;
}
return -1;
}
function uroAddCurrentCamWatchData(idx, lat, lon, type, azymuth, speed, validated, server)
{
var camObj = uroCamWatchObjects[idx];
camObj.loaded = true;
camObj.server = server;
camObj.current = new uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon);
return(uroCamDataChanged(idx));
}
function uroAddCamToWatchList()
{
if(uroIsCamOnWatchList(uroShownFID) == -1)
{
var camObj = W.model.cameras.objects[uroShownFID];
uroCamWatchObjects.push(new uroCamWatchObj(true, uroShownFID, camObj.geometry.x, camObj.geometry.y, camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, 0, W.location.code));
uroAddCurrentCamWatchData(uroCamWatchObjects.length-1, camObj.geometry.y, camObj.geometry.x, camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, W.location.code);
uroAddLog('added camera '+uroShownFID+' to watchlist');
uroOWLUpdateHTML();
}
}
function uroRemoveCamFromWatchList()
{
var camidx = uroIsCamOnWatchList(uroShownFID);
if(camidx != -1)
{
uroCamWatchObjects.splice(camidx,1);
uroAddLog('removed camera '+uroShownFID+' from watchlist');
uroOWLUpdateHTML();
}
}
function uroUpdateCamWatchList()
{
var camIdx = uroIsCamOnWatchList(uroShownFID);
if(camIdx != -1)
{
var camObj = W.model.cameras.objects[uroShownFID];
uroCamWatchObjects[camIdx].watch = new uroCamWatchObjCheckProps(camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, camObj.geometry.y, camObj.geometry.x);
}
}
function uroClearCamWatchList()
{
uroShowAlertBox("fa-warning", "URO+ Warning", "Removing all cameras from the OWL <b>cannot</b> be undone.<br>Are you <i>sure</i> you want to do this?", true, "Delete ALL Cameras", "Keep Cameras", uroClearCamWatchListAction, null);
}
function uroClearCamWatchListAction()
{
uroCamWatchObjects = [];
uroOWLUpdateHTML();
}
function uroRetrieveCameras(lat, lon)
{
var camPos = new OpenLayers.LonLat();
var camChanged = false;
camPos.lon = lon;
camPos.lat = lat;
camPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
var camURL = 'https://' + document.location.host;
camURL += Waze.Config.api_base;
camURL += '/Features?language=en&cameras=true&bbox=';
var latl = camPos.lat - 0.25;
var latu = camPos.lat + 0.25;
var lonl = camPos.lon - 0.25;
var lonr = camPos.lon + 0.25;
camURL += lonl+','+latl+','+lonr+','+latu;
uroAddLog('retrieving camera data around '+camPos.lon+','+camPos.lat);
var camReq = new XMLHttpRequest();
camReq.open('GET',camURL,false);
try
{
camReq.send();
uroAddLog('response '+camReq.status+' received');
if (camReq.status === 200)
{
var camData = JSON.parse(camReq.responseText);
for(var camIdx = 0; camIdx < camData.cameras.objects.length; camIdx++)
{
var camObj = camData.cameras.objects[camIdx];
var listIdx = uroIsCamOnWatchList(camObj.id);
if(listIdx != -1)
{
camPos.lon = camObj.geometry.coordinates[0];
camPos.lat = camObj.geometry.coordinates[1];
camPos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
camPos.lon = uroTruncate(camPos.lon);
camPos.lat = uroTruncate(camPos.lat);
camChanged |= uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, camObj.validated, W.location.code);
}
else if(camObj.validated === false)
{
}
}
}
else
{
uroAddLog('request failed (status != 200)');
}
}
catch(err)
{
uroAddLog('camera load request failed (exception '+err+' caught)');
}
return camChanged;
}
function uroGetCurrentCamWatchListObjects()
{
var camChanged = false;
var camsChanged = [];
var camsDeleted = [];
var camidx;
var camObj;
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
camObj = uroCamWatchObjects[camidx];
if((camObj.loaded === false) && ((camObj.server == W.location.code) || (camObj.server == '??')))
{
if(typeof W.model.cameras.objects[camObj.fid] == 'object')
{
if(W.model.cameras.objects[camObj.fid].state != "Delete")
{
var wazeObj = W.model.cameras.objects[camObj.fid];
camChanged |= uroAddCurrentCamWatchData(camidx, wazeObj.geometry.y, wazeObj.geometry.x, wazeObj.attributes.type, wazeObj.attributes.azymuth, wazeObj.attributes.speed, wazeObj.attributes.validated);
}
else
{
camChanged |= uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon);
}
}
else
{
camChanged |= uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon);
}
}
}
if(camChanged)
{
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
if(uroCamDataChanged(camidx))
{
camsChanged.push(uroCamWatchObjects[camidx]);
}
}
}
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
camObj = uroCamWatchObjects[camidx];
if((camObj.loaded === false) && (camObj.server == W.location.code))
{
camsDeleted.push(camObj);
}
}
if((camsChanged.length > 0) || (camsDeleted.length > 0))
{
var alertStr = '';
for(camidx=0;camidx<camsChanged.length;camidx++)
{
alertStr += 'Camera ID '+camsChanged[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsChanged[camidx].groupID)+'" has been changed<br>';
}
alertStr += '<br>';
for(camidx=0;camidx<camsDeleted.length;camidx++)
{
alertStr += 'Camera ID '+camsDeleted[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsDeleted[camidx].groupID)+'" has been deleted<br>';
}
uroShowAlertBox("fa-info-circle", "URO+ Camera Watchlist Alert", alertStr, false, "OK", null, null, null);
}
}
function uroClearDeletedCameras()
{
for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
{
if(uroCamWatchObjects[camidx].loaded === false)
{
uroShownFID = uroCamWatchObjects[camidx].fid;
uroRemoveCamFromWatchList();
}
}
}
function uroClearUnknownServerCameras()
{
var confirmMsg = '<p>Cameras with an unknown server <i>cannot</i> be automatically verified by URO+</p>';
confirmMsg += 'It is recommended that you manually load WME from each server (World, USA/Canada and Israel) to give URO+ a chance of locating these cameras.<br>';
confirmMsg += 'If the cameras then continue to show up as an unknown server, it is safe to delete them...<br><br>';
confirmMsg += 'Do you still wish to proceed with deleting all unknown server cameras?';
uroShowAlertBox("fa-warning", "URO+ Warning", confirmMsg, true, "Delete unknown cameras", "Keep unknown cameras", uroClearUnknownServerCamerasAction, null);
}
function uroClearUnknownServerCamerasAction()
{
for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
{
if(uroCamWatchObjects[camidx].server == '??')
{
uroShownFID = uroCamWatchObjects[camidx].fid;
uroRemoveCamFromWatchList();
}
}
}
function uroRescanCamWatchList()
{
for(var camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
uroCamWatchObjects[camidx].loaded = false;
}
uroGetCurrentCamWatchListObjects();
uroOWLUpdateHTML();
}
function uroGotoCam()
{
var camidx = this.id.substr(13);
var camPos = new OpenLayers.LonLat();
camPos.lon = uroCamWatchObjects[camidx].watch.lon;
camPos.lat = uroCamWatchObjects[camidx].watch.lat;
W.map.setCenter(camPos,4);
W.map.camerasLayer.setVisibility(true);
return false;
}
// Segment Functions
/*
function uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn)
{
if(left !== null) left = uroTruncate(uroTypeCast(left));
if(right !== null) right = uroTruncate(uroTypeCast(right));
if(bottom !== null) bottom = uroTruncate(uroTypeCast(bottom));
if(top !== null) top = uroTruncate(uroTypeCast(top));
if(fromNode !== null) fromNode = uroTypeCast(fromNode);
if(toNode !== null) toNode = uroTypeCast(toNode);
if(fwdDir !== null) fwdDir = uroTypeCast(fwdDir);
if(revDir !== null) revDir = uroTypeCast(revDir);
if(length !== null) length = uroTypeCast(length);
if(level !== null) level = uroTypeCast(level);
if(rank !== null) rank = uroTypeCast(rank);
if(roadType !== null) roadType = uroTypeCast(roadType);
if(updatedOn !== null) updatedOn = uroTypeCast(updatedOn);
this.left = left;
this.right = right;
this.bottom = bottom;
this.top = top;
this.fromNode = fromNode;
this.toNode = toNode;
this.fwdDir = fwdDir;
this.revDir = revDir;
this.length = length;
this.level = level;
this.rank = rank;
this.roadType = roadType;
this.updatedOn = updatedOn;
}
function uroSegWatchObj(persistent, fid, left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn, groupID, server)
{
fid = uroTypeCast(fid);
groupID = uroTypeCast(groupID);
if(typeof persistent == "string") persistent = (persistent == "true");
this.fid = fid;
this.persistent = persistent;
this.loaded = false;
this.server = server;
this.groupID = groupID;
this.watch = new uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn);
this.current = new uroSegWatchObjCheckProps(null, null, null, null, null, null, null, null, null, null, null, null, null);
}
function uroSegDataChanged(idx)
{
var segObj = uroSegWatchObjects[idx];
if(segObj.loaded === false) return false;
if(segObj.current.left != segObj.watch.left) return true;
if(segObj.current.right != segObj.watch.right) return true;
if(segObj.current.bottom != segObj.watch.bottom) return true;
if(segObj.current.top != segObj.watch.top) return true;
if(segObj.current.fromNode != segObj.watch.fromNode) return true;
if(segObj.current.toNode != segObj.watch.toNode) return true;
if(segObj.current.fwdDir != segObj.watch.fwdDir) return true;
if(segObj.current.revDir != segObj.watch.revDir) return true;
if(segObj.current.length != segObj.watch.length) return true;
if(segObj.current.level != segObj.watch.level) return true;
if(segObj.current.rank != segObj.watch.rank) return true;
if(segObj.current.roadType != segObj.watch.roadType) return true;
if(segObj.current.updatedOn != segObj.watch.updatedOn) return true;
return false;
}
function uroIsSegOnWatchList(fid)
{
for(var loop=0;loop<uroSegWatchObjects.length;loop++)
{
if(uroSegWatchObjects[loop].fid == fid) return loop;
}
return -1;
}
function uroAddCurrentSegWatchData(idx, left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn, server)
{
var segObj = uroSegWatchObjects[idx];
segObj.loaded = true;
segObj.server = server;
segObj.current = new uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn);
return(uroSegDataChanged(idx));
}
function uroClearSegWatchList()
{
if(confirm('Removing all segments from the OWL cannot be undone\nAre you sure you want to do this?') === true)
{
uroSegWatchObjects = [];
uroOWLUpdateHTML();
}
}
function uroAddUpdateSegWatchList()
{
var selectedCount = W.selectionManager.selectedItems.length;
if(selectedCount === 0)
{
return;
}
for(var loop=0;loop < selectedCount; loop++)
{
var segObj = W.selectionManager.selectedItems[loop].model.attributes;
var fid = segObj.id;
var idx = uroIsSegOnWatchList(fid);
if(idx != -1)
{
uroSegWatchObjects[idx].watch = new uroSegWatchObjCheckProps(segObj.geometry.bounds.left, segObj.geometry.bounds.right, segObj.geometry.bounds.bottom, segObj.geometry.bounds.top, segObj.fromNodeID, segObj.toNodeID, segObj.fwdDirection, segObj.revDirection, segObj.length, segObj.level, segObj.rank, segObj.roadType, segObj.updatedOn);
uroAddLog('updated watchlist details for segment '+fid);
}
else
{
uroSegWatchObjects.push(new uroSegWatchObj(true, fid, segObj.geometry.bounds.left, segObj.geometry.bounds.right, segObj.geometry.bounds.bottom, segObj.geometry.bounds.top, segObj.fromNodeID, segObj.toNodeID, segObj.fwdDirection, segObj.revDirection, segObj.length, segObj.level, segObj.rank, segObj.roadType, segObj.updatedOn, 0, W.location.code));
uroAddCurrentSegWatchData(uroSegWatchObjects.length-1, segObj.geometry.bounds.left, segObj.geometry.bounds.right, segObj.geometry.bounds.bottom, segObj.geometry.bounds.top, segObj.fromNodeID, segObj.toNodeID, segObj.fwdDirection, segObj.revDirection, segObj.length, segObj.level, segObj.rank, segObj.roadType, segObj.updatedOn, W.location.code);
uroAddLog('added segment '+fid+' to watchlist');
}
}
//uroOWLUpdateHTML();
}
function uroRemoveSegFromWatchList()
{
var selectedCount = W.selectionManager.selectedItems.length;
if(selectedCount === 0)
{
return;
}
for(var loop=0;loop < selectedCount; loop++)
{
var fid = W.selectionManager.selectedItems[loop].model.attributes.id;
var idx = uroIsSegOnWatchList(fid);
if(idx != -1)
{
uroSegWatchObjects.splice(idx,1);
uroAddLog('removed segment '+fid+' from watchlist');
}
}
//uroOWLUpdateHTML();
}
function uroRetrieveSegments(lat, lon)
{
var pos = new OpenLayers.LonLat();
var changed = false;
pos.lon = lon;
pos.lat = lat;
pos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
var URL = 'https://' + document.location.host;
URL += Waze.Config.api_base;
URL += '/Features?roadTypes=1%2C2%2C3%2C4%2C5%2C6%2C7%2C8%2C9%2C10%2C11%2C12%2C13%2C14%2C15%2C16%2C17%2C18%2C19%2C20%2C21';
URL += '&bbox=';
var latl = pos.lat - 0.25;
var latu = pos.lat + 0.25;
var lonl = pos.lon - 0.25;
var lonr = pos.lon + 0.25;
URL += lonl+','+latl+','+lonr+','+latu;
URL += '&language=en';
uroAddLog('retrieving segment data around '+pos.lon+','+pos.lat);
var req = new XMLHttpRequest();
req.open('GET',URL,false);
try
{
req.send();
uroAddLog('response '+req.status+' received');
if (req.status === 200)
{
var data = JSON.parse(req.responseText);
for(var idx = 0; idx < data.segments.objects.length; idx++)
{
var obj = data.segments.objects[idx];
var listIdx = uroIsSegOnWatchList(obj.id);
if(listIdx != -1)
{
//pos.lon = obj.geometry.coordinates[0];
//pos.lat = obj.geometry.coordinates[1];
//pos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
//camPos.lon = uroTruncate(camPos.lon);
//camPos.lat = uroTruncate(camPos.lat);
//camChanged |= uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, camObj.validated, W.location.code);
}
else if(obj.validated === false)
{
}
}
}
else
{
uroAddLog('request failed (status != 200)');
}
}
catch(err)
{
uroAddLog('segment load request failed (exception '+err+' caught)');
}
return changed;
}
function uroGetCurrentSegWatchListObjects()
{
var segChanged = false;
var segsChanged = [];
var segsDeleted = [];
var idx;
var segObj;
for(idx=0;idx<uroSegWatchObjects.length;idx++)
{
segObj = uroSegWatchObjects[idx];
if((segObj.loaded === false) && ((segObj.server == W.location.code) || (segObj.server == '??')))
{
var segLat = (segObj.watch.top + segObj.watch.bottom) / 2;
var segLon = (segObj.watch.right + segObj.watch.left) / 2;
if(typeof W.model.segments.objects[segObj.fid] == 'object')
{
if(W.model.segments.objects[segObj.fid].state != "Delete")
{
var wazeObj = W.model.segments.objects[segObj.fid];
segChanged |= uroAddCurrentSegWatchData(idx, wazeObj.geometry.bounds.left, wazeObj.geometry.bounds.right, wazeObj.geometry.bounds.bottom, wazeObj.geometry.bounds.top, wazeObj.fromNodeID, wazeObj.toNodeID, wazeObj.fwdDirection, wazeObj.revDirection, wazeObj.length, wazeObj.level, wazeObj.rank, wazeObj.roadType, wazeObj.updatedOn, W.location.code);
}
else
{
segChanged |= uroRetrieveSegments(segLat, segLon);
}
}
else
{
segChanged |= uroRetrieveSegments(segLat, segLon);
}
}
}
if(segChanged)
{
for(idx=0;idx<uroSegWatchObjects.length;idx++)
{
if(uroSegDataChanged(idx))
{
segsChanged.push(uroSegWatchObjects[idx]);
}
}
}
for(idx=0;idx<uroSegWatchObjects.length;idx++)
{
segObj = uroSegWatchObjects[idx];
if((segObj.loaded === false) && (segObj.server == W.location.code))
{
segsDeleted.push(segObj);
}
}
if((segsChanged.length > 0) || (segsDeleted.length > 0))
{
var alertStr = 'Segment WatchList Alert!!!\r\n';
for(idx=0;idx<segsChanged.length;idx++)
{
alertStr += 'Segment ID '+segsChanged[idx].fid+' in group "'+uroFindCWLGroupByIdx(segsChanged[idx].groupID)+'" has been changed\r\n';
}
for(idx=0;idx<segsDeleted.length;idx++)
{
alertStr += 'Segment ID '+segsDeleted[idx].fid+' in group "'+uroFindCWLGroupByIdx(segsDeleted[idx].groupID)+'" has been deleted\r\n';
}
alert(alertStr);
}
}
// Places Functions
function uroPlaceWatchObjCheckProps(left, right, bottom, top, name, imageCount, residential, updatedOn)
{
if(left !== null) left = uroTruncate(uroTypeCast(left));
if(right !== null) right = uroTruncate(uroTypeCast(right));
if(bottom !== null) bottom = uroTruncate(uroTypeCast(bottom));
if(top !== null) top = uroTruncate(uroTypeCast(top));
if(imageCount !== null) imageCount = uroTypeCast(imageCount);
if(typeof residential == "string") residential = (residential == "true");
if(updatedOn !== null) updatedOn = uroTypeCast(updatedOn);
this.left = left;
this.right = right;
this.bottom = bottom;
this.top = top;
this.name = name;
this.imageCount = imageCount;
this.residential = residential;
this.updatedOn = updatedOn;
}
function uroPlaceWatchObj(persistent, fid, left, right, bottom, top, imageCount, name, residential, updatedOn, groupID, server)
{
groupID = uroTypeCast(groupID);
if(typeof persistent == "string") persistent = (persistent == "true");
this.fid = fid;
this.persistent = persistent;
this.loaded = false;
this.server = server;
this.groupID = groupID;
this.watch = new uroPlaceWatchObjCheckProps(left, right, bottom, top, name, imageCount, residential, updatedOn);
this.current = new uroPlaceWatchObjCheckProps(null, null, null, null, null, null, null, null);
}
function uroPlaceDataChanged(idx)
{
var placeObj = uroPlaceWatchObjects[idx];
if(placeObj.loaded === false) return false;
if(placeObj.current.left != placeObj.watch.left) return true;
if(placeObj.current.right != placeObj.watch.right) return true;
if(placeObj.current.bottom != placeObj.watch.bottom) return true;
if(placeObj.current.top != placeObj.watch.top) return true;
if(placeObj.current.name != placeObj.watch.name) return true;
if(placeObj.current.imageCount != placeObj.watch.imageCount) return true;
if(placeObj.current.residential != placeObj.watch.residential) return true;
if(placeObj.current.updatedOn != placeObj.watch.updatedOn) return true;
return false;
}
function uroIsPlaceOnWatchList(fid)
{
for(var loop=0;loop<uroPlaceWatchObjects.length;loop++)
{
if(uroPlaceWatchObjects[loop].fid == fid) return loop;
}
return -1;
}
function uroClearPlaceWatchList()
{
if(confirm('Removing all places from the OWL cannot be undone\nAre you sure you want to do this?') === true)
{
uroPlaceWatchObjects = [];
uroOWLUpdateHTML();
}
}
*/
function uroHighlightCWLEntry()
{
this.style.backgroundColor = '#FFFFAA';
return false;
}
function uroUnhighlightCWLEntry()
{
var camidx = this.id.substr(8);
var changed = uroCamDataChanged(camidx);
var deleted = (uroCamWatchObjects[camidx].loaded === false);
if(uroCamWatchObjects[camidx].server != W.location.code)
{
if(uroCamWatchObjects[camidx].server == '??') this.style.backgroundColor = '#A0A0A0';
else this.style.backgroundColor = '#AAFFAA';
}
else if(changed) this.style.backgroundColor = '#AAAAFF';
else if(deleted) this.style.backgroundColor = '#FFAAAA';
else this.style.backgroundColor = '#FFFFFF';
return false;
}
function uroCWLIconHighlight()
{
this.style.color="#0000ff";
return false;
}
function uroCWLIconLowlight()
{
this.style.color="#ccccff";
return false;
}
function uroPopulateCWLGroupSelect()
{
var selector = document.getElementById('_uroCWLGroupSelect');
while(selector.options.length > 0)
{
selector.options.remove(0);
}
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
var groupObj = uroCWLGroups[loop];
if(groupObj.groupID != -1)
{
selector.options.add(new Option(groupObj.groupName,groupObj.groupID));
}
}
}
function uroGetNextCWLGroupID()
{
var nextID = 1;
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
if(uroCWLGroups[loop].groupID >= nextID)
{
nextID = uroCWLGroups[loop].groupID + 1;
}
}
return nextID;
}
function uroFindCWLGroupByName(groupName)
{
var groupID = -1;
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
if((uroCWLGroups[loop].groupName == groupName) && (uroCWLGroups[loop].groupID != -1))
{
groupID = uroCWLGroups[loop].groupID;
break;
}
}
return groupID;
}
function uroAddCWLGroup()
{
var groupID = uroGetNextCWLGroupID();
var groupName = uroGetElmValue('_uroCWLGroupEntry');
if(uroFindCWLGroupByName(groupName) == -1)
{
uroCWLGroups.push(new uroOWLGroupObj(groupID,groupName,false));
uroPopulateCWLGroupSelect();
}
}
function uroRemoveCWLGroup()
{
var loop;
var selector = document.getElementById('_uroCWLGroupSelect');
var groupID = parseInt(selector.selectedOptions[0].value);
if(groupID === 0) return false; // prevent deletion of the default group
for(loop=0;loop<uroCamWatchObjects.length;loop++)
{
var cwObj = uroCamWatchObjects[loop];
if(cwObj.groupID == groupID)
{
cwObj.groupID = 0;
}
}
for(loop=0;loop<uroCWLGroups.length;loop++)
{
var groupObj = uroCWLGroups[loop];
if(groupObj.groupID == groupID)
{
groupObj.groupID = -1;
}
}
uroOWLUpdateHTML();
}
function uroAssignCameraToGroup()
{
var camidx = this.id.substr(13);
var selector = document.getElementById('_uroCWLGroupSelect');
uroCamWatchObjects[camidx].groupID = parseInt(selector.selectedOptions[0].value);
uroOWLUpdateHTML();
return false;
}
function uroAddBtnEvl(btnID, evlType, evlFunction)
{
var btnObj = document.getElementById(btnID);
if(btnObj !== null)
{
btnObj.addEventListener(evlType, evlFunction, true);
}
}
function uroCWLGroupCollapseExpand()
{
var groupidx = this.id.substr(18);
if(uroCWLGroups[groupidx].groupCollapsed === true) uroCWLGroups[groupidx].groupCollapsed = false;
else uroCWLGroups[groupidx].groupCollapsed = true;
uroOWLUpdateHTML();
return false;
}
var uroSelectedOWLGroup = null;
function uroOWLUpdateHTML(doFullUpdate)
{
var camTypes = new Array("","","Speed", "Dummy", "Red Light");
var iHTML = '';
if(document.getElementById('_uroCWLGroupSelect') !== null)
{
uroSelectedOWLGroup = document.getElementById('_uroCWLGroupSelect').selectedIndex;
}
iHTML = '<br><b>Camera Watchlist:</b><br><br>';
iHTML += '<div id="_uroCWLCamList" style="height:65%;overflow:auto;">';
if(uroCWLGroups.length > 0)
{
var camidx;
for(var groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
{
var groupObj = uroCWLGroups[groupidx];
iHTML += '<div id="_uroCWLGroup-'+groupidx+'">';
if(groupObj.groupCollapsed === true)
{
iHTML += '<i class="fa fa-plus-square-o" style="cursor:pointer;font-size:14px;" id="_uroCWLGroupState-'+groupidx+'"></i>';
}
else
{
iHTML += '<i class="fa fa-minus-square-o" style="cursor:pointer;font-size:14px;" id="_uroCWLGroupState-'+groupidx+'"></i>';
}
iHTML += '<b>'+groupObj.groupName+'</b><br>';
groupObj.groupCount = 0;
if(uroCamWatchObjects.length > 0)
{
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
var camObj = uroCamWatchObjects[camidx];
if(camObj.groupID == groupObj.groupID)
{
groupObj.groupCount++;
var changed = uroCamDataChanged(camidx);
var deleted = (camObj.loaded === false);
iHTML += '<div id="_uroCWL-'+camidx+'" style="padding:3px;border-width:2px;border-style:solid;border-color:#FFFFFF;background-color:';
if(camObj.server != W.location.code)
{
if(camObj.server == '??') iHTML += '#A0A0A0;';
else iHTML += '#AAFFAA;';
}
else if(changed) iHTML += '#AAAAFF;';
else if(deleted) iHTML += '#FFAAAA;';
else iHTML += '#FFFFFF;';
if(groupObj.groupCollapsed === true) iHTML += 'display:none;">';
else iHTML += 'display:block;">';
iHTML += 'ID: '+camObj.fid;
iHTML += ' ('+camObj.server+')';
iHTML += ' Type: '+camTypes[camObj.watch.type];
if(camObj.server != W.location.code)
{
if(camObj.server == '??')
{
iHTML += '<br><i>Unknown server</i>';
}
else
{
iHTML += '<br><i>Not on this server</i>';
}
}
else if(deleted)
{
iHTML += '<br>DELETED';
}
else if(changed)
{
if(camObj.current.type != camObj.watch.type)
{
iHTML += '<br> Type changed';
iHTML += ' ('+camObj.watch.type+' to '+camObj.current.type+')';
}
if(camObj.current.azymuth != camObj.watch.azymuth)
{
iHTML += '<br> Azimuth changed';
iHTML += ' ('+camObj.watch.azymuth+' to '+camObj.current.azymuth+')';
}
if(camObj.current.speed != camObj.watch.speed)
{
iHTML += '<br> Speed changed';
iHTML += ' ('+camObj.watch.speed+' to '+camObj.current.speed+')';
}
if(camObj.current.validated != camObj.watch.validated)
{
iHTML += '<br> Approval state changed';
iHTML += ' ('+camObj.watch.validated+' to '+camObj.current.validated+')';
}
if(camObj.current.lat != camObj.watch.lat)
{
iHTML += '<br> Latitude changed';
iHTML += ' ('+camObj.watch.lat+' to '+camObj.current.lat+')';
}
if(camObj.current.lon != camObj.watch.lon)
{
iHTML += '<br> Longitude changed';
iHTML += ' ('+camObj.watch.lon+' to '+camObj.current.lon+')';
}
}
if(camObj.server == W.location.code)
{
if(deleted === false)
{
iHTML += ' <i class="fa fa-group" style="cursor:pointer;font-size:14px;color:#ccccff;" id="_uroCWLIcon1-'+camidx+'"></i>';
}
iHTML += ' <i class="fa fa-arrow-circle-right" style="cursor:pointer;font-size:14px;color:#ccccff;" id="_uroCWLIcon2-'+camidx+'"></i>';
}
iHTML += '</div>';
}
}
}
iHTML += '</div>';
}
}
iHTML += '</div><div id="_uroCWLControls">';
iHTML += '<hr>Group control:<br>';
iHTML += '<select id="_uroCWLGroupSelect" style="width:40%;height:22px;"></select> <input type="button" id="_btnCWLGroupDel" value="Delete group"><br>';
iHTML += '<input type="text" id="_uroCWLGroupEntry" style="width:40%;height:22px;"> <input type="button" id="_btnCWLGroupAdd" value="Add group">';
iHTML += '<br><input type="button" id="_btnRescanCamWatchList" value="Refresh Camera Data"><br><br>';
iHTML += '<b>Remove cameras from OWL:</b><br>';
iHTML += '<input type="button" id="_btnRemoveDeletedCameras" value="Deleted"> ';
iHTML += '<input type="button" id="_btnRemoveUnknownServerCameras" value="Unknown Server"> ';
iHTML += '<input type="button" id="_btnClearCamWatchList" value="ALL Cameras">';
iHTML += '</div>';
uroOWL.innerHTML = iHTML;
uroFinaliseOWLHTMLUpdate();
}
function uroFinaliseOWLHTMLUpdate()
{
if(uroCamWatchObjects.length > 0)
{
if(document.getElementById("_uroCWL-0") == null)
{
setTimeout(uroFinaliseOWLHTMLUpdate,100);
return;
}
for(var camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
document.getElementById("_uroCWL-"+camidx).onmouseover = uroHighlightCWLEntry;
document.getElementById("_uroCWL-"+camidx).onmouseleave = uroUnhighlightCWLEntry;
if(uroCamWatchObjects[camidx].server == W.location.code)
{
var icon1 = document.getElementById("_uroCWLIcon1-"+camidx);
var icon2 = document.getElementById("_uroCWLIcon2-"+camidx);
if(icon1 !== null)
{
icon1.onmouseover = uroCWLIconHighlight;
icon1.onmouseleave = uroCWLIconLowlight;
icon1.onclick = uroAssignCameraToGroup;
}
if(icon2 !== null)
{
icon2.onmouseover = uroCWLIconHighlight;
icon2.onmouseleave = uroCWLIconLowlight;
icon2.onclick = uroGotoCam;
}
}
}
}
if(document.getElementById('_btnClearCamWatchList') == null)
{
setTimeout(uroFinaliseOWLHTMLUpdate,100);
return;
}
uroAddBtnEvl('_btnClearCamWatchList', 'click', uroClearCamWatchList);
uroAddBtnEvl('_btnRemoveDeletedCameras', 'click', uroClearDeletedCameras);
uroAddBtnEvl('_btnRemoveUnknownServerCameras', 'click', uroClearUnknownServerCameras);
uroAddBtnEvl('_btnRescanCamWatchList', 'click', uroRescanCamWatchList);
uroAddBtnEvl('_btnCWLGroupDel', 'click', uroRemoveCWLGroup);
uroAddBtnEvl('_btnCWLGroupAdd', 'click', uroAddCWLGroup);
if(document.getElementById('_uroCWLGroupSelect') !== null)
{
uroAddLog('populating CWL group list');
uroPopulateCWLGroupSelect();
var selector = document.getElementById('_uroCWLGroupSelect');
if(uroSelectedOWLGroup >= selector.length)
{
uroSelectedOWLGroup = 0;
}
selector.selectedIndex = uroSelectedOWLGroup;
}
if(uroCWLGroups.length > 0)
{
for(var groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
{
if(uroCWLGroups[groupidx].groupCount === 0)
{
uroSetStyleDisplay('_uroCWLGroup-'+groupidx,'none');
}
else
{
uroSetOnClick('_uroCWLGroupState-'+groupidx,uroCWLGroupCollapseExpand);
}
}
}
}
// --------------------------------------------------------------------------------------------------------------------
// END OF WATCHLIST STUFF
// --------------------------------------------------------------------------------------------------------------------
function uroIsOnIgnoreList(fid)
{
if(sessionStorage.UROverview_FID_IgnoreList.indexOf('fid:'+fid) == -1) return false;
else return true;
}
function uroEnableIgnoreListControls()
{
var btnState = "visible";
if(sessionStorage.UROverview_FID_IgnoreList === '')
{
btnState = "hidden";
}
document.getElementById('_btnUndoLastHide').style.visibility = btnState;
document.getElementById('_btnClearSessionHides').style.visibility = btnState;
uroFilterItems();
}
function uroAddToIgnoreList()
{
if(!uroIsOnIgnoreList(uroShownFID))
{
sessionStorage.UROverview_FID_IgnoreList += 'fid:'+uroShownFID;
uroAddLog('added fid '+uroShownFID+' to ignore list');
uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
uroDiv.style.visibility = 'hidden';
uroEnableIgnoreListControls();
W.map.events.register("mousemove", null, uroFilterItemsOnMove);
}
return false;
}
function uroRemoveLastAddedIgnore()
{
var ignorelist = sessionStorage.UROverview_FID_IgnoreList;
var fidpos = ignorelist.lastIndexOf('fid:');
if(fidpos != -1)
{
ignorelist = ignorelist.slice(0,fidpos);
sessionStorage.UROverview_FID_IgnoreList = ignorelist;
uroAddLog('removed last fid from ignore list');
uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
uroEnableIgnoreListControls();
}
}
function uroRemoveAllIgnores()
{
sessionStorage.UROverview_FID_IgnoreList = '';
uroEnableIgnoreListControls();
}
function uroKeywordPresent(desc, keyword)
{
var re;
if(uroGetCBChecked('_cbCaseInsensitive') === true) re = RegExp(keyword,'i');
else re = RegExp(keyword);
if(desc.search(re) != -1) return true;
else return false;
}
function uroClickify(desc)
{
var linkStartPos = desc.indexOf('http://');
if(linkStartPos == -1) linkStartPos = desc.indexOf('https://');
if(linkStartPos != -1)
{
var descPreLink = desc.slice(0,linkStartPos);
var descURL = desc.slice(linkStartPos);
var linkEndPos = descURL.indexOf(' ');
var descPostLink = '';
if(linkEndPos != -1)
{
descPostLink = descURL.slice(linkEndPos);
descURL = descURL.slice(0,linkEndPos);
}
var linkTarget = '';
if(descURL.indexOf('cryosphere') != -1) linkTarget = '_cryosphere';
else if(descURL.indexOf('waze.com') != -1) linkTarget = '_wazeUR';
desc = descPreLink + '<a target="'+linkTarget+'" href="'+descURL+'">here</a>' + descPostLink;
}
return desc;
}
function uroGetUpdateRequestSessions()
{
var idList = [];
while((idList.length < 50) && (uroPendingURSessionIDs.length))
{
var id = uroPendingURSessionIDs.shift();
idList.push(id);
}
if(idList.length > 0)
{
uroAddLog('grabbing '+idList.length+' updateRequestSessions, IDs: '+idList);
W.model.updateRequestSessions.get(idList);
}
if((uroPendingURSessionIDs.length) || (uroRequestedURSessionIDs.length))
{
setTimeout(uroGetUpdateRequestSessions,1000);
}
}
function uroRefreshUpdateRequestSessions()
{
var urcount = 0;
uroPendingURSessionIDs = [];
uroRequestedURSessionIDs = [];
for (var urID in W.model.mapUpdateRequests.objects)
{
if(W.model.mapUpdateRequests.objects.hasOwnProperty(urID))
{
if(W.model.updateRequestSessions.objects[urID] === undefined)
{
uroPendingURSessionIDs.push(urID);
}
urcount++;
}
}
uroGetUpdateRequestSessions();
}
function uroURHasMyComments(fid)
{
if(uroUserID === -1) return false;
var nComments = W.model.updateRequestSessions.objects[fid].comments.length;
if(nComments === 0) return false;
for(var cidx=0; cidx<nComments; cidx++)
{
if(W.model.updateRequestSessions.objects[fid].comments[cidx].userID == uroUserID) return true;
}
return false;
}
function uroACMObj(urID, markerType, customType, hasMyComments, nComments)
{
this.urID = urID;
this.markerType = markerType;
this.customType = customType;
this.hasMyComments = hasMyComments;
this.nComments = nComments;
}
function uroAddCustomMarkers(urID, markerType, customType, hasMyComments, nComments)
{
var useCustomMarker = false;
if(uroGetCBChecked('_cbMasterEnable') === true)
{
if(customType === 0) useCustomMarker = (uroGetCBChecked('_cbCustomRoadworksMarkers'));
else if(customType === 1) useCustomMarker = (uroGetCBChecked('_cbCustomConstructionMarkers'));
else if(customType === 2) useCustomMarker = (uroGetCBChecked('_cbCustomClosuresMarkers'));
else if(customType === 3) useCustomMarker = (uroGetCBChecked('_cbCustomEventsMarkers'));
else if(customType === 4) useCustomMarker = (uroGetCBChecked('_cbCustomNotesMarkers'));
else if(customType === 5) useCustomMarker = (uroGetCBChecked('_cbCustomWSLMMarkers'));
else if(customType === 6) useCustomMarker = (uroGetCBChecked('_cbCustomBOGMarkers'));
else if(customType === 7) useCustomMarker = (uroGetCBChecked('_cbCustomDifficultMarkers'));
else if(customType === 98) useCustomMarker = (uroGetCBChecked('_cbCustomNativeSLMarkers'));
else if(customType === 99) useCustomMarker = (uroGetCBChecked('_cbCustomKeywordMarkers'));
else if(customType === 100) useCustomMarker = (uroGetCBChecked('_cbCustomElginMarkers'));
else if(customType === 101) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficCastMarkers'));
else if(customType === 102) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficMasterMarkers'));
else if(customType === 103) useCustomMarker = (uroGetCBChecked('_cbCustomCaltransMarkers'));
else if(customType === 104) useCustomMarker = (uroGetCBChecked('_cbCustomTFLMarkers'));
}
if(!useCustomMarker) customType = -1;
uroCustomMarkerList.push(new uroACMObj(urID, markerType, customType, hasMyComments, nComments));
}
function uroRenderCustomMarkers(markerType)
{
var urID;
var elmID;
var newSpan;
var divElem;
var objIdx;
var customType;
var cmlObj;
var defaultMarkerURL = "url('"+document.location.origin + '/assets-editor/sprites/vectors/' + uroNativeMarkerImage+"')";
var touchedByURO = false;
if(markerType == 'ur')
{
var useDefaultConvoMarker = false;
var addCommentCount = false;
if(uroGetCBChecked('_cbMasterEnable') === true)
{
if((uroGetCBChecked('_cbNativeConvoMarkers')) && (uroBetaEditor === false)) useDefaultConvoMarker = true;
if((uroGetCBChecked('_cbNativeBetaConvoMarkers')) && (uroBetaEditor === true)) useDefaultConvoMarker = true;
if(uroGetCBChecked('_cbCommentCount')) addCommentCount = true;
}
else
{
useDefaultConvoMarker = true;
}
var uRCM_masterEnable = uroGetCBChecked('_cbMasterEnable');
divElem = document.getElementById(W.map.updateRequestLayer.id);
if(divElem.childNodes.length > 0)
{
for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
{
customType = -1;
cmlObj = uroCustomMarkerList[objIdx];
if(cmlObj.markerType == 'ur')
{
if(uRCM_masterEnable === true)
{
customType = cmlObj.customType;
}
if(customType < 100)
{
urID = cmlObj.urID;
var nComments = cmlObj.nComments;
var iconObj = W.map.updateRequestLayer.markers[urID].icon;
newSpan = '';
if(nComments !== 0)
{
var classList = iconObj.imageDiv.classList;
elmID = "commentCount_"+urID;
if(addCommentCount)
{
// add a new comment count bubble if the UR doesn't already have one
if(document.getElementById(elmID) === null)
{
newSpan += '<span id="'+elmID+'" style="position:absolute;top:-9px;left:-11px;pointer-events:none;z-index:1">';
// define the comment-count holding span within the span used to hold the empty bubble image, and before the image is
// added to the HTML, to avoid z-indexing issues when adjacent comment count bubbles are overlapped...
newSpan += '<span id="'+elmID+"_inner"+'" style="position:absolute;top:4px;left:11px;font-size:11px;;pointer-events:none"></span>';
newSpan += '<img src="'+uroMarkers[0]+'">';
newSpan += '</span>';
}
}
else
{
// remove comment count bubble from this UR marker if one has previously been
// added and the user has now disabled the option...
if(document.getElementById(elmID) !== null)
{
document.getElementById(elmID).remove();
}
if(document.getElementById(elmID+"_inner") !== null)
{
document.getElementById(elmID+"_inner").remove();
}
}
elmID = "convoMarker_"+urID;
if(useDefaultConvoMarker === false)
{
if(document.getElementById(elmID) === null)
{
var hasMyComments = cmlObj.hasMyComments;
// z-index needs to be set to 1 here so that when a new comment is added to a UR and WME re-renders the native
// conversation marker, the custom marker remains on top...
newSpan += '<span id="'+elmID+'" style="position:absolute;top:-9px;left:18px;pointer-events:none;z-index:1">';
if(hasMyComments) newSpan += '<img src="'+uroMarkers[2]+'">';
else newSpan += '<img src="'+uroMarkers[1]+'">';
newSpan += '</span>';
classList.remove("has-comments");
}
}
else
{
// remove custom conversation marker from this UR if one has previously been
// added and the user has now disabled this option
if(document.getElementById(elmID) !== null)
{
document.getElementById(elmID).remove();
}
if(nComments > 0)
{
// only replace the native marker class if the UR has comments - if we're just clearing the custom
// marker following a master enable switchoff, we don't then want to add native markers to URs which
// didn't have them in the first place...
classList.add("has-comments");
}
}
}
// change main marker if required
touchedByURO = W.map.updateRequestLayer.markers[urID].touchedByURO;
if(customType != -1)
{
if((touchedByURO === undefined) || (touchedByURO === false))
{
customType = uroGetCustomMarkerIdx(customType);
W.map.updateRequestLayer.markers[urID].icon.imageDiv.style.backgroundImage = uroAltMarkers[customType];
W.map.updateRequestLayer.markers[urID].touchedByURO = true;
}
}
else
{
if((touchedByURO === undefined) || (touchedByURO === true))
{
W.map.updateRequestLayer.markers[urID].icon.imageDiv.style.backgroundImage = defaultMarkerURL;
W.map.updateRequestLayer.markers[urID].touchedByURO = false;
}
}
if(newSpan !== '')
{
iconObj.$div.prepend(newSpan);
if(addCommentCount)
{
var styleLeft;
if(nComments < 10) styleLeft = '11px';
else if(nComments < 100) styleLeft = '8px';
else styleLeft = '5px';
elmID = "commentCount_"+urID;
if(document.getElementById(elmID+"_inner") !== null)
{
document.getElementById(elmID+"_inner").innerHTML = nComments;
document.getElementById(elmID+"_inner").style.left = styleLeft;
}
}
}
}
}
}
}
}
else if(markerType == 'mp')
{
divElem = document.getElementById(W.map.problemLayer.id);
if(divElem.childNodes.length > 0)
{
for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
{
cmlObj = uroCustomMarkerList[objIdx];
if(cmlObj.markerType == 'mp')
{
customType = cmlObj.customType;
if((customType >= 100) || (customType == -1))
{
urID = cmlObj.urID;
// change main marker if required
touchedByURO = W.map.problemLayer.markers[urID].touchedByURO;
if(customType != -1)
{
if((touchedByURO === undefined) || (touchedByURO === false))
{
customType = uroGetCustomMarkerIdx(customType);
W.map.problemLayer.markers[urID].icon.imageDiv.style.backgroundImage = uroAltMarkers[customType];
W.map.problemLayer.markers[urID].touchedByURO = true;
}
}
else
{
if((touchedByURO === undefined) || (touchedByURO === true))
{
W.map.problemLayer.markers[urID].icon.imageDiv.style.backgroundImage = defaultMarkerURL;
W.map.problemLayer.markers[urID].touchedByURO = false;
}
}
}
}
}
}
}
}
function uroFilterPlaces()
{
if(uroFilterPreamble() === false) return;
if(uroPlaceSelected === true) return;
if(uroGetCBChecked('_cbDisablePlacesFiltering') === true) return;
var filterCats = [];
for(var i=0; i<W.Config.venues.categories.length; i++)
{
var parentCategory = W.Config.venues.categories[i];
var subCategory;
if(uroGetCBChecked('_cbPlacesFilter-'+parentCategory) === true)
{
filterCats.push(parentCategory);
for(var i1=0; i1<W.Config.venues.subcategories[parentCategory].length; i1++)
{
subCategory = W.Config.venues.subcategories[parentCategory][i1];
filterCats.push(subCategory);
}
}
else
{
for(var i2=0; i2<W.Config.venues.subcategories[parentCategory].length; i2++)
{
subCategory = W.Config.venues.subcategories[parentCategory][i2];
if(uroGetCBChecked('_cbPlacesFilter-'+subCategory) === true)
{
filterCats.push(subCategory);
}
}
}
}
var placeStyle;
var uFP_filterEditedLessThan = uroGetCBChecked('_cbPlaceFilterEditedLessThan');
var uFP_filterEditedMoreThan = uroGetCBChecked('_cbPlaceFilterEditedMoreThan');
var uFP_filterL0 = uroGetCBChecked('_cbHidePlacesL0');
var uFP_filterL1 = uroGetCBChecked('_cbHidePlacesL1');
var uFP_filterL2 = uroGetCBChecked('_cbHidePlacesL2');
var uFP_filterL3 = uroGetCBChecked('_cbHidePlacesL3');
var uFP_filterL4 = uroGetCBChecked('_cbHidePlacesL4');
var uFP_filterL5 = uroGetCBChecked('_cbHidePlacesL5');
var uFP_filterStaff = uroGetCBChecked('_cbHidePlacesStaff');
var uFP_filterAL = uroGetCBChecked('_cbHidePlacesAdLocked');
var uFP_filterOnLockLevel = (uFP_filterL0 || uFP_filterL1 || uFP_filterL2 || uFP_filterL3 || uFP_filterL4 || uFP_filterL5 || uFP_filterStaff);
var uFP_filterNoPhotos = uroGetCBChecked('_cbHideNoPhotoPlaces');
var uFP_filterWithPhotos = uroGetCBChecked('_cbHidePhotoPlaces');
var uFP_filterNoLinks = uroGetCBChecked('_cbHideNoLinkedPlaces');
var uFP_filterWithLinks = uroGetCBChecked('_cbHideLinkedPlaces');
var uFP_filterNoKeyword = uroGetCBChecked('_cbHideKeywordPlaces');
var uFP_filterKeyword = uroGetCBChecked('_cbHideNoKeywordPlaces');
var uFP_filterPrivate = uroGetCBChecked('_cbFilterPrivatePlaces');
var uFP_invertFilters = uroGetCBChecked('_cbInvertPlacesFilter');
var uFP_masterEnable = uroGetCBChecked('_cbMasterEnable');
var uFP_filterAreaPlaces = uroGetCBChecked('_cbHideAreaPlaces');
var uFP_filterPointPlaces = uroGetCBChecked('_cbHidePointPlaces');
var uFP_NameKeyword = document.getElementById('_textKeywordPlace').value.toLowerCase();
var uFP_thresholdMinDays = document.getElementById('_inputFilterPlaceEditMinDays').value;
var uFP_thresholdMaxDays = document.getElementById('_inputFilterPlaceEditMaxDays').value;
for(var v=0; v<W.map.landmarkLayer.features.length; v++)
{
placeStyle = 'visible';
if(uFP_masterEnable === true)
{
var lmObj = W.map.landmarkLayer.features[v];
// when an area place is selected, the drag points for editing the place outline now get added as objects into W.map.landmarkLayer.features,
// however none of these objects have the .model property - we must therefore check each entry in features[] to see if it has .model before
// attempting to filter it...
if(lmObj.model != null)
{
if(lmObj.model.attributes.id < 0)
{
// don't apply filtering to newly-created places - this allows the user to leave their filtering settings unchanged whilst
// adding a new place which, once saved, would then be hidden...
break;
}
if(uFP_filterAreaPlaces)
{
if(lmObj.model.attributes.geometry.id.indexOf('Polygon') !== -1)
{
placeStyle = 'hidden';
}
}
if(uFP_filterPointPlaces)
{
if(lmObj.model.attributes.geometry.id.indexOf('Point') !== -1)
{
placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
if((uFP_filterEditedLessThan) || (uFP_filterEditedMoreThan))
{
var editDate = lmObj.model.attributes.updatedOn;
if(editDate === undefined)
{
// where a place has never been edited since its creation, use the creation date instead...
editDate = lmObj.model.attributes.createdOn;
}
if(editDate != null)
{
var editDaysAgo = uroDateToDays(editDate);
if(uFP_filterEditedLessThan)
{
if(editDaysAgo < uFP_thresholdMinDays)
{
placeStyle = 'hidden';
}
}
if(uFP_filterEditedMoreThan)
{
if(editDaysAgo > uFP_thresholdMaxDays)
{
placeStyle = 'hidden';
}
}
}
}
}
if(placeStyle == 'visible')
{
if(uFP_filterOnLockLevel)
{
var lockLevel = lmObj.model.attributes.lockRank;
if ((uFP_filterL0) && (lockLevel === 0)) placeStyle = 'hidden';
if ((uFP_filterL1) && (lockLevel === 1)) placeStyle = 'hidden';
if ((uFP_filterL2) && (lockLevel === 2)) placeStyle = 'hidden';
if ((uFP_filterL3) && (lockLevel === 3)) placeStyle = 'hidden';
if ((uFP_filterL4) && (lockLevel === 4)) placeStyle = 'hidden';
if ((uFP_filterL5) && (lockLevel === 5)) placeStyle = 'hidden';
if ((uFP_filterStaff) && (lockLevel === 6)) placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
if(uFP_filterAL)
{
if(lmObj.model.attributes.adLocked) placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
if(uFP_filterNoPhotos || uFP_filterWithPhotos)
{
var nPhotos = 0;
for(var loop=0; loop<lmObj.model.attributes.images.length; loop++)
{
if(lmObj.model.attributes.images[loop].attributes.approved) nPhotos++;
}
if((uFP_filterNoPhotos) && (nPhotos === 0)) placeStyle = 'hidden';
if((uFP_filterWithPhotos) && (nPhotos !== 0)) placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
if(uFP_filterNoLinks || uFP_filterWithLinks)
{
var nLinks = lmObj.model.attributes.externalProviderIDs.length;
if((uFP_filterNoLinks) && (nLinks === 0)) placeStyle = 'hidden';
if((uFP_filterWithLinks) && (nLinks !== 0)) placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
if((uFP_filterPrivate === true) && (lmObj.model.attributes.residential === true))
{
placeStyle = 'hidden';
}
else
{
for(var cat=0; cat<filterCats.length; cat++)
{
if(lmObj.model.attributes.categories.contains(filterCats[cat]))
{
placeStyle = 'hidden';
break;
}
}
}
}
if(placeStyle == 'visible')
{
if(uFP_filterNoKeyword || uFP_filterKeyword)
{
var venueName = lmObj.model.attributes.name.toLowerCase();
var noKeywordMatch = true;
if(uFP_NameKeyword === '')
{
noKeywordMatch = (venueName !== '');
}
else
{
noKeywordMatch = (venueName.indexOf(uFP_NameKeyword) === -1);
}
if(!noKeywordMatch && uFP_filterNoKeyword) placeStyle = 'hidden';
if(noKeywordMatch && uFP_filterKeyword) placeStyle = 'hidden';
}
}
}
if(uFP_invertFilters === true)
{
if(placeStyle == 'hidden') placeStyle = 'visible';
else placeStyle = 'hidden';
}
}
var geoID = W.map.landmarkLayer.features[v].geometry.id;
if(document.getElementById(geoID) !== null)
{
document.getElementById(geoID).style.visibility = placeStyle;
}
}
var uFP_filterUneditable = uroGetCBChecked('_cbFilterUneditablePlaceUpdates');
var uFP_filterLockRanked = uroGetCBChecked('_cbFilterLockRankedPlaceUpdates');
var uFP_filterFlagged = uroGetCBChecked("_cbFilterFlaggedPUR");
var uFP_filterNewPlace = uroGetCBChecked("_cbFilterNewPlacePUR");
var uFP_filterUpdatedDetails = uroGetCBChecked("_cbFilterUpdatedDetailsPUR");
var uFP_filterNewPhoto = uroGetCBChecked("_cbFilterNewPhotoPUR");
var uFP_filterMinPURAge = uroGetCBChecked('_cbEnablePURMinAgeFilter');
var uFP_filterMaxPURAge = uroGetCBChecked('_cbEnablePURMaxAgeFilter');
var uFP_invertPURFilters = uroGetCBChecked('_cbInvertPURFilters');
var uFP_filterHighSeverity = uroGetCBChecked('_cbPURFilterHighSeverity');
var uFP_filterMedSeverity = uroGetCBChecked('_cbPURFilterMediumSeverity');
var uFP_filterLowSeverity = uroGetCBChecked('_cbPURFilterLowSeverity');
var uFP_leavePURGeos = uroGetCBChecked('_cbLeavePURGeos');
var uFP_thresholdMinPURDays = uroGetElmValue('_inputPURFilterMinDays');
var uFP_thresholdMaxPURDays = uroGetElmValue('_inputPURFilterMaxDays');
var uFP_isLoggedIn = W.loginManager.isLoggedIn();
var uFP_userRank = W.loginManager.user.rank;
var purAge = null;
for(var pu in W.map.placeUpdatesLayer.markers)
{
if(W.map.placeUpdatesLayer.markers.hasOwnProperty(pu))
{
var puObj = W.map.placeUpdatesLayer.markers[pu];
if(W.map.placeUpdatesLayer.getVisibility() === true)
{
placeStyle = 'visible';
if(uFP_masterEnable === true)
{
if(uFP_masterEnable === true)
{
if(uFP_filterUneditable === true)
{
if(puObj.model.attributes.permissions === 0)
{
placeStyle = 'hidden';
}
if((placeStyle == 'visible') && (uFP_isLoggedIn))
{
if(uFP_userRank < puObj.model.attributes.lockRank)
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (puObj.model.attributes.adLocked))
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterLockRanked === true))
{
if(puObj.model.attributes.lockRank !== 0)
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterFlagged === true))
{
if(puObj.icon.imageDiv.className.indexOf('flag') != -1)
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterNewPlace === true))
{
if(puObj.icon.imageDiv.className.indexOf('add_venue') != -1)
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterUpdatedDetails === true))
{
if((puObj.icon.imageDiv.className.indexOf('update_venue') != -1) || (puObj.icon.imageDiv.className.indexOf('multiple') != -1))
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterNewPhoto === true))
{
if(puObj.icon.imageDiv.className.indexOf('add_image') != -1)
{
placeStyle = 'hidden';
}
}
if(uFP_invertPURFilters === true)
{
if(placeStyle == 'hidden') placeStyle = 'visible';
else placeStyle = 'hidden';
}
if(uFP_filterMinPURAge || uFP_filterMaxPURAge)
{
purAge = uroGetPURAge(puObj.model);
if(uFP_filterMinPURAge === true)
{
if(purAge < uFP_thresholdMinPURDays) placeStyle = 'hidden';
}
if(uFP_filterMaxPURAge === true)
{
if(purAge > uFP_thresholdMaxPURDays) placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
var purSeverity = puObj._getSeverity();
if((uFP_filterHighSeverity) && (purSeverity == "high")) placeStyle = 'hidden';
if((placeStyle == 'visible') && (uFP_filterMedSeverity) && (purSeverity == "medium")) placeStyle = 'hidden';
if((placeStyle == 'visible') && (uFP_filterLowSeverity) && (purSeverity == "low")) placeStyle = 'hidden';
}
}
}
puObj.icon.imageDiv.style.visibility = placeStyle;
if(uFP_leavePURGeos === false)
{
if(puObj.model != null)
{
if(puObj.model.geometry != null)
{
var puGeo = document.getElementById(puObj.model.geometry.id);
if(puGeo !== null)
{
puGeo.style.visibility = placeStyle;
}
}
}
}
}
}
}
}
function uroFilterCameras()
{
if(uroFilterPreamble() === false) return;
var camLayer = document.getElementById(uroRootContainer+'_svgRoot');
if(camLayer === null)
{
if(uroNullCamLayer === false)
{
uroAddLog('caught null camLayer');
uroNullCamLayer = true;
}
return;
}
uroNullCamLayer = false;
if(uroMouseIsDown === false) W.map.camerasLayer.redraw();
if(uroGetCBChecked('_cbMasterEnable') === true)
{
uroUpdateCamEditorList();
var filterNameID = null;
var tbUserName = uroGetElmValue('_textCameraEditor');
var selector = document.getElementById('_selectCameraUserID');
if(selector.selectedIndex > 0)
{
var selUserName = document.getElementById('_selectCameraUserID').selectedOptions[0].innerHTML;
if(selUserName == tbUserName)
{
filterNameID = document.getElementById('_selectCameraUserID').selectedOptions[0].value;
}
}
if(filterNameID === null)
{
var userObj = W.model.users.getByAttributes({userName:tbUserName})[0];
if(userObj !== undefined)
{
filterNameID = userObj.id;
}
}
for (var uroCamObj in W.model.cameras.objects)
{
if(W.model.cameras.objects.hasOwnProperty(uroCamObj))
{
var uroCamUpdater = '';
var uroCamUpdaterRank = -1;
var uroCamCreator = '';
var uroCamCreatorRank = -1;
var uroCam = W.model.cameras.objects[uroCamObj];
var uroCamStyle = 'visible';
if(uroCam.attributes.createdBy !== null)
{
if(W.model.users.objects[uroCam.attributes.createdBy] != null)
{
uroCamCreator = W.model.users.objects[uroCam.attributes.createdBy].userName;
uroCamCreatorRank = W.model.users.objects[uroCam.attributes.createdBy].rank;
}
}
if(uroCam.attributes.updatedBy !== null)
{
if(W.model.users.objects[uroCam.attributes.updatedBy] != null)
{
uroCamUpdater = W.model.users.objects[uroCam.attributes.updatedBy].userName;
uroCamUpdaterRank = W.model.users.objects[uroCam.attributes.updatedBy].rank;
}
}
var uroCamApproved = uroCam.attributes.validated;
var uroCamType = uroCam.attributes.type;
if(filterNameID != null)
{
if(uroGetCBChecked('_cbShowOnlyCamsCreatedBy') === true)
{
if(filterNameID != uroCam.attributes.createdBy) uroCamStyle = 'hidden';
}
if(uroGetCBChecked('_cbShowOnlyCamsEditedBy') === true)
{
if(filterNameID != uroCam.attributes.updatedBy) uroCamStyle = 'hidden';
}
}
if(uroGetCBChecked('_cbShowOnlyMyCams') === true)
{
if((uroUserID != uroCam.attributes.createdBy)&&(uroUserID != uroCam.attributes.updatedBy)) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbShowWorldCams') === false) || (uroGetCBChecked('_cbShowUSACams') === false) || (uroGetCBChecked('_cbShowNonWorldCams') === false))
{
var posWorld = uroCamCreator.indexOf('world_');
var posUSA = uroCamCreator.indexOf('usa_');
if((uroGetCBChecked('_cbShowWorldCams') === false) && (posWorld === 0)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowUSACams') === false) && (posUSA === 0)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowNonWorldCams') === false) && (posWorld !== 0) && (posUSA !== 0)) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbShowApprovedCams') === false) || (uroGetCBChecked('_cbShowNonApprovedCams') === false))
{
if((uroGetCBChecked('_cbShowApprovedCams') === false) && (uroCamApproved === true)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowNonApprovedCams') === false) && (uroCamApproved === false)) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbShowNonApprovedCams') === true) && (uroCamApproved === false))
{
if(((uroGetCBChecked('_cbShowOlderCreatedNonApproved') === true)) && (uroGetCameraAge(uroCam,1) <= uroGetElmValue('_inputCameraMinCreatedDays'))) uroCamStyle = 'hidden';
if(((uroGetCBChecked('_cbShowOlderUpdatedNonApproved') === true)) && (uroGetCameraAge(uroCam,0) <= uroGetElmValue('_inputCameraMinUpdatedDays'))) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbShowSpeedCams') === false) || (uroGetCBChecked('_cbShowRedLightCams') === false) || (uroGetCBChecked('_cbShowDummyCams') === false))
{
if((uroGetCBChecked('_cbShowSpeedCams') === false) && (uroCamType == 2)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowRedLightCams') === false) && (uroCamType == 4)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowDummyCams') === false) && (uroCamType == 3)) uroCamStyle = 'hidden';
}
if(uroGetCBChecked('_cbShowSpeedCams') === true)
{
if((uroGetCBChecked('_cbShowIfNoSpeedSet') === false) && (uroCam.attributes.speed === null)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowIfSpeedSet') === false) && (uroCam.attributes.speed !== null)) uroCamStyle = 'hidden';
}
if(uroGetCBChecked('_cbHideCreatedByMe') === true)
{
if(uroUserID == uroCam.attributes.createdBy) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbHideCreatedByRank0') === true) && (uroCamCreatorRank === 0)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank1') === true) && (uroCamCreatorRank == 1)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank2') === true) && (uroCamCreatorRank == 2)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank3') === true) && (uroCamCreatorRank == 3)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank4') === true) && (uroCamCreatorRank == 4)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank5') === true) && (uroCamCreatorRank == 5)) uroCamStyle = 'hidden';
if(uroGetCBChecked('_cbHideUpdatedByMe') === true)
{
if(uroUserID == uroCam.attributes.updatedBy) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbHideUpdatedByRank0') === true) && (uroCamUpdaterRank === 0)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank1') === true) && (uroCamUpdaterRank == 1)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank2') === true) && (uroCamUpdaterRank == 2)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank3') === true) && (uroCamUpdaterRank == 3)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank4') === true) && (uroCamUpdaterRank == 4)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank5') === true) && (uroCamUpdaterRank == 5)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCWLCams') === true) && (uroIsCamOnWatchList(uroCam.attributes.id) != -1)) uroCamStyle = 'hidden';
var uroCamGeometryID = uroCam.geometry.id;
if(camLayer.getElementById(uroCamGeometryID) !== null)
{
if(uroCamStyle == "hidden")
{
camLayer.getElementById(uroCamGeometryID).remove();
}
}
}
}
}
}
function uroFilterURs_onObjectsChanged()
{
if(uroFilterPreamble())
{
if(uroBackfilling === false)
{
if(uroURDialogIsOpen === false)
{
uroURBackfill();
}
else
{
uroFilterURs();
}
}
}
}
function uroFilterURs_onObjectsAdded()
{
if(uroFilterPreamble())
{
if(uroBackfilling === false)
{
uroURBackfill();
}
}
}
function uroFilterURs_onObjectsRemoved()
{
if(uroFilterPreamble())
{
if(uroBackfilling === false)
{
uroURBackfill();
}
}
}
function uroBackfillQueueObj(lon, lat, blockSize)
{
this.lon = lon;
this.lat = lat;
this.blockSize = blockSize;
}
function uroURBackfill_GetData()
{
if(uroBackfillQueue.length === 0)
{
uroBackfilling = false;
uroFilterURs();
return;
}
var nextBFQueueObj = uroBackfillQueue.shift();
var lon = parseFloat(nextBFQueueObj.lon);
var lat = parseFloat(nextBFQueueObj.lat);
var blockSize = parseFloat(nextBFQueueObj.blockSize);
uroAddLog('Backfill square '+lon+','+lat);
var backfillReq = new XMLHttpRequest();
backfillReq.onreadystatechange = function ()
{
if (backfillReq.readyState == 4)
{
uroAddLog('backfill data request, response '+backfillReq.status+' received');
if (backfillReq.status == 200)
{
var tResp = JSON.parse(backfillReq.responseText);
var urCount = tResp.mapUpdateRequests.objects.length;
uroAddLog(urCount+' URs loaded for backfill processing');
if(urCount == 500)
{
uroAddLog('WARNING - backfill data may have been pre-filtered by server');
}
var backfilled = 0;
for(var i=0; i<urCount; i++)
{
var urID = tResp.mapUpdateRequests.objects[i].id;
if(W.model.mapUpdateRequests.objects[urID] === undefined)
{
var newUR = require('Waze/Feature/Vector/UpdateRequest');
var tUR = new newUR(tResp.mapUpdateRequests.objects[i]);
var tPoint = new OpenLayers.Geometry.Point();
tPoint.x = tResp.mapUpdateRequests.objects[i].geometry.coordinates[0];
tPoint.y = tResp.mapUpdateRequests.objects[i].geometry.coordinates[1];
tPoint.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
tUR.geometry = tPoint;
var tReqBounds = new OpenLayers.Geometry.Polygon();
var tBounds = new OpenLayers.Bounds();
tBounds.left = tPoint.x;
tBounds.right = tPoint.x;
tBounds.top = tPoint.y;
tBounds.bottom = tPoint.y;
tReqBounds.bounds = tBounds;
tUR.requestBounds = tReqBounds;
W.model.mapUpdateRequests.put(tUR);
backfilled++;
}
}
uroAddLog(backfilled+' URs backfilled');
}
uroURBackfill_GetData();
}
};
var tURL = 'https://' + document.location.host;
tURL += Waze.Config.api_base;
tURL += '/Features?language=en&mapUpdateRequestFilter=0';
tURL += '&bbox='+(lon)+','+(lat)+','+(lon + blockSize)+','+(lat + blockSize);
backfillReq.open('GET',tURL,true);
backfillReq.send();
}
function uroURBackfill()
{
if((uroGetCBChecked('_cbURBackfill') === false) || (uroGetCBChecked('_cbMasterEnable') === false))
{
uroFilterURs();
return;
}
var nativeURCount = Object.keys(W.model.mapUpdateRequests.objects).length;
if(nativeURCount < 500)
{
uroAddLog(nativeURCount+' URs loaded natively, no backfilling required');
uroFilterURs();
return;
}
uroAddLog('exactly 500 URs loaded, possible server-side filtering requiring backfill...');
var subSize = 0.1;
var vpWidth = W.map.getExtent().getWidth();
var vpHeight = W.map.getExtent().getHeight();
var vpCentre = W.map.getCenter();
var vpLL = new OpenLayers.LonLat();
var vpUR = new OpenLayers.LonLat();
vpLL.lon = vpCentre.lon - (vpWidth / 2);
vpLL.lat = vpCentre.lat - (vpHeight / 2);
vpUR.lon = vpCentre.lon + (vpWidth / 2);
vpUR.lat = vpCentre.lat + (vpHeight / 2);
vpLL = vpLL.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
vpUR = vpUR.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
vpLL.lon -= (subSize / 2);
vpLL.lat -= (subSize / 2);
vpUR.lon += (subSize / 2);
vpUR.lat += (subSize / 2);
vpLL.lon = +vpLL.lon.toFixed(1);
vpLL.lat = +vpLL.lat.toFixed(1);
vpUR.lon = +vpUR.lon.toFixed(1);
vpUR.lat = +vpUR.lat.toFixed(1);
uroBackfilling = true;
uroBackfillQueue = [];
for(var bfLat = vpLL.lat; bfLat <= vpUR.lat; bfLat += subSize)
{
for(var bfLon = vpLL.lon; bfLon <= vpUR.lon; bfLon += subSize)
{
uroBackfillQueue.push(new uroBackfillQueueObj(bfLon, bfLat, subSize));
}
}
uroURBackfill_GetData();
}
function uroFilterURs()
{
if(uroUserID === -1) return;
// compatibility fix for URComments - based on code supplied by RickZabel
var hasActiveURFilters = false;
if(uroGetCBChecked('_cbMasterEnable') === true)
{
var urTabInputs = document.getElementById('uroCtrlURs').getElementsByTagName('input');
for(var loop = 0; loop < urTabInputs.length; loop++)
{
if(urTabInputs[loop].type == 'checkbox')
{
var ignoreCB = false;
ignoreCB = ignoreCB || (urTabInputs[loop].id == '_cbCaseInsensitive');
ignoreCB = ignoreCB || (urTabInputs[loop].id == '_cbNoFilterForTaggedURs');
if((urTabInputs[loop].checked) && (ignoreCB === false))
{
hasActiveURFilters = true;
break;
}
}
}
}
sessionStorage.UROverview_hasActiveURFilters = hasActiveURFilters;
if(uroFilterPreamble() === false) return;
uroRefreshUpdateRequestSessions();
var selectorResolver = document.getElementById('_selectURResolverID');
var selectorCommentUser = document.getElementById('_selectURUserID');
if(uroGetCBChecked('_cbURResolverIDFilter') === false)
{
while(selectorResolver.options.length > 0)
{
selectorResolver.options.remove(0);
}
}
if(uroGetCBChecked('_cbURUserIDFilter') === false)
{
while(selectorCommentUser.options.length > 0)
{
selectorCommentUser.options.remove(0);
}
}
if(Object.keys(W.model.updateRequestSessions.objects).length === 0)
{
return;
}
var commenterUser = null;
if(uroGetCBChecked('_cbURUserIDFilter') === true)
{
if(selectorCommentUser.options.length === 0)
{
uroUpdateUserList();
}
if(selectorCommentUser.selectedOptions[0] != null)
{
commenterUser = parseInt(selectorCommentUser.selectedOptions[0].value);
}
}
var resolverUser = null;
if(uroGetCBChecked('_cbURResolverIDFilter') === true)
{
if(selectorResolver.options.length === 0)
{
uroUpdateResolverList();
}
if(selectorResolver.selectedOptions[0] != null)
{
resolverUser = parseInt(selectorResolver.selectedOptions[0].value);
}
}
uroCustomMarkerList = [];
var uFURs_masterEnable = uroGetCBChecked('_cbMasterEnable');
var filterOutsideEditableArea = uroGetCBChecked('_cbURFilterOutsideArea');
var filterSolved = uroGetCBChecked('_cbFilterSolved');
var filterUnidentified = uroGetCBChecked('_cbFilterUnidentified');
var filterClosed = uroGetCBChecked('_cbFilterClosedUR');
var filterOpen = uroGetCBChecked('_cbFilterOpenUR');
var filterDescMustBePresent = uroGetCBChecked('_cbURDescriptionMustBePresent');
var filterDescMustBeAbsent = uroGetCBChecked('_cbURDescriptionMustBeAbsent');
var filterKeywordMustBePresent = uroGetCBChecked('_cbEnableKeywordMustBePresent');
var filterKeywordMustBeAbsent = uroGetCBChecked('_cbEnableKeywordMustBeAbsent');
var filterMinURAge = uroGetCBChecked('_cbEnableMinAgeFilter');
var filterMaxURAge = uroGetCBChecked('_cbEnableMaxAgeFilter');
var filterMinComments = uroGetCBChecked('_cbEnableMinCommentsFilter');
var filterMaxComments = uroGetCBChecked('_cbEnableMaxCommentsFilter');
var filterReporterLastCommenter = uroGetCBChecked('_cbHideIfReporterLastCommenter');
var filterReporterNotLastCommenter = uroGetCBChecked('_cbHideIfReporterNotLastCommenter');
var filterHideAnyComments = uroGetCBChecked('_cbHideAnyComments');
var filterHideNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
var filterHideMyComments = uroGetCBChecked('_cbHideMyComments');
var filterIfLastCommenter = uroGetCBChecked('_cbHideIfLastCommenter');
var filterIfNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
var filterCommentMinAge = uroGetCBChecked('_cbEnableCommentAgeFilter2');
var filterCommentMaxAge = uroGetCBChecked('_cbEnableCommentAgeFilter');
var filterUserID = uroGetCBChecked('_cbURUserIDFilter');
var filterMyFollowed = uroGetCBChecked('_cbHideMyFollowed');
var filterMyUnfollowed = uroGetCBChecked('_cbHideMyUnfollowed');
var filterWazeAuto = uroGetCBChecked('_cbFilterWazeAuto');
var filterRoadworks = uroGetCBChecked('_cbFilterRoadworks');
var filterConstruction = uroGetCBChecked('_cbFilterConstruction');
var filterClosure = uroGetCBChecked('_cbFilterClosure');
var filterEvent = uroGetCBChecked('_cbFilterEvent');
var filterNote = uroGetCBChecked('_cbFilterNote');
var filterWSLM = uroGetCBChecked('_cbFilterWSLM');
var filterBOG = uroGetCBChecked('_cbFilterBOG');
var filterDifficult = uroGetCBChecked('_cbFilterDifficult');
var filterIncorrectTurn = uroGetCBChecked('_cbFilterIncorrectTurn');
var filterIncorrectAddress = uroGetCBChecked('_cbFilterIncorrectAddress');
var filterIncorrectRoute = uroGetCBChecked('_cbFilterIncorrectRoute');
var filterMissingRoundabout = uroGetCBChecked('_cbFilterMissingRoundabout');
var filterGeneralError = uroGetCBChecked('_cbFilterGeneralError');
var filterTurnNotAllowed = uroGetCBChecked('_cbFilterTurnNotAllowed');
var filterIncorrectJunction = uroGetCBChecked('_cbFilterIncorrectJunction');
var filterMissingBridgeOverpass = uroGetCBChecked('_cbFilterMissingBridgeOverpass');
var filterWrongDrivingDirection = uroGetCBChecked('_cbFilterWrongDrivingDirection');
var filterMissingExit = uroGetCBChecked('_cbFilterMissingExit');
var filterMissingRoad = uroGetCBChecked('_cbFilterMissingRoad');
var filterMissingLandmark = uroGetCBChecked('_cbFilterMissingLandmark');
var filterNativeSpeedLimit = uroGetCBChecked('_cbFilterSpeedLimits');
var filterBlockedRoad = uroGetCBChecked('_cbFilterBlockedRoad');
var filterUndefined = uroGetCBChecked('_cbFilterUndefined');
var invertURFilters = uroGetCBChecked('_cbInvertURFilter');
var invertURStateFilters = uroGetCBChecked('_cbInvertURStateFilter');
var noFilterTaggedURs = uroGetCBChecked('_cbNoFilterForTaggedURs');
var noFilterURInURL = uroGetCBChecked('_cbNoFilterForURInURL');
var keywordPresent = uroGetElmValue('_textKeywordPresent');
var keywordAbsent = uroGetElmValue('_textKeywordAbsent');
var thresholdMinAge = uroGetElmValue('_inputFilterMinDays');
var thresholdMaxAge = uroGetElmValue('_inputFilterMaxDays');
var thresholdMinComments = uroGetElmValue('_inputFilterMinComments');
var thresholdMaxComments = uroGetElmValue('_inputFilterMaxComments');
var thresholdMaxCommentAge = uroGetElmValue('_inputFilterCommentDays');
var thresholdMinCommentAge = uroGetElmValue('_inputFilterCommentDays2');
var ignoreOtherEditorComments = uroGetCBChecked('_cbIgnoreOtherEditorComments');
var urcFilteringIsActive = false;
var urcCB = document.getElementById('URCommentsFilterEnabled');
if(urcCB !== null)
{
if(urcCB.checked)
{
urcFilteringIsActive = true;
}
}
urcCB = document.getElementById('URCommentUROOnlyMyUR');
if(urcCB !== null)
{
if(urcCB.checked)
{
urcFilteringIsActive = true;
}
}
urcCB = document.getElementById('URCommentUROHideTagged');
if(urcCB !== null)
{
if(urcCB.checked)
{
urcFilteringIsActive = true;
}
}
for (var urobj in W.model.mapUpdateRequests.objects)
{
if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
{
var ureq = W.model.mapUpdateRequests.objects[urobj];
var ureqID = null;
if(ureq.fid === null) ureqID = ureq.attributes.id;
else ureqID = ureq.fid;
var urStyle = 'visible';
var inhibitFiltering = ((ureqID == uroURIDInURL) && (noFilterURInURL));
var hasMyComments = false;
var nComments = 0;
var customType = uroGetCustomType(ureqID, "ur");
if(W.model.updateRequestSessions.objects[ureqID] != null)
{
nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
if((uFURs_masterEnable === false) && (nComments === 0))
{
// when master enable is turned off, we want to make sure that all URs, including ones that were previously hidden, are correctly
// displayed in their native form - i.e. no comment count or custom conversation bubbles. The easiest way to achieve this is to
// force the uroRenderCustomMarkers code to test for the presence of these bubbles on each UR, which we do by setting a non-zero
// comment count for each UR... For URs which genuinely do have no comments we use -1 to indicate that we're not really setting
// a comment count, but that we still need to do something that wouldn't be achieved by using 0.
nComments = -1;
}
}
if((uFURs_masterEnable === true) && (inhibitFiltering === false))
{
var wazeauto_ur = false;
var ukroadworks_ur = false;
var construction_ur = false;
var closure_ur = false;
var event_ur = false;
var note_ur = false;
var wslm_ur = false;
var bog_ur = false;
var difficult_ur = false;
var filterByNotIncludedKeyword = false;
var filterByIncludedKeyword = true;
var desc = '';
if(ureq.attributes.description !== null) desc = ureq.attributes.description.replace(/<\/?[^>]+(>|$)/g, "");
if(customType === 0) ukroadworks_ur = true;
else if(customType === 1) construction_ur = true;
else if(customType === 2) closure_ur = true;
else if(customType === 3) event_ur = true;
else if(customType === 4) note_ur = true;
else if(customType === 5) wslm_ur = true;
else if(customType === 6) bog_ur = true;
else if(customType === 7) difficult_ur = true;
// check UR against editable area...
if(filterOutsideEditableArea === true)
{
if(ureq.canEdit() === false) urStyle = 'hidden';
}
// check UR against current session ignore list...
if(uroIsOnIgnoreList(ureqID)) urStyle = 'hidden';
// state-age filtering
if(urStyle == 'visible')
{
// check against closed/not identified filtering if enabled...
if(filterSolved === true)
{
if(ureq.attributes.resolution === 0) urStyle = 'hidden';
}
if(filterUnidentified === true)
{
if(ureq.attributes.resolution == 1) urStyle = 'hidden';
}
if((ureq.attributes.resolvedOn !== null) && (filterClosed === true))
{
urStyle = 'hidden';
}
if((ureq.attributes.resolvedOn === null) && (filterOpen === true))
{
urStyle = 'hidden';
}
if(urStyle == 'visible')
{
// check UR against keyword filtering if enabled...
if(filterDescMustBePresent === true)
{
if(desc === '') urStyle = 'hidden';
}
if(filterDescMustBeAbsent === true)
{
if(desc !== '') urStyle = 'hidden';
}
if(filterKeywordMustBePresent === true)
{
var keywordIsPresentInDesc = uroKeywordPresent(desc,keywordPresent);
filterByIncludedKeyword &= (!keywordIsPresentInDesc);
}
if(filterKeywordMustBeAbsent === true)
{
var keywordIsAbsentInDesc = uroKeywordPresent(desc,keywordAbsent);
filterByNotIncludedKeyword |= keywordIsAbsentInDesc;
}
}
if(urStyle == 'visible')
{
// do age-based filtering if enabled
if(filterMinURAge === true)
{
if(uroGetURAge(ureq,0,false) < thresholdMinAge) urStyle = 'hidden';
}
if(filterMaxURAge === true)
{
if(uroGetURAge(ureq,0,false) > thresholdMaxAge) urStyle = 'hidden';
}
}
if(urStyle == 'visible')
{
if(resolverUser !== null)
{
if(ureq.attributes.resolvedBy != resolverUser) urStyle = 'hidden';
}
}
if(urStyle == 'visible')
{
// do comments/following filtering
if(W.model.updateRequestSessions.objects[ureqID] != null)
{
nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
var commentDaysOld = -1;
if(filterMinComments === true)
{
if(nComments < thresholdMinComments) urStyle = 'hidden';
}
if(filterMaxComments === true)
{
if(nComments > thresholdMaxComments) urStyle = 'hidden';
}
if(nComments > 0)
{
var reporterIsLastCommenter = false;
if(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1].userID == -1) reporterIsLastCommenter = true;
if(filterReporterLastCommenter === true)
{
if(reporterIsLastCommenter === true) urStyle = 'hidden';
}
else if(filterReporterNotLastCommenter === true)
{
if(reporterIsLastCommenter === false) urStyle = 'hidden';
}
hasMyComments = uroURHasMyComments(ureqID);
if(hasMyComments === false)
{
if(filterHideAnyComments === true) urStyle = 'hidden';
if(filterHideNotLastCommenter === true) urStyle = 'hidden';
}
else
{
if(filterHideMyComments === true) urStyle = 'hidden';
var userIsLastCommenter = false;
if(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1].userID == uroUserID) userIsLastCommenter = true;
if(filterIfLastCommenter === true)
{
if(userIsLastCommenter === true) urStyle = 'hidden';
}
else if(filterIfNotLastCommenter === true)
{
if(userIsLastCommenter === false) urStyle = 'hidden';
}
}
var cidx;
if(ignoreOtherEditorComments === false)
{
commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1]);
}
else
{
for(cidx=0; cidx<nComments; cidx++)
{
var cObj = W.model.updateRequestSessions.objects[ureqID].comments[cidx];
if((cObj.userID == uroUserID) || (cObj.userID == -1))
{
commentDaysOld = uroGetCommentAge(cObj);
}
}
}
if((filterCommentMinAge === true) && (commentDaysOld != -1))
{
if(thresholdMinCommentAge > commentDaysOld) urStyle = 'hidden';
}
if((filterCommentMaxAge === true) && (commentDaysOld != -1))
{
if(thresholdMaxCommentAge < commentDaysOld) urStyle = 'hidden';
}
if((commenterUser !== null) && (urStyle != 'hidden'))
{
urStyle = 'hidden';
for(cidx=0; cidx<nComments; cidx++)
{
if(W.model.updateRequestSessions.objects[ureqID].comments[cidx].userID == commenterUser)
{
urStyle = 'visible';
break;
}
}
}
var commentText = '';
for(cidx=0; cidx<nComments; cidx++)
{
commentText += W.model.updateRequestSessions.objects[ureqID].comments[cidx].text;
}
if(filterKeywordMustBePresent === true)
{
var keywordIsPresentInComments = uroKeywordPresent(commentText,keywordPresent);
filterByIncludedKeyword &= (!keywordIsPresentInComments);
}
if(filterKeywordMustBeAbsent === true)
{
var keywordIsAbsentInComments = uroKeywordPresent(commentText,keywordAbsent);
filterByNotIncludedKeyword |= keywordIsAbsentInComments;
}
}
else
{
if(filterUserID === true)
{
urStyle = 'hidden';
}
}
filterByNotIncludedKeyword &= filterKeywordMustBeAbsent;
filterByIncludedKeyword &= filterKeywordMustBePresent;
if(filterByNotIncludedKeyword || filterByIncludedKeyword)
{
urStyle = 'hidden';
}
if(W.model.updateRequestSessions.objects[ureqID].isFollowing === true)
{
if(filterMyFollowed === true) urStyle = 'hidden';
}
else
{
if(filterMyUnfollowed === true) urStyle = 'hidden';
}
}
}
if(invertURStateFilters === true)
{
if(urStyle == 'hidden') urStyle = 'visible';
else urStyle = 'hidden';
}
}
// type filtering
if(urStyle == 'visible')
{
// Test for Waze automatic URs before any others - these always (?) get inserted as General Error URs,
// so we can't filter them by type...
if(desc.indexOf('Waze Automatic:') != -1)
{
wazeauto_ur = true;
}
if(wazeauto_ur === true)
{
if(filterWazeAuto === true) urStyle = 'hidden';
}
else if(ukroadworks_ur === true)
{
if(filterRoadworks === true) urStyle = 'hidden';
}
else if(construction_ur === true)
{
if(filterConstruction === true) urStyle = 'hidden';
}
else if(closure_ur === true)
{
if(filterClosure === true) urStyle = 'hidden';
}
else if(event_ur === true)
{
if(filterEvent === true) urStyle = 'hidden';
}
else if(note_ur === true)
{
if(filterNote === true) urStyle = 'hidden';
}
else if(wslm_ur === true)
{
if(filterWSLM === true) urStyle = 'hidden';
}
else if(bog_ur === true)
{
if(filterBOG === true) urStyle = 'hidden';
}
else if(difficult_ur === true)
{
if(filterDifficult === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 6)
{
if(filterIncorrectTurn === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 7)
{
if (filterIncorrectAddress === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 8)
{
if(filterIncorrectRoute === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 9)
{
if(filterMissingRoundabout === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 10)
{
if(filterGeneralError === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 11)
{
if(filterTurnNotAllowed === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 12)
{
if(filterIncorrectJunction === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 13)
{
if(filterMissingBridgeOverpass === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 14)
{
if(filterWrongDrivingDirection === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 15)
{
if(filterMissingExit === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 16)
{
if(filterMissingRoad === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 18)
{
if(filterMissingLandmark === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 19)
{
if(filterBlockedRoad === true) urStyle = 'hidden';
}
else if(ureq.attributes.type == 23)
{
if(filterNativeSpeedLimit === true) urStyle = 'hidden';
}
else if(filterUndefined === true) urStyle = 'hidden';
if(invertURFilters === true)
{
if(urStyle == 'hidden') urStyle = 'visible';
else urStyle = 'hidden';
}
}
// stage-age filtering override for tagged URs
if(noFilterTaggedURs === true)
{
if(ukroadworks_ur === true)
{
if(filterRoadworks === false) urStyle = 'visible';
}
else if(construction_ur === true)
{
if(filterConstruction === false) urStyle = 'visible';
}
else if(closure_ur === true)
{
if(filterClosure === false) urStyle = 'visible';
}
else if(event_ur === true)
{
if(filterEvent === false) urStyle = 'visible';
}
else if(note_ur === true)
{
if(filterNote === false) urStyle = 'visible';
}
else if(wslm_ur === true)
{
if(filterWSLM === false) urStyle = 'visible';
}
}
}
// only touch marker visibility if we've got active filter settings, or if URComments is not
// doing any filtering of its own
if((hasActiveURFilters === true) || (urcFilteringIsActive === false) || (uFURs_masterEnable === false))
{
W.map.updateRequestLayer.markers[urobj].icon.imageDiv.style.visibility = urStyle;
}
if(urStyle != 'hidden')
{
uroAddCustomMarkers(ureqID,'ur',customType, hasMyComments, nComments);
}
}
}
uroRenderCustomMarkers('ur');
}
function uroFilterProblems()
{
if(uroFilterPreamble() === false) return;
var selector;
if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === false) && (uroGetCBChecked('_cbMPClosedUserIDFilter') === false))
{
selector = document.getElementById('_selectMPUserID');
while(selector.options.length > 0)
{
selector.options.remove(0);
}
}
var solverUser = null;
if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === true) || (uroGetCBChecked('_cbMPClosedUserIDFilter') === true))
{
selector = document.getElementById('_selectMPUserID');
if(selector.options.length === 0)
{
uroUpdateMPSolverList();
}
if(selector.selectedOptions[0] != null)
{
solverUser = parseInt(selector.selectedOptions[0].value);
}
}
var urobj;
var problem;
var problemStyle;
var problem_marker_img;
var uFP_masterEnable = uroGetCBChecked('_cbMasterEnable');
var filter_OutsideEditableArea = uroGetCBChecked('_cbMPFilterOutsideArea');
var filter_Solved = uroGetCBChecked('_cbMPFilterSolved');
var filter_Unidentified = uroGetCBChecked('_cbMPFilterUnidentified');
var filter_Closed = uroGetCBChecked('_cbMPFilterClosed');
var filter_NotClosedUserID = uroGetCBChecked('_cbMPNotClosedUserIDFilter');
var filter_ClosedUserID = uroGetCBChecked('_cbMPClosedUserIDFilter');
var filter_Reopened = uroGetCBChecked('_cbMPFilterReopenedProblem');
var filter_LowSeverity = uroGetCBChecked('_cbMPFilterLowSeverity');
var filter_MediumSeverity = uroGetCBChecked('_cbMPFilterMediumSeverity');
var filter_HighSeverity = uroGetCBChecked('_cbMPFilterHighSeverity');
var filter_TurnProblems = uroGetCBChecked('_cbMPFilterTurnProblem');
var filter_Type101 = uroGetCBChecked('_cbMPFilterDrivingDirectionMismatch');
var filter_Type102 = uroGetCBChecked('_cbMPFilterMissingJunction');
var filter_Type103 = uroGetCBChecked('_cbMPFilterMissingRoad');
var filter_Type104 = uroGetCBChecked('_cbMPFilterCrossroadsJunctionMissing');
var filter_Type105 = uroGetCBChecked('_cbMPFilterRoadTypeMismatch');
var filter_Type106 = uroGetCBChecked('_cbMPFilterRestrictedTurn');
var filter_Type200 = uroGetCBChecked('_cbMPFilterTurnProblem');
var filter_Type300 = uroGetCBChecked('_cbMPFilterRoadClosureProblem');
var filter_TypeUnknown = uroGetCBChecked('_cbMPFilterUnknownProblem');
var filter_TaggedElgin = uroGetCBChecked('_cbFilterElgin');
var filter_TaggedTrafficCast = uroGetCBChecked('_cbFilterTrafficCast');
var filter_TaggedTrafficMaster = uroGetCBChecked('_cbFilterTrafficMaster');
var filter_TaggedCaltrans = uroGetCBChecked('_cbFilterCaltrans');
var filter_TaggedTFL = uroGetCBChecked('_cbFilterTFL');
var filter_Invert = uroGetCBChecked('_cbInvertMPFilter');
for (urobj in W.model.problems.objects)
{
if(W.model.problems.objects.hasOwnProperty(urobj))
{
problem = W.model.problems.objects[urobj];
problemStyle = 'visible';
var ureqID = null;
var customType = null;
if(uFP_masterEnable === true)
{
var elgin_mp = false;
var trafficcast_mp = false;
var trafficmaster_mp = false;
var caltrans_mp = false;
var tfl_mp = false;
ureqID = problem.attributes.id;
customType = uroGetCustomType(ureqID, "mp");
if(customType === 100) elgin_mp = true;
else if(customType === 101) trafficcast_mp = true;
else if(customType === 102) trafficmaster_mp = true;
else if(customType === 103) caltrans_mp = true;
else if(customType === 104) tfl_mp = true;
// check problem against current session ignore list...
if(uroIsOnIgnoreList(ureqID)) problemStyle = 'hidden';
if(filter_OutsideEditableArea === true)
{
if(problem.canEdit() === false)
{
problemStyle = 'hidden';
}
}
// check against closed/not identified filtering if enabled...
problem_marker_img = '';
if(problem.geometry.id !== null)
{
if(document.getElementById(problem.geometry.id) !== null)
{
problem_marker_img = document.getElementById(problem.geometry.id).href.baseVal;
if(filter_Solved === true)
{
if(problem_marker_img.indexOf('_solved') != -1) problemStyle = 'hidden';
}
if(filter_Unidentified === true)
{
if(problem_marker_img.indexOf('_rejected') != -1) problemStyle = 'hidden';
}
}
}
if(filter_Closed === true)
{
if(problem.attributes.open === false)
{
problemStyle = 'hidden';
}
}
if(problemStyle == 'visible')
{
if(solverUser !== null)
{
if((filter_NotClosedUserID === true) && (problem.attributes.resolvedBy == solverUser)) problemStyle = 'hidden';
if((filter_ClosedUserID === true) && (problem.attributes.resolvedBy != solverUser)) problemStyle = 'hidden';
}
}
if(problemStyle == 'visible')
{
var problemType = null;
if(uroDOMHasTurnProblems)
{
problemType = problem.attributes.problemType;
}
else
{
problemType = problem.attributes.subType;
}
if(elgin_mp === true)
{
if(filter_TaggedElgin === true) problemStyle = 'hidden';
}
else if(trafficcast_mp === true)
{
if(filter_TaggedTrafficCast === true) problemStyle = 'hidden';
}
else if(trafficmaster_mp === true)
{
if(filter_TaggedTrafficMaster === true) problemStyle = 'hidden';
}
else if(caltrans_mp === true)
{
if(filter_TaggedCaltrans === true) problemStyle = 'hidden';
}
else if(tfl_mp === true)
{
if(filter_TaggedTFL === true) problemStyle = 'hidden';
}
else if(problemType == 101)
{
if(filter_Type101 === true) problemStyle = 'hidden';
}
else if(problemType == 102)
{
if(filter_Type102 === true) problemStyle = 'hidden';
}
else if(problemType == 103)
{
if(filter_Type103 === true) problemStyle = 'hidden';
}
else if(problemType == 104)
{
if(filter_Type104 === true) problemStyle = 'hidden';
}
else if(problemType == 105)
{
if(filter_Type105 === true) problemStyle = 'hidden';
}
else if(problemType == 106)
{
if(filter_Type106 === true) problemStyle = 'hidden';
}
else if(problemType == 200)
{
if(filter_Type200 === true) problemStyle = 'hidden';
}
else if(problemType == 300)
{
if(filter_Type300 === true) problemStyle = 'hidden';
}
else if(filter_TypeUnknown === true) problemStyle = 'hidden';
if(filter_Reopened === true)
{
if((problem.attributes.open === true) && (problem.attributes.resolvedOn !== null))
{
problemStyle = 'hidden';
}
}
if(filter_Invert === true)
{
if(problemStyle == 'hidden') problemStyle = 'visible';
else problemStyle = 'hidden';
}
if(problem.attributes.weight <= 3)
{
if(filter_LowSeverity === true) problemStyle = 'hidden';
}
else if(problem.attributes.weight <= 7)
{
if(filter_MediumSeverity === true) problemStyle = 'hidden';
}
else if(filter_HighSeverity === true) problemStyle = 'hidden';
}
}
W.map.problemLayer.markers[urobj].icon.imageDiv.style.visibility = problemStyle;
if((problemStyle != 'hidden') && (ureqID !== null) && (customType !== null))
{
uroAddCustomMarkers(ureqID,'mp',customType, false, 0);
}
}
}
if(uroDOMHasTurnProblems)
{
for (urobj in W.model.turnProblems.objects)
{
if(W.model.turnProblems.objects.hasOwnProperty(urobj))
{
problem = W.model.turnProblems.objects[urobj];
problemStyle = 'visible';
if(uFP_masterEnable === true)
{
// check problem against current session ignore list...
if(uroIsOnIgnoreList(problem.attributes.id)) problemStyle = 'hidden';
// check against closed/not identified filtering if enabled...
problem_marker_img = '';
if(problem.geometry.id !== null)
{
if(document.getElementById(problem.geometry.id) !== null)
{
problem_marker_img = document.getElementById(problem.geometry.id).href.baseVal;
if(filter_Solved === true)
{
if(problem_marker_img.indexOf('_solved') != -1) problemStyle = 'hidden';
}
if(filter_Unidentified === true)
{
if(problem_marker_img.indexOf('_rejected') != -1) problemStyle = 'hidden';
}
}
}
if(filter_Closed === true)
{
if(problem.attributes.open === false)
{
problemStyle = 'hidden';
}
}
if(problemStyle == 'visible')
{
if(filter_TurnProblems === true) problemStyle = 'hidden';
if(filter_Reopened === true)
{
if((problem.attributes.open === true) && (problem.attributes.resolvedOn !== null))
{
problemStyle = 'hidden';
}
}
if(filter_Invert === true)
{
if(problemStyle == 'hidden') problemStyle = 'visible';
else problemStyle = 'hidden';
}
}
}
W.map.problemLayer.markers[urobj].icon.imageDiv.style.visibility = problemStyle;
}
}
}
uroRenderCustomMarkers('mp');
}
function uroToHex(decValue,digits)
{
var modifier = 1;
for(var i=0; i<digits; i++)
{
modifier *= 16;
}
decValue = parseInt(decValue);
decValue += modifier;
var retval = decValue.toString(16);
retval = retval.substr(-digits);
retval = retval.toUpperCase();
return retval;
}
function uroFilterPreamble()
{
var mapviewport = document.getElementsByClassName("olMapViewport")[0];
if(mapviewport === null)
{
if(uroNullMapViewport === false)
{
uroAddLog('caught null mapviewport');
uroNullMapViewport = true;
}
return false;
}
var uiElms = document.getElementById('uroCtrlMisc');
if(uiElms == null)
{
uroAddLog('caught missing UI');
return false;
}
if(uiElms.innerHTML.length === 0)
{
uroAddLog('caught empty UI');
return false;
}
uroNullMapViewport = false;
return true;
}
function uroFilterItems_URTabClick()
{
uroFilterURs();
}
function uroFilterItems_MPTabClick()
{
uroFilterProblems();
}
function uroFilterItems_PlacesTabClick()
{
uroFilterPlaces();
}
function uroFilterItems_CamerasTabClick()
{
uroFilterCameras();
}
function uroFilterItems_MiscTabClick()
{
uroFilterItems();
}
function uroFilterItems_MasterEnableClick()
{
if(uroGetCBChecked('_cbMasterEnable') === false)
{
uroHidePopup();
}
uroFilterItems();
}
function uroScaleTheScaleBar()
{
// adjust the scale bar to more accurately reflect true distances at all latitudes
var currLat = W.map.getCenter().transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326")).lat;
if((currLat < 85) && (currLat > -85))
{
var cosLat = Math.cos((currLat * Math.PI) / 180);
var scaleElm;
var elmWidth;
scaleElm = document.getElementsByClassName('olControlScaleLineTop')[0];
if(scaleElm.innerHTML.indexOf(' ') !== -1)
{
elmWidth = Math.round((scaleElm.clientWidth + 2) / cosLat);
scaleElm.innerHTML = scaleElm.innerHTML.replace(' ','');
scaleElm.style.width = elmWidth + 'px';
}
scaleElm = document.getElementsByClassName('olControlScaleLineBottom')[0];
if(scaleElm.innerHTML.indexOf(' ') !== -1)
{
elmWidth = Math.round((scaleElm.clientWidth + 2) / cosLat);
scaleElm.innerHTML = scaleElm.innerHTML.replace(' ','');
scaleElm.style.width = elmWidth + 'px';
}
}
}
function uroFilterItems()
{
uroScaleTheScaleBar();
uroFilterProblems();
uroFilterPlaces();
uroFilterCameras();
uroFilterURs();
}
function uroFilterItemsOnMove()
{
W.map.events.unregister('mousemove',null,uroFilterItemsOnMove);
uroFilterItems();
}
function uroDeleteObject()
{
uroAddLog('delete camera ID '+uroShownFID);
if(W.model.cameras.objects[uroShownFID] === null)
{
uroAddLog('camera object not found...');
return false;
}
uroRemoveCamFromWatchList();
var actionObj = require('Waze/Action/DeleteObject');
var deleteAction = new actionObj(W.model.cameras.objects[uroShownFID], null);
W.model.actionManager.add(deleteAction);
uroExitPopup();
uroHidePopup();
return false;
}
function uroGetUserNameAndRank(userID)
{
var userName;
var userLevel;
if(W.model.users.objects[userID] != null)
{
userName = W.model.users.objects[userID].userName;
if(userName === undefined)
{
userName = userID;
}
userLevel = W.model.users.objects[userID].rank + 1;
}
else
{
userName = userID;
userLevel = '?';
}
return userName + ' (' + userLevel + ')';
}
function uroCheckCommentsForTag(idSrc, customText)
{
var ursObj = W.model.updateRequestSessions.objects[idSrc];
if(typeof(ursObj) == 'undefined') return -1;
if(ursObj.comments.length === 0) return -1;
for(var idx=ursObj.comments.length-1; idx>=0; idx--)
{
for(var tag=0; tag<uroCustomURTags.length; tag++)
{
var keyword = uroCustomURTags[tag];
if(ursObj.comments[idx].text.indexOf(keyword) != -1)
{
return tag;
}
}
if(customText !== '')
{
if(ursObj.comments[idx].text.toLowerCase().indexOf(customText) != -1)
{
return 99;
}
}
}
return -1;
}
function uroGetCustomMarkerIdx(customType)
{
if(customType === 0) return 1; // ROADWORKS
if(customType === 1) return 1; // CONSTRUCTION
if(customType === 2) return 0; // CLOSURE
if(customType === 3) return 4; // EVENT
if(customType === 4) return 3; // NOTE
if(customType === 5) return 5; // WSLM
if(customType === 6) return 11; // BOG
if(customType === 7) return 12; // DIFFICULT
if(customType === 98) return 5; // Native speed limit URs
if(customType === 99) return 2; // custom text
if(customType === 100) return 6; // ELGIN
if(customType === 101) return 7; // TRAFFICCAST
if(customType === 102) return 8; // TRAFFICMASTER
if(customType === 103) return 9; // CALTRANS
if(customType === 104) return 10; // TFL
return -1;
}
function uroGetCustomType(idSrc, markerType)
{
var desc = '';
var customText = '';
if(uroGetCBChecked('_cbCustomKeywordMarkers')) customText = document.getElementById('_textCustomKeyword').value.toLowerCase();
if(markerType == "ur")
{
var ureq = W.model.mapUpdateRequests.objects[idSrc];
// early test for native speed limit URs
if(ureq.attributes.type == 23) return 98;
// UR objects always have a .description attribute, which is set to null if empty...
if(ureq.attributes.description !== null)
{
desc = ureq.attributes.description;
}
}
else if(markerType == "mp")
{
var mp = W.model.problems.objects[idSrc];
// ...whereas MP objects with a blank description don't even get the attribute
if(mp.attributes.description != null)
{
desc = mp.attributes.description;
}
}
if(desc !== '')
{
if(markerType == 'ur')
{
for(var tag=0; tag<uroCustomURTags.length; tag++)
{
var keyword = uroCustomURTags[tag];
if(desc.indexOf(keyword) != -1)
{
return tag;
}
}
}
if((uroGetCBChecked('_cbCustomKeywordMarkers')) && (customText !== ''))
{
if(desc.toLowerCase().indexOf(customText) != -1) return 99;
}
if(markerType == 'mp')
{
if(desc.indexOf('[Elgin]') != -1) return 100;
if(desc.indexOf('[TrafficCast]') != -1) return 101;
if(desc.indexOf('[TM]') != -1) return 102;
if(desc.indexOf('[Caltrans]') != -1) return 103;
if(desc.indexOf('[TfL Open Data]') != -1) return 104;
}
}
if(markerType == "ur")
{
return uroCheckCommentsForTag(idSrc, customText);
}
return -1;
}
function uroFormatRestriction(restObj)
{
var retval = '<tr>';
retval += '<td style="text-align:center;">';
if((restObj.days & 1) == 1) retval += 'S';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 2) == 2) retval += 'M';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 4) == 4) retval += 'T';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 8) == 8) retval += 'W';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 16) == 16) retval += 'T';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 32) == 32) retval += 'F';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 64) == 64) retval += 'S';
else retval += '-';
retval += '</td><td>';
if(restObj.fromDate === null) retval += 'All dates';
else retval += restObj.fromDate+' to '+restObj.toDate;
retval += '</td><td>';
if(restObj.allDay === true) retval += 'All day';
else retval += restObj.fromTime+' to '+restObj.toTime;
retval += '</td><td>';
if(restObj.allVehicleTypes == restObj.vehicleTypes) retval += 'All vehicles';
else retval += 'Some vehicles';
retval += '</td><td>';
if(restObj.description !== null)
{
var desc = restObj.description.replace(/<\/?[^>]+(>|$)/g, "");
desc = uroClickify(desc);
retval += desc;
}
retval += '</td></tr>';
return retval;
}
function uroHidePopup()
{
if(uroPopupShown)
{
uroDiv.style.visibility = 'hidden';
uroPopupShown = false;
uroPopupTimer = -2;
uroShownFID = -1;
}
uroPopupSuppressed = false;
}
function uroSuppressPopup()
{
uroDiv.style.visibility = 'hidden';
window.getSelection().removeAllRanges();
uroPopupSuppressed = true;
}
function uroRecentreSessionOnUR()
{
W.map.updateRequestLayer.markers[uroShownFID].icon.imageDiv.click();
W.map.moveTo(W.map.updateRequestLayer.markers[uroShownFID].lonlat, 5);
uroHidePopup();
return false;
}
function uroRecentreSessionOnMP()
{
W.map.problemLayer.markers[uroShownFID].icon.imageDiv.click();
W.map.moveTo(W.map.problemLayer.markers[uroShownFID].lonlat, 5);
uroHidePopup();
return false;
}
function uroRecentreSessionOnPUR()
{
W.map.placeUpdatesLayer.markers[uroShownFID].icon.imageDiv.click();
W.map.moveTo(W.map.placeUpdatesLayer.markers[uroShownFID].lonlat, 5);
uroHidePopup();
return false;
}
function uroRecentreSessionOnVenueNavPoint()
{
W.map.moveTo(uroGetVenueNavPoint(uroShownFID), 5);
uroHidePopup();
return false;
}
function uroGetDateTimeString(ts)
{
var tDateObj = new Date(ts);
var dateLocale;
var timeLocale;
if(uroGetCBChecked('_cbDateFmtDDMMYY')) dateLocale = 'en-gb';
if(uroGetCBChecked('_cbDateFmtMMDDYY')) dateLocale = 'en-us';
if(uroGetCBChecked('_cbDateFmtYYMMDD')) dateLocale = 'ja';
if(uroGetCBChecked('_cbTimeFmt24H')) timeLocale = 'en-gb';
if(uroGetCBChecked('_cbTimeFmt12H')) timeLocale = 'en-us';
return tDateObj.toLocaleDateString(dateLocale) + ' ' + tDateObj.toLocaleTimeString(timeLocale);
}
function uroParsePxString(pxString)
{
return parseInt(pxString.split("px")[0]);
}
function uroStackListObj(fid,x,y)
{
this.fid = fid;
this.x = uroTypeCast(x);
this.y = uroTypeCast(y);
}
function uroRestackMarkers()
{
if(uroStackList.length === 0) return;
var markerCollection = null;
if(uroStackType == 1) markerCollection = W.map.updateRequestLayer.markers;
else if(uroStackType == 2) markerCollection = W.map.problemLayer.markers;
else if(uroStackType == 3) markerCollection = W.map.placeUpdatesLayer.markers;
if(markerCollection !== null)
{
uroAddLog('restacking markers...');
// strip off the .realX/realY attributes from any UR object we've previously added it to, to allow
// the native recentering to work again...
for(var marker in markerCollection)
{
if(markerCollection.hasOwnProperty(marker))
{
var testMarkerObj = markerCollection[marker];
if(testMarkerObj.model.attributes.geometry.realX != null)
{
testMarkerObj.model.attributes.geometry.x = testMarkerObj.model.attributes.geometry.realX;
testMarkerObj.model.attributes.geometry.y = testMarkerObj.model.attributes.geometry.realY;
delete(testMarkerObj.model.attributes.geometry.realX);
delete(testMarkerObj.model.attributes.geometry.realY);
}
}
}
// now restack any markers that were repositioned...
for(var idx=0; idx<uroStackList.length; idx++)
{
var orig_x = uroStackList[idx].x + 'px';
var orig_y = uroStackList[idx].y + 'px';
var fid = uroStackList[idx].fid;
if(markerCollection[fid] != null)
{
markerCollection[fid].icon.imageDiv.style.left = orig_x;
markerCollection[fid].icon.imageDiv.style.top = orig_y;
}
}
uroStackList = [];
uroUnstackedMasterID = null;
uroStackType = null;
}
}
function uroIsIDAlreadyUnstacked(idSrc)
{
if(uroStackList.length === 0) return false;
for(var idx=0; idx<uroStackList.length; idx++)
{
if(uroStackList[idx].fid == idSrc) return true;
}
return false;
}
function uroCheckStacking(stackType, masterID, unstackedX, unstackedY)
{
if(uroIsIDAlreadyUnstacked(masterID) === true) return;
if(uroStackType !== null) return;
if(uroPopupDwellTimer > 0) return;
uroAddLog('checking for marker stack, masterID: '+masterID+', stackType: '+stackType);
var stackList = [];
stackList.push(masterID);
var threshSquared = uroGetElmValue('_inputUnstackSensitivity');
threshSquared *= threshSquared;
var markerCollection = null;
var marker;
if(stackType == 1) markerCollection = W.map.updateRequestLayer.markers;
else if(stackType == 2) markerCollection = W.map.problemLayer.markers;
else if(stackType == 3) markerCollection = W.map.placeUpdatesLayer.markers;
var offset = 1000000000;
if(markerCollection !== null)
{
for(marker in markerCollection)
{
if(markerCollection.hasOwnProperty(marker))
{
var testMarkerObj = markerCollection[marker];
var includeInStack = (testMarkerObj.icon.imageDiv.style.visibility != 'hidden');
var suppressClosed = (testMarkerObj.icon.imageDiv.classList.contains("recently-closed") & (W.map.updateRequestLayer.showHidden === false));
// if multiple markers are stacked exactly on top of one another, WME will always open up the one which it would have rendered on the
// top of the stack in the absence of any URO+ filtering, regardless of which UR pin actually receives the click event. To prevent
// this, we give each pin in the stack a unique set of false coordinates, storing the original coordinates in newly created
// properties so they can be restored later on
//
// originally this fix changed the x coordinate for each UR in the stack to be a unique value that ought to have been well out of the range
// of any real coordinate and therefore unable to clash with any UR marker coordinates that weren't in the current stack. However it now
// appears this could then cause WME to think a completely different UR was being opened - possibly as a result of some change in coordinate
// handling allowing out of range values to be wrapped around back into the normal range? As a workaround for this new WME behaviour, both the
// x and y coordinates are now set to valid values somewhere in the North Atlantic Ocean - the likelihood of there being any real URs in this
// area is so vanishingly small as to be not worth worrying about...
if(testMarkerObj.model.attributes.geometry.realX === undefined)
{
testMarkerObj.model.attributes.geometry.realX = testMarkerObj.model.attributes.geometry.x;
testMarkerObj.model.attributes.geometry.x += offset;
testMarkerObj.model.attributes.geometry.realY = testMarkerObj.model.attributes.geometry.y;
testMarkerObj.model.attributes.geometry.y += offset;
offset += 1000;
}
if((includeInStack) && (!suppressClosed))
{
if(testMarkerObj.id != masterID)
{
var xdiff = unstackedX - uroParsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.left);
var ydiff = unstackedY - uroParsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.top);
var distSquared = ((xdiff * xdiff) + (ydiff * ydiff));
if(distSquared < threshSquared)
{
stackList.push(testMarkerObj.id);
}
}
}
}
}
}
// as it's the fiddling with .geometry.x which seems to inhibit the autocentering behaviour when a UR/MP marker is clicked, we need to
// allow this to occur even if unstacking isn't required at this zoom level. To then reenable recentering when clicking on the crosshairs
// or feed entry, we need to reinstate the correct .x value once the marker is no longer being highlighted, which means we pretty much need
// to do all of the unstacking *except* for actually unstacking the stack and hiding the other markers...
var inhibitUnstacking = (W.map.getZoom() < uroGetElmValue('_inputUnstackZoomLevel'));
// also inhibit unstacking if there's only a single marker in the list - this will be true if we're highlighting an isolated marker that
// doesn't need to be unstacked, and where we're only using the .geometry.x fiddle to prevent autocentering...
inhibitUnstacking |= (stackList.length == 1);
uroStackType = stackType;
if(stackList.length > 0)
{
if(inhibitUnstacking) uroAddLog('single marker highlighted, adjusting geometry properties to prevent recentering...');
else uroAddLog('markers are stacked!');
if(uroUnstackedMasterID != masterID)
{
uroAddLog('unstacked ID mismatch, relocating markers...');
uroRestackMarkers();
uroUnstackedMasterID = masterID;
uroStackList = [];
// push the highlighted marker onto the stacklist so uroIsIDAlreadyUnstacked() will return true
uroStackList.push(new uroStackListObj(masterID,unstackedX,unstackedY));
for(var shoveIdx=0; shoveIdx < stackList.length; shoveIdx++)
{
var fid = stackList[shoveIdx];
var x = uroParsePxString(markerCollection[fid].icon.imageDiv.style.left);
var y = uroParsePxString(markerCollection[fid].icon.imageDiv.style.top);
// store the unstacked marker positions so they can be reinstated later
uroStackList.push(new uroStackListObj(fid,x,y));
if(!inhibitUnstacking)
{
markerCollection[fid].icon.imageDiv.style.left = unstackedX + 'px';
markerCollection[fid].icon.imageDiv.style.top = unstackedY + 'px';
unstackedX += 10;
unstackedY -= 30;
}
}
if(!inhibitUnstacking)
{
// hide other markers to prevent confusion with the unstacked markers
for(marker in markerCollection)
{
if(markerCollection.hasOwnProperty(marker))
{
var toHideID = markerCollection[marker].id;
if(uroIsIDAlreadyUnstacked(toHideID) === false)
{
markerCollection[toHideID].icon.imageDiv.style.visibility = 'hidden';
}
}
}
}
}
}
else
{
uroRestackMarkers();
}
}
function uroGetVenueNavPoint(uroFID)
{
for(var vObj in W.model.venues.objects)
{
if(W.model.venues.objects.hasOwnProperty(vObj))
{
if(uroFID == vObj)
{
return W.model.venues.objects[vObj].getNavigationPoint().point.toLonLat();
}
}
}
// just in case... return a safe value if the requested venue object wasn't found
return W.map.getCenter();
}
function uroOpenNewTab()
{
// flush the current settings into localStorage before the new tab opens, so that when its instance of
// URO+ fires up it'll have the same settings as this one
uroSaveSettings();
return true;
}
function uroEditTBR()
{
if(uroTBRObj === null)
{
return;
}
uroTBRObj.click();
return false;
}
function uroKillCentering()
{
return W.map.getExtent();
}
function uroRestoreCentering()
{
if(uroAutoCentreDisabledOn.length > 0)
{
if(uroAutoCentreDisabledOn[0] == 'PUR')
{
if(W.map.placeUpdatesLayer.markers[uroAutoCentreDisabledOn[1]] != null)
{
W.map.placeUpdatesLayer.markers[uroAutoCentreDisabledOn[1]].model.geometry.getBounds = W.map.placeUpdatesLayer.markers[uroAutoCentreDisabledOn[1]].model.geometry.origGetBounds;
}
}
else if(uroAutoCentreDisabledOn[0] == 'MP')
{
if(W.map.problemLayer.markers[uroAutoCentreDisabledOn[1]] != null)
{
W.map.problemLayer.markers[uroAutoCentreDisabledOn[1]].model.getDisconnectBounds = W.map.problemLayer.markers[uroAutoCentreDisabledOn[1]].model.origGetDisconnectBounds;
}
}
uroAutoCentreDisabledOn = [];
}
}
function uroAddClosureRowToTable(rcObj)
{
var result = '';
if(rcObj.active === true)
{
result += '<tr>';
}
else
{
result += '<tr bgcolor="#C0C0C0">';
}
var startDate = rcObj.startDate;
var endDate = "unknown";
if(rcObj.endDate !== null)
{
endDate = rcObj.endDate;
}
var provider = "---";
if(rcObj.provider !== null)
{
provider = rcObj.provider;
}
else if(rcObj.createdBy !== null)
{
provider = uroGetUserNameAndRank(rcObj.createdBy);
}
var reason = "---";
if(rcObj.reason !== null)
{
reason = rcObj.reason;
}
result += '<td>' + startDate + ' to ' + endDate + '</td>';
result += '<td>' + provider + '</td>';
result += '<td>' + reason + '</td>';
result += '</td></tr>';
return result;
}
function uroNewLookHighlightedItemsCheck(e)
{
if(uroMTEMode) return;
if(!uroInitialised) return;
if(e == 'dwellTimeout')
{
}
else
{
if((uroMouseIsDown) && (e.buttons === 0))
{
uroAddLog('trapped erroneous mousedown state');
uroMouseIsDown = false;
}
}
if(uroMouseIsDown)
{
return;
}
if(OpenLayers === null)
{
if(uroNullOpenLayers === false)
{
uroAddLog('caught null OpenLayers');
uroNullOpenLayers = true;
}
return;
}
uroNullOpenLayers = false;
if(W.map.updateRequestLayer === null)
{
if(uroNullURLayer === false)
{
uroAddLog('caught null UR layer');
uroNullURLayer = true;
}
return;
}
uroNullURLayer = false;
if(W.map.problemLayer === null)
{
if(uroNullProblemLayer === false)
{
uroAddLog('caught null problem layer');
uroNullProblemLayer = true;
}
return;
}
uroNullProblemLayer = false;
if(uroGetCBChecked('_cbMasterEnable') === false)
{
return;
}
var mouseX;
var mouseY;
if(e == 'dwellTimeout')
{
mouseX = uroPrevMouseX;
mouseY = uroPrevMouseY;
}
else
{
mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top;
var maxJitter = uroGetElmValue('_inputMaxJitter');
if((Math.abs(uroPrevMouseX - mouseX) > maxJitter) || (Math.abs(uroPrevMouseY - mouseY) > maxJitter))
{
uroPopupDwellTimer = uroGetElmValue('_inputPopupDwellTimeout');
}
uroPrevMouseX = mouseX;
uroPrevMouseY = mouseY;
}
var result = '';
var rw;
var rh;
var popupXOffset = uroParsePxString(window.getComputedStyle(document.getElementById('sidebar')).getPropertyValue("width"));
var popupYOffset = $(document.getElementById("WazeMap")).offset().top - 80;
var uroPopupX = mouseX + popupXOffset + 10;
var uroPopupY = mouseY + popupYOffset - 10;
var objHasIgnoreLink = false;
var objHasDeleteLink = false;
var objHasAddWatchLink = false;
var objHasRemoveWatchLink = false;
var objHasUpdateWatchLink = false;
var objHasRecentreSessionLink = false;
var objHasOpenInNewTabLink = false;
var objHasCloneLink = false;
var isVenue = false;
var newPopupType = null;
var markerObj;
var markerPos;
var markerImg;
var ureq;
var idx;
var hovered;
// popup for segment restrictions
if(uroGetCBChecked('_cbInhibitSegPopup') === false)
{
for(var slIdx=0; slIdx < W.map.segmentLayer.features.length; slIdx++)
{
if(W.map.segmentLayer.features[slIdx].renderIntent == 'highlight')
{
if(W.map.getExtent().intersectsBounds(W.map.segmentLayer.features[slIdx].geometry.getBounds()))
{
var doPopUp = false;
var segObj;
var restObj;
if(W.map.segmentLayer.features[slIdx].fid === null) segObj = W.map.segmentLayer.features[slIdx].model;
else segObj = W.map.segmentLayer.features[slIdx];
// generic segment data
if(uroGetCBChecked('_cbInhibitSegGenericPopup') === false)
{
doPopUp = true;
var streetID = segObj.attributes.primaryStreetID;
var streetName = W.model.streets.objects[streetID].name;
if(streetName !== null)
{
result += '<b>'+streetName+'</b><br>';
}
result += '<b>ID: </b>'+segObj.attributes.id+'<br>';
var fwdSpeed = segObj.attributes.fwdMaxSpeed;
var revSpeed = segObj.attributes.revMaxSpeed;
var fwdUnverified = segObj.attributes.fwdMaxSpeedUnverified;
var revUnverified = segObj.attributes.revMaxSpeedUnverified;
if(segObj.attributes.fwdDirection)
{
result += '<b>A-B speed: </b>'+uroGetLocalisedSpeedString(fwdSpeed);
if(fwdUnverified) result += ' (unverified)';
result += '<br>';
}
if(segObj.attributes.revDirection)
{
result += '<b>B-A speed: </b>'+uroGetLocalisedSpeedString(revSpeed);
if(revUnverified) result += ' (unverified)';
result += '<br>';
}
if((segObj.attributes.fwdDirection) && (segObj.attributes.revDirection) && (fwdSpeed != revSpeed) && (!fwdUnverified) && (!revUnverified))
{
result += 'Two-way segment has different verified speed limits...<br>';
}
}
// segment restrictions
result += '<table cellpadding=4 border=1">';
if(segObj.attributes.fwdRestrictions.length > 0)
{
doPopUp = true;
result += '<tr><td colspan=11><b>A-B restrictions:</b></td></tr>';
for(idx = 0; idx < segObj.attributes.fwdRestrictions.length; idx++)
{
restObj = segObj.attributes.fwdRestrictions[idx];
result += uroFormatRestriction(restObj);
}
}
if (segObj.attributes.revRestrictions.length > 0)
{
doPopUp = true;
result += '<tr><td colspan=11><b>B-A restrictions:</b></td></tr>';
for(idx = 0; idx < segObj.attributes.revRestrictions.length; idx++)
{
restObj = segObj.attributes.revRestrictions[idx];
result += uroFormatRestriction(restObj);
}
}
result += '</table>';
if(W.map.closuresMarkerLayer.getVisibility() === true)
{
result += '<table cellpadding=4 border=1" width="100%">';
if(segObj.attributes.hasClosures === true)
{
var hasFwd = false;
var hasRev = false;
var rcObj;
var roadClosure;
for(roadClosure in W.model.roadClosures.objects)
{
if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
{
rcObj = W.model.roadClosures.objects[roadClosure];
if(rcObj.segID == segObj.attributes.id)
{
if(rcObj.forward === true)
{
if(hasFwd === false)
{
result += '<tr><td colspan=3><b>A-B closures:</b></td></tr>';
hasFwd = true;
}
result += uroAddClosureRowToTable(rcObj);
}
else
{
hasRev = true;
}
}
}
}
if(hasRev === true)
{
result += '<tr><td colspan=3><b>B-A closures:</b></td></tr>';
for(roadClosure in W.model.roadClosures.objects)
{
if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
{
rcObj = W.model.roadClosures.objects[roadClosure];
if(rcObj.segID == segObj.attributes.id)
{
if(rcObj.forward === false)
{
result += uroAddClosureRowToTable(rcObj);
}
}
}
}
}
if((hasFwd === true) || (hasRev === true))
{
doPopUp = true;
}
}
result += '</table>';
}
if(doPopUp === true)
{
if(segObj.attributes.id === null) uroFID = segObj.id;
else uroFID = segObj.attributes.id;
newPopupType = 'segment_restriction';
}
break;
}
else
{
uroAddLog('segment '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup');
}
}
}
}
// popup for restricted turns
if(newPopupType === null)
{
if(uroGetCBChecked('_cbInhibitTurnsPopup') === false)
{
var turnMarkerCount = W.map.layers[uroTurnsLayerIdx].markers.length;
if(turnMarkerCount > 0)
{
for(idx=0; idx<turnMarkerCount; idx++)
{
markerObj = W.map.layers[uroTurnsLayerIdx].markers[idx];
var arrowElm = markerObj.icon.imageDiv.childNodes[0];
markerImg = window.getComputedStyle(arrowElm).getPropertyValue("background-image");
markerPos = window.getComputedStyle(arrowElm).getPropertyValue("background-position");
markerPos = markerPos.split(' ');
markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
hovered = false;
if(markerImg.indexOf('turns-se0f8d1e954.png') != -1)
{
if(markerPos == -141)
{
hovered = true;
}
}
if(hovered === true)
{
uroAddLog('hover over restricted turn marker');
uroTBRObj = arrowElm.childNodes[0];
var trObj = ($(arrowElm).data('model'));
var resObj = null;
if(trObj.fromSeg.attributes.fromRestrictions != null)
{
resObj = trObj.fromSeg.attributes.fromRestrictions[trObj.toSeg.attributes.id];
}
if(resObj === null)
{
if(trObj.fromSeg.attributes.toRestrictions != null)
{
resObj = trObj.fromSeg.attributes.toRestrictions[trObj.toSeg.attributes.id];
}
}
result += '<label id="_editTBR">Click to edit</label><br>';
result += '<table cellpadding=4 border=1">';
for(var resIdx=0; resIdx < resObj.length; resIdx++)
{
result += uroFormatRestriction(resObj[resIdx]);
}
result += '</table>';
uroFID = markerObj.icon.imageDiv._eventCacheID;
newPopupType = 'turn_restriction';
break;
}
}
}
}
}
var targetTab = '';
// popup for landmarks
if((newPopupType === null) && (uroGetCBChecked('_cbInhibitLandmarkPopup') === false))
{
uroPlaceSelected = false;
var venueObj = null;
var renderIntent = null;
var navpointPos=new OpenLayers.LonLat();
for(var llFeatureIdx=0; llFeatureIdx < W.map.landmarkLayer.features.length; llFeatureIdx++)
{
renderIntent = W.map.landmarkLayer.features[llFeatureIdx].renderIntent;
if(renderIntent == 'highlight')
{
if(W.map.getExtent().intersectsBounds(W.map.landmarkLayer.features[llFeatureIdx].geometry.getBounds()))
{
if(W.map.landmarkLayer.features[llFeatureIdx].fid === null) venueObj = W.map.landmarkLayer.features[llFeatureIdx].model;
else venueObj = W.map.landmarkLayer.features[llFeatureIdx];
if(newPopupType === null)
{
if(venueObj.attributes.id === null) uroFID = venueObj.id;
else uroFID = venueObj.attributes.id;
navpointPos = uroGetVenueNavPoint(uroFID);
navpointPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
result += '<b>';
if(venueObj.attributes.name === '')
{
if(venueObj.attributes.residential === true) result += '<i>Residential</i>';
else result += '<i>Unnamed</i>';
}
else result += venueObj.attributes.name;
if(venueObj.attributes.externalProviderIDs.length > 0)
{
result += ' <i>(linked)</i>';
}
if(venueObj.attributes.adLocked)
{
result += ' <i>(AdLocked)</i>';
}
result += '</b><br>';
result += '<ul>';
for(idx = 0; idx < venueObj.attributes.categories.length; idx++)
{
result += '<li>' + I18n.lookup("venues.categories." + venueObj.attributes.categories[idx]);
}
result += '</ul>';
if(venueObj.attributes.residential === true)
{
if(venueObj.geometry.CLASS_NAME == 'OpenLayers.Geometry.Point')
{
result += '<a href="#" id="_cloneRP">Clone place</a>';
objHasCloneLink = true;
}
}
var npLink = document.location.href;
var npLayers = '&layers='+uroGetVisibilityBitmask();
npLink = npLink.substr(0,npLink.indexOf('?zoom'));
npLink += '?zoom=5&lat='+navpointPos.lat+'&lon='+navpointPos.lon+npLayers;
targetTab = "_uroTab_" + Math.round(Math.random()*1000000);
result += '<hr>Jump to nav point: <a href="'+npLink+'" id="_openInNewTab" target="'+targetTab+'">in new tab</a> - ';
objHasOpenInNewTabLink = true;
result += '<a href="#" id="_recentreSession">in this tab</a>';
objHasRecentreSessionLink = true;
newPopupType = 'venue';
isVenue = true;
break;
}
else
{
var otherID;
if(venueObj.attributes.id === null) otherID = venueObj.id;
else otherID = venueObj.attributes.id;
uroAddLog('venue '+otherID+' is also highlighted');
}
}
else
{
uroAddLog('landmark '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup');
}
}
else if((renderIntent == 'select') || (renderIntent == 'highlightselected'))
{
uroPlaceSelected = true;
}
}
}
var unstackedX;
var unstackedY;
var ureqID = null;
var isUR = false;
var isProblem = false;
var isTurnProb = false;
var isPlaceUpdate = false;
// look for URs, place updates and problems
if(newPopupType === null)
{
var idSrc = null;
if(uroGetCBChecked('_cbInhibitURPopup') === false)
{
hovered = false;
for(var markerURL in W.map.updateRequestLayer.markers)
{
if(W.map.updateRequestLayer.markers.hasOwnProperty(markerURL))
{
markerObj = W.map.updateRequestLayer.markers[markerURL];
markerImg = markerObj.icon.$div.css('background-image');
markerPos = markerObj.icon.$div.css('background-position');
markerPos = markerPos.split(' ');
markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
var urIDSrc = markerObj.id;
if((markerPos == -403) || (markerPos == -483) || (markerPos == -563) || (markerPos == -643))
{
hovered = true;
isUR = true;
newPopupType = 'ur';
uroAddLog('hover over UR ID '+urIDSrc);
unstackedX = uroParsePxString(W.map.updateRequestLayer.markers[urIDSrc].icon.imageDiv.style.left);
unstackedY = uroParsePxString(W.map.updateRequestLayer.markers[urIDSrc].icon.imageDiv.style.top);
// override popup base position
uroPopupX = unstackedX + popupXOffset + 6;
uroPopupY = unstackedY + popupYOffset + 66;
uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
// check for stacking...
if(uroShownFID != idSrc)
{
uroCheckStacking(1,urIDSrc, unstackedX, unstackedY);
}
idSrc = urIDSrc;
break;
}
}
}
if((hovered === false) && (uroStackType == 1))
{
uroRestackMarkers();
uroFilterURs();
}
}
if((newPopupType === null) && (uroGetCBChecked('_cbInhibitPUPopup') === false))
{
hovered = false;
for(var markerPUL in W.map.placeUpdatesLayer.markers)
{
if(W.map.placeUpdatesLayer.markers.hasOwnProperty(markerPUL))
{
markerObj = W.map.placeUpdatesLayer.markers[markerPUL];
markerImg = markerObj.icon.$div.css('background-image');
markerPos = markerObj.icon.$div.css('background-position');
markerPos = markerPos.split(' ');
markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
if(markerImg.indexOf('placeUpdates-sb30471988c.png') != -1)
{
// absolute offsets: 0 = new place, -120 = flagged, -240 = new photo, -360 = updated details
// relative offsets: 0 = green, -40 = highlighted, -80 = default
if(((markerPos + 40) % 120) === 0)
{
hovered = true;
uroAddLog('PUR marker type 1');
}
}
else if((markerImg.indexOf('placeUpdates-s2e8d9c5ce4.png') != -1)||(markerImg.indexOf('placeUpdates-s58d24ce3b0.png') != -1))
{
if
(
(markerObj.icon.$div.css("filter") == "brightness(110%)") ||
(markerObj.icon.$div.css("webkit-filter") == "brightness(1.1)")
)
{
hovered = true;
uroAddLog('PUR marker type 2');
}
}
if(hovered === true)
{
idSrc = markerObj.id;
unstackedX = uroParsePxString(W.map.placeUpdatesLayer.markers[idSrc].icon.imageDiv.style.left);
unstackedY = uroParsePxString(W.map.placeUpdatesLayer.markers[idSrc].icon.imageDiv.style.top);
// override popup base position
uroPopupX = unstackedX + popupXOffset + 6;
uroPopupY = unstackedY + popupYOffset + 66;
uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
if(uroShownFID != idSrc)
{
// check for stacking...
uroCheckStacking(3,idSrc, unstackedX, unstackedY);
}
isPlaceUpdate = true;
newPopupType = 'pur';
uroAddLog('hover over placeUpdate ID '+idSrc);
// to inhibit auto-centering only when the PUR marker is clicked, we wait for the marker to get highlighted, then
// make a copy of the original getBounds() function before replacing it with a call to W.map.getExtent(). Clicking
// the marker causes a call to getBounds() which will then return the current map extent, and thus no change in the
// map view will occur...
uroRestoreCentering();
W.map.placeUpdatesLayer.markers[markerPUL].model.geometry.origGetBounds = W.map.placeUpdatesLayer.markers[markerPUL].model.geometry.getBounds;
W.map.placeUpdatesLayer.markers[markerPUL].model.geometry.getBounds = uroKillCentering;
uroAutoCentreDisabledOn.push('PUR', markerPUL);
break;
}
}
}
if((hovered === false) && (uroStackType == 3))
{
uroRestackMarkers();
uroFilterPlaces();
}
}
if((newPopupType === null) && (uroGetCBChecked('_cbInhibitMPPopup') === false))
{
hovered = false;
for(var markerPL in W.map.problemLayer.markers)
{
if(W.map.problemLayer.markers.hasOwnProperty(markerPL))
{
markerObj = W.map.problemLayer.markers[markerPL];
markerImg = markerObj.icon.$div.css('background-image');
markerPos = markerObj.icon.$div.css('background-position');
markerPos = markerPos.split(' ');
markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
if((markerPos == -65) || (markerPos == -145) || (markerPos == -225) || (markerPos == -305))
{
hovered = true;
idSrc = null;
if(markerObj.model.fid === null) idSrc = markerObj.id;
else idSrc = markerObj.model.fid;
unstackedX = uroParsePxString(W.map.problemLayer.markers[idSrc].icon.imageDiv.style.left);
unstackedY = uroParsePxString(W.map.problemLayer.markers[idSrc].icon.imageDiv.style.top);
// override popup base position
uroPopupX = unstackedX + popupXOffset + 6;
uroPopupY = unstackedY + popupYOffset + 66;
uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
// check for stacking...
if(uroShownFID != idSrc)
{
uroCheckStacking(2,idSrc, unstackedX, unstackedY);
}
isProblem = true;
newPopupType = 'map_problem';
uroAddLog('hover over problem ID '+idSrc);
// same method of disabling the on-click auto-centre behaviour as for PURs above...
uroRestoreCentering();
W.map.problemLayer.markers[markerPL].model.origGetDisconnectBounds = W.map.problemLayer.markers[markerPL].model.getDisconnectBounds;
W.map.problemLayer.markers[markerPL].model.getDisconnectBounds = uroKillCentering;
uroAutoCentreDisabledOn.push('MP', markerPL);
break;
}
}
}
if((hovered === false) && (uroStackType == 2))
{
uroRestackMarkers();
uroFilterProblems();
}
}
if (idSrc !== null)
{
ureq = null;
if(isUR) ureq = W.model.mapUpdateRequests.objects[idSrc];
else if(isProblem)
{
ureq = W.model.problems.objects[idSrc];
if(ureq === undefined)
{
if(uroDOMHasTurnProblems)
{
ureq = W.model.turnProblems.objects[idSrc];
if(ureq != null) isTurnProb = true;
}
}
}
else if(isPlaceUpdate) ureq = W.map.placeUpdatesLayer.markers[idSrc].model;
if(ureq.fid !== null) ureqID = ureq.fid;
else if(ureq.id !== null) ureqID = ureq.id;
else if(ureq.attributes.id !== null) ureqID = ureq.attributes.id;
uroFID = ureqID;
if(isUR)
{
uroHoveredURID = uroFID;
}
else
{
uroHoveredURID = null;
}
}
else
{
if(uroFID != -1)
{
uroFID = -1;
}
}
if(uroFID != -1)
{
var uroDaysResolved;
if(isUR)
{
uroAddLog('building popup for UR '+idSrc);
result = '<b>Update Request ('+idSrc+'): ' + I18n.lookup("update_requests.types." + ureq.attributes.type) + '</b><br>';
if(ureq.attributes.description !== null)
{
var desc = ureq.attributes.description.replace(/<\/?[^>]+(>|$)/g, "");
if(desc != "null")
{
desc = uroClickify(desc);
result += desc + '<br>';
}
}
var uroDaysOld = uroGetURAge(ureq,0,false);
var uroSubmittedTS = uroGetURAge(ureq,0,true);
if(uroSubmittedTS != -1)
{
uroSubmittedTS = uroGetDateTimeString(uroSubmittedTS);
}
if(uroDaysOld != -1)
{
result += '<i>Submitted ' + uroParseDaysAgo(uroDaysOld) + ' ';
if(uroSubmittedTS != -1) result += '(' + uroSubmittedTS + ') ';
if(ureq.attributes.guestUserName != null)
{
result += 'via Livemap';
if(ureq.attributes.guestUserName !== '')
{
result += ' by '+ureq.attributes.guestUserName.replace(/<\/?[^>]+(>|$)/g, "");
}
}
result += '</i>';
}
if(ureq.attributes.resolvedOn !== null)
{
uroDaysResolved = uroGetURAge(ureq,1,false);
var uroResolvedTS = uroGetURAge(ureq,1,true);
if(uroResolvedTS != -1)
{
uroResolvedTS = uroGetDateTimeString(uroResolvedTS);
}
if(uroDaysResolved != -1)
{
result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' ';
if(uroResolvedTS != -1) result += '(' + uroResolvedTS + ')</i>';
result += '<br><i>Marked as ';
if(ureq.attributes.resolution === 0) result += 'solved';
else if(ureq.attributes.resolution == 1) result += 'not identified';
else result += 'unknown';
if(ureq.attributes.resolvedBy !== null)
{
result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy);
}
result += '</i>';
}
}
if(W.model.updateRequestSessions.objects[ureqID] != null)
{
var hasMyComments = uroURHasMyComments(ureqID);
var nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
result += '<br>' + nComments + ' comment';
if(nComments != 1) result += 's';
if((hasMyComments === false) && (nComments > 0)) result += ' (none by me)';
if(nComments > 0)
{
var commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1]);
if(commentDaysOld != -1)
{
result += ', last update '+uroParseDaysAgo(commentDaysOld);
}
}
}
}
else if(isProblem)
{
uroAddLog('building popup for problem '+idSrc);
if(isTurnProb) result = '<b>Turn Problem ('+idSrc+'): ' + I18n.lookup("problems.types.turn.title");
else
{
result = '<b>Map Problem ('+idSrc+'): ';
var problemType = null;
if(uroDOMHasTurnProblems)
{
problemType = ureq.attributes.problemType;
}
else
{
problemType = ureq.attributes.subType;
}
if(problemType == 300)
{
result += I18n.lookup("problems.panel.closure.title");
}
else
{
if(I18n.lookup("problems.types." + problemType) === undefined) result += 'Unknown problem type ('+problemType+')';
else result += I18n.lookup("problems.types." + problemType + ".title");
}
}
result += '</b><br>';
if(ureq.attributes.description != null)
{
result += 'Description: ' + ureq.attributes.description + '<br>';
}
if(ureq.attributes.extraInfo != null)
{
result += 'ExtraInfo: ' + ureq.attributes.extraInfo + '<br>';
}
if(ureq.attributes.provider != null)
{
result += 'Provider: ' + ureq.attributes.provider + '<br>';
}
if(ureq.attributes.resolvedOn != null)
{
uroDaysResolved = uroGetURAge(ureq,1,false);
if(uroDaysResolved != -1)
{
result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' ';
if(ureq.attributes.resolvedBy != null)
{
result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy);
}
if((ureq.attributes.open === true) && (ureq.attributes.resolvedOn != null))
{
result += '<br>Reopened by Waze';
}
result += '</i>';
}
}
}
else if(isPlaceUpdate)
{
uroAddLog('building popup for placeUpdate '+idSrc);
result = '<b>';
if(ureq.attributes.name === '') result += 'Unnamed landmark';
else result += ureq.attributes.name;
result += '</b><br>';
result += '<ul>';
for(idx = 0; idx < ureq.attributes.categories.length; idx++)
{
result += '<li>' + I18n.lookup("venues.categories." + ureq.attributes.categories[idx]);
}
result += '</ul>';
if(ureq.attributes.residential === true)
{
result += '<i>Residential</i>';
}
var daysOld = uroGetPURAge(ureq);
if(daysOld != -1)
{
result += '<br><i>Submitted '+uroParseDaysAgo(daysOld)+'</i>';
}
}
// add "open new WME tab" link
var urPos=new OpenLayers.LonLat();
if(isPlaceUpdate)
{
urPos=ureq.geometry.bounds.centerLonLat.clone();
}
else
{
if(ureq.geometry.realX === undefined)
{
urPos.lon = ureq.geometry.x;
urPos.lat = ureq.geometry.y;
}
else
{
urPos.lon = ureq.geometry.realX;
urPos.lat = ureq.geometry.realY;
}
}
urPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
var urLink = document.location.href;
var urLayers = '&layers='+uroGetVisibilityBitmask();
urLink = urLink.substr(0,urLink.indexOf('?zoom'));
urLink += '?zoom=5&lat='+urPos.lat+'&lon='+urPos.lon+urLayers;
if(isUR) urLink += '&mapUpdateRequest='+idSrc;
else if(isTurnProb) urLink += '&showturn='+idSrc+'&endshow';
else if(isProblem) urLink += '&mapProblem='+idSrc;
else if(isPlaceUpdate) urLink += '&showpur='+idSrc+'&endshow';
targetTab = "_uroTab_" + Math.round(Math.random()*1000000);
result += '<hr><ul><li><a href="'+urLink+'" id="_openInNewTab" target="'+targetTab+'">Open in new tab</a> - ';
objHasOpenInNewTabLink = true;
result += '<a href="#" id="_recentreSession">centre in current tab</a>';
objHasRecentreSessionLink = true;
// add "open new livemap tab" link
var lmLink = null;
if(document.getElementById("livemap-link") != null)
{
uroAddLog('Livemap link in livemap-link id element');
lmLink = document.getElementById("livemap-link").href;
}
else if(document.getElementsByClassName("livemap-link") != null)
{
uroAddLog('Livemap link in livemap-link class element');
lmLink = document.getElementsByClassName("livemap-link")[0].href;
}
else
{
uroAddLog('Livemap link not found...');
}
if(lmLink !== null)
{
var zpos = lmLink.indexOf('?');
if(zpos > -1) lmLink = lmLink.substr(0,zpos);
lmLink += '?zoom=17&lat='+urPos.lat+'&lon='+urPos.lon+'&layers=BTTTT';
result += '<li><a href="'+lmLink+'" target="_lmTab">Open in new livemap tab</a>';
}
if(!isPlaceUpdate)
{
// add "ignore for this session" link
result += '<li><a href="#" id="_addtoignore">Hide for this session</a></ul>';
objHasIgnoreLink = true;
}
}
}
if((newPopupType != 'map_problem') && (newPopupType != 'pur'))
{
uroRestoreCentering();
}
// look for cameras
if((newPopupType === null) && (uroGetCBChecked('_cbInhibitCamPopup') === false))
{
for(var clFeatureIdx = 0; clFeatureIdx < W.map.camerasLayer.features.length; clFeatureIdx++)
{
if(W.map.camerasLayer.features[clFeatureIdx].renderIntent == 'highlight')
{
if(W.map.camerasLayer.features[clFeatureIdx].fid === null) ureq = W.map.camerasLayer.features[clFeatureIdx].model;
else ureq = W.map.camerasLayer.features[clFeatureIdx];
if(ureq.fid === null) ureqID = ureq.attributes.id;
else ureqID = ureq.fid;
// test isSelected() so that we only do overview data on cameras that are being hovered over
if(ureq.isSelected() === false)
{
uroPopupY -= 20;
newPopupType = 'camera';
uroFID = ureqID;
uroAddLog('generating popup for camera '+uroFID);
if(I18n.lookup("edit.camera.fields.type") === undefined)
{
result += '<b>Camera: ' + ureq.TYPES[ureq.attributes.type] + '</b><br>';
}
else
{
result += '<b>Camera: ' + I18n.lookup("edit.camera.fields.type." + ureq.attributes.type) + '</b><br>';
}
result += 'ID: '+uroFID+'<br>';
result += 'Created by ';
var userID;
if(W.model.users.get(ureq.attributes.createdBy) != null)
{
userID = ureq.attributes.createdBy;
result += uroGetUserNameAndRank(userID);
}
else result += 'unknown';
result += ', ';
var camAge = uroGetCameraAge(ureq,1);
if(camAge != -1)
{
result += uroParseDaysAgo(camAge);
}
else result += 'unknown days ago';
result += '<br>Updated by ';
if(W.model.users.get(ureq.attributes.updatedBy) != null)
{
userID = ureq.attributes.updatedBy;
var userName = W.model.users.objects[userID].userName;
var userLevel = W.model.users.objects[userID].rank + 1;
result += userName + ' (' + userLevel + ')';
}
else result += 'unknown';
result += ', ';
camAge = uroGetCameraAge(ureq,0);
if(camAge != -1)
{
result += uroParseDaysAgo(camAge);
}
else result += 'unknown days ago';
result += '<br>Speed data: ';
result += uroGetLocalisedSpeedString(ureq.attributes.speed);
result += '<hr><ul>';
if(uroIsCamOnWatchList(uroFID) != -1)
{
result += '<li><a href="#" id="_updatewatchlist">Update watchlist entry</a>';
result += '<li><a href="#" id="_removefromwatchlist">Remove from watchlist</a>';
objHasUpdateWatchLink = true;
objHasRemoveWatchLink = true;
}
else
{
result += '<li><a href="#" id="_addtowatchlist">Add to watchlist</a>';
objHasAddWatchLink = true;
}
if(ureq.attributes.permissions !== 0)
{
result += '<li><a href="#" id="_deleteobject">Delete Camera</a>';
objHasDeleteLink = true;
}
result += '</ul>';
}
break;
}
}
}
if((newPopupType !== null) && (uroPopupDwellTimer === 0) && (uroPopupSuppressed === false))
{
if((uroFID != uroShownFID) || (newPopupType != uroShownPopupType))
{
if(uroFID != uroShownFID) uroAddLog('FID mismatch, show popup: '+uroFID+'/'+uroShownFID);
else uroAddLog('Popup type mismatch: '+newPopupType+'/'+uroShownPopupType);
uroShownFID = uroFID;
uroShownPopupType = newPopupType;
uroPopupShown = false;
}
if(uroPopupShown === false)
{
uroAddLog('display popup at '+uroPopupX+','+uroPopupY);
uroPopupShown = true;
uroDiv.style.height = "auto";
uroDiv.style.width = "auto";
uroDiv.innerHTML = result;
if((uroFID != -1) && (objHasIgnoreLink === true))
{
uroAddEventListener('_addtoignore','click', uroAddToIgnoreList, true);
}
if(objHasDeleteLink === true)
{
uroAddEventListener('_deleteobject','click', uroDeleteObject, true);
}
if(objHasRemoveWatchLink === true)
{
uroAddEventListener('_removefromwatchlist','click', uroRemoveCamFromWatchList, true);
}
if(objHasAddWatchLink === true)
{
uroAddEventListener('_addtowatchlist','click', uroAddCamToWatchList, true);
}
if(objHasUpdateWatchLink === true)
{
uroAddEventListener('_updatewatchlist','click', uroUpdateCamWatchList, true);
}
if(objHasOpenInNewTabLink === true)
{
uroAddEventListener('_openInNewTab','mouseup', uroOpenNewTab, true);
}
if(objHasRecentreSessionLink === true)
{
if(isUR) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnUR, true);
else if((isProblem)||(isTurnProb)) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnMP, true);
else if(isPlaceUpdate) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnPUR, true);
else if(isVenue) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnVenueNavPoint, true);
}
if(objHasCloneLink === true)
{
uroAddEventListener('_cloneRP', 'click', uroCloneResidentialPlace, true);
}
if(newPopupType == 'turn_restriction')
{
uroAddEventListener('_editTBR','click', uroEditTBR, true);
}
// restrict the popup width to be no wider than just under half the window width to avoid it
// completely overlapping the marker it's associated with - by keeping it to just below half
// the window width we guarantee that it'll fit either to the left or the right of the marker
// no matter how far across the screen the marker is located...
rw = parseInt(uroDiv.clientWidth);
if(rw > (window.innerWidth * 0.45))
{
rw = (window.innerWidth * 0.45);
uroDiv.style.width = rw+'px';
}
// get the div height after any adjustment of the width above, to account for whatever content
// reflow may have occurred as a result of reducing the width...
rh = parseInt(uroDiv.clientHeight);
if((uroPopupX + rw) > window.innerWidth)
{
// where the popup would be off the right hand side of the screen, move it completely over to the
// other side of the mouse pointer
uroPopupX -= (rw + 20);
if(uroPopupX < 0) uroPopupX = 0;
}
if((uroPopupY + rh) > window.innerHeight)
{
// where the popup would be off the bottom of the screen, shift it up just far enough to be
// fully visible
uroPopupY -= (((uroPopupY + rh) - window.innerHeight) + 30);
if(uroPopupY < 0) uroPopupY = 0;
}
uroDiv.style.top = uroPopupY+'px';
uroDiv.style.left = uroPopupX+'px';
uroDiv.style.visibility = 'visible';
}
uroPopupTimer = -1;
}
else if((newPopupType === null) && (uroPopupDwellTimer !== 0) && (uroPopupShown === true))
{
uroHidePopup();
}
else
{
if((uroPopupTimer == -1) && (uroFID != uroShownFID))
{
uroPopupTimer = uroGetElmValue('_inputPopupEntryTimeout');
}
}
}
function uroExclusiveCB()
{
var cbChecked = uroGetCBChecked(this.id);
if(cbChecked === true)
{
var pairedList = this.attributes.pairedWith.value.split(',');
for(var i=0; i<pairedList.length; i++)
{
uroSetCBChecked(pairedList[i], false);
}
}
}
function uroGetAMs(e)
{
if(uroMTEMode) return;
if(!uroFilterPreamble) return;
if(!uroInitialised) return;
var amList = '';
if(W.map.managedAreasLayer.getVisibility() === true)
{
var mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
var mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top - document.getElementById('toolbar').clientHeight;
var mousePixel = new OL.Pixel(mouseX, mouseY);
var mousePoint = W.map.getLonLatFromPixel(mousePixel).toPoint();
for(var amObj in W.model.managedAreas.objects)
{
if(W.model.managedAreas.objects[amObj].geometry.containsPoint(mousePoint))
{
if(amList !== '') amList += ', ';
amList += uroGetUserNameAndRank(W.model.managedAreas.objects[amObj].userID);
}
}
if(amList === '')
{
amList = 'none';
}
amList = "<b>Area Managers:</b> "+amList;
}
document.getElementById("uroAMList").innerHTML = amList;
}
function uroMouseDown()
{
uroMouseIsDown = true;
}
function uroMouseUp()
{
uroMouseIsDown = false;
}
function uroMouseOut()
{
//console.debug('Elvis has left the building...');
}
function uroUREvent_onObjectsChanged()
{
}
function uroUREvent_onObjectsAdded()
{
if(uroGetCBChecked('_cbURResolverIDFilter') === true)
{
uroUpdateResolverList();
}
uroFilterURs();
}
function uroUREvent_onObjectsRemoved()
{
}
function uroGetSelectedURCommentCount()
{
if(W.model.updateRequestSessions.objects[uroSelectedURID] != null)
{
var cachedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
uroAddLog(uroSelectedURID+':'+cachedCommentCount+' '+uroExpectedCommentCount);
// if there aren't the same number of cached comments as there are comments in the UR dialog list, initiate
// a refresh of the comment data...
if(cachedCommentCount != uroExpectedCommentCount)
{
if(uroPendingCommentDataRefresh === true)
{
if(cachedCommentCount > 0)
{
uroCachedLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].comments[cachedCommentCount-1].id;
}
else
{
uroCachedLastCommentID = null;
}
uroAddLog('updateRequestSessions refresh required for UR '+uroSelectedURID);
if(uroCachedLastCommentID !== null)
{
uroAddLog('last comment ID for this UR is '+uroCachedLastCommentID);
}
else
{
uroAddLog('first comment for this UR, no previous comment to ID');
}
var idList = [];
idList.push(uroSelectedURID);
// need to delete the existing cache object first, as .get() is only capable of creating new objects,
// it doesn't seem able to update an existing object with new data
W.model.updateRequestSessions.remove(W.model.updateRequestSessions.objects[uroSelectedURID]);
W.model.updateRequestSessions.get(idList);
// the call to .get() initiates a XMLHttpRequest for the data, so we now need to switch modes - the
// refresh process has started so we're no longer pending, but we are now waiting for the XMLHttpRequest
// to return something...
uroPendingCommentDataRefresh = false;
uroWaitingCommentDataRefresh = true;
}
else
{
if(cachedCommentCount > 0)
{
var currentLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].comments[cachedCommentCount-1].id;
if(currentLastCommentID == uroCachedLastCommentID)
{
// most recent comment loaded for this UR is the same one that was present at the start of this
// refresh process, so kick back into pending mode so we can retry the .get()...
uroAddLog('latest comment ID still the same, reverting to pending mode...');
uroPendingCommentDataRefresh = true;
}
else
{
// something may have gone awry here - the most recent comment loaded for this UR doesn't have the
// same ID as the one present at the start of the refresh process, yet the comment counts still don't
// match up, which suggests either a comment got lost along the way or someone else has commented on
// the same UR at almost the same time. To get out of the loop this would create, assume that a
// mismatch in the IDs means the .get() has completed successfully no matter what the new comment
// count is, and take this new count to be the count we were expecting all along...
uroAddLog('latest comment ID different, but expected count not correct...');
uroExpectedCommentCount = cachedCommentCount;
}
}
else
{
uroAddLog('first comment on this UR not received yet, reverting to pending mode...');
uroPendingCommentDataRefresh = true;
}
}
}
else
{
// if the WME session is loaded with a UR already selected, such that WME has opened the UR dialog as part
// of the session startup process, adding new comments to the UR cause the cached data to be updated immediately.
// This prevents URO+ from switching into waiting mode in the above block of code, so we have to instead do
// it here by comparing the cached count against the expected count following the Send click event.
if(cachedCommentCount >= uroExpectedCommentCount)
{
uroPendingCommentDataRefresh = false;
uroWaitingCommentDataRefresh = true;
uroExpectedCommentCount = null;
}
// once the cached data has been updated, refilter the URs so that the new comment count is taken into account
// immediately for filtering and display purposes
if(uroWaitingCommentDataRefresh === true)
{
uroWaitingCommentDataRefresh = false;
uroFilterURs();
uroAddLog('refresh complete');
}
}
}
}
function uroAddedComment()
{
// when the user clicks the Send button to submit a new UR comment, this event handler fires before the new comment is
// posted to the server and thus also before the comment list gets updated in the UR dialog. So we take the current
// comment count and, if the new comment edit box isn't empty, increment it by 1 to get the expected count. Then we
// set the pending flag true to initiate a session refresh on the next 100ms tick
uroExpectedCommentCount = W.map.panelRegion.currentView.conversationView.viewModel.attributes.commentCount;
if(document.getElementsByClassName('new-comment-text')[0].value !== '')
{
uroExpectedCommentCount++;
uroAddLog('new comment added to UR '+uroSelectedURID+', cache refresh required...');
uroPendingCommentDataRefresh = true;
}
else
{
uroPendingCommentDataRefresh = false;
}
}
function uroInhibitNextUpdateRequestButton(e)
{
var doClick = true;
e.stopPropagation();
if(document.getElementsByClassName('form-control new-comment-text').length > 0)
{
if(document.getElementsByClassName('form-control new-comment-text')[0].textLength > 0)
{
doClick = (confirm('Comment not sent, close report panel anyway?'));
}
}
if(doClick)
{
document.getElementsByClassName('close-panel')[0].click();
}
}
function uroAddLZ(valueToPad, newLength)
{
var padString = '';
for(var i=0; i<newLength; i++)
{
padString += '0';
}
padString += valueToPad.toString();
return padString.slice(-newLength);
}
function uroIncrementClosureDate(oldDate, incByDays)
{
var dateBits = oldDate.split('-');
var year = parseInt(dateBits[0]);
var month = parseInt(dateBits[1])-1;
var date = parseInt(dateBits[2])+1;
var incrementedDate = new Date(year, month, date);
return (uroAddLZ(incrementedDate.getFullYear(),4) + '-' + uroAddLZ(incrementedDate.getMonth()+1,2) + '-' + uroAddLZ(incrementedDate.getDate(),2));
}
function uroGetElementProperty(elmName, elmOffset, elmProperty)
{
var retval = null;
if(document.getElementsByName(elmName).length !== 0)
{
retval = document.getElementsByName(elmName)[elmOffset][elmProperty];
}
return retval;
}
// Residential Place Cloning
//{
var uroCRPStreetID;
var uroCRPHouseNumber;
function uroCompleteRPClone()
{
// as with closure cloning, the place details edit form requires us to push the new value into the relevant
// edit field and then generate a change event on that field, otherwise WME doesn't bother reading the value...
// street name
var streetObj = W.model.streets.get(uroCRPStreetID);
if(streetObj !== undefined)
{
document.getElementsByClassName('street-name')[0].value = streetObj.name;
document.getElementsByClassName('street-name')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
// city name
var cityObj = W.model.cities.get(streetObj.cityID);
if(cityObj !== undefined)
{
document.getElementsByClassName('city-name')[0].value = cityObj.name;
document.getElementsByClassName('city-name')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
// county
document.getElementsByClassName('state-id')[0].value = cityObj.stateID;
document.getElementsByClassName('state-id')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
// country
document.getElementsByClassName('country-id')[0].value = cityObj.countryID;
document.getElementsByClassName('country-id')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
}
// house number
document.getElementsByClassName('house-number')[0].value = parseInt(uroCRPHouseNumber);
document.getElementsByClassName('house-number')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
// now wait for the user to confirm everything and click Apply...
}
function uroConvertToRP()
{
// panel isn't open yet, which means the user either hasn't clicked yet or WME is still processing the
// placement of the venue, so wait a while and then check again...
if(document.getElementById('edit-panel').getElementsByClassName('landmark').length === 0)
{
setTimeout(uroConvertToRP, 100);
return;
}
// panel is open, so move to the next step of the cloning procedure by converting the newly created
// place to residential by generating a click event on the "convert to residential" link...
document.getElementsByClassName("toggle-residential")[0].click();
// and then click on the address edit icon...
document.getElementsByClassName('waze-icon-edit')[0].click();
// now click on the "none" checkbox for the street name edit field so we can enter the street name
document.getElementById('empty-street').click();
// WME automatically clears the checkbox associated with the city name edit field if we set the street
// name to be one that has a city associated with it, which is nice :-)
// the click event seems to take a while to execute, and if we call dispatchEvent on the edit field whilst
// it's still tagged as disabled then it gets ignored, causing the value in that field to be dropped when
// we apply the changes to the place. Trying to programatically detect when the field has been activated
// doesn't seem to be reliable, however a fixed delay of 1s seems to work nicely
setTimeout(uroCompleteRPClone, 1000);
}
function uroCloneResidentialPlace()
{
// trying to clone a RPP when one is already selected causes the selected one to be changed back to
// a non-residential, as uroConvertToRP() thinks the user has already clicked to place the new RPP...
if(document.getElementById('edit-panel').getElementsByClassName('landmark').length === 0)
{
var venueObj = W.model.venues.objects[uroFID];
if(venueObj !== undefined)
{
// copy address from highlighted residential place
uroCRPHouseNumber = venueObj.attributes.houseNumber;
uroCRPStreetID = venueObj.attributes.streetID;
// generate a click event on the first new point venue entry in the menu in order to generate a
// new point venue object that we can manipulate...
document.getElementsByClassName("drawing-control main-control point")[0].click();
// now wait for the user to click on the map to place the new point venue
uroConvertToRP();
}
}
}
//}
// Closure Cloning
//{
var uroConfirmClosureDelete = true;
var uroClosuresToDelete = 0;
function uroCloneClosure()
{
var closureOffset = parseInt(this.id.split('-')[1]);
// grab the current closure details from the UI...
document.getElementsByClassName('closure-item')[closureOffset].children[0].children[0].children[1].click();
var cLocation = uroGetElementProperty('closure_location', 0, 'value');
var cReason = uroGetElementProperty('closure_reason', 0, 'value');
var cEvent = uroGetElementProperty('closure_mteId', 0, 'selectedIndex');
var cDirection = uroGetElementProperty('closure_direction', 0, 'selectedIndex');
var cHasStartDate = uroGetElementProperty('closure_hasStartDate', 0, 'checked');
var cStartDate = uroGetElementProperty('closure_startDate', 0, 'value');
var cStartTime = uroGetElementProperty('closure_startTime', 0, 'value');
var cEndDate = uroGetElementProperty('closure_endDate', 0, 'value');
var cEndTime = uroGetElementProperty('closure_endTime', 0, 'value');
var cIgnoreTraffic = uroGetElementProperty('closure_permanent', 0, 'checked');
document.getElementsByClassName('closures')[0].getElementsByClassName('cancel-button')[0].click();
// auto-increment the start and end dates...
cStartDate = uroIncrementClosureDate(cStartDate,1);
cEndDate = uroIncrementClosureDate(cEndDate,1);
// now open the add closure UI and populate it with the values we've just grabbed...
document.getElementsByClassName('add-closure-button')[0].click();
// need to generate a change event on each of the form fields, because WME appears to be silently populating some hidden
// closure object with the details as they're entered manually, and if we just set the form values without then forcing
// the change event as well then WME will end up using its default values instead of the ones we've so lovingly copied...
if(cLocation !== null)
{
document.getElementsByName('closure_location')[0].value = cLocation;
document.getElementsByName('closure_location')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cReason !== null)
{
document.getElementsByName('closure_reason')[0].value = cReason;
document.getElementsByName('closure_reason')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cEvent !== null)
{
document.getElementsByName('closure_mteId')[0].selectedIndex = cEvent;
document.getElementsByName('closure_mteId')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cDirection !== null)
{
document.getElementsByName('closure_direction')[0].selectedIndex = cDirection;
document.getElementsByName('closure_direction')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cHasStartDate !== null)
{
document.getElementsByName('closure_hasStartDate')[0].checked = cHasStartDate;
document.getElementsByName('closure_hasStartDate')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cStartDate !== null)
{
document.getElementsByName('closure_startDate')[0].value = cStartDate;
document.getElementsByName('closure_startDate')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cStartTime !== null)
{
document.getElementsByName('closure_startTime')[0].value = cStartTime;
document.getElementsByName('closure_startTime')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cEndDate !== null)
{
document.getElementsByName('closure_endDate')[0].value = cEndDate;
document.getElementsByName('closure_endDate')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cEndTime !== null)
{
document.getElementsByName('closure_endTime')[0].value = cEndTime;
document.getElementsByName('closure_endTime')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
if(cIgnoreTraffic !== null)
{
document.getElementsByName('closure_permanent')[0].checked = cIgnoreTraffic;
document.getElementsByName('closure_permanent')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
}
}
function uroDeleteNextClosureOnList()
{
var nClosures = document.getElementsByClassName('closure-item').length;
if(nClosures > 0)
{
if (nClosures != uroClosuresToDelete)
{
uroClosuresToDelete = nClosures;
document.getElementsByClassName('closure-item')[0].getElementsByClassName('delete')[0].click();
}
setTimeout(uroDeleteNextClosureOnList,100);
}
else
{
uroConfirmClosureDelete = true;
}
}
function uroDeleteAllClosures()
{
uroConfirmClosureDelete = true;
if(window.confirm(I18n.lookup("closures.delete_confirm_no_reason")+' ('+I18n.lookup("closures.apply_to_all")+')'))
{
uroConfirmClosureDelete = false;
var nClosures = document.getElementsByClassName('closure-item').length;
if(nClosures > 0)
{
uroClosuresToDelete = -1;
uroDeleteNextClosureOnList();
}
else
{
uroConfirmClosureDelete = true;
}
}
}
//}
// Feed Filtering
//{
function uroToggleFFCtrls()
{
if(uroShowFeedFilter === false)
{
uroShowFeedFilter = true;
document.getElementById('_uroFFCtrlVisibility').className = "fa fa-minus-square-o";
document.getElementById('uroFFCtrls').style.display = "block";
}
else
{
uroShowFeedFilter = false;
document.getElementById('_uroFFCtrlVisibility').className = "fa fa-plus-square-o";
document.getElementById('uroFFCtrls').style.display = "none";
}
}
function uroAddFeedFilterControls()
{
if(document.getElementById('sidepanel-feed') != null)
{
if(document.getElementById('sidepanel-feed').childNodes[0] != null)
{
var nDiv = document.createElement('div');
var iHTML = '';
nDiv.id = "uroFeedFilter";
iHTML += '<i class="fa fa-plus-square-o" style="cursor:pointer;font-size:14px;" id="_uroFFCtrlVisibility"> </i><b>Feed Filter Controls</b><br>';
iHTML += '<div id="uroFFCtrls">';
iHTML += '<b>Filter feed by listing type:</b><br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_TypeUR" />UR<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_TypePUR" />PUR<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_TypePM" />PM notifications<br>';
iHTML += '<br><b>Filter feed by listing reason:</b><br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotCBSBR" />'+I18n.lookup("feed.issues.motivations.CAN_BE_SOLVED_BY_RANK")+'<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotCTF" />'+I18n.lookup("feed.issues.motivations.CLOSE_TO_FAVORITES")+'<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotIA" />'+I18n.lookup("feed.issues.motivations.ISSUE_AGE")+'<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotIR" />'+I18n.lookup("feed.issues.motivations.ISSUE_REOPENED")+'<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotND" />'+I18n.lookup("feed.issues.motivations.NEAR_DRIVES")+'<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotRBU" />'+I18n.lookup("feed.issues.motivations.REPORTED_BY_USER")+'<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotUFI" />'+I18n.lookup("feed.issues.motivations.USER_FOLLOWS_ISSUE")+'<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotUFILC" />'+I18n.lookup("feed.issues.motivations.USER_FOLLOWS_ISSUE_LAST_COMMENT")+'<br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_MotNone" />None of the above...<br>';
iHTML += '<br><input type="checkbox" id="_cbFeedFilter_Invert" />Invert behaviour of above filters<br>';
iHTML += '<br><b>Filter feed by keyword/phrase:</b><br>';
iHTML += '<input type="checkbox" id="_cbFeedFilter_HideKeyword" pairedWith="_cbFeedFilter_ShowKeyword" />Hide or ';
iHTML += '<input type="checkbox" id="_cbFeedFilter_ShowKeyword" pairedWith="_cbFeedFilter_HideKeyword" />show if keyword is present<br>';
iHTML += '<input type="text" id="_cbFeedFilter_Keyword" />';
iHTML += '</div>';
nDiv.innerHTML = iHTML;
document.getElementById('sidepanel-feed').insertBefore(nDiv,document.getElementById('sidepanel-feed').childNodes[0]);
uroAddEventListener('_uroFFCtrlVisibility','click',uroToggleFFCtrls, true);
uroShowFeedFilter = true;
uroToggleFFCtrls();
}
}
}
function uroTSTFeedFilter()
{
var feedEntries = document.getElementsByClassName('feed-item');
var feedLength = feedEntries.length;
if(feedLength === 0) return;
if(document.getElementById('uroFeedFilter') === null) return;
var hideFI;
var iHTML;
var fClass;
var ffKeyword = uroGetElmValue('_cbFeedFilter_Keyword');
var ffShowKW = uroGetCBChecked('_cbFeedFilter_ShowKeyword');
var ffHideKW = uroGetCBChecked('_cbFeedFilter_HideKeyword');
var kwPresent;
var transMotCBSBR = I18n.lookup("feed.issues.motivations.CAN_BE_SOLVED_BY_RANK");
var transMotCTF = I18n.lookup("feed.issues.motivations.CLOSE_TO_FAVORITES");
var transMotIA = I18n.lookup("feed.issues.motivations.ISSUE_AGE");
var transMotIR = I18n.lookup("feed.issues.motivations.ISSUE_REOPENED");
var transMotND = I18n.lookup("feed.issues.motivations.NEAR_DRIVES");
var transMotRBU = I18n.lookup("feed.issues.motivations.REPORTED_BY_USER");
var transMotUFI = I18n.lookup("feed.issues.motivations.USER_FOLLOWS_ISSUE");
var transMotUFILC = I18n.lookup("feed.issues.motivations.USER_FOLLOWS_ISSUE_LAST_COMMENT");
var isChecked_cbFeedFilter_TypeUR = uroGetCBChecked('_cbFeedFilter_TypeUR');
var isChecked_cbFeedFilter_TypePUR = uroGetCBChecked('_cbFeedFilter_TypePUR');
var isChecked_cbFeedFilter_TypePM = uroGetCBChecked('_cbFeedFilter_TypePM');
var isChecked_cbFeedFilter_MotCBSBR = uroGetCBChecked('_cbFeedFilter_MotCBSBR');
var isChecked_cbFeedFilter_MotCTF = uroGetCBChecked('_cbFeedFilter_MotCTF');
var isChecked_cbFeedFilter_MotIA = uroGetCBChecked('_cbFeedFilter_MotIA');
var isChecked_cbFeedFilter_MotIR = uroGetCBChecked('_cbFeedFilter_MotIR');
var isChecked_cbFeedFilter_MotND = uroGetCBChecked('_cbFeedFilter_MotND');
var isChecked_cbFeedFilter_MotRBU = uroGetCBChecked('_cbFeedFilter_MotRBU');
var isChecked_cbFeedFilter_MotUFI = uroGetCBChecked('_cbFeedFilter_MotUFI');
var isChecked_cbFeedFilter_MotUFILC = uroGetCBChecked('_cbFeedFilter_MotUFILC');
var isChecked_cbFeedFilter_MotNone = uroGetCBChecked('_cbFeedFilter_MotNone');
var isChecked_cbFeedFilter_Invert = uroGetCBChecked('_cbFeedFilter_Invert');
for(var i=0; i<feedLength; i++)
{
hideFI = false;
iHTML = feedEntries[i].innerHTML;
fClass = feedEntries[i].className;
if(isChecked_cbFeedFilter_TypeUR)
{
hideFI |= (fClass.indexOf('feed-issue-ur') != -1);
}
if(isChecked_cbFeedFilter_TypePUR)
{
hideFI |= (fClass.indexOf('feed-issue-pu') != -1);
}
if(isChecked_cbFeedFilter_TypePM)
{
hideFI |= (fClass.indexOf('feed-notification-pm') != -1);
}
if(isChecked_cbFeedFilter_MotCBSBR)
{
hideFI |= (iHTML.indexOf(transMotCBSBR) != -1);
}
if(isChecked_cbFeedFilter_MotCTF)
{
hideFI |= (iHTML.indexOf(transMotCTF) != -1);
}
if(isChecked_cbFeedFilter_MotIA)
{
hideFI |= (iHTML.indexOf(transMotIA) != -1);
}
if(isChecked_cbFeedFilter_MotIR)
{
hideFI |= (iHTML.indexOf(transMotIR) != -1);
}
if(isChecked_cbFeedFilter_MotND)
{
hideFI |= (iHTML.indexOf(transMotND) != -1);
}
if(isChecked_cbFeedFilter_MotRBU)
{
hideFI |= (iHTML.indexOf(transMotRBU) != -1);
}
if(isChecked_cbFeedFilter_MotUFI)
{
hideFI |= (iHTML.indexOf(transMotUFI) != -1);
}
if(isChecked_cbFeedFilter_MotUFILC)
{
hideFI |= (iHTML.indexOf(transMotUFILC) != -1);
}
if(isChecked_cbFeedFilter_MotNone)
{
hideFI |= (iHTML.indexOf('motivation') == -1);
}
if(isChecked_cbFeedFilter_Invert)
{
hideFI = !hideFI;
}
if((ffShowKW) || (ffHideKW))
{
kwPresent = (iHTML.indexOf(ffKeyword) != -1);
hideFI |= (kwPresent & ffHideKW);
hideFI |= ((!kwPresent) & ffShowKW);
}
if(hideFI) feedEntries[i].style.display = 'none';
else feedEntries[i].style.display = 'block';
}
}
//}
function uroFinalizeListenerSetup()
{
uroFinalisingListenerSetup = true;
// filter markers when the marker objects are modified (this happens whenever WME needs to load fresh marker data
// due to having panned/zoomed the map beyond the extents of the previously loaded data)
W.model.mapUpdateRequests.on("objectschanged", uroFilterURs_onObjectsChanged);
W.model.mapUpdateRequests.on("objectsadded", uroFilterURs_onObjectsAdded);
W.model.mapUpdateRequests.on("objectsremoved", uroFilterURs_onObjectsRemoved);
W.model.updateRequestSessions.on("objectschanged", uroUREvent_onObjectsChanged);
W.model.updateRequestSessions.on("objectsadded", uroUREvent_onObjectsAdded);
W.model.updateRequestSessions.on("objectsremoved", uroUREvent_onObjectsRemoved);
W.model.cameras.on("objectschanged", uroFilterCameras);
W.model.cameras.on("objectsadded", uroFilterCameras);
W.model.cameras.on("objectsremoved", uroFilterCameras);
W.model.problems.on("objectschanged", uroFilterProblems);
W.model.problems.on("objectsadded", uroFilterProblems);
W.model.problems.on("objectsremoved", uroFilterProblems);
W.model.venues.on("objectschanged", uroFilterPlaces);
W.model.venues.on("objectsadded", uroFilterPlaces);
W.model.venues.on("objectsremoved", uroFilterPlaces);
var userTabs = document.getElementById(uroUserTabId);
var tabContent = null;
var navTabs = userTabs.getElementsByClassName('nav-tabs')[0];
tabContent = document.getElementById('user-info').getElementsByClassName('tab-content')[0];
var newtabUR = document.createElement('li');
newtabUR.innerHTML = '<a href="#sidepanel-uroverview" data-toggle="tab">URO+</a>';
navTabs.appendChild(newtabUR);
uroControls.id = "sidepanel-uroverview";
uroControls.className = "tab-pane";
tabContent.appendChild(uroControls);
uroAddEventListener('_btnUndoLastHide',"click", uroRemoveLastAddedIgnore, true);
uroAddEventListener('_btnClearSessionHides',"click", uroRemoveAllIgnores, true);
uroEnableIgnoreListControls();
uroAddEventListener('_btnClearCamWatchList',"click", uroClearCamWatchList, true);
uroAddEventListener('_btnSettingsToText',"click", uroSettingsToText, true);
uroAddEventListener('_btnTextToSettings',"click", uroTextToSettings, true);
uroAddEventListener('_btnResetSettings',"click", uroDefaultSettings, true);
uroAddEventListener('_btnClearSettingsText',"click", uroClearSettingsText, true);
uroAddEventListener('_cbMasterEnable',"click", uroFilterItems_MasterEnableClick, true);
/*
uroAddEventListener('_btnDebugToScreen',"click", uroDumpDebug, true);
*/
uroAddEventListener('uroDiv',"dblclick",uroSuppressPopup,true);
uroAddEventListener('_selectCameraUserID',"change", uroCamEditorSelected, true);
uroAddEventListener('uroAlertTickBtn','click',uroCloseAlertBoxWithTick,true);
uroAddEventListener('uroAlertCrossBtn','click',uroCloseAlertBoxWithCross,true);
uroSetOnClick("_linkSelectUserRequests",uroShowURTab);
uroSetOnClick("_linkSelectMapProblems",uroShowMPTab);
uroSetOnClick("_linkSelectPlaces",uroShowPlacesTab);
uroSetOnClick("_linkSelectCameras",uroShowCameraTab);
uroSetOnClick("_linkSelectMisc",uroShowMiscTab);
uroSetOnClick("_linkSelectOWL",uroShowOWLTab);
for(var idx=0;idx<W.Config.venues.categories.length;idx++)
{
uroSetOnClick('_uroPlacesGroupState-'+idx,uroPlacesGroupCollapseExpand);
}
uroAddLog('finalise onload');
uroLoadSettings();
uroNewLookCheckDetailsRequest();
if(uroGetCBChecked('_cbEnableDTE'))
{
if(dteControlsIdx != -1)
{
dteSetNewTabLength();
}
else
{
uroAddLog('ERROR - archive panel not found!');
uroSetStyleDisplay(uroUserTabId,'');
}
}
// filter markers as and when the map is moved
W.map.events.register("moveend", null, uroFilterItems);
W.map.events.register("mousemove", null, uroGetAMs);
W.map.events.register("mousemove", null, uroNewLookHighlightedItemsCheck);
W.map.events.registerPriority("mousedown", null, uroMouseDown);
// trap mousedown on Streetview marker drag
document.getElementsByClassName('street-view-control')[0].onmousedown = uroMouseDown;
W.map.events.register("mouseup", null, uroMouseUp);
W.map.events.register("mouseout", null, uroMouseOut);
uroSetStyles(uroCtrlURs);
uroSetStyles(uroCtrlMPs);
uroSetStyles(uroCtrlPlaces);
uroSetStyles(uroCtrlCameras);
uroSetStyles(uroCtrlMisc);
uroSetStyles(uroOWL);
uroAddFeedFilterControls();
uroShowURTab();
uroUserID = W.loginManager.getLoggedInUser().id;
uroFilterItems();
uroShowDebugOutput = uroPersistentDebugOutput;
var dbgMode = "none";
if(uroShowDebugOutput)
{
dbgMode = "inline";
}
document.getElementById('_uroDebugMode').style.display = dbgMode;
uroAddEventListener('_uroVersion',"click", uroToggleDebug, true);
// add exclusiveCB click handlers to all checkboxes with a pairedWith attribute
var cbList = document.getElementsByTagName('input');
for (var optIdx=0;optIdx<cbList.length;optIdx++)
{
if((cbList[optIdx].id.indexOf('_cb') === 0) && (cbList[optIdx].attributes.pairedWith != null))
{
uroSetOnClick(cbList[optIdx].id,uroExclusiveCB);
}
}
uroSetupListeners = false;
uroInitialised = true;
}
function uroTSTPopupHandler()
{
if(document.getElementsByClassName('panel')[0] === undefined)
{
uroHidePopupOnPanelOpen = true;
}
if(uroPopupShown === true)
{
var hidePopup = false;
if(document.getElementsByClassName('dropdown action open').length > 0.5)
{
hidePopup = true;
}
else
{
if(window.getComputedStyle(document.getElementById('layer-switcher-list').parentNode).getPropertyValue('opacity') > 0.5)
{
hidePopup = true;
}
else
{
if(window.getComputedStyle(document.getElementsByClassName('toolbar-group-drawing')[0].childNodes[0]).getPropertyValue('opacity')> 0.5)
{
hidePopup = true;
}
else
{
if(window.getComputedStyle(document.getElementsByClassName('toolbar-group-venues')[0].childNodes[0]).getPropertyValue('opacity') > 0.5)
{
hidePopup = true;
}
}
}
}
if(document.getElementsByClassName('panel')[0] != null)
{
if(uroHidePopupOnPanelOpen === true)
{
hidePopup = true;
uroHidePopupOnPanelOpen = false;
}
}
if(hidePopup === true)
{
uroHidePopup();
}
}
if((uroAreaNameHoverObj !== null) && (uroAreaNameHoverTime != -1) && (uroAreaNameOverlayShown === false))
{
if(++uroAreaNameHoverTime > 5)
{
uroAreaNameOverlaySetup();
}
}
uroReplaceAreaNames(false);
if(uroPopupTimer > 0)
{
if(uroMouseInPopup === false)
{
uroPopupTimer--;
}
}
if(uroPopupTimer === 0)
{
uroHidePopup();
}
if(uroPopupDwellTimer > 0)
{
uroPopupDwellTimer--;
if(uroPopupDwellTimer === 0)
{
uroNewLookHighlightedItemsCheck('dwellTimeout');
}
}
}
function uroTSTDTEHandler()
{
if(document.getElementsByClassName("archive-panel")[0] === undefined)
{
if(dteClearHighlightsOnPanelClose)
{
dteClearListHighlight();
dteClearHighlightsOnPanelClose = false;
}
}
else
{
if(dteArmClearHighlightsOnPanelClose)
{
dteArmClearHighlightsOnPanelClose = false;
dteClearHighlightsOnPanelClose = true;
}
}
}
function uroTSTNextBtnHandler()
{
// replace the "next xxx" button on UR, MP and PUR editing UIs
if(W.map.panelRegion.hasView() === true)
{
var nurButton = W.map.panelRegion.$el[0].getElementsByClassName('btn btn-block')[0];
if(nurButton != null)
{
var doneString = I18n.lookup('problems.panel.done');
var nextURString = (nurButton.innerHTML.indexOf(I18n.lookup('update_requests.panel.next')) !== -1);
var nextMPString = (nurButton.innerHTML.indexOf(I18n.lookup('problems.panel.next')) !== -1);
var nextPURString = (nurButton.innerHTML.indexOf(I18n.lookup('venues.update_requests.panel.next_venue')) !== -1);
var nextIssueString = (nurButton.innerHTML.indexOf(I18n.lookup('feed.issues.next')) !== -1);
var updateButton = false;
var panelClass = W.map.panelRegion.$el[0].childNodes[0].childNodes[0].className;
var isURorMPPanel = (panelClass.indexOf('problem-edit') !== -1);
var isPURPanel = (panelClass.indexOf('place-update') !== -1);
// "next" button class used for UR and MP edit panel
if(isURorMPPanel === true)
{
// user has enabled UR button mod?
if(uroGetCBChecked('_cbInhibitNURButton') === true)
{
// mod the button if this is a UR/MP panel rather than a PUR panel, and if the "Next update request" or "Next issue"
// text is currently being used on the button - WME uses the former text when a UR is selected directly from the map
// pin, and the latter text when a UR is selected from the feed. However, it *also* uses the latter text for PURs
// selected from the feed, hence the need to test the panel type here as well - having to work around odd
// inconsistencies in WME behaviour, whatever next...
updateButton = updateButton || ((nextURString === true) || (nextIssueString === true));
}
// user has enabled MP button mod?
if(uroGetCBChecked('_cbInhibitNMPButton') === true)
{
// although there's no easy way to determine whether the panel is opened for a UR or MP, since the feed doesn't yet seem
// to provide a means of accessing MPs then we don't yet have to worry about finding the "Next issue" text in a MP panel,
// which makes life slightly easier here...
updateButton = updateButton || (nextMPString === true);
}
}
else if(isPURPanel === true)
{
// "next-venue" class only used for PUR edit panel
if(uroGetCBChecked('_cbInhibitNPURButton') === true)
{
// as noted above, PURs opened from the feed may also use the "Next issue" text, so we need to test for that as well,
// knowing that in order to get this far we've already determined that the user really does want to mod the PUR
// next button *and* that the currently open panel is a PUR panel...
updateButton = updateButton || ((nextPURString === true) || (nextIssueString === true));
}
}
if(updateButton === true)
{
// if we need to change the button label back to "Done", do it here...
nurButton.innerHTML = doneString;
uroAddLog('inhibit Next UR/MP/PUR button');
}
// if updateButton isn't already set here, it suggests the panel is using the native "Done" button...
if(updateButton === false)
{
nurButton = W.map.panelRegion.$el[0].getElementsByClassName('btn btn-block done')[0];
if(nurButton != null)
{
updateButton = true;
}
}
if(updateButton === true)
{
// Add a new click handler to override the native one - this acts both to prevent the normal action of the "Next UR/MP/PUR" button in
// moving to the next UR/MP/PUR, and also allows us to warn about closing the UR panel if there's an unsent comment...
nurButton.addEventListener("click", uroInhibitNextUpdateRequestButton, false);
}
}
}
}
function uroTSTCommentAddedHandler()
{
// test for the opening or closing of the UR editing dialog so we can detect when a new comment is added
var URDialogIsOpen = (document.getElementsByClassName('new-comment-form').length == 1);
if(URDialogIsOpen)
{
var thisSelectedURID = document.getElementsByClassName('permalink')[0].href.split('&mapUpdateRequest=')[1].split('&')[0];
if(thisSelectedURID != uroSelectedURID)
{
// if the user selects a new UR whilst the editing dialog is still open, treat it in the
// same way as if the user had selected that UR with the dialog closed
uroURDialogIsOpen = false;
}
if(uroURDialogIsOpen === false)
{
// user is editing a new UR
uroSelectedURID = thisSelectedURID;
// add our own click event handler to the Send button, so we can do stuff whenever a new comment is added
document.getElementsByClassName('new-comment-form')[0].getElementsByClassName('btn')[0].addEventListener("click", uroAddedComment, false);
uroAddLog('user is editing UR '+uroSelectedURID);
uroExpectedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
if((uroHoveredURID !== null) && (uroSelectedURID !== null) && (parseInt(uroHoveredURID) !== parseInt(uroSelectedURID)))
{
if(++uroURReclickAttempts < 3)
{
uroAddLog('DANGER, WILL ROBINSON! You clicked on UR ID '+uroHoveredURID+' but WME has loaded the details for UR ID '+uroSelectedURID+' instead, attempting to fix...');
W.map.updateRequestLayer.markers['+uroHoveredURID+'].icon.$div.click();
return;
}
else
{
uroAddLog('Woe is me, attempting to open UR ID '+uroHoveredURID+' has failed...');
uroShowAlertBox('fa-warning', 'URO+ Warning', 'WME may have opened the details panel for a different UR to the one you selected, proceed with caution', false, "OK", "", null, null);
}
}
uroURReclickAttempts = 0;
}
}
else if(uroURDialogIsOpen === true)
{
// dialog was open and has now been closed
uroSelectedURID = null;
}
uroURDialogIsOpen = URDialogIsOpen;
if(((uroPendingCommentDataRefresh === true) || (uroWaitingCommentDataRefresh === true)) && (uroSelectedURID !== null))
{
uroAddLog('check completion of comment data refresh for UR '+uroSelectedURID+' ('+uroPendingCommentDataRefresh+','+uroWaitingCommentDataRefresh+')');
uroGetSelectedURCommentCount();
}
}
function uroTSTOWLHandler()
{
/*
var selectedTotal = W.selectionManager.selectedItems.length;
if((selectedTotal > 0) && (document.getElementById('_uroDivOWLBtns') === null))
{
var selectedClass = W.selectionManager.selectedItems[0].model.CLASS_NAME;
var displayAddToOWLBtn = false;
var displayUpdateOWLBtn = false;
var displayRemoveFromOWLBtn = false;
var selectedSegments = false;
var selectedLandmarks = false;
var fid;
var loop;
// WME only seems to allow multi-object selections for segments, so testing the class of the first object in the
// selection list tells us the class of any other objects in the list too...
if(selectedClass == "Waze.Feature.Vector.Segment")
{
selectedSegments = true;
for(loop=0; loop<selectedTotal; loop++)
{
fid = W.selectionManager.selectedItems[loop].model.attributes.id;
var segIdx = uroIsSegOnWatchList(fid);
if(segIdx == -1)
{
displayAddToOWLBtn = true;
}
else
{
if(uroSegDataChanged(segIdx))
{
displayUpdateOWLBtn = true;
}
displayRemoveFromOWLBtn = true;
}
}
}
else if(selectedClass == "Waze.Feature.Vector.Landmark")
{
selectedLandmarks = true;
for(loop=0; loop<selectedTotal; loop++)
{
fid = W.selectionManager.selectedItems[loop].model.attributes.id;
var placeIdx = uroIsPlaceOnWatchList(fid);
if(placeIdx == -1)
{
displayAddToOWLBtn = true;
}
else
{
if(uroPlaceDataChanged(placeIdx))
{
displayUpdateOWLBtn = true;
}
displayRemoveFromOWLBtn = true;
}
}
}
var btnHTML = '<div id="_uroDivOWLBtns">';
if((displayAddToOWLBtn === true) && (displayUpdateOWLBtn === false))
{
btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Add to OWL</button>';
}
else if((displayUpdateOWLBtn === true) && (displayAddToOWLBtn === false))
{
btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Update OWL</button>';
}
else if((displayAddToOWLBtn === true) && (displayUpdateOWLBtn === true))
{
btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Add to & Update OWL</button>';
}
if(displayRemoveFromOWLBtn === true)
{
btnHTML += '<button class="btn btn-default" id="_btnRemoveOWL">Remove from OWL</button>';
}
btnHTML += '</div>';
// note to self... altering the inner HTML of the segment-edit-general panel when the selected
// segment is part of a roundabout always used to disable the onclick handler for the select
// roundabout button. will need to see how this behaves in the current WME given the changes in
// panel arrangement and the introduction of the native select roundabout button
if(selectedSegments === true)
{
document.getElementById("segment-edit-general").innerHTML += btnHTML;
}
else if(selectedLandmarks === true)
{
document.getElementById("landmark-edit-general").innerHTML += btnHTML;
}
if((displayAddToOWLBtn === true)||(displayUpdateOWLBtn === true))
{
if(selectedSegments === true)
{
uroAddEventListener('_btnAddUpdateOWL','click', uroAddUpdateSegWatchList, true);
}
else
{
uroAddEventListener('_btnAddUpdateOWL','click', uroAddUpdatePlaceWatchList, true);
}
}
if(displayRemoveFromOWLBtn === true)
{
if(selectedSegments === true)
{
uroAddEventListener('_btnRemoveOWL','click', uroRemoveSegFromWatchList, true);
}
else
{
uroAddEventListener('_btnRemoveOWL','click', uroRemovePlaceFromWatchList, true);
}
}
}
*/
}
function uroTSTClosureCloningHandler()
{
// closure cloning support...
//
// has the closures tab been generated?
if(document.getElementById('segment-edit-closures') !== null)
{
// and is it active?
if(document.getElementById('segment-edit-closures').className === 'tab-pane active')
{
// and are there any closures defined for all of the selected segment(s)...
var nClosures = document.getElementsByClassName('full-closures')[0].childNodes.length;
if(nClosures > 0)
{
// and last but by no means least, have we already added the clone icon to this closure?
for(var cLoop = 0; cLoop < nClosures; cLoop++)
{
var btnElm = document.getElementsByClassName('full-closures')[0].childNodes[cLoop].children[0].children[0];
if(btnElm.innerHTML.indexOf('_uroCloneClosure-') == -1)
{
var newAnchor = document.createElement('a');
var anchorID = '_uroCloneClosure-'+cLoop;
newAnchor.href="#";
newAnchor.innerHTML = "<i class='fa fa-copy'></i>";
newAnchor.id = anchorID;
btnElm.appendChild(newAnchor);
uroAddEventListener(anchorID,"click",uroCloneClosure,false);
}
}
}
// if there's more than one closure (full or partial) listed, also add the delete all button if not already present...
if(document.getElementsByClassName('closure-item').length > 1)
{
if(document.getElementById('_btnDeleteAllClosures') === null)
{
var daDiv = document.createElement('div');
daDiv.className = 'delete-all-button btn btn-primary';
daDiv.id = '_btnDeleteAllClosures';
daDiv.innerHTML = '<i class="fa fa-trash"></i> '+I18n.lookup("closures.delete_confirm_no_reason")+' ('+I18n.lookup("closures.apply_to_all")+')';
daDiv.style.width = '100%';
daDiv.style.marginBottom = '10px';
var acBtn = document.getElementsByClassName('add-closure-button')[0];
acBtn.parentNode.insertBefore(daDiv, acBtn.nextSibling);
uroAddEventListener('_btnDeleteAllClosures',"click", uroDeleteAllClosures, false);
}
}
}
}
}
function uroMiscUITweaksHandler()
{
if(uroFilterPreamble())
{
// give user the option of setting their own background colour...
{
var mapviewport = document.getElementsByClassName("olMapViewport")[0];
if((uroGetCBChecked('_cbWhiteBackground') === true) && (uroGetCBChecked('_cbMasterEnable') === true))
{
var customColour = '#' + uroToHex(uroGetElmValue('_inputCustomBackgroundRed'),2);
customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundGreen'),2);
customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundBlue'),2);
mapviewport.style.backgroundColor = customColour;
}
else
{
mapviewport.style.backgroundColor = "#C2C2C2";
}
}
// allows user to hide the area managers layer without switching off the layer completely...
{
// ...if this sounds like a weird option - why not just switch off the layer from the layers menu? - then
// remember that in order for URO+ to be able to display in its own tab the list of AMs under the current
// mouse pointer location, which is somewhat more useful than the list given in the topbar, it needs the
// AM layer to be activated so that the AM areas data is loaded into WME. It doesn't however need the layer
// to then be visible, and since having a bunch of purple polygons covering the map can make for a rather
// difficult editing experience, being able to hide the polys whilst retaining the area information is
// of real benefit...
if((uroGetCBChecked('_cbHideAMLayer')) && (uroGetCBChecked('_cbMasterEnable')))
{
W.map.managedAreasLayer.setOpacity(0);
}
else
{
W.map.managedAreasLayer.setOpacity(1);
}
}
// gives user the option of minimising the size of the sidebar tabs to save space
{
if(!uroGetCBChecked('_cbDisableTabStyling'))
{
// The nav-tabs class is now also used for the General/Closures tabs on the segment edit panel, so we have
// to restrict the scope of this code to just those nav-tab classed elements within the user-tabs element.
var navTabs = document.getElementById('user-tabs').getElementsByClassName("nav-tabs")[0].children;
for(var loop = 0; loop<navTabs.length; loop++)
{
navTabs[loop].children[0].style.padding = "4px";
}
}
}
// gives user the option of hiding the somewhat unnecessary editor info panel at the top of the sidebar
{
var panelDisplay = '';
if(uroGetCBChecked('_cbHideEditorInfo'))
{
panelDisplay = "none";
}
document.getElementById("user-details").style.display = panelDisplay;
}
}
}
function uroTenthSecondTick()
{
if(uroMTEMode) return;
if(uroSetupListeners)
{
if(uroFinalisingListenerSetup === false)
{
if(W.loginManager.isLoggedIn())
{
uroFinalizeListenerSetup();
}
}
}
else
{
uroTSTPopupHandler();
uroTSTDTEHandler();
uroTSTNextBtnHandler();
uroTSTCommentAddedHandler();
uroTSTOWLHandler();
uroTSTClosureCloningHandler();
uroTSTFeedFilter();
uroMiscUITweaksHandler();
}
}
function uroActiveTab(_id)
{
var e = document.getElementById(_id);
e.style.backgroundColor = "aliceblue";
e.style.borderTop = "1px solid";
e.style.borderLeft = "1px solid";
e.style.borderRight = "1px solid";
e.style.borderBottom = "0px solid";
}
function uroInactiveTab(_id)
{
var e = document.getElementById(_id);
e.style.backgroundColor = "white";
e.style.borderTop = "0px solid";
e.style.borderLeft = "0px solid";
e.style.borderRight = "0px solid";
e.style.borderBottom = "1px solid";
}
function uroInactiveAllTabs()
{
uroInactiveTab("_tabSelectCameras");
uroInactiveTab("_tabSelectMapProblems");
uroInactiveTab("_tabSelectMisc");
uroInactiveTab("_tabSelectUserRequests");
uroInactiveTab("_tabSelectCWL");
uroInactiveTab("_tabSelectPlaces");
if(!uroCtrlsHidden)
{
uroSetStyleDisplay('uroCtrlURs','none');
uroSetStyleDisplay('uroCtrlMPs','none');
uroSetStyleDisplay('uroCtrlCameras','none');
uroSetStyleDisplay('uroCtrlMisc','none');
uroSetStyleDisplay('uroOWL','none');
uroSetStyleDisplay('uroCtrlPlaces','none');
}
}
function uroShowURTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectUserRequests");
uroCurrentTab = 1;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlURs','block');
return false;
}
function uroShowMPTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectMapProblems");
uroCurrentTab = 2;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMPs','block');
return false;
}
function uroShowPlacesTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectPlaces");
uroCurrentTab = 3;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlPlaces','block');
for(var idx=0;idx<uroPlacesGroupsCollapsed.length;idx++)
{
uroPlacesGroupCEHandler(idx);
}
return false;
}
function uroShowCameraTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectCameras");
uroCurrentTab = 4;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlCameras','block');
return false;
}
function uroShowOWLTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectCWL");
uroCurrentTab = 5;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroOWL','block');
uroOWLUpdateHTML();
return false;
}
function uroShowMiscTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectMisc");
uroCurrentTab = 6;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMisc','block');
return false;
}
function uroNewLookCheckDetailsRequest()
{
var thisurl = document.location.href;
var doRetry = true;
var urID;
var endmarkerpos = thisurl.indexOf('&endshow');
var showmarkerpos = thisurl.indexOf('&showturn=');
if((endmarkerpos != -1) && (showmarkerpos != -1))
{
showmarkerpos += 10;
uroAddLog('showturn tab opened');
urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
uroAddLog(' turn problem ID = '+urID);
try
{
W.map.problemLayer.markers[urID].icon.imageDiv.click();
doRetry = false;
}
catch(err)
{
uroAddLog('problems not fully loaded, retrying...');
}
if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
}
else
{
showmarkerpos = thisurl.indexOf('&showpur=');
if((endmarkerpos != -1) && (showmarkerpos != -1))
{
showmarkerpos += 9;
uroAddLog('showPUR tab opened');
urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
uroAddLog(' PUR ID = '+urID);
try
{
W.map.placeUpdatesLayer.markers[urID].icon.imageDiv.click();
doRetry = false;
}
catch(err)
{
uroAddLog('PURs not fully loaded, retrying...');
}
if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
}
}
}
function uroUpdateMPSolverList()
{
if(Object.keys(W.model.problems.objects).length === 0)
{
return;
}
var resolverList = [];
var selector = document.getElementById('_selectMPUserID');
var selectedUser = null;
if(selector.selectedOptions[0] != null)
{
selectedUser = parseInt(selector.selectedOptions[0].value);
}
while(selector.options.length > 0)
{
selector.options.remove(0);
}
var selectedIdx = 0;
var idx = 0;
for (var mpobj in W.model.problems.objects)
{
if(W.model.problems.objects.hasOwnProperty(mpobj))
{
var prob = W.model.problems.objects[mpobj];
if(prob.attributes.resolvedBy !== null)
{
var userID = prob.attributes.resolvedBy;
var userName = W.model.users.objects[userID].userName;
if(resolverList.indexOf(userName) == -1)
{
resolverList.push(userName);
selector.options.add(new Option(userName, userID));
if(userID == selectedUser)
{
selectedIdx = idx;
}
idx++;
}
}
}
}
if(selectedIdx !== null)
{
selector.selectedIndex = selectedIdx;
}
}
function uroUpdateResolverList()
{
if(Object.keys(W.model.mapUpdateRequests.objects).length === 0)
{
return;
}
var resolverList = [];
var selector = document.getElementById('_selectURResolverID');
var selectedUser = null;
if(selector.selectedOptions[0] != null)
{
selectedUser = parseInt(selector.selectedOptions[0].value);
}
while(selector.options.length > 0)
{
selector.options.remove(0);
}
var selectedIdx = 0;
var idx = 0;
for (var urobj in W.model.mapUpdateRequests.objects)
{
if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
{
var ureq = W.model.mapUpdateRequests.objects[urobj];
if(ureq.attributes.resolvedBy !== null)
{
var userID = ureq.attributes.resolvedBy;
var userName = W.model.users.objects[userID].userName;
if(resolverList.indexOf(userName) == -1)
{
resolverList.push(userName);
selector.options.add(new Option(userName, userID));
if(userID == selectedUser)
{
selectedIdx = idx;
}
idx++;
}
}
}
}
if(selectedIdx !== null)
{
selector.selectedIndex = selectedIdx;
}
}
function uroUpdateUserList()
{
if(Object.keys(W.model.updateRequestSessions.objects).length === 0) return;
var selector = document.getElementById('_selectURUserID');
var selectedUser = null;
if(selector.selectedOptions[0] != null)
{
selectedUser = parseInt(selector.selectedOptions[0].value);
}
while(selector.options.length > 0)
{
selector.options.remove(0);
}
var selectedIdx = null;
var listedIDs = [];
for(var ursIdx in W.model.updateRequestSessions.objects)
{
if(W.model.updateRequestSessions.objects.hasOwnProperty(ursIdx))
{
var ursObj = W.model.updateRequestSessions.objects[ursIdx];
if(ursObj.comments.length > 0)
{
for(var cidx=0; cidx < ursObj.comments.length; cidx++)
{
var userID = ursObj.comments[cidx].userID;
if((listedIDs.indexOf(userID) == -1) && (userID != -1))
{
listedIDs.push(userID);
}
}
}
}
}
if(listedIDs.length > 0)
{
var users = W.model.users.getByIds(listedIDs);
for(var idx=0; idx<listedIDs.length; idx++)
{
selector.options.add(new Option(users[idx].userName, listedIDs[idx]));
if(listedIDs[idx] == selectedUser)
{
selectedIdx = idx;
}
}
}
if(selectedIdx !== null)
{
selector.selectedIndex = selectedIdx;
}
}
function uroUpdateCamEditorList()
{
if(Object.keys(W.model.cameras.objects).length === 0) return;
var selector = document.getElementById('_selectCameraUserID');
var selectedUser = null;
if(selector.selectedOptions[0] != null)
{
selectedUser = parseInt(selector.selectedOptions[0].value);
}
while(selector.options.length > 0)
{
selector.options.remove(0);
}
var selectedIdx = null;
var listedIDs = [];
for(var camIdx in W.model.cameras.objects)
{
if(W.model.cameras.objects.hasOwnProperty(camIdx))
{
var camObj = W.model.cameras.objects[camIdx].attributes;
var cbID = camObj.createdBy;
var ubID = camObj.updatedBy;
if((cbID !== null) && (listedIDs.indexOf(cbID) == -1))
{
listedIDs.push(cbID);
}
if((ubID !== null) && (ubID !== cbID) && (listedIDs.indexOf(ubID) == -1))
{
listedIDs.push(ubID);
}
}
}
selector.options.add(new Option('<select a user>', null));
if(listedIDs.length > 0)
{
var users = W.model.users.getByIds(listedIDs);
var selectorEntry = '';
for(var idx=0; idx<users.length; idx++)
{
if(users[idx].userName === undefined)
{
selectorEntry = users[idx].id;
}
else
{
selectorEntry = users[idx].userName;
}
selector.options.add(new Option(selectorEntry, users[idx].id));
if(users[idx].id == selectedUser)
{
selectedIdx = idx+1;
}
}
}
if(selectedIdx !== null)
{
selector.selectedIndex = selectedIdx;
}
}
function uroCamEditorSelected()
{
var selector = document.getElementById('_selectCameraUserID');
if(selector.selectedIndex > 0)
{
document.getElementById('_textCameraEditor').value = document.getElementById('_selectCameraUserID').selectedOptions[0].innerHTML;
}
}
function uroSetStyles(obj)
{
obj.style.fontSize = '12px';
obj.style.lineHeight = '100%';
obj.style.overflow = 'auto';
obj.style.height = (window.innerHeight * 0.55) + 'px';
}
function uroPlacesGroupCEHandler(groupidx)
{
if(uroPlacesGroupsCollapsed[groupidx] === false)
{
document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "block";
document.getElementById('_uroPlacesGroupState-'+groupidx).className = "fa fa-minus-square-o";
}
else
{
document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "none";
document.getElementById('_uroPlacesGroupState-'+groupidx).className = "fa fa-plus-square-o";
}
}
function uroPlacesGroupCollapseExpand()
{
var groupidx = this.id.substr(21);
if(uroPlacesGroupsCollapsed[groupidx] === true) uroPlacesGroupsCollapsed[groupidx] = false;
else uroPlacesGroupsCollapsed[groupidx] = true;
uroPlacesGroupCEHandler(groupidx);
return false;
}
function uroPopulatePlacesTab()
{
var tHTML = '';
tHTML += '<b>Filter PURs by category/status:</b><br>';
tHTML += '<input type="checkbox" id="_cbFilterUneditablePlaceUpdates">Ones I can\'t edit</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterLockRankedPlaceUpdates">Ones with non-zero lockRanks</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterNewPlacePUR">Ones for new places</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterUpdatedDetailsPUR">Ones for updated place details</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterNewPhotoPUR">Ones for new photos</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterFlaggedPUR">Ones flagged for attention</input><br>';
tHTML += '<br><input type="checkbox" id="_cbLeavePURGeos">Don\'t hide place polygons/points</input><br>';
tHTML += '<br><input type="checkbox" id="_cbInvertPURFilters">Invert PUR filters</input><br>';
tHTML += '<br><b>Filter PURs by severity:</b><br>';
tHTML += '<input type="checkbox" id="_cbPURFilterLowSeverity">Low</input> ';
tHTML += '<input type="checkbox" id="_cbPURFilterMediumSeverity">Medium</input> ';
tHTML += '<input type="checkbox" id="_cbPURFilterHighSeverity">High</input>';
tHTML += '<br><b>Filter PURs by age of submission:</b><br>';
tHTML += '<input type="checkbox" id="_cbEnablePURMinAgeFilter">Hide PURs less than </input>';
tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPURFilterMinDays"> days old<br>';
tHTML += '<input type="checkbox" id="_cbEnablePURMaxAgeFilter">Hide PURs more than </input>';
tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPURFilterMaxDays"> days old<br>';
tHTML += '<hr>';
tHTML += '<br><b>Filter Places by state:</b><br>';
tHTML += 'Hide if last edited<br>';
tHTML += '<input type="checkbox" id="_cbPlaceFilterEditedLessThan"> less than </input>';
tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterPlaceEditMinDays"> days ago<br>';
tHTML += '<input type="checkbox" id="_cbPlaceFilterEditedMoreThan"> more than </input>';
tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterPlaceEditMaxDays"> days ago<br>';
tHTML += '<br>Hide if locked at level:<br>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL0">1</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL1">2</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL2">3</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL3">4</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL4">5</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL5">6</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesStaff">Staff</input>';
tHTML += '<br> <input type="checkbox" id="_cbHidePlacesAdLocked">AdLocked</input><br>';
tHTML += '<br>Hide by geometry:<br>';
tHTML += ' <input type="checkbox" id="_cbHideAreaPlaces">Areas</input>';
tHTML += ' <input type="checkbox" id="_cbHidePointPlaces">Points</input>';
tHTML += '<br><br><input type="checkbox" id="_cbHidePhotoPlaces" pairedWith="_cbHideNoPhotoPlaces">Hide or </input>';
tHTML += '<input type="checkbox" id="_cbHideNoPhotoPlaces" pairedWith="_cbHidePhotoPlaces">show ones with photos</input><br>';
tHTML += '<input type="checkbox" id="_cbHideLinkedPlaces" pairedWith="_cbHideNoLinkedPlaces">Hide or </input>';
tHTML += '<input type="checkbox" id="_cbHideNoLinkedPlaces" pairedWith="_cbHideLinkedPlaces">show ones with external links</input><br>';
tHTML += '<input type="checkbox" id="_cbHideKeywordPlaces" pairedWith="_cbHideNoKeywordPlaces">Hide or </input>';
tHTML += '<input type="checkbox" id="_cbHideNoKeywordPlaces" pairedWith="_cbHideKeywordPlaces">show ones with a name including</input><br>';
tHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPlace"><br>';
tHTML += '<br><br><b>Filter Places by category:</b><br>';
var nCategories = W.Config.venues.categories.length;
var i;
if(uroPlacesGroupsCollapsed.length != nCategories)
{
for(i=0; i<nCategories; i++)
{
uroPlacesGroupsCollapsed.push(false);
}
}
for(i=0; i<nCategories; i++)
{
var parentCategory = W.Config.venues.categories[i];
var localisedName = I18n.lookup("venues.categories." + parentCategory);
if(uroPlacesGroupsCollapsed[i] === true)
{
tHTML += '<i class="fa fa-plus-square-o" style="cursor:pointer;font-size:14px;" id="_uroPlacesGroupState-'+i+'"></i>';
}
else
{
tHTML += '<i class="fa fa-minus-square-o" style="cursor:pointer;font-size:14px;" id="_uroPlacesGroupState-'+i+'"></i>';
}
tHTML += ' <input type="checkbox" id="_cbPlacesFilter-'+parentCategory+'"><b>'+localisedName+'</b></input><br>';
tHTML += '<div id="_uroPlacesGroup-'+i+'" style="padding:3px;border-width:2px;border-style:solid;border-color:#FFFFFF">';
for(var ii=0; ii<W.Config.venues.subcategories[parentCategory].length; ii++)
{
var subCategory = W.Config.venues.subcategories[parentCategory][ii];
localisedName = I18n.lookup("venues.categories." + subCategory);
tHTML += ' <input type="checkbox" id="_cbPlacesFilter-'+subCategory+'">'+localisedName+'</input><br>';
}
tHTML += '</div>';
}
tHTML += '<input type="checkbox" id="_cbFilterPrivatePlaces"><b>Residential Places</b></input><br>';
tHTML += '<br><input type="checkbox" id="_cbInvertPlacesFilter">Invert Place filters?</input>';
uroCtrlPlaces.innerHTML = tHTML;
}
function uroWazeBits()
{
// "fake" uroWazeBits() function which only performs layer scan, to stop the uroWazeBits() call in WMETB from
// messing around with other stuff in the actual uroWazeBits() function (now renamed uroRealWazeBits...) that
// really only ought to be called once.
var i;
for(i=0;i<W.map.layers.length;i++)
{
if(W.map.layers[i].name == 'Spotlight') uroMaskLayer = i;
if(W.map.layers[i].CLASS_NAME == 'OpenLayers.Layer.Vector.RootContainer') uroRootContainer = W.map.layers[i].div.id;
if(W.map.layers[i].name == 'Node Connections') uroTurnsLayerIdx = i;
}
uroPlacesRoot = W.map.landmarkLayer.id + '_vroot';
for(i=0;i<W.map.controls.length;i++)
{
if(W.map.controls[i].CLASS_NAME == 'Waze.View.ArchivePanel') dteControlsIdx = i;
else if(W.map.controls[i].CLASS_NAME == 'Waze.Control.Archive') dteControlsIdx = i;
if(W.map.controls[i].id !== null)
{
if(W.map.controls[i].id.indexOf('UpdateRequests') != -1) uroURControlsIdx = i;
if(W.map.controls[i].id.indexOf('MapProblems') != -1) uroProblemControlsIdx = i;
}
}
uroAddLog('uroMaskLayer at idx '+uroMaskLayer);
uroAddLog('Turns layer at idx '+uroTurnsLayerIdx);
uroAddLog('uroRootContainer = '+uroRootContainer);
uroAddLog('Places root layer = '+uroPlacesRoot);
}
function uroRealWazeBits()
{
if(document.getElementsByClassName("sandbox").length > 0)
{
uroAddLog('WME practice mode detected, script is disabled...');
return;
}
if(document.location.href.indexOf('user') !== -1)
{
uroAddLog('User profile page detected, script is disabled...');
return;
}
uroAddLog('adding WazeBits...'+uroToHex(uroWazeBitsPresent,4));
if((uroWazeBitsPresent & 0x0001) === 0)
{
if(typeof W != "undefined")
{
if(typeof W.map != "undefined")
{
uroAddLog(' W.map OK');
uroWazeBitsPresent |= 0x0001;
}
}
}
if((uroWazeBitsPresent & 0x0002) === 0)
{
if(typeof W != "undefined")
{
if(typeof W.model != "undefined")
{
uroAddLog(' W.model OK');
uroWazeBitsPresent |= 0x0002;
}
}
}
if((uroWazeBitsPresent & 0x0004) === 0)
{
if(typeof W != "undefined")
{
if(typeof W.loginManager != "undefined")
{
uroAddLog(' loginManager OK');
uroWazeBitsPresent |= 0x0004;
}
}
}
if((uroWazeBitsPresent & 0x0008) === 0)
{
if(typeof W != "undefined")
{
if(typeof W.selectionManager != "undefined")
{
uroAddLog(' selectionManager OK');
uroWazeBitsPresent |= 0x0008;
}
}
}
if((uroWazeBitsPresent & 0x0010) === 0)
{
if(typeof OpenLayers != "undefined")
{
uroAddLog(' OpenLayers OK');
uroWazeBitsPresent |= 0x0010;
}
}
if((uroWazeBitsPresent & 0x0020) === 0)
{
if(typeof Waze != "undefined")
{
uroAddLog(' Waze OK');
uroWazeBitsPresent |= 0x0020;
}
}
if((uroWazeBitsPresent & 0x0040) === 0)
{
if(document.getElementById('user-tabs') !== null)
{
uroUserTabId = 'user-tabs';
uroAddLog(' user-tabs OK');
uroWazeBitsPresent |= 0x0040;
}
}
if((uroWazeBitsPresent & 0x0080) === 0)
{
if(document.getElementById('sidepanel-drives') !== null)
{
uroAddLog(' sidepanel-drives OK');
uroWazeBitsPresent |= 0x0080;
}
}
if((uroWazeBitsPresent & 0x0100) === 0)
{
if(typeof I18n != "undefined")
{
uroAddLog(' I18n OK');
uroWazeBitsPresent |= 0x0100;
}
}
if(uroWazeBitsPresent !== 0x01FF)
{
setTimeout(uroRealWazeBits,250);
}
else if(W.loginManager.isLoggedIn() === false)
{
uroAddLog('Waiting for user log-in...');
setTimeout(uroRealWazeBits,1000);
}
else
{
uroAddLog('All WazeBits present and correct...');
W.app.modeController.model.bind("change:mode",uroInitialise);
if(W.app.modeController.mode.mteModeState !== undefined)
{
uroMTEMode = true;
uroSetupListeners = true;
uroFinalisingListenerSetup = false;
uroHidePopup();
uroAddLog('MTE mode, sleeping until normal service is resumed...');
return;
}
uroMTEMode = false;
uroSetupUI();
uroDOMHasTurnProblems = (W.model.turnProblems != null);
uroPopulatePlacesTab();
uroControls.appendChild(uroCtrlURs);
uroControls.appendChild(uroCtrlMPs);
uroControls.appendChild(uroCtrlPlaces);
uroControls.appendChild(uroCtrlCameras);
uroControls.appendChild(uroOWL);
uroControls.appendChild(uroCtrlMisc);
uroControls.appendChild(uroCtrlHides);
uroControls.appendChild(uroAMList);
uroCtrlURs.onclick = uroFilterItems_URTabClick;
uroCtrlMPs.onclick = uroFilterItems_MPTabClick;
uroCtrlPlaces.onclick = uroFilterItems_PlacesTabClick;
uroCtrlCameras.onclick = uroFilterItems_CamerasTabClick;
uroCtrlMisc.onclick = uroFilterItems_MiscTabClick;
uroWazeBits();
uroDiv.addEventListener("mouseover", uroEnterPopup, false);
uroDiv.addEventListener("mouseout", uroExitPopup, false);
if(sessionStorage.UROverview_FID_IgnoreList === undefined) sessionStorage.UROverview_FID_IgnoreList = '';
if(sessionStorage.UROverview_FID_WatchList === undefined) sessionStorage.UROverview_FID_WatchList = '';
if(uroConfirmIntercepted === false) uroAddInterceptor();
setInterval(uroTenthSecondTick,100);
}
}
function uroAddInterceptor()
{
uroAddLog('Adding interceptor function...');
// add interceptor function for confirm(), so that we can auto-select the "OK" option when solving URs
// which have pending question...
var _confirm = window.confirm;
window.confirm = function(msg)
{
var cm_delete_confirm = I18n.lookup("closures.delete_confirm").split('"')[0].trimRight(1);
if((I18n.lookup("update_requests.panel.confirm") == msg) && (uroGetCBChecked('_cbDisablePendingQuestions') === true))
{
uroAddLog('Intercepted pending comments confirmation...');
return true;
}
else if(msg.indexOf(cm_delete_confirm) != -1)
{
uroAddLog('intercepted closure delete confirmation...');
if(uroConfirmClosureDelete)
{
return _confirm(msg);
}
else
{
return true;
}
}
else if(typeof(msg) == 'undefined')
{
uroAddLog('Intercepted blank confirmation...');
return true;
}
else
{
return _confirm(msg);
}
};
uroConfirmIntercepted = true;
}
function uroEnterPopup()
{
uroMouseInPopup = true;
}
function uroExitPopup()
{
uroMouseInPopup = false;
}
function uroToggleDebug()
{
uroShowDebugOutput = !uroShowDebugOutput;
var dbgMode = "none";
if(uroShowDebugOutput)
{
dbgMode = "inline";
}
document.getElementById('_uroDebugMode').style.display = dbgMode;
}
function uroInitialise()
{
uroInitialised = false;
if(document.URL.indexOf('beta') != -1) uroBetaEditor = true;
var urlBits = document.URL.split("&mapUpdateRequest=");
if(urlBits.length == 2)
{
uroURIDInURL = parseInt(urlBits[1].split('&')[0]);
uroAddLog('found UR ID '+uroURIDInURL+' in URL');
}
uroRealWazeBits();
}
function uroSetupUI()
{
// create a new div to display the UR details floaty-box
uroDiv = document.createElement('div');
uroDiv.id = "uroDiv";
uroDiv.style.position = 'absolute';
uroDiv.style.visibility = 'hidden';
uroDiv.style.top = '0';
uroDiv.style.left = '0';
uroDiv.style.zIndex = 100;
uroDiv.style.backgroundColor = 'aliceblue';
uroDiv.style.borderWidth = '3px';
uroDiv.style.borderStyle = 'solid';
uroDiv.style.borderRadius = '10px';
uroDiv.style.boxShadow = '5px 5px 10px Silver';
uroDiv.style.padding = '4px';
document.body.appendChild(uroDiv);
// create a new div to display script alerts
uroAlerts = document.createElement('div');
uroAlerts.id = "uroAlerts";
uroAlerts.style.position = 'fixed';
uroAlerts.style.visibility = 'hidden';
uroAlerts.style.top = '50%';
uroAlerts.style.left = '50%';
uroAlerts.style.zIndex = 100;
uroAlerts.style.backgroundColor = 'aliceblue';
uroAlerts.style.borderWidth = '3px';
uroAlerts.style.borderStyle = 'solid';
uroAlerts.style.borderRadius = '10px';
uroAlerts.style.boxShadow = '5px 5px 10px Silver';
uroAlerts.style.padding = '4px';
uroAlerts.style.webkitTransform = "translate(-50%, -50%)";
uroAlerts.style.transform = "translate(-50%, -50%)";
var alertsHTML = '<div id="header" style="padding: 4px; background-color:LightGreen; font-weight: bold;">Alert title goes here...</div>';
alertsHTML += '<div id="content" style="padding: 4px; background-color:White">Alert content goes here...</div>';
alertsHTML += '<div id="controls" style="padding: 4px;">';
alertsHTML += '<span id="uroAlertTickBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px;">';
alertsHTML += '<i class="fa fa-check"> </i>';
alertsHTML += '<span id="uroAlertTickBtnCaption" style="font-weight: bold;"></span>';
alertsHTML += '</span>';
alertsHTML += ' ';
alertsHTML += '<span id="uroAlertCrossBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px;">';
alertsHTML += '<i class="fa fa-times"> </i>';
alertsHTML += '<span id="uroAlertCrossBtnCaption" style="font-weight: bold;"></span>';
alertsHTML += '</span>';
alertsHTML += '</div>';
uroAlerts.innerHTML = alertsHTML;
document.body.appendChild(uroAlerts);
uroControls = document.createElement('section');
uroControls.style.fontSize = '12px';
uroControls.id = 'uroControls';
var updateURL;
if(navigator.userAgent.indexOf('Chrome') == -1)
{
updateURL = 'https://gf.qytechs.cn/scripts/1952-uroverview-plus-uro';
}
else
{
updateURL = 'https://chrome.google.com/webstore/detail/uroverview/amdamgkgchnbaopmphhjapmjcdghdphi';
}
var tabbyHTML = '<b><a href="'+updateURL+'" target="_blank">UROverview Plus</a></b> <label id="_uroVersion">'+uroVersion+'</label>';
tabbyHTML += '<label id="_uroDebugMode">(dbg)</label>';
tabbyHTML += ' <input type="checkbox" id="_cbMasterEnable" checked>Enabled</input>';
tabbyHTML += '<p><table border=0 width="100%"><tr>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectUserRequests"><a href="#" id="_linkSelectUserRequests" style="text-decoration:none;font-size:12px">URs</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectMapProblems"><a href="#" id="_linkSelectMapProblems" style="text-decoration:none;font-size:12px">MPs</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectPlaces"><a href="#" id="_linkSelectPlaces" style="text-decoration:none;font-size:12px">Places</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectCameras"><a href="#" id="_linkSelectCameras" style="text-decoration:none;font-size:12px">Cams</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectCWL"><a href="#" id="_linkSelectOWL" style="text-decoration:none;font-size:12px">OWL</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectMisc"><a href="#" id="_linkSelectMisc" style="text-decoration:none;font-size:12px">Misc</a></td>';
tabbyHTML += '</tr></table>';
uroControls.innerHTML = tabbyHTML;
uroCtrlURs = document.createElement('p');
uroCtrlMPs = document.createElement('p');
uroCtrlCameras = document.createElement('p');
uroOWL = document.createElement('p');
uroCtrlMisc = document.createElement('p');
uroAMList = document.createElement('div');
uroCtrlHides = document.createElement('div');
uroCtrlPlaces = document.createElement('p');
// UR controls tab
uroCtrlURs.id = "uroCtrlURs";
uroCtrlURs.innerHTML = '<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURFilterOutsideArea">Hide URs outside my editable area</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbNoFilterForURInURL">Don\'t filter UR in URL</input><br><br>';
uroCtrlURs.innerHTML += '<b>Filter by type:</b><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterWazeAuto">Waze Automatic</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectTurn">Incorrect turn</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectAddress">Incorrect address</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectRoute">Incorrect route</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingRoundabout">Missing roundabout</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterGeneralError">General error</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterTurnNotAllowed">Turn not allowed</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectJunction">Incorrect junction</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingBridgeOverpass">Missing bridge overpass</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterWrongDrivingDirection">Wrong driving direction</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingExit">Missing exit</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingRoad">Missing road</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterBlockedRoad">Blocked road</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingLandmark">Missing Landmark</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterSpeedLimits">Missing or Invalid Speed limit</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterUndefined">Undefined</input><br>';
uroCtrlURs.innerHTML += ' <i>Specially tagged types</i><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterRoadworks">[ROADWORKS]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterConstruction">[CONSTRUCTION]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterClosure">[CLOSURE]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterEvent">[EVENT]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterNote">[NOTE]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterBOG">[BOG]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterDifficult">[DIFFICULT]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterWSLM">[WSLM]</input><br><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbInvertURFilter">Invert operation of type filters?</input><br>';
uroCtrlURs.innerHTML += '<hr>';
uroCtrlURs.innerHTML += '<br><b>Hide by state:</b><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterOpenUR">Open</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterClosedUR">Closed</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterSolved">Solved</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterUnidentified">Not identified</input><br><br>';
uroCtrlURs.innerHTML += '<br><b>Filter by age of submission:</b><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMinAgeFilter">Hide URs less than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinDays"> days old<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMaxAgeFilter">Hide URs more than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxDays"> days old<br>';
uroCtrlURs.innerHTML += '<br><b>Filter by description/comments/following:</b><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyFollowed" pairedWith="_cbHideMyUnfollowed">Ones I am or </input>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyUnfollowed" pairedWith="_cbHideMyFollowed">am not following</input><br><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURDescriptionMustBePresent" pairedWith="_cbURDescriptionMustBeAbsent">Hide</input> or ';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURDescriptionMustBeAbsent" pairedWith="_cbURDescriptionMustBePresent">show</input> URs with no description<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableKeywordMustBePresent">Hide URs not including </input>';
uroCtrlURs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPresent"><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableKeywordMustBeAbsent">Hide URs including </input>';
uroCtrlURs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordAbsent"><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbCaseInsensitive"><i>Case-insensitive matches?</i></input><br><br>';
uroCtrlURs.innerHTML += 'With comments from me?<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyComments" pairedWith="_cbHideAnyComments">Yes </input>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideAnyComments" pairedWith="_cbHideMyComments">No</input><br>';
uroCtrlURs.innerHTML += 'If last comment made by me?<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfLastCommenter" pairedWith="_cbHideIfNotLastCommenter">Yes </input>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfNotLastCommenter" pairedWith="_cbHideIfLastCommenter">No </input><br>';
uroCtrlURs.innerHTML += 'If last comment made by UR reporter?<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfReporterLastCommenter" pairedWith="_cbHideIfReporterNotLastCommenter">Yes </input>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfReporterNotLastCommenter" pairedWith="_cbHideIfReporterLastCommenter">No</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMinCommentsFilter">With less than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinComments"> comments<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMaxCommentsFilter">With more than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="0" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxComments"> comments<br><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter2">Last comment less than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays2"> days ago<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter">Last comment more than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays"> days ago<br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbIgnoreOtherEditorComments"><i>Ignore other editor comments?</i></input><br><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURUserIDFilter">Without comments from user</input>';
uroCtrlURs.innerHTML += '<select id="_selectURUserID" style="width:80%; height:22px;"></select><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURResolverIDFilter">Not resolved by user</input>';
uroCtrlURs.innerHTML += '<select id="_selectURResolverID" style="width:80%; height:22px;"></select>';
uroCtrlURs.innerHTML += '<br><br><input type="checkbox" id="_cbInvertURStateFilter">Invert operation of state/age filters?</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbNoFilterForTaggedURs">Don\'t apply state/age filters to tagged URs</input><br>';
// Map problems controls tab
uroCtrlMPs.id = "uroCtrlMPs";
uroCtrlMPs.innerHTML = '<br>';
uroCtrlMPs.innerHTML += '<b>Filter MPs by type:</b><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMissingJunction">Missing junction</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMissingRoad">Missing road</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterCrossroadsJunctionMissing">Missing crossroads</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterDrivingDirectionMismatch">Driving direction mismatch</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRoadTypeMismatch">Road type mismatch</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRestrictedTurn">Restricted turn might be allowed</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRoadClosureProblem">Road closure</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterUnknownProblem">Unknown/other problem type</input><br><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterTurnProblem">Turn Problems</input><br><br>';
uroCtrlMPs.innerHTML += ' <i>Specially tagged types</i><br>';
uroCtrlMPs.innerHTML += ' <input type="checkbox" id="_cbFilterElgin">[Elgin]</input><br>';
uroCtrlMPs.innerHTML += ' <input type="checkbox" id="_cbFilterTrafficCast">[TrafficCast]</input><br>';
uroCtrlMPs.innerHTML += ' <input type="checkbox" id="_cbFilterTrafficMaster">[TM]</input><br>';
uroCtrlMPs.innerHTML += ' <input type="checkbox" id="_cbFilterCaltrans">[Caltrans]</input><br>';
uroCtrlMPs.innerHTML += ' <input type="checkbox" id="_cbFilterTFL">[TfL Open Data]</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterReopenedProblem">Reopened Problems</input><br><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbInvertMPFilter">Invert operation of type filters?</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterOutsideArea">Hide MPs outside my editable area</input><br>';
uroCtrlMPs.innerHTML += '<br><b>Hide closed/solved/unidentified Problems:</b><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterClosed">Closed</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterSolved">Solved</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterUnidentified">Not identified</input><br><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPClosedUserIDFilter" pairedWith="_cbMPNotClosedUserIDFilter">Closed</input> or ';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPNotClosedUserIDFilter" pairedWith="_cbMPClosedUserIDFilter">Not Closed</input> by user';
uroCtrlMPs.innerHTML += '<select id="_selectMPUserID" style="width:80%; height:22px;"></select><br>';
uroCtrlMPs.innerHTML += '<br><b>Hide problems (not turn) by severity:</b><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterLowSeverity">Low</input> ';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMediumSeverity">Medium</input> ';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterHighSeverity">High</input><br>';
// Places filtering tab
uroCtrlPlaces.id = "uroCtrlPlaces";
uroCtrlPlaces.innerHTML = 'Places filter list being populated, please wait...';
// Camera controls tab
uroCtrlCameras.id = "uroCtrlCameras";
uroCtrlCameras.innerHTML = '<br><b>Show Cameras created by:</b><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowWorldCams" checked>world_* users</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowUSACams" checked>usa_* users</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowNonWorldCams" checked>other users</input><br>';
uroCtrlCameras.innerHTML += '<br><b>Show Cameras touched by a specific editor:</b><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowOnlyCamsCreatedBy">Created by</input> / ';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowOnlyCamsEditedBy">edited by</input><br>';
uroCtrlCameras.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textCameraEditor"><br>';
uroCtrlCameras.innerHTML += '<select id="_selectCameraUserID" style="width:80%; height:22px;"></select><br>';
uroCtrlCameras.innerHTML += '<br><input type="checkbox" id="_cbShowOnlyMyCams">Show ONLY cameras created/edited by me</input><br>';
uroCtrlCameras.innerHTML += '<br><b>Show Cameras by approval status:</b><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowApprovedCams" checked>approved</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowNonApprovedCams" checked>non-approved</input><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbShowOlderCreatedNonApproved"> if created more than </input>';
uroCtrlCameras.innerHTML += '<input type="number" min="1" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCameraMinCreatedDays"> days ago<br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbShowOlderUpdatedNonApproved"> if updated more than </input>';
uroCtrlCameras.innerHTML += '<input type="number" min="1" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCameraMinUpdatedDays"> days ago<br>';
uroCtrlCameras.innerHTML += '<br><b>Show Cameras by type:</b><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowSpeedCams" checked>Speed</input><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbShowIfSpeedSet" checked> with speed data</input><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbShowIfNoSpeedSet" checked> with no speed data</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowRedLightCams" checked>Red Light</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowDummyCams" checked>Dummy</input><br>';
uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by creator:</b><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByMe">me</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank0">L1</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank1">L2</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank2">L3</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank3">L4</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank4">L5</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank5">L6</input>';
uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by updater:</b><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByMe">me</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank0">L1</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank1">L2</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank2">L3</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank3">L4</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank4">L5</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank5">L6</input>';
uroCtrlCameras.innerHTML += '<br><br><b><input type="checkbox" id="_cbHideCWLCams">Hide cameras on watchlist</input></b><br>';
// Object watchlist tab
uroOWL.id = "uroOWL";
uroCWLGroups = [];
uroOWLUpdateHTML();
// Misc controls tab
uroCtrlMisc.id = "uroCtrlMisc";
uroCtrlMisc.innerHTML = '<br><b>Use default conversation markers:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbNativeConvoMarkers" checked />in public WME<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbNativeBetaConvoMarkers" checked />in beta WME<br>';
uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbCommentCount" />Show comment count on UR markers</b><br>';
uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbURBackfill" />Backfill UR data</b><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Marker Unstacking:</b><br>';
uroCtrlMisc.innerHTML += 'Distance threshold: <input type="number" min="1" max="30" value="15" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackSensitivity" /><br>';
uroCtrlMisc.innerHTML += 'Disable below zoom: <input type="number" min="0" max="10" value="3" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackZoomLevel" /><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Use custom marker for URs tagged as:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomRoadworksMarkers" />[ROADWORKS]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomConstructionMarkers" />[CONSTRUCTION]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomClosuresMarkers" />[CLOSURE]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomEventsMarkers" />[EVENT]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomNotesMarkers" />[NOTE]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomBOGMarkers" />[BOG]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomDifficultMarkers" />[DIFFICULT]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomWSLMMarkers" />[WSLM]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomNativeSLMarkers" />Native speed limit reports<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomKeywordMarkers" />';
uroCtrlMisc.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textCustomKeyword" /><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Use custom marker for MPs tagged as:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomElginMarkers" />[Elgin]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTrafficMasterMarkers" />[TM]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTrafficCastMarkers" />[TrafficCast]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomCaltransMarkers" />[Caltrans]<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTFLMarkers" />[TfL Open Data]<br>';
uroCtrlMisc.innerHTML += '<br><br><b>Popup mouse behaviour:</b><br>';
uroCtrlMisc.innerHTML += 'Mouse idle <input type="number" min="1" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupDwellTimeout" /> *100ms<br>';
uroCtrlMisc.innerHTML += 'Mouse over <input type="number" min="1" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupEntryTimeout" /> *100ms<br>';
uroCtrlMisc.innerHTML += 'Distance <input type="number" min="0" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMaxJitter" /> pixels<br>';
uroCtrlMisc.innerHTML += '<br><br><b>Disable popup for:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitURPopup" />URs<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitMPPopup" />MPs<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitCamPopup" />Cameras<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitSegPopup" />Segments<br>';
uroCtrlMisc.innerHTML += ' <input type="checkbox" id="_cbInhibitSegGenericPopup" />Speed limit info<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitTurnsPopup" />Restricted Turns<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitLandmarkPopup" />Places<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitPUPopup" />Place Updates<br>';
uroCtrlMisc.innerHTML += '<br><br><b>Date/Time formatting for popups:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtDDMMYY" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtYYMMDD" checked />day/month/year<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtMMDDYY" pairedWith="_cbDateFmtDDMMYY,_cbDateFmtYYMMDD" />month/day/year<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtYYMMDD" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtDDMMYY" />year/month/day<br><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbTimeFmt24H" pairedWith="_cbTimeFmt12H" checked />24 hour<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbTimeFmt12H" pairedWith="_cbTimeFmt24H" />12 hour<br><br>';
uroCtrlMisc.innerHTML += '<i>Unticked uses browser default setting</i>';
uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbWhiteBackground" />Use custom background colour</b><br>';
uroCtrlMisc.innerHTML += 'R:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundRed" />';
uroCtrlMisc.innerHTML += 'G:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundGreen" />';
uroCtrlMisc.innerHTML += 'B:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundBlue" /><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Replace "Next ..." button with "Done" for:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNURButton" />URs<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNMPButton" />MPs<br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNPURButton" />PURs<br>';
uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbHideAMLayer" />Hide Area Manager polygons</b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisablePlacesFiltering" />Disable Places filtering</b><br>';
////uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisablePendingQuestions">Disable UR Pending Questions confirmation</input></b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisableTabStyling" />Use default tab styling</b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbHideEditorInfo" />Hide sidebar editor info</b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbEnableDTE" />Drive Tab Enhancement (DTE)</b><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Settings backup/restore/reset:</b><br>';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnSettingsToText" value="Backup" /> ';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnTextToSettings" value="Restore" /> | ';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnResetSettings" value="Reset" /><br><br>';
uroCtrlMisc.innerHTML += '<textarea id="_txtSettings" value="" /><br>';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnClearSettingsText" value="Clear" /><br>';
/*
uroCtrlMisc.innerHTML += '<br><br><b>Debug:</b><br>';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnDebugToScreen" value="Show debug data" />';
*/
// footer for tabs container
uroCtrlHides.id = 'uroCtrlHides';
uroCtrlHides.innerHTML = '<input type="button" id="_btnUndoLastHide" value="Undo last hide" /> ';
uroCtrlHides.innerHTML += '<input type="button" id="_btnClearSessionHides" value="Undo all hides" /><p>';
// footer for AM list
uroAMList.id = 'uroAMList';
window.addEventListener("beforeunload", uroSaveSettings, false);
//uroRealWazeBits();
}
function dteAddHeader()
{
if(uroMTEMode) return;
if(!uroInitialised) return;
var rlcObj = document.getElementsByClassName("result-list-container");
if(typeof rlcObj == "undefined") return;
if(typeof rlcObj[0].children[0] == "undefined") return;
if(typeof rlcObj[0].children[0].innerHTML == "undefined") return;
var thtml = rlcObj[0].children[0].innerHTML;
if(thtml.indexOf('Full drive history') == -1)
{
thtml += '<br><br><i><small>Full drive history goes back to '+dteOldestFullDrive.toDateString()+'</small></i>';
rlcObj[0].children[0].innerHTML = thtml;
}
}
function dteGetData()
{
var loc = 'https://'+window.location.hostname+Waze.Config.api_base+'/Archive/MyList?minDistance=1000';
loc += '&offset='+dteOffset+'&count=5';
var dteReq = new XMLHttpRequest();
dteReq.onreadystatechange = function()
{
var foundMissingDrive = false;
if(dteReq.readyState == 4)
{
uroAddLog('drive data request, response '+dteReq.status+' received');
if(dteReq.status == 200)
{
if(dteReq.responseText !== "")
{
var drives = JSON.parse(dteReq.responseText);
var loadedDrives = drives.archives.objects.length;
uroAddLog('received '+loadedDrives+' drives');
if(loadedDrives != 5) foundMissingDrive = true;
for(var loop=0; loop < loadedDrives; loop++)
{
if(drives.archives.objects[loop].hasFullSession === false)
{
foundMissingDrive = true;
}
else
{
dteOffset++;
dteOldestFullDrive = new Date(drives.archives.objects[loop].startTime);
}
}
}
else
{
foundMissingDrive = true;
}
}
if(foundMissingDrive === false)
{
dteGetData();
}
else
{
uroAddLog(dteOffset+' full drives in history');
uroAddLog('oldest drives are on '+dteOldestFullDrive.toDateString());
if(dteOffset < 5)
{
dteOffset = 5;
uroAddLog('insufficient full drives, using standard drives tab');
}
else if(dteOffset > 50)
{
var nPages = Math.ceil(dteOffset / 50);
uroAddLog('too many full drives for a single tab page, splitting over '+nPages+' pages...');
dteOffset = Math.ceil(dteOffset/nPages);
}
if((dteOldestFullDrive - dteEpoch) > 0)
{
var totalDrives = 0;
if(W.model.archives.additionalInfo !== null)
{
totalDrives = W.model.archives.additionalInfo.totalSessions;
}
if(totalDrives !== null)
{
uroAddLog('updating drives tab...');
W.map.controls[dteControlsIdx].sidePanelView.ResultsPerPage = dteOffset;
uroAddLog(totalDrives+' drives in history');
W.map.controls[dteControlsIdx].sidePanelView.setSessions(totalDrives);
W.map.controls[dteControlsIdx].loadSessions(0);
}
setInterval(dteAddHeader,250);
setInterval(dteCheckDriveListChanges,250);
}
}
}
};
dteReq.open('GET',loc,true);
dteReq.send();
}
function dteSetNewTabLength()
{
uroAddLog('altering ResultsPerPage parameter...');
var t = document.getElementById('sidepanel-drives');
t.style.overflow = 'auto';
t.style.height = (window.innerHeight * 0.6) + 'px';
dteOffset = 0;
dteGetData();
}
function dteListClick()
{
dteClearListHighlight();
this.style.backgroundColor = "lightgreen";
dteArmClearHighlightsOnPanelClose = true;
}
function dteClearListHighlight()
{
var drivesShown = document.getElementById('sidepanel-drives').getElementsByClassName('result session').length;
if(drivesShown > 0)
{
for(var loop = 0;loop < drivesShown; loop++)
{
var listEntry = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[loop];
listEntry.style.backgroundColor = "";
}
}
}
function dteCheckDriveListChanges()
{
if(uroMTEMode) return;
if(!uroInitialised) return;
var drivesShown = document.getElementById('sidepanel-drives').getElementsByClassName('result session').length;
if(drivesShown > 0)
{
var topID = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[0].getAttribute('data-id');
if(topID != dteTopID)
{
dteTopID = topID;
for(var loop = 0;loop < drivesShown; loop++)
{
var listEntry = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[loop];
var driveID = listEntry.getAttribute('data-id');
var driveObj = W.model.archives.objects[driveID];
var driveSecs = Math.floor((driveObj.endTime - driveObj.startTime) / 1000);
var driveHours = Math.floor(driveSecs / 3600);
driveSecs -= (driveHours * 3600);
var driveMins = Math.floor(driveSecs / 60);
driveSecs -= (driveMins * 60);
var trueTime = (driveHours+':'+("0"+driveMins).slice(-2)+'.'+("0"+driveSecs).slice(-2));
listEntry.getElementsByTagName('span')[1].innerHTML = trueTime;
listEntry.addEventListener("click", dteListClick, false);
}
}
}
}
uroInitialise();