UROverview Plus (URO+)

Adds a whole bunch of features to WME, which someday I may get around to documenting properly...

目前為 2017-11-23 提交的版本,檢視 最新版本

// ==UserScript==
// @name                UROverview Plus (URO+)
// @namespace           http://greasemonkey.chizzum.com
// @description         Adds a whole bunch of features to WME, which someday I may get around to documenting properly...
// @include             https://*.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.120
// ==/UserScript==

/*

TO-DO ITEMS
=======================================================================================================================
Bug fixes - MUST BE CLEARED BEFORE RELEASE
=======================================================================================================================



=======================================================================================================================
Things to be checked
=======================================================================================================================

positioning of TBR popup relative to the restriction & TIO controls


=======================================================================================================================
Things to consider working on, in no order of priority
=======================================================================================================================

Unify handling of parking lot and other place PURs for unstacking - the lack of any visual differentiation between their
native markers when both layers are visible leads to undesirable effects when unstacking, as well as confusion over
whether or not a "stack" of PUR markers will actually unstack in the first place (i.e. if it consists of one place
marker and one parking lot marker...)

Fade out filtered markers instead of completely hiding them

Adjust default settings for places & MP tabs to account for dynamic nature of tab building

Convert camera XHR code to async operation

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)
 
Filter URs by age of first comment

Ignore comments from anyone other than reporter and self

Filter all camera types by speed limit/unset limit

Further filtering options for RTCs - filter by active/future, filter by creator, MTE etc.

Single click to update all watched camera data

Copy saved camera data back to camera object/create new camera with same properties?


=======================================================================================================================
New functionality in progress
=======================================================================================================================

Enhanced object history - replace native history entries with a full breakdown of the data available from the server

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 map: true */
/* globals WazeMap: true */
/* globals require: */
/* jshint bitwise: false */
/* jshint eqnull: true */


var uroVersion = "3.120";
var uroReleaseDate = "20171122";

// list of changes affecting all users
var uroChanges =
[
   "Corrected default diagnostics output setting",
   "Disabling marker popups no longer disables marker unstacking",
   "AM list no longer shows stale data if the map is moved without using the mouse"
];
// 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;
// true enables performance monitoring debug output
var uroPerformanceMonitoringOutput = 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 uroPopupAutoHideTimer = 0;
var uroPopupShown = false;
var uroPopupSuppressed = false;
var uroSetupListeners = true;
var uroRootContainer = null;
var uroPlacesRoot = null;
var uroConfirmIntercepted = false;
var uroCustomMarkerList = [];
var uroPendingURSessionIDs = [];
var uroRequestedURSessionIDs = [];
var uroPlacesGroupsCollapsed = [];
var uroKnownProblemTypeIDs = [];
var uroKnownProblemTypeNames = [];

var uroMouseInPopup = false;
var uroURControlsIdx = null;
var uroProblemControlsIdx = null;
var uroTurnsLayerIdx = null;
var uroMCLayerIdx = 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 uroMCSelected = false;
var uroPlaceSelected = false;
var uroAutoCentreDisabledOn = [];
var uroMouseIsDown = false;
var uroBackfilling = false;
var uroHidePopupOnPanelOpen = false;
var uroPointerWithinMap = 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 uroCtrlMCs = 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 uroMousedOverMapComment = null;
var uroMousedOverOtherObjectWithinMapComment = false;

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 uroDoFeedFilter = true;
var uroPreviousFeedLength = 0;

var uroEnhanceHistorySegID = null;
var uroSegHistoryDetails = null;
var uroSegHistoryEntries = 0;
var uroSegHistoryLoaded = false;

var uroTBRObj = null;

var uroBackfillQueue = [];

var uroUnstackedMasterID = null;
var uroStackList = [];
var uroStackType = null;

var uroMousedOverMarkerID = null;
var uroMousedOverMarkerType = null;
var uroClickedOnMarkerID = null;
var uroClickedOnMarkerType = null;

var uroAlertBoxStack = [];
var uroAlertBoxTickAction = null;
var uroAlertBoxCrossAction = null;
var uroAlertBoxInUse = false;

var uroCustomURTags = ['[ROADWORKS]','[CONSTRUCTION]','[CLOSURE]','[EVENT]','[NOTE]','[WSLM]','[BOG]','[DIFFICULT]'];

var uroAltMarkers =
[
   // each altMarker has 4 variants: 0 = normal open, 1 = selected open, 2 = normal closed, 3 = selected closed

   //  0: closure UR
   [
      "",
      "",
      "",
      ""
   ],
   //  1: roadworks UR
   [
      "",
      "",
      "",
      ""
   ],
   // 2: custom keyword UR
   [
      "",
      "",
      "",
      ""
   ],
   //  3: note UR
   [
      "",
      "",
      "",
      ""
   ],
   //  4: event UR
   [
      "",
      "",
      "",
      ""
   ],
   // 5: WMSL/SLUR UR
   [
      "",
      "",
      "",
      ""      
   ],
   // 6: Elgin MP
   [
      "",
      "",
      "",
      ""
   ],
   // 7: TrafficCast MP
   [
      "",
      "",
      "",
      ""
   ],
   // 8: TrafficMaster MP
   [
      "",
      "",
      "",
      ""
   ],
   // 9: CalTrans
   [
      '',
      '',
      '',
      ''
   ],
   // 10: TfL
   [
      '',
      '',
      '',
      ''
   ],
   // 11: BOG
   [
      '',
      '',
      '',
      ''
   ],
   // 12: Difficult turn
   [
      '',
      '',
      '',
      ''
   ]
];


var uroMarkers =
[
   // 0 = comment count circle
   [""],
   // 1 = green comment marker
   [""],
   // 2 = yellow (own) comment marker
   [""]
];

var uroFeedFilterReloads = 0;
var uroDeleteAutoRepeat = false;
var uroFeedEntriesDeleted = 0;
var uroFeedFilterFilters =
[
   ['feed.issues.motivations.CAN_BE_SOLVED_BY_RANK',           'motivation'],
   ['feed.issues.motivations.CLOSE_TO_FAVORITES',              'motivation'],
   ['feed.issues.motivations.ISSUE_AGE',                       'motivation'],
   ['feed.issues.motivations.ISSUE_REOPENED',                  'motivation'],
   ['feed.issues.motivations.NEAR_DRIVES',                     'motivation'],
   ['feed.issues.motivations.REPORTED_BY_USER',                'motivation'],
   ['feed.issues.motivations.USER_FOLLOWS_ISSUE',              'motivation'],
   ['feed.issues.motivations.USER_FOLLOWS_ISSUE_LAST_COMMENT', 'motivation'],
   ['venues.update_requests.panel.flag_title.IMAGE',           'title'],
   ['venues.update_requests.panel.flag_title.VENUE',           'title'],
   ['venues.update_requests.panel.title.ADD_IMAGE',            'title'],
   ['venues.update_requests.panel.title.ADD_VENUE',            'title'],
   ['venues.update_requests.panel.title.DELETE_VENUE',         'title'],
   ['venues.update_requests.panel.title.UPDATE_VENUE',         'title']
];


/*
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 uroTempFixMTEDropDown()
{
   // temporary fix for that bloody annoying bug in the closure event dropdown...  sort it out devs!
   // also removes the "Choose Event" non-option from the list, so that it now always starts with "None"
   if(document.getElementsByName('closure_eventId').length > 0)
   {
      if(document.getElementsByName('closure_eventId')[0].selectedOptions.length > 0)
      {
         document.getElementsByName('closure_eventId')[0].required = false;
         if(document.getElementsByName('closure_eventId')[0].selectedOptions[0].text == I18n.lookup('closures.choose_event'))
         {
            document.getElementsByName('closure_eventId')[0].selectedOptions[0].remove();
         }
      }
   }
}

function uroAddLog(logtext)
{
   if(uroShowDebugOutput) console.log('URO+: '+logtext);
}
function uroGetCBChecked(cbID)
{
   try
   {
      return(document.getElementById(cbID).checked);
   }
   catch(err)
   {
      uroAddLog('uroGetCBChecked() - '+cbID+' not found!');
      return null;
   }
}
function uroSetCBChecked(cbID, state)
{
   try
   {
      document.getElementById(cbID).checked = state;
   }
   catch(err)
   {
      uroAddLog('uroSetCBChecked() - '+cbID+' not found!');
   }
}
function uroGetElmValue(elmID)
{
   try
   {
      return(document.getElementById(elmID).value);
   }
   catch(err)
   {
      uroAddLog('uroGetElmValue() - '+elmID+' not found!');
      return null;
   }
}
function uroSetStyleDisplay(elm,style)
{
   try
   {
      document.getElementById(elm).style.display = style;
   }
   catch(err)
   {
      uroAddLog('uroSetStyleDisplay() - '+elm+' not found!');
   }
}
function uroSetOnClick(elm,fn)
{
   try
   {
      document.getElementById(elm).onclick = fn;
   }
   catch(err)
   {
      uroAddLog('uroSetOnClick() - '+elm+' not found!');
   }
}
function uroAddEventListener(elm,eventType,eventFn,eventBool)
{
   try
   {
      document.getElementById(elm).addEventListener(eventType, eventFn, eventBool);
   }
   catch(err)
   {
      uroAddLog('uroAddEventListener() - '+elm+' not found!');
   }
}


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>&nbsp;';
   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>';
   }
   else
   {
      if(!uroBetaEditor)
      {
         releaseNotes += '<ul><li>Nothing of interest, unless you\'re a WME beta tester, in which case log into the beta and find out...</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 += '<img src="'+uroAltMarkers[1][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[0][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[4][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[3][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[5][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[11][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[12][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[10][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[6][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[8][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[7][0]+'">';
   confirmMsg += '<img src="'+uroAltMarkers[9][0]+'">';
   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.UROverviewMCOptions = uroGatherSettings('uroCtrlMCs');
      localStorage.UROverviewCameraOptions = uroGatherSettings('uroCtrlCameras');
      localStorage.UROverviewMiscOptions = uroGatherSettings('uroCtrlMisc');
      localStorage.UROverviewPlacesOptions = uroGatherSettings('uroCtrlPlaces');
      localStorage.UROverviewFeedFilterOptions = uroGatherSettings('uroFeedFilter');
      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 uroTranslateLegacyMPTabSettings()
{
   var options = localStorage.UROverviewMPOptions.split(':');
   for(var optIdx=0;optIdx<options.length;optIdx++)
   {
      var fields = options[optIdx].split(',');
      if(fields[0].indexOf('_cb') === 0)
      {
         if(fields[0] == '_cbMPFilterParkingLotInputAsPoint') uroSetCBChecked('_cbMPFilter_T50', (fields[1] == 'true'));
         if(fields[0] == '_cbMPMissingPLP_T70') uroSetCBChecked('_cbMPFilter_T70', (fields[1] == 'true'));
         if(fields[0] == '_cbMPMissingPLP_T71') uroSetCBChecked('_cbMPFilter_T71', (fields[1] == 'true'));
         if(fields[0] == '_cbMPFilterDrivingDirectionMismatch') uroSetCBChecked('_cbMPFilter_T101', (fields[1] == 'true'));
         if(fields[0] == '_cbMPFilterMissingJunction') uroSetCBChecked('_cbMPFilter_T102', (fields[1] == 'true'));
         if(fields[0] == '_cbMPFilterMissingRoad') uroSetCBChecked('_cbMPFilter_T103', (fields[1] == 'true'));
         if(fields[0] == '_cbMPFilterCrossroadsJunctionMissing') uroSetCBChecked('_cbMPFilter_T104', (fields[1] == 'true'));
         if(fields[0] == '_cbMPFilterRoadTypeMismatch') uroSetCBChecked('_cbMPFilter_T105', (fields[1] == 'true'));
         if(fields[0] == '_cbMPFilterRestrictedTurn') uroSetCBChecked('_cbMPFilter_T106', (fields[1] == 'true'));
         if(fields[0] == '_cbMPFilterTurnProblem') uroSetCBChecked('_cbMPFilter_T200', (fields[1] == 'true'));
         if(fields[0] == '_cbMPFilterRoadClosureProblem') uroSetCBChecked('_cbMPFilter_T300', (fields[1] == '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');
      uroTranslateLegacyMPTabSettings();
      uroApplySettings(localStorage.UROverviewMPOptions);
      isNewInstall = false;
   }

   if (localStorage.UROverviewMCOptions != null)
   {
      uroAddLog('recover MC tab settings');
      uroApplySettings(localStorage.UROverviewMCOptions);
      isNewInstall = false;
   }
   
   if (localStorage.UROverviewPlacesOptions != null)
   {
      uroAddLog('recover Places tab settings');
      uroApplySettings(localStorage.UROverviewPlacesOptions);
      isNewInstall = false;
   }

   if (localStorage.UROverviewFeedFilterOptions != null)
   {
      uroAddLog('recover Feed Filter settings');
      uroApplySettings(localStorage.UROverviewFeedFilterOptions);
      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=1197]:_cbMPFilterOutsideArea,false:_cbMPFilter_T1,false:_cbMPFilter_T2,false:_cbMPFilter_T3,false:_cbMPFilter_T5,false:_cbMPFilter_T6,false:_cbMPFilter_T7,false:_cbMPFilter_T8,false:_cbMPFilter_T10,false:_cbMPFilter_T11,false:_cbMPFilter_T12,false:_cbMPFilter_T13,false:_cbMPFilter_T14,false:_cbMPFilter_T15,false:_cbMPFilter_T16,false:_cbMPFilter_T17,false:_cbMPFilter_T19,false:_cbMPFilter_T20,false:_cbMPFilter_T21,false:_cbMPFilter_T22,false:_cbMPFilter_T50,false:_cbMPFilter_T51,false:_cbMPFilter_T52,false:_cbMPFilter_T53,false:_cbMPFilter_T70,false:_cbMPFilter_T71,false:_cbMPFilter_T101,false:_cbMPFilter_T102,false:_cbMPFilter_T103,false:_cbMPFilter_T104,false:_cbMPFilter_T105,false:_cbMPFilter_T106,false:_cbMPFilter_T200,false:_cbMPFilter_T300,false:_cbMPFilterUnknownProblem,false:_cbFilterElgin,false:_cbFilterTrafficCast,false:_cbFilterTrafficMaster,false:_cbFilterCaltrans,false:_cbFilterTFL,false:_cbMPFilterReopenedProblem,false:_cbInvertMPFilter,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=65]false: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=1477]:_cbHideUserRTCs,false:_cbHideEditorRTCs,false:_cbNativeConvoMarkers,true:_cbNativeBetaConvoMarkers,true:_cbCommentCount,false:_cbEnableDeleteFeedEntries,false:_cbURBackfill,false:_inputUnstackSensitivity,15:_inputUnstackZoomLevel,3:_cbCustomRoadworksMarkers,false:_cbCustomConstructionMarkers,false:_cbCustomClosuresMarkers,false:_cbCustomEventsMarkers,false:_cbCustomNotesMarkers,false:_cbCustomBOGMarkers,false:_cbCustomDifficultMarkers,false:_cbCustomWSLMMarkers,false:_cbCustomNativeSLMarkers,false:_cbCustomKeywordMarkers,false:_textCustomKeyword,:_cbCustomElginMarkers,false:_cbCustomTrafficMasterMarkers,false:_cbCustomTrafficCastMarkers,false:_cbCustomCaltransMarkers,false:_cbCustomTFLMarkers,false:_inputPopupDwellTimeout,2:_inputPopupEntryTimeout,2:_inputMaxJitter,2:_inputPopupAutoHideTimeout,0:_cbInhibitURPopup,false:_cbInhibitMPPopup,false:_cbInhibitCamPopup,false:_cbInhibitSegPopup,false:_cbInhibitSegGenericPopup,false:_cbInhibitTurnsPopup,false:_cbInhibitLandmarkPopup,false:_cbInhibitPUPopup,false:_cbInhibitMapCommentPopup,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=1756]:_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:_cbFilterSpeedLimits,false:_cbFilterUndefined,false:_cbFilterRoadworks,false:_cbFilterConstruction,false:_cbFilterClosure,false:_cbFilterEvent,false:_cbFilterNote,false:_cbFilterBOG,false:_cbFilterDifficult,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 += '[UROverviewFeedFilterOptions][len=596]:_cbReloadFeedAfterDelete,false:_cbFeedFilter_TypeUR,false:_cbFeedFilter_TypeMP,false:_cbFeedFilter_TypePUR,false:_cbFeedFilter_TypePM,false:_cbFeedFilter_0,false:_cbFeedFilter_1,false:_cbFeedFilter_2,false:_cbFeedFilter_3,false:_cbFeedFilter_4,false:_cbFeedFilter_5,false:_cbFeedFilter_6,false:_cbFeedFilter_7,false:_cbFeedFilter_8,false:_cbFeedFilter_9,false:_cbFeedFilter_10,false:_cbFeedFilter_11,false:_cbFeedFilter_12,false:_cbFeedFilter_13,false:_cbFeedFilter_MotNone,false:_cbFeedFilter_Invert,false:_cbFeedFilter_HideKeyword,false:_cbFeedFilter_ShowKeyword,false:_textFeedFilter_Keyword,[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=5557]:_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:_cbHidePlacesStaff,false:_cbHidePlacesAdLocked,false:_cbHideAreaPlaces,false:_cbHidePointPlaces,false:_cbHidePhotoPlaces,false:_cbHideNoPhotoPlaces,false:_cbHideLinkedPlaces,false:_cbHideNoLinkedPlaces,false:_cbHideDescribedPlaces,false:_cbHideNonDescribedPlaces,false:_cbHideKeywordPlaces,false:_cbHideNoKeywordPlaces,false:_textKeywordPlace,:_cbShowOnlyPlacesCreatedBy,false:_cbShowOnlyPlacesEditedBy,false:_textPlacesEditor,:_cbPlacesFilter-CAR_SERVICES,false:_cbPlacesFilter-GAS_STATION,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_URGENT_CARE,false:_cbPlacesFilter-DOCTOR_CLINIC,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-EMERGENCY_SHELTER,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:_cbPlacesFilter-PARKING_LOT,false:_cbFilterPrivatePlaces,false:_cbInvertPlacesFilter,false[END]';
   defaultSettings += '[UROverviewCurrentVersion][len=5]3.104[END]';
   defaultSettings += '[UROverviewCWLGroups][len=16]0,No group,false[END]';  
   defaultSettings += '[UROverviewMCOptions][len=493]:_cbMCFilterRoadworks,false:_cbMCFilterConstruction,false:_cbMCFilterClosure,false:_cbMCFilterEvent,false:_cbMCFilterNote,false:_cbMCFilterBOG,false:_cbMCFilterDifficult,false:_cbMCFilterWSLM,false:_cbInvertMCFilter,false:_cbMCHideMyFollowed,false:_cbMCHideMyUnfollowed,false:_cbMCDescriptionMustBePresent,false:_cbMCDescriptionMustBeAbsent,false:_cbMCEnableKeywordMustBePresent,false:_textMCKeywordPresent,:_cbMCEnableKeywordMustBeAbsent,false:_textMCKeywordAbsent,:_cbMCCaseInsensitive,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 uroGetMCAge(mcObj,ageType,getRaw)
{
   if(ageType === 0)
   {
      if((mcObj.attributes.createdOn === null)||(mcObj.attributes.createdOn === 0)) return -1;
      if(getRaw) return mcObj.attributes.createdOn;
      else return uroDateToDays(mcObj.attributes.createdOn);
   }
   else if(ageType === 1)
   {
      if((mcObj.attributes.updatedOn === null)||(mcObj.attributes.updatedOn === 0)) return -1;
      if(getRaw) return mcObj.attributes.updatedOn;
      else return uroDateToDays(mcObj.attributes.updatedOn);
   }
   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, includeValidity)
{
   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(includeValidity === true)
      {
         // special handling for the 7KM/H spielstrasse found in Germany...
         if(country == "Germany")
         {
            if(speed != 7)
            {
               if(speed % multipleFactor !== 0) retval += " (not valid?)";
            }
         }
         else
         {
            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 for camera data request');
      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) || camChanged);
            }
            else if(camObj.validated === false)
            {

            }
         }
      }
      else
      {
         uroAddLog('camera data request failed (status != 200)');
      }
   }
   catch(err)
   {
      uroAddLog('camera data 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) || camChanged);
            }
            else
            {
               camChanged = (uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon) || camChanged);
            }
         }
         else
         {
            camChanged = (uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon) || camChanged);
         }
      }
   }

   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 uroAcceptCameraChanges()
{
   for(var camidx=0; camidx < uroCamWatchObjects.length; camidx++)
   {
      if(uroCamDataChanged(camidx))
      {
         uroCamWatchObjects[camidx].watch.type = uroCamWatchObjects[camidx].current.type;
         uroCamWatchObjects[camidx].watch.azymuth = uroCamWatchObjects[camidx].current.azymuth;
         uroCamWatchObjects[camidx].watch.speed = uroCamWatchObjects[camidx].current.speed;
         uroCamWatchObjects[camidx].watch.validated = uroCamWatchObjects[camidx].current.validated;
         uroCamWatchObjects[camidx].watch.lat = uroCamWatchObjects[camidx].current.lat;
         uroCamWatchObjects[camidx].watch.lon = uroCamWatchObjects[camidx].current.lon;
      }
   }
   uroOWLUpdateHTML();
}
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>&nbsp;&nbsp;Type changed';
                        iHTML += ' ('+camObj.watch.type+' to '+camObj.current.type+')';
                     }
                     if(camObj.current.azymuth != camObj.watch.azymuth)
                     {
                        iHTML += '<br>&nbsp;&nbsp;Azimuth changed';
                        iHTML += ' ('+camObj.watch.azymuth+' to '+camObj.current.azymuth+')';
                     }
                     if(camObj.current.speed != camObj.watch.speed)
                     {
                        iHTML += '<br>&nbsp;&nbsp;Speed changed';
                        iHTML += ' ('+camObj.watch.speed+' to '+camObj.current.speed+')';
                     }
                     if(camObj.current.validated != camObj.watch.validated)
                     {
                        iHTML += '<br>&nbsp;&nbsp;Approval state changed';
                        iHTML += ' ('+camObj.watch.validated+' to '+camObj.current.validated+')';
                     }
                     if(camObj.current.lat != camObj.watch.lat)
                     {
                        iHTML += '<br>&nbsp;&nbsp;Latitude changed';
                        iHTML += ' ('+camObj.watch.lat+' to '+camObj.current.lat+')';
                     }
                     if(camObj.current.lon != camObj.watch.lon)
                     {
                        iHTML += '<br>&nbsp;&nbsp;Longitude changed';
                        iHTML += ' ('+camObj.watch.lon+' to '+camObj.current.lon+')';
                     }
                  }

                  if(camObj.server == W.location.code)
                  {
                     if(deleted === false)
                     {
                        iHTML += '&nbsp;<i class="fa fa-group" style="cursor:pointer;font-size:14px;color:#ccccff;" id="_uroCWLIcon1-'+camidx+'"></i>';
                     }
                     iHTML += '&nbsp;<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>&nbsp;<input type="button" id="_btnCWLGroupDel" value="Delete group"><br>';
   iHTML += '<input type="text" id="_uroCWLGroupEntry" style="width:40%;height:22px;">&nbsp;<input type="button" id="_btnCWLGroupAdd" value="Add group">';
   iHTML += '<br><input type="button" id="_btnRescanCamWatchList" value="Refresh camera data"><br><br>';
   iHTML += '<input type="button" id="_btnUpdateCamValues" value="Accept all changes"><br><br>';
   iHTML += '<b>Remove cameras from OWL:</b><br>';
   iHTML += '<input type="button" id="_btnRemoveDeletedCameras" value="Deleted">&nbsp;&nbsp;';
   iHTML += '<input type="button" id="_btnRemoveUnknownServerCameras" value="Unknown Server">&nbsp;&nbsp;';
   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('_btnUpdateCamValues', 'click', uroAcceptCameraChanges);
   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, caseInsensitive)
{
   var re;
   if(caseInsensitive) re = RegExp(keyword,'i');
   else re = RegExp(keyword);

   if(desc.search(re) != -1) return true;
   else return false;
}
function uroClickify(desc, suffix)
{
   var terminators = [' ',',',')',']','\r','\n'];
   if(desc === null) return '';
   if(desc === undefined) return '';
   if(desc === '') return '';
   
   desc = desc.replace(/<\/?[^>]+(>|$)/g, "");
   if(desc !== "null")
   {   
      var links = desc.split("http");
      desc = '';
      var i, j, linkEndPos, descPostLink;
      for(i=0; i<links.length; i++)
      {
         if(links[i][2] === '/')
         {
            links[i] = "http" + links[i];
            linkEndPos = links[i].length + 1;
            for(j=0; j<terminators.length; j++)
            {
               if(links[i].indexOf(terminators[j]) !== -1)
               {
                  linkEndPos = Math.min(linkEndPos, links[i].indexOf(terminators[j]));
               }
            }
            
            descPostLink = '';
            if(linkEndPos < links[i].length)
            {
               descPostLink = links[i].slice(linkEndPos);
               links[i] = links[i].slice(0,linkEndPos);
            }
            desc += '<a target="_wazeUR" href="'+links[i]+'">'+links[i]+'</a>' + descPostLink;
         }         
         else
         {
            desc += links[i];
         }
      }
      return desc + suffix;
   }
   else
   {
      return '';
   }
}
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 uroChangeCustomMarkers(urID,isHighlighted,markerType)
{
   if(document.getElementById('customMarker_'+urID) !== null)
   {
      var customType = null;
      var customVariant = 0;
      if(markerType == "ur")
      {
         customType = W.map.updateRequestLayer.markers[urID].uroCustomType;
         if(W.model.mapUpdateRequests.objects[urID].attributes.open === false) customVariant = 2;
      }
      else if(markerType == "mp")
      {
         customType = W.map.problemLayer.markers[urID].uroCustomType;
         if(W.model.problems.objects[urID].attributes.open === false) customVariant = 2;
      }
      if(isHighlighted === true)
      {
         customVariant += 1;
      }
      if((customType !== null) && (customType !== undefined))
      {
         document.getElementById('customMarker_'+urID).innerHTML = '<img src="'+uroAltMarkers[customType][customVariant]+'">';
      }
   }
}
function uroRenderCustomMarkers(markerType)
{
   var urID;
   var elmID;
   var newSpan;
   var divElem;
   var objIdx;
   var customType;
   var customVariant;
   var cmlObj;
   var customMarker;
   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;
                  elmID = "customMarker_"+urID;
                  customMarker = '';
                  if(customType != -1)
                  {
                     if(document.getElementById(elmID) === null)
                     {
                        newSpan += '<span id="'+elmID+'" style="position:absolute;pointer-events:none;top:-3px;left:-2px;"></span>';
                     }
                     customType = uroGetCustomMarkerIdx(customType);
                     W.map.updateRequestLayer.markers[urID].uroCustomType = customType;
                     customVariant = 0;
                     if(W.model.mapUpdateRequests.objects[urID] !== undefined)
                     {
                        if(W.model.mapUpdateRequests.objects[urID].attributes.open === false) customVariant = 2;
                     }
                     customMarker = '<img src="'+uroAltMarkers[customType][customVariant]+'">';
                  }
                  else
                  {
                     if(document.getElementById(elmID) !== null)
                     {
                        document.getElementById(elmID).remove();
                     }
                  }
                  
                  if(newSpan !== '')
                  {
                     iconObj.$div.prepend(newSpan);
                  }
                     
                  if((customMarker !== '') && (document.getElementById(elmID) !== null))
                  {
                     document.getElementById(elmID).innerHTML = customMarker;
                  }
                     
                  if(addCommentCount)
                  {                    
                     elmID = "commentCount_"+urID+"_inner";
                     if(document.getElementById(elmID) !== null)
                     {
                        var styleLeft;
                        if(nComments < 10) styleLeft = '11px';
                        else if(nComments < 100) styleLeft = '8px';
                        else styleLeft = '5px';
                        document.getElementById(elmID).innerHTML = nComments;
                        document.getElementById(elmID).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;
                  elmID = "customMarker_"+urID;
                  customMarker = '';
                  if(customType != -1)
                  {
                     if(document.getElementById(elmID) === null)
                     {
                        newSpan = '<span id="'+elmID+'" style="position:absolute;pointer-events:none;top:-3px;left:-2px;"></span>';
                        if(W.map.problemLayer.markers[urID] !== undefined)
                        {
                           W.map.problemLayer.markers[urID].icon.$div.prepend(newSpan);
                        }                        
                     }
                     if(document.getElementById(elmID) !== null)
                     {
                        customType = uroGetCustomMarkerIdx(customType);
                        W.map.problemLayer.markers[urID].uroCustomType = customType;
                        customVariant = 0;
                        if(W.model.problems.objects[urID] !== undefined)
                        {
                           if(W.model.problems.objects[urID].attributes.open === false) customVariant = 2;
                        }
                        customMarker = '<img src="'+uroAltMarkers[customType][customVariant]+'">';
                        document.getElementById(elmID).innerHTML = customMarker;
                     }
                  }
                  else
                  {
                     if(document.getElementById(elmID) !== null)
                     {
                        document.getElementById(elmID).remove();
                     }
                  }
               }
            }
         }
      }
   }
}


function uroFilterRTCs()
{
   var t1=performance.now();
   if(uroFilterPreamble() === false) return;

   var uFR_filterFromApp = uroGetCBChecked('_cbHideUserRTCs');
   var uFR_filterFromWME = uroGetCBChecked('_cbHideEditorRTCs');
   var uFP_masterEnable = uroGetCBChecked('_cbMasterEnable');

   for (var rtcObj in W.map.closuresMarkerLayer.markers)
   {         
      if(W.map.closuresMarkerLayer.markers.hasOwnProperty(rtcObj))
      {
         var rtc = W.map.closuresMarkerLayer.markers[rtcObj];
         var rtcStyle = 'visible';      
         if(uFP_masterEnable === true)
         {
            var fromApp = (rtc.model.startDate.indexOf('1970-01-01') != -1);
            if(uFR_filterFromApp && fromApp)
            {
               rtcStyle = 'hidden';
            }
            if(uFR_filterFromWME && !fromApp)
            {
               rtcStyle = 'hidden';
            }
         }
         rtc.icon.imageDiv.style.visibility = rtcStyle;
      }
   }
   if(uroPerformanceMonitoringOutput === true) console.debug('uroFilterRTCs: '+(performance.now() - t1));
}


function uroFilterPlaces()
{
   var t1=performance.now();
   if(uroFilterPreamble() === false) return;

   if(uroPlaceSelected === true) return;

   if(uroGetCBChecked('_cbDisablePlacesFiltering') === true) return;

   uroUpdateVenueEditorList();

   var filterNameID = null;
   var tbUserName = uroGetElmValue('_textPlacesEditor');
   var selector = document.getElementById('_selectPlacesUserID');
   if(selector.selectedIndex > 0)
   {
      var selUserName = document.getElementById('_selectPlacesUserID').selectedOptions[0].innerHTML;
      if(selUserName == tbUserName)
      {
         filterNameID = document.getElementById('_selectPlacesUserID').selectedOptions[0].value;
      }
   }      
   if(filterNameID === null)
   {
      var userObj = W.model.users.getByAttributes({userName:tbUserName})[0];
      if(userObj !== undefined)
      {
         filterNameID = userObj.id;
      }
   }

   
   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_filterNoDescription = uroGetCBChecked('_cbHideNonDescribedPlaces');
   var uFP_filterWithDescription = uroGetCBChecked('_cbHideDescribedPlaces');
   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_filterCreatedBy = uroGetCBChecked('_cbShowOnlyPlacesCreatedBy');
   var uFP_filterEditedBy = uroGetCBChecked('_cbShowOnlyPlacesEditedBy');
   
   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_filterNoDescription || uFP_filterWithDescription)
              {
                var lDesc = lmObj.model.attributes.description.length;
                if((uFP_filterNoDescription) && (lDesc === 0)) placeStyle = 'hidden';
                if((uFP_filterWithDescription) && (lDesc !== 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(placeStyle == 'visible')
            {
               if(filterNameID != null)
               {
                  if(uFP_filterCreatedBy === true)
                  {
                     if(filterNameID != lmObj.model.attributes.createdBy) placeStyle = 'hidden';
                  }
                  if(uFP_filterEditedBy === true)
                  {
                     if(filterNameID != lmObj.model.attributes.updatedBy) 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;
                     }
                  }
               }
            }
         }
      }
   }
   if(uroPerformanceMonitoringOutput === true) console.debug('uroFilterPlaces: '+(performance.now() - t1));
}
function uroFilterCameras()
{
   var t1 = performance.now();
   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;
         }
      }

      var isChecked_cbShowOnlyCamsCreatedBy = uroGetCBChecked('_cbShowOnlyCamsCreatedBy');
      var isChecked_cbShowOnlyCamsEditedBy = uroGetCBChecked('_cbShowOnlyCamsEditedBy');
      var isChecked_cbShowOnlyMyCams = uroGetCBChecked('_cbShowOnlyMyCams');
      var isChecked_cbShowWorldCams = uroGetCBChecked('_cbShowWorldCams');
      var isChecked_cbShowUSACams = uroGetCBChecked('_cbShowUSACams');
      var isChecked_cbShowNonWorldCams = uroGetCBChecked('_cbShowNonWorldCams');
      var isChecked_cbShowApprovedCams = uroGetCBChecked('_cbShowApprovedCams');
      var isChecked_cbShowNonApprovedCams = uroGetCBChecked('_cbShowNonApprovedCams');
      var isChecked_cbShowOlderCreatedNonApproved = uroGetCBChecked('_cbShowOlderCreatedNonApproved');
      var isChecked_cbShowOlderUpdatedNonApproved = uroGetCBChecked('_cbShowOlderUpdatedNonApproved');
      var isChecked_cbShowSpeedCams = uroGetCBChecked('_cbShowSpeedCams');
      var isChecked_cbShowRedLightCams = uroGetCBChecked('_cbShowRedLightCams');
      var isChecked_cbShowDummyCams = uroGetCBChecked('_cbShowDummyCams');
      var isChecked_cbShowIfNoSpeedSet = uroGetCBChecked('_cbShowIfNoSpeedSet');
      var isChecked_cbShowIfSpeedSet = uroGetCBChecked('_cbShowIfSpeedSet');
      var isChecked_cbShowRLCIfNoSpeedSet = uroGetCBChecked('_cbShowRLCIfNoSpeedSet');
      var isChecked_cbShowRLCIfNonZeroSpeedSet = uroGetCBChecked('_cbShowRLCIfNonZeroSpeedSet');
      var isChecked_cbShowRLCIfZeroSpeedSet = uroGetCBChecked('_cbShowRLCIfZeroSpeedSet');
      var isChecked_cbHideCreatedByMe = uroGetCBChecked('_cbHideCreatedByMe');
      var isChecked_cbHideCreatedByRank0 = uroGetCBChecked('_cbHideCreatedByRank0');
      var isChecked_cbHideCreatedByRank1 = uroGetCBChecked('_cbHideCreatedByRank1');
      var isChecked_cbHideCreatedByRank2 = uroGetCBChecked('_cbHideCreatedByRank2');
      var isChecked_cbHideCreatedByRank3 = uroGetCBChecked('_cbHideCreatedByRank3');
      var isChecked_cbHideCreatedByRank4 = uroGetCBChecked('_cbHideCreatedByRank4');
      var isChecked_cbHideCreatedByRank5 = uroGetCBChecked('_cbHideCreatedByRank5');
      var isChecked_cbHideUpdatedByMe = uroGetCBChecked('_cbHideUpdatedByMe');
      var isChecked_cbHideUpdatedByRank0 = uroGetCBChecked('_cbHideUpdatedByRank0');
      var isChecked_cbHideUpdatedByRank1 = uroGetCBChecked('_cbHideUpdatedByRank1');
      var isChecked_cbHideUpdatedByRank2 = uroGetCBChecked('_cbHideUpdatedByRank2');
      var isChecked_cbHideUpdatedByRank3 = uroGetCBChecked('_cbHideUpdatedByRank3');
      var isChecked_cbHideUpdatedByRank4 = uroGetCBChecked('_cbHideUpdatedByRank4');
      var isChecked_cbHideUpdatedByRank5 = uroGetCBChecked('_cbHideUpdatedByRank5');
      var isChecked_cbHideCWLCams = uroGetCBChecked('_cbHideCWLCams');
      
      var hasValue_inputCameraMinCreatedDays = uroGetElmValue('_inputCameraMinCreatedDays');
      var hasValue_inputCameraMinUpdatedDays = uroGetElmValue('_inputCameraMinUpdatedDays');
      
      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(isChecked_cbShowOnlyCamsCreatedBy === true)
               {
                  if(filterNameID != uroCam.attributes.createdBy) uroCamStyle = 'hidden';
               }
               if(isChecked_cbShowOnlyCamsEditedBy === true)
               {
                  if(filterNameID != uroCam.attributes.updatedBy) uroCamStyle = 'hidden';
               }
            }
            
            if(isChecked_cbShowOnlyMyCams === true)
            {
               if((uroUserID != uroCam.attributes.createdBy)&&(uroUserID != uroCam.attributes.updatedBy)) uroCamStyle = 'hidden';
            }

            if((isChecked_cbShowWorldCams === false) || (isChecked_cbShowUSACams === false) || (isChecked_cbShowNonWorldCams === false))
            {
               var posWorld = uroCamCreator.indexOf('world_');
               var posUSA = uroCamCreator.indexOf('usa_');

               if((isChecked_cbShowWorldCams === false) && (posWorld === 0)) uroCamStyle = 'hidden';
               if((isChecked_cbShowUSACams === false) && (posUSA === 0)) uroCamStyle = 'hidden';
               if((isChecked_cbShowNonWorldCams === false) && (posWorld !== 0) && (posUSA !== 0)) uroCamStyle = 'hidden';
            }

            if((isChecked_cbShowApprovedCams === false) || (isChecked_cbShowNonApprovedCams === false))
            {
               if((isChecked_cbShowApprovedCams === false) && (uroCamApproved === true)) uroCamStyle = 'hidden';
               if((isChecked_cbShowNonApprovedCams === false) && (uroCamApproved === false)) uroCamStyle = 'hidden';
            }

            if((isChecked_cbShowNonApprovedCams === true) && (uroCamApproved === false))
            {
               if(((isChecked_cbShowOlderCreatedNonApproved === true)) && (uroGetCameraAge(uroCam,1) <= hasValue_inputCameraMinCreatedDays)) uroCamStyle = 'hidden';
               if(((isChecked_cbShowOlderUpdatedNonApproved === true)) && (uroGetCameraAge(uroCam,0) <= hasValue_inputCameraMinUpdatedDays)) uroCamStyle = 'hidden';
            }

            if((isChecked_cbShowSpeedCams === false) || (isChecked_cbShowRedLightCams === false) || (isChecked_cbShowDummyCams === false))
            {
               if((isChecked_cbShowSpeedCams === false) && (uroCamType == 2)) uroCamStyle = 'hidden';
               if((isChecked_cbShowRedLightCams === false) && (uroCamType == 4)) uroCamStyle = 'hidden';
               if((isChecked_cbShowDummyCams === false) && (uroCamType == 3)) uroCamStyle = 'hidden';
            }

            
            if((isChecked_cbShowSpeedCams === true) && (uroCamType == 2))
            {
               if((isChecked_cbShowIfNoSpeedSet === false) && (uroCam.attributes.speed === null)) uroCamStyle = 'hidden';
               if((isChecked_cbShowIfSpeedSet === false) && (uroCam.attributes.speed !== null)) uroCamStyle = 'hidden';
            }
            
            if((isChecked_cbShowRedLightCams === true) && (uroCamType == 4))
            {
               if((isChecked_cbShowRLCIfNoSpeedSet === false) && (uroCam.attributes.speed === null)) uroCamStyle = 'hidden';
               if((isChecked_cbShowRLCIfNonZeroSpeedSet === false) && (uroCam.attributes.speed > 0)) uroCamStyle = 'hidden';
               if((isChecked_cbShowRLCIfZeroSpeedSet === false) && (uroCam.attributes.speed === 0)) uroCamStyle = 'hidden';
            }

            if(isChecked_cbHideCreatedByMe === true)
            {
               if(uroUserID == uroCam.attributes.createdBy) uroCamStyle = 'hidden';
            }
            if((isChecked_cbHideCreatedByRank0 === true) && (uroCamCreatorRank === 0)) uroCamStyle = 'hidden';
            if((isChecked_cbHideCreatedByRank1 === true) && (uroCamCreatorRank == 1)) uroCamStyle = 'hidden';
            if((isChecked_cbHideCreatedByRank2 === true) && (uroCamCreatorRank == 2)) uroCamStyle = 'hidden';
            if((isChecked_cbHideCreatedByRank3 === true) && (uroCamCreatorRank == 3)) uroCamStyle = 'hidden';
            if((isChecked_cbHideCreatedByRank4 === true) && (uroCamCreatorRank == 4)) uroCamStyle = 'hidden';
            if((isChecked_cbHideCreatedByRank5 === true) && (uroCamCreatorRank == 5)) uroCamStyle = 'hidden';

            if(isChecked_cbHideUpdatedByMe === true)
            {
               if(uroUserID == uroCam.attributes.updatedBy) uroCamStyle = 'hidden';
            }
            if((isChecked_cbHideUpdatedByRank0 === true) && (uroCamUpdaterRank === 0)) uroCamStyle = 'hidden';
            if((isChecked_cbHideUpdatedByRank1 === true) && (uroCamUpdaterRank == 1)) uroCamStyle = 'hidden';
            if((isChecked_cbHideUpdatedByRank2 === true) && (uroCamUpdaterRank == 2)) uroCamStyle = 'hidden';
            if((isChecked_cbHideUpdatedByRank3 === true) && (uroCamUpdaterRank == 3)) uroCamStyle = 'hidden';
            if((isChecked_cbHideUpdatedByRank4 === true) && (uroCamUpdaterRank == 4)) uroCamStyle = 'hidden';
            if((isChecked_cbHideUpdatedByRank5 === true) && (uroCamUpdaterRank == 5)) uroCamStyle = 'hidden';

            if((isChecked_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();
               }
            }
         }
      }
   }
   if(uroPerformanceMonitoringOutput === true) console.debug('uroFilterCameras: '+(performance.now() - t1));
}
function uroFilterMapComments()
{
   var t1 = performance.now();
   if(uroFilterPreamble() === false) return;

   var uFURs_masterEnable = uroGetCBChecked('_cbMasterEnable');
   var filterDescMustBePresent = uroGetCBChecked('_cbMCDescriptionMustBePresent');
   var filterDescMustBeAbsent = uroGetCBChecked('_cbMCDescriptionMustBeAbsent');
   var filterKeywordMustBePresent = uroGetCBChecked('_cbMCEnableKeywordMustBePresent');
   var filterKeywordMustBeAbsent = uroGetCBChecked('_cbMCEnableKeywordMustBeAbsent');
   var filterMyFollowed = uroGetCBChecked('_cbMCHideMyFollowed');
   var filterMyUnfollowed = uroGetCBChecked('_cbMCHideMyUnfollowed');
   var filterRoadworks = uroGetCBChecked('_cbMCFilterRoadworks');
   var filterConstruction = uroGetCBChecked('_cbMCFilterConstruction');
   var filterClosure = uroGetCBChecked('_cbMCFilterClosure');
   var filterEvent = uroGetCBChecked('_cbMCFilterEvent');
   var filterNote = uroGetCBChecked('_cbMCFilterNote');
   var filterWSLM = uroGetCBChecked('_cbMCFilterWSLM');
   var filterBOG = uroGetCBChecked('_cbMCFilterBOG');
   var filterDifficult = uroGetCBChecked('_cbMCFilterDifficult');
   var invertFilters = uroGetCBChecked('_cbInvertMCFilter');   
   var keywordPresent = uroGetElmValue('_textMCKeywordPresent');
   var keywordAbsent = uroGetElmValue('_textMCKeywordAbsent');   
   var caseInsensitive = uroGetCBChecked('_cbMCCaseInsensitive');  
   
   for (var mcIdx = 0; mcIdx < W.map.layers[uroMCLayerIdx].features.length; mcIdx++)
   {
      {
         var mcObj = W.map.layers[uroMCLayerIdx].features[mcIdx].model;
         
         var desc = '';
         if(mcObj.attributes.subject !== null) desc += mcObj.attributes.subject.replace(/<\/?[^>]+(>|$)/g, "");
         if(mcObj.attributes.body !== null) desc += mcObj.attributes.body.replace(/<\/?[^>]+(>|$)/g, "");
         for(var cIdx=0; cIdx < mcObj.attributes.conversation.length; cIdx++)
         {
            desc += mcObj.attributes.conversation[cIdx].text.replace(/<\/?[^>]+(>|$)/g, "");
         }

         var mcStyle = 'visible';         
         var customType = uroGetCustomType(null, "mc", desc);
         
         if(uFURs_masterEnable === true)
         {
            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;


            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;

            // keywords
            if(mcStyle == 'visible')
            {
               if(filterDescMustBePresent === true)
               {
                  if(desc === '') mcStyle = 'hidden';
               }
               if(filterDescMustBeAbsent === true)
               {
                  if(desc !== '') mcStyle = 'hidden';
               }

               if(filterKeywordMustBePresent === true)
               {
                  var keywordIsPresentInDesc = uroKeywordPresent(desc,keywordPresent,caseInsensitive);
                  filterByIncludedKeyword = (filterByIncludedKeyword && (!keywordIsPresentInDesc));
               }
               if(filterKeywordMustBeAbsent === true)
               {
                  var keywordIsAbsentInDesc = uroKeywordPresent(desc,keywordAbsent,caseInsensitive);
                  filterByNotIncludedKeyword = (filterByNotIncludedKeyword || keywordIsAbsentInDesc);
               }
               
               filterByNotIncludedKeyword = (filterByNotIncludedKeyword && filterKeywordMustBeAbsent);
               filterByIncludedKeyword = (filterByIncludedKeyword && filterKeywordMustBePresent);
               if(filterByNotIncludedKeyword || filterByIncludedKeyword)
               {
                  mcStyle = 'hidden';
               }
               
            }

            // is following?
            if(mcStyle == 'visible')
            {
               if(mcObj.attributes.isFollowing === true)
               {
                  if(filterMyFollowed === true) mcStyle = 'hidden';
               }
               else
               {
                  if(filterMyUnfollowed === true) mcStyle = 'hidden';
               }
            }

            // custom tags
            if(mcStyle == 'visible')
            {
               if(ukroadworks_ur === true)
               {
                  if(filterRoadworks === true) mcStyle = 'hidden';
               }
               else if(construction_ur === true)
               {
                  if(filterConstruction === true) mcStyle = 'hidden';
               }
               else if(closure_ur === true)
               {
                  if(filterClosure === true) mcStyle = 'hidden';
               }
               else if(event_ur === true)
               {
                  if(filterEvent === true) mcStyle = 'hidden';
               }
               else if(note_ur === true)
               {
                  if(filterNote === true) mcStyle = 'hidden';
               }
               else if(wslm_ur === true)
               {
                  if(filterWSLM === true) mcStyle = 'hidden';
               }
               else if(bog_ur === true)
               {
                  if(filterBOG === true) mcStyle = 'hidden';
               }
               else if(difficult_ur === true)
               {
                  if(filterDifficult === true) mcStyle = 'hidden';
               }

               if(invertFilters === true)
               {
                 if(mcStyle == 'hidden') mcStyle = 'visible';
                 else mcStyle = 'hidden';
               }
            }
         }

         var geoID = W.map.layers[uroMCLayerIdx].features[mcIdx].geometry.id;
         if(document.getElementById(geoID) !== null)
         {
            document.getElementById(geoID).style.visibility = mcStyle;
         }

      }
   }
   if(uroPerformanceMonitoringOutput === true) console.debug('uroFilterMapComments: '+(performance.now() - t1));
}


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()
{
   var t1 = performance.now();
   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 caseInsensitive = uroGetCBChecked('_cbCaseInsensitive');
   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 desc = ureq.attributes.description;
         var customType = uroGetCustomType(ureqID, "ur", desc);
         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;

            if(desc !== null) desc = desc.replace(/<\/?[^>]+(>|$)/g, "");
            else desc = '';

            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,caseInsensitive);
                     filterByIncludedKeyword = (filterByIncludedKeyword && (!keywordIsPresentInDesc));
                  }
                  if(filterKeywordMustBeAbsent === true)
                  {
                     var keywordIsAbsentInDesc = uroKeywordPresent(desc,keywordAbsent,caseInsensitive);
                     filterByNotIncludedKeyword = (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,caseInsensitive);
                           filterByIncludedKeyword = (filterByIncludedKeyword && (!keywordIsPresentInComments));
                        }
                        if(filterKeywordMustBeAbsent === true)
                        {
                           var keywordIsAbsentInComments = uroKeywordPresent(commentText,keywordAbsent,caseInsensitive);
                           filterByNotIncludedKeyword = (filterByNotIncludedKeyword || keywordIsAbsentInComments);
                        }
                     }
                     else
                     {
                        if(filterUserID === true)
                        {
                           urStyle = 'hidden';
                        }
                     }

                     filterByNotIncludedKeyword = (filterByNotIncludedKeyword && filterKeywordMustBeAbsent);
                     filterByIncludedKeyword = (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');
   if(uroPerformanceMonitoringOutput === true) console.debug('uroFilterURs: '+(performance.now() - t1));
}
function uroGetProblemTypes()
{
   uroKnownProblemTypeIDs = [];
   uroKnownProblemTypeNames = [];
   var tProblemList = I18n.lookup("problems.types");
   for(var tObj in tProblemList)
   {
      if(tObj !== undefined)
      {
         uroKnownProblemTypeIDs.push(parseInt(tObj));
         uroKnownProblemTypeNames.push(tProblemList[tObj].title);
      }
   }    
}
function uroFilterProblems()
{
   var t1 = performance.now();
   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('_cbMPFilter_T200');
   
   var filterTypes = [];
   var i;
   for(i=0; i<uroKnownProblemTypeIDs.length; i++)
   {
      if(uroGetCBChecked('_cbMPFilter_T'+uroKnownProblemTypeIDs[i])) filterTypes.push(uroKnownProblemTypeIDs[i]);
   }    
   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;
            var desc = '';
            if(problem.attributes.description != null)
            {
               desc = problem.attributes.description;
            }            
            customType = uroGetCustomType(ureqID, "mp", desc);
            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(uroKnownProblemTypeIDs.indexOf(problemType) !== -1)
               {
                  if(filterTypes.indexOf(problemType) !== -1)
                  {
                     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');
   if(uroPerformanceMonitoringOutput === true) console.debug('uroFilterProblems: '+(performance.now() - t1));
}

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_MCTabClick()
{
   uroFilterMapComments();
}
function uroFilterItems_PlacesTabClick()
{
   uroFilterPlaces();
}
function uroFilterItems_CamerasTabClick()
{
   uroFilterCameras();
}
function uroFilterItems_MiscTabClick()
{
   uroFilterItems();
}
function uroFilterItems_MasterEnableClick()
{
   if(uroGetCBChecked('_cbMasterEnable') === false)
   {
      uroHidePopup('uroFilterItems_MasterEnableClick');
   }
   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();
   uroFilterRTCs();
}
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('uroDeleteObject');
   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;
      }
      else
      {
         userName = '<a href="' + (W.Config.user_profile.url + userName) + '">' + userName + '</a>';
      }
      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, desc)
{
   var provider = '';
   var customText = '';
   if(uroGetCBChecked('_cbCustomKeywordMarkers')) customText = document.getElementById('_textCustomKeyword').value.toLowerCase();
   
   if(desc === null) desc = '';
   
   if(markerType == "ur")
   {
      var ureq = W.model.mapUpdateRequests.objects[idSrc];
      // early test for native speed limit URs
      if(ureq.attributes.type == 23) return 98;      
   }
   else if(markerType == "mp")
   {
      var mp = W.model.problems.objects[idSrc];
      if(mp.attributes.provider != null)
      {
         provider = mp.attributes.provider;
      }
   }

   if(desc !== '')
   {
      if((markerType == 'ur') || (markerType == 'mc'))
      {
         for(var tag=0; tag<uroCustomURTags.length; tag++)
         {
            var keyword = uroCustomURTags[tag];
            if(desc.indexOf(keyword) != -1)
            {
               return tag;
            }
         }         
      }

      if(markerType == 'ur')
      {
         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(provider.indexOf('London TFL Closures') != -1) return 104;
      }
   }
   if(markerType == "ur")
   {
      return uroCheckCommentsForTag(idSrc, customText);
   }

   return -1;
}

function uroFormatRestriction(restObj)
{ 
   var retval = '<tr>';
   
   if(restObj._defaultType == "DIFFICULT")
   {
      retval += '<td colspan=11>Difficult Turn';
   }
   else
   {
      var roDays = null;
      var roFromDate = null;
      var roToDate = null;
      var roFromTime = null;
      var roToTime = null;   
      if(restObj._days !== undefined)
      {
         roDays = restObj._days;
         roFromDate = restObj._fromDate;
         roToDate = restObj._toDate;
         roFromTime = restObj._fromTime;
         roToTime = restObj._toTime;
      }
      else if(restObj._timeFrames.length > 0)
      {
         if(restObj._timeFrames[0]._weekdays !== undefined)
         {
            roDays = restObj._timeFrames[0]._weekdays;
            roFromDate = restObj._timeFrames[0]._startDate;
            roToDate = restObj._timeFrames[0]._endDate;
            roFromTime = restObj._timeFrames[0]._fromTime;
            roToTime = restObj._timeFrames[0]._toTime;
         }
      }
      
      if(roDays !== null)
      {
         retval += '<td style="text-align:center;">';
         if((roDays & 1) == 1) retval += 'M';
         else retval += '-';
         retval += '</td><td style="text-align:center;">';
         if((roDays & 2) == 2) retval += 'Tu';
         else retval += '-';
         retval += '</td><td style="text-align:center;">';
         if((roDays & 4) == 4) retval += 'W';
         else retval += '-';
         retval += '</td><td style="text-align:center;">';
         if((roDays & 8) == 8) retval += 'Th';
         else retval += '-';
         retval += '</td><td style="text-align:center;">';
         if((roDays & 16) == 16) retval += 'F';
         else retval += '-';
         retval += '</td><td style="text-align:center;">';
         if((roDays & 32) == 32) retval += 'Sa';
         else retval += '-';
         retval += '</td><td style="text-align:center;">';
         if((roDays & 64) == 64) retval += 'Su';
         else retval += '-';
         retval += '</td>';
      }

      retval += '<td>';

      if(roFromDate === null) retval += 'All dates';
      else retval += roFromDate+' to '+roToDate;

      retval += '</td><td>';

      if((restObj._allDay === true) || ((roFromTime === null) && (roToTime === null))) retval += 'All day';
      else retval += roFromTime+' to '+roToTime;

      retval += '</td><td>';

      if((restObj.allVehicleType !== undefined) && (restObj._vehicleTypes !== undefined))
      {
         if(restObj.allVehicleTypes == restObj._vehicleTypes) retval += 'All vehicles';
         else retval += 'Some vehicles';
      }

      retval += '</td><td>';
      retval += uroClickify(restObj._description, '');
   }

   retval += '</td></tr>';

   return retval;
}

function uroHidePopup(caller)
{
   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('uroRecentreSessionOnUR');
   return false;
}
function uroRecentreSessionOnMP()
{
   W.map.problemLayer.markers[uroShownFID].icon.imageDiv.click();
   W.map.moveTo(W.map.problemLayer.markers[uroShownFID].lonlat, 5);
   uroHidePopup('uroRecentreSessionOnMP');
   return false;
}
function uroRecentreSessionOnPUR()
{
   W.map.placeUpdatesLayer.markers[uroShownFID].icon.imageDiv.click();
   W.map.moveTo(W.map.placeUpdatesLayer.markers[uroShownFID].lonlat, 5);
   uroHidePopup('uroRecentreSessionOnPUR');
   return false;
}
function uroRecentreSessionOnPPUR()
{
   W.map.parkingPlaceUpdatesLayer.markers[uroShownFID].icon.imageDiv.click();
   W.map.moveTo(W.map.parkingPlaceUpdatesLayer.markers[uroShownFID].lonlat, 5);
   uroHidePopup('uroRecentreSessionOnPPUR');
   return false;
}
function uroRecentreSessionOnVenueNavPoint()
{
   W.map.moveTo(uroGetVenueNavPoint(uroShownFID), 5);
   uroHidePopup('uroRecentreSessionOnVenueNavPoint');
   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;
   else if(uroStackType == 4) markerCollection = W.map.parkingPlaceUpdatesLayer.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;
   else if(stackType == 4) markerCollection = W.map.parkingPlaceUpdatesLayer.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 = (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)
         {
            if(W.model.venues.objects[vObj].getNavigationPoints().length > 0)
            {
               // if the venue has any navpoints defined, use the position of the first one
               return W.model.venues.objects[vObj].getNavigationPoints()[0]._point.toLonLat();
            }
            else
            {
               // otherwise use the centrepoint of the venue point or polygon
               if(W.model.venues.objects[vObj].attributes.geometry.id.indexOf("Point") === -1)
               {
                  return W.model.venues.objects[vObj].attributes.geometry.getCentroid().toLonLat();
               }
               else
               {
                  return W.model.venues.objects[vObj].attributes.geometry.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.getElementsByClassName('waze-icon-clock')[0].click();
   return false;
}

function uroKillCentering()
{
   return W.map.getExtent();
}
function uroRestoreCentering()
{
   if(uroAutoCentreDisabledOn.length > 0)
   {
      var acType = uroAutoCentreDisabledOn[0];
      var acID = uroAutoCentreDisabledOn[1];
      
      if(acType == 'PUR')
      {
         if(W.map.placeUpdatesLayer.markers[acID] != null)
         {
            W.map.placeUpdatesLayer.markers[acID].model.geometry.getBounds = W.map.placeUpdatesLayer.markers[acID].model.geometry.origGetBounds;
         }
      }
      else if(acType == 'PPUR')
      {
         if(W.map.parkingPlaceUpdatesLayer.markers[acID] != null)
         {
            W.map.parkingPlaceUpdatesLayer.markers[acID].model.geometry.getBounds = W.map.parkingPlaceUpdatesLayer.markers[acID].model.geometry.origGetBounds;
         }
      }
      else if(acType == 'MP')
      {
         if(W.map.problemLayer.markers[acID] != null)
         {
            W.map.problemLayer.markers[acID].model.getDisconnectBounds = W.map.problemLayer.markers[acID].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;
   }
   var mte = "---";
   if(rcObj.eventId !== null)
   {
      try
      {
         mte = W.model.majorTrafficEvents.objects[rcObj.eventId].attributes.names[0].value;
      }
      catch(err)
      {
      }
   }
   
   result += '<td>' + startDate + ' to ' + endDate + '</td>';
   result += '<td>' + provider + '</td>';
   result += '<td>' + reason + '</td>';
   result += '<td>' + mte + '</td>';
   result += '</td></tr>';
   return result;
}


function uroGetAddress(streetID, houseNumber, formatForSegmentPopup)
{
   var result = '';
   if((houseNumber !== undefined) && (houseNumber !== null))
   {
      result += houseNumber + ' ';
   }

   if(streetID != null)
   {
      var streetName = I18n.lookup('edit.address.no_street');
      var doesStreetIDExist = true;
      if(W.model.streets.objects[streetID] === undefined)
      {
         streetName = 'non-existent streetID';
         doesStreetIDExist = false;
      }
      else
      {
         if((streetName !== null) && (W.model.streets.objects[streetID].isEmpty === false))
         {
            streetName = W.model.streets.objects[streetID].name;
         }
      }
      if(formatForSegmentPopup === true)
      {
         result += '<b>'+streetName+'</b><br>';         
      }
      else
      {
         result += streetName + ', ';
      }
      
      if(doesStreetIDExist === true)
      {
         var cityName = I18n.lookup('edit.address.no_city');
         var doesCityIDExist = true;
         var cityID = W.model.streets.objects[streetID].cityID;
         if(W.model.cities.objects[cityID] === undefined)
         {
            cityName = 'non-existent cityID';
            doesCityIDExist = false;
         }
         else
         {
            if(W.model.cities.objects[cityID].attributes.name !== "")
            {
               cityName = W.model.cities.objects[cityID].attributes.name;
            }
         }
         result += cityName + ', ';
         
         if(doesCityIDExist === true)
         {
            var stateID = W.model.cities.objects[cityID].attributes.stateID;
            if(W.model.states.objects[stateID] === undefined)
            {
               result += 'non-existent stateID';
            }
            else
            {
               result += W.model.states.objects[stateID].name;
            }
         }
      }
   }
   result += '<br>';
   
   return result;
}


function uroNewLookHighlightedItemsCheck(e)
{
   var result = '';
   var rw;
   var rh;
   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 isMapComment = false;
   var newPopupType = null;
   var markerObj;
   var markerPos;
   var markerImg;
   var ureq = null;
   var idx;
   var hovered = false;
   var targetTab = '';
   var unstackedX;
   var unstackedY;
   var ureqID = null;
   var isUR = false;
   var isProblem = false;
   var isTurnProb = false;
   var isPlaceUpdate = false;
   var mouseX;
   var mouseY;
   var uroDaysResolved;
   var renderIntent;
   
   // function preamble...
   {
      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;
      }
      
      if(e == 'dwellTimeout')
      {
         mouseX = uroPrevMouseX;
         mouseY = uroPrevMouseY;
         
         if(uroPointerWithinMap === false)
         {
            return;
         }
      }
      else
      {
         if(uroTestPointerOutsideMap(e.clientX, e.clientY))
         {
            return;
         }      
         
         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 mouseLonLat = W.map.getLonLatFromViewPortPx(new OpenLayers.Pixel(mouseX,mouseY));
   var mousePoint = new OpenLayers.Geometry.Point(mouseLonLat.lon, mouseLonLat.lat);
   if(uroMousedOverMapComment !== null)
   {
      if(W.model.mapComments.objects[uroMousedOverMapComment] === undefined)
      {
         uroAddLog('clearing uroMousedOverMapComment: object no longer exists in current map view');
         uroMousedOverMapComment = null;
      }
      else if(W.model.mapComments.objects[uroMousedOverMapComment].attributes.geometry.containsPoint(mousePoint) === false)
      {
         uroAddLog('clearing uroMousedOverMapComment: pointer no longer within comment boundary');
         uroMousedOverMapComment = null;
      }
   }
   
   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;   
   
   // popup for segment restrictions
   if((uroMousedOverMarkerType === null) && (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(uroMousedOverMapComment !== null) 
               {
                  uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for segment highlight');
                  uroMousedOverOtherObjectWithinMapComment = true;
               }
               
               if(W.map.segmentLayer.features[slIdx].fid === null) segObj = W.map.segmentLayer.features[slIdx].model;
               else segObj = W.map.segmentLayer.features[slIdx];

               var streetID = segObj.attributes.primaryStreetID;
               if(streetID !== null)
               {
               
                  // generic segment data
                  if(uroGetCBChecked('_cbInhibitSegGenericPopup') === false)
                  {                  
                     doPopUp = true;
                     uroAddLog('building popup for segment '+streetID);
                     
                     result += uroGetAddress(streetID, null, true);                     
                     result += '<b>ID: </b>'+segObj.attributes.id+'<br>';
                     
                     var autoLock = segObj.attributes.rank;
                     var userLock = segObj.attributes.lockRank;
                     result += '<b>Lock: </b>';
                     if(userLock !== null)
                     {
                        result += 'M' + (userLock+1) + ' / ';
                     }
                     result += 'A' + (autoLock+1) + '<br>';

                     var fwdSpeed = segObj.attributes.fwdMaxSpeed;
                     var revSpeed = segObj.attributes.revMaxSpeed;
                     var fwdUnverified = segObj.attributes.fwdMaxSpeedUnverified;
                     var revUnverified = segObj.attributes.revMaxSpeedUnverified;
                     var roadType = segObj.attributes.roadType;
                     var verifyLimits = true;
                     if((roadType === 17) || (roadType === 20))
                     {
                        verifyLimits = false;
                     }
                     if(segObj.attributes.fwdDirection)
                     {
                        result += '<b>A-B speed: </b>'+uroGetLocalisedSpeedString(fwdSpeed, verifyLimits);
                        if(fwdUnverified) result += ' (unverified)';
                        result += '<br>';
                     }
                     if(segObj.attributes.revDirection)
                     {
                        result += '<b>B-A speed: </b>'+uroGetLocalisedSpeedString(revSpeed, verifyLimits);
                        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.restrictions.length > 0)
                  {
                     doPopUp = true;
                     var fwdResult = '<tr><td colspan=11><b>A-B restrictions:</b></td></tr>';
                     var revResult = '<tr><td colspan=11><b>B-A restrictions:</b></td></tr>';
                     var bothResult = '<tr><td colspan=11><b>Two-way restrictions:</b></td></tr>';
                     
                     //result += '<tr><td colspan=11><b>Restrictions:</b></td></tr>';
                     var nABRestrictions = 0;
                     var nBARestrictions = 0;
                     var nBothRestrictions = 0;
                     for(idx = 0; idx < segObj.attributes.restrictions.length; idx++)
                     {
                        restObj = segObj.attributes.restrictions[idx];
                        if(restObj._direction === "FWD")
                        {
                           nABRestrictions++;
                           fwdResult += uroFormatRestriction(restObj);
                        }
                        else if(restObj._direction === "REV")
                        {
                           nBARestrictions++;
                           revResult += uroFormatRestriction(restObj);
                        }
                        else if(restObj._direction === "BOTH")
                        {
                           nBothRestrictions++;
                           bothResult += uroFormatRestriction(restObj);
                        }
                        else
                        {
                           uroAddLog("unknown restriction direction...");
                        }
                     }
                     if(nABRestrictions > 0)
                     {
                        result += fwdResult;
                     }
                     if(nBARestrictions > 0)
                     {
                        result += revResult;
                     }
                     if(nBothRestrictions > 0)
                     {
                        result += bothResult;
                     }
                  }



                  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=4><b>A-B closures:</b></td></tr>';
                                       hasFwd = true;
                                    }
                                    result += uroAddClosureRowToTable(rcObj);
                                 }
                                 else
                                 {
                                    hasRev = true;
                                 }
                              }
                           }
                        }
                        if(hasRev === true)
                        {
                           result += '<tr><td colspan=4><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((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitTurnsPopup') === false))
   {
      var rescanTurnLayer = false;
      
      if(uroTurnsLayerIdx === null)
      {
         // always scan for the turn layer if we haven't yet found it...
         rescanTurnLayer = true;
      }
      else
      {
         if(W.map.layers[uroTurnsLayerIdx] === undefined) 
         {
            // also scan for it if the layer index we found previously is now invalid...
            rescanTurnLayer = true;
         }
         else
         {
            // and also scan for it if the previous layer index is still valid but no longer points to the
            // turns layer - this usually means another script has added its own layer at the end of the 
            // permanent native layer stack, so that when WME next generates the temporary turns layer, it
            // ends up with a higher layer index than when it was previously detected...
            if(W.map.layers[uroTurnsLayerIdx].name.indexOf('marker_drawing_context') != -1)
            {
               rescanTurnLayer = true;
            }
         }
      }
      if(rescanTurnLayer === true)
      {
         uroWazeBits();
      }
      if(uroTurnsLayerIdx !== null)
      {     
         if(W.map.layers[uroTurnsLayerIdx].markers !== undefined)
         {
            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");

                  hovered = false;
                  if(markerImg.indexOf('turns-difficult11fe96ffa43a348c0820d9723e52503e.png') !== -1)
                  {
                     if((markerPos === '0px -37px') || (markerPos === '-36px 0px'))
                     {
                        hovered = true;
                     }
                  }
                  else if(markerImg.indexOf('turns96765f551688fd5082b619129499bbb3.png') !== -1)
                  {
                     if(markerPos === '-72px 0px')
                     {
                        hovered = true;
                     }
                  }
                  if(hovered === true)
                  {
                     if(uroMousedOverMapComment !== null) 
                     {
                        uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for turn arrow highlight');
                        uroMousedOverOtherObjectWithinMapComment = true;
                     }
                     
                     uroAddLog('hover over restricted turn marker');
                     uroTBRObj = arrowElm.childNodes[0];
                     var trObj = ($(arrowElm).data('model'));
                     
                     var Vertex = new require("Waze/Model/Graph/Vertex");
                     var turnGraph = W.model.getTurnGraph().getTurn(Vertex.forwardOf(trObj.fromSeg.attributes.id),Vertex.reverseOf(trObj.toSeg.attributes.id));
                     var resObj = turnGraph._turnData._restrictions;
                     if(resObj.length === 0)
                     {
                        turnGraph = W.model.getTurnGraph().getTurn(Vertex.forwardOf(trObj.fromSeg.attributes.id),Vertex.forwardOf(trObj.toSeg.attributes.id));
                        resObj = turnGraph._turnData._restrictions;
                     }
                     if(resObj.length === 0)
                     {
                        turnGraph = W.model.getTurnGraph().getTurn(Vertex.reverseOf(trObj.fromSeg.attributes.id),Vertex.forwardOf(trObj.toSeg.attributes.id));
                        resObj = turnGraph._turnData._restrictions;
                     }
                     if(resObj.length === 0)
                     {
                        turnGraph = W.model.getTurnGraph().getTurn(Vertex.reverseOf(trObj.fromSeg.attributes.id),Vertex.reverseOf(trObj.toSeg.attributes.id));
                        resObj = turnGraph._turnData._restrictions;
                     }
                     
                     
                     uroAddLog('building popup for turn restriction');

                     ////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;
                  }
               }
            }
         }
      }
   }

   // popup for landmarks
   if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitLandmarkPopup') === false))
   {
      uroPlaceSelected = false;
      
      var venueObj = null;
      renderIntent = null;
      var navpointPos=new OpenLayers.LonLat();
      
      for(var llFeatureIdx=0; llFeatureIdx < W.map.landmarkLayer.features.length; llFeatureIdx++)
      {
         if(W.map.landmarkLayer.features[llFeatureIdx] !== undefined)
         {
            renderIntent = W.map.landmarkLayer.features[llFeatureIdx].renderIntent;
            if(renderIntent == 'highlight')
            {
               if(W.map.getExtent().intersectsBounds(W.map.landmarkLayer.features[llFeatureIdx].geometry.getBounds()))
               {
                  if(uroMousedOverMapComment !== null) 
                  {
                     uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for place highlight');
                     uroMousedOverOtherObjectWithinMapComment = true;
                  }

                  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;
                     
                     uroAddLog('building popup for place '+uroFID);
                     
                     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>';
                     var vDesc = venueObj.attributes.description;
                     if(vDesc !== '')
                     {
                        result += '"<i>' + uroClickify(vDesc, '') + '</i>"<br>';
                     }
                     result += '<hr>';
                     result += uroGetAddress(venueObj.attributes.streetID, venueObj.attributes.houseNumber, false);                  
                     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='+...
                     var npLayers = '';
                     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;
            }
         }
      }
   }
   
   // popup for map comments
   if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitMapCommentPopup') === false))
   {
      if (W.map.layers[uroMCLayerIdx].featureType !== 'mapComment')
      {
         uroWazeBits();
      }
      if(uroMCLayerIdx !== null)
      {
         uroMCSelected = false;
         
         var mcObj = null;
         renderIntent = null;
         
         for(var mcFeatureIdx=0; mcFeatureIdx < W.map.layers[uroMCLayerIdx].features.length; mcFeatureIdx++)
         {
            renderIntent = W.map.layers[uroMCLayerIdx].features[mcFeatureIdx].renderIntent;
            if(renderIntent == 'highlight')
            {
               if(W.map.getExtent().intersectsBounds(W.map.layers[uroMCLayerIdx].features[mcFeatureIdx].geometry.getBounds()))
               {
                  mcObj = W.map.layers[uroMCLayerIdx].features[mcFeatureIdx].model;
                  if(newPopupType === null)
                  {  
                     if((uroMousedOverMapComment === mcObj.attributes.id) && (uroMousedOverOtherObjectWithinMapComment === true))
                     {
                        uroAddLog('inhibit popup for map comment '+uroMousedOverMapComment);
                     }
                     else
                     {
                        uroMousedOverOtherObjectWithinMapComment = false;
                        if(mcObj.attributes.geometry.id.indexOf('Polygon') !== -1)
                        {
                           // only capture ID for area comments...
                           uroMousedOverMapComment = mcObj.attributes.id;
                        }
                        uroFID = mcObj.attributes.id;
                        uroAddLog('building popup for map comment '+uroFID);
                        
                        result += '<b>';
                        if(mcObj.attributes.subject === '') 
                        {
                           result += '<i>No subject</i>';
                        }
                        else result += mcObj.attributes.subject;
                        result += '</b><br>';
                        result += uroClickify(mcObj.attributes.body, '<br>');
                        
                        var mcDaysOld = uroGetMCAge(mcObj,0,false);
                        var mcSubmittedTS = uroGetMCAge(mcObj,0,true);
                        if(mcSubmittedTS != -1)
                        {
                           mcSubmittedTS = uroGetDateTimeString(mcSubmittedTS);
                        }
                        if(mcDaysOld != -1)
                        {
                           result += '<i>Submitted ' + uroParseDaysAgo(mcDaysOld) + ' ';
                           if(mcSubmittedTS != -1) result += '(' + mcSubmittedTS + ') ';
                           if(mcObj.attributes.createdBy != null)
                           {
                              result += ' by '+uroGetUserNameAndRank(mcObj.attributes.createdBy);
                           }
                           result += '</i><br>';
                        }
                        mcDaysOld = uroGetMCAge(mcObj,1,false);
                        mcSubmittedTS = uroGetMCAge(mcObj,1,true);
                        if(mcSubmittedTS != -1)
                        {
                           mcSubmittedTS = uroGetDateTimeString(mcSubmittedTS);
                        }
                        if(mcDaysOld != -1)
                        {
                           result += '<i>Updated ' + uroParseDaysAgo(mcDaysOld) + ' ';
                           if(mcSubmittedTS != -1) result += '(' + mcSubmittedTS + ') ';
                           if(mcObj.attributes.createdBy != null)
                           {
                              result += ' by '+uroGetUserNameAndRank(mcObj.attributes.updatedBy);
                           }
                           result += '</i><br>';
                        }
                        
                        var mcHasMyComments = false;
                        var mcNComments = mcObj.attributes.conversation.length;
                        if(mcNComments > 0)
                        {
                           for(var i=0; i<mcNComments; i++)
                           {
                              if(mcObj.attributes.conversation[i].userID == uroUserID)
                              {
                                 mcHasMyComments = true;
                                 break;
                              }
                           }
                        }
                        result += '<br>' + mcNComments +' comment';
                        if(mcNComments != 1) result += 's';
                        if((mcHasMyComments === false) && (mcNComments > 0)) result += ' (none by me)';

                        newPopupType = 'map_comment';
                        isMapComment = true;
                        break;
                     }
                  }
                  else
                  {
                     var mcOtherID = mcObj.attributes.id;
                     uroAddLog('map comment '+mcOtherID+' is also highlighted');
                  }
               }
               else
               {
                  uroAddLog('map comment '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup');
               }               
            }
            else if((renderIntent == 'select') || (renderIntent == 'highlightselected'))
            {
               uroMCSelected = true;
            }
         }
      }
   }   

   // look for URs, place updates and problems
   if(newPopupType === null)
   {
      if((uroMousedOverMarkerType == 'ur') && (newPopupType === null))
      {
         unstackedX = uroParsePxString(W.map.updateRequestLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.left);
         unstackedY = uroParsePxString(W.map.updateRequestLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.top);
         // check for stacking...
         if(uroShownFID != uroMousedOverMarkerID)
         {
            uroCheckStacking(1,uroMousedOverMarkerID, unstackedX, unstackedY);
         }
         hovered = true;
         
         if(uroGetCBChecked('_cbInhibitURPopup') === false)
         {
            if(uroMousedOverMapComment !== null) 
            {
               uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for UR highlight');
               uroMousedOverOtherObjectWithinMapComment = true;
            }

            isUR = true;
            newPopupType = uroMousedOverMarkerType;
            ureq = W.model.mapUpdateRequests.objects[uroMousedOverMarkerID];

            // 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);

            uroFID = uroMousedOverMarkerID;
            
            uroAddLog('building popup for UR '+uroMousedOverMarkerID);
            result = '<b>Update Request ('+uroMousedOverMarkerID+'): ' + I18n.lookup("update_requests.types." + ureq.attributes.type) + '</b><br>';
            
            result += uroClickify(ureq.attributes.description, '<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[uroMousedOverMarkerID] != null)
            {
               var hasMyComments = uroURHasMyComments(uroMousedOverMarkerID);
               var nComments = W.model.updateRequestSessions.objects[uroMousedOverMarkerID].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[uroMousedOverMarkerID].comments[nComments-1]);
                  if(commentDaysOld != -1)
                  {
                     result += ', last update '+uroParseDaysAgo(commentDaysOld);
                  }
               }
            }         
         }
      }

      if(((uroMousedOverMarkerType == 'pur') || (uroMousedOverMarkerType == 'ppur')) && (newPopupType === null))
      {
         
         if(uroMousedOverMarkerType == 'pur')
         {
            ureq = W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].model;
            unstackedX = uroParsePxString(W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.left);
            unstackedY = uroParsePxString(W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.top);
         }
         else
         {
            ureq = W.map.parkingPlaceUpdatesLayer.markers[uroMousedOverMarkerID].model;
            unstackedX = uroParsePxString(W.map.parkingPlaceUpdatesLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.left);
            unstackedY = uroParsePxString(W.map.parkingPlaceUpdatesLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.top);
         }         
         if(uroShownFID != uroMousedOverMarkerID)
         {
            // check for stacking...
            if(uroMousedOverMarkerType == 'pur')
            {
               uroCheckStacking(3,uroMousedOverMarkerID, unstackedX, unstackedY);
            }
            else
            {
               uroCheckStacking(4,uroMousedOverMarkerID, unstackedX, unstackedY);
            }               
         }
         hovered = true;

         if(uroGetCBChecked('_cbInhibitPUPopup') === false)
         {
            newPopupType = uroMousedOverMarkerType;
            if(uroMousedOverMapComment !== null) 
            {
               uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for PUR highlight');
               uroMousedOverOtherObjectWithinMapComment = true;
            }
                  
            isPlaceUpdate = true;

            // 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);
            
            uroFID = uroMousedOverMarkerID;
            
            // 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();
            if(uroMousedOverMarkerType == 'pur')
            {
               W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.origGetBounds = W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.getBounds;
               W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.getBounds = uroKillCentering;
               uroAutoCentreDisabledOn.push('PUR', uroMousedOverMarkerID);
            }
            else
            {
               W.map.parkingPlaceUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.origGetBounds = W.map.parkingPlaceUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.getBounds;
               W.map.parkingPlaceUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.getBounds = uroKillCentering;
               uroAutoCentreDisabledOn.push('PPUR', uroMousedOverMarkerID);
            }
            
            if(uroMousedOverMarkerType == 'pur')
            {
               uroAddLog('building popup for placeUpdate '+uroMousedOverMarkerID);
            }
            else
            {
               uroAddLog('building popup for parkingPlaceUpdate '+uroMousedOverMarkerID);
            }
               
            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>';
            }
         }
      }
      
      if((uroMousedOverMarkerType == 'mp') && (newPopupType === null))
      {
         unstackedX = uroParsePxString(W.map.problemLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.left);
         unstackedY = uroParsePxString(W.map.problemLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.top);
         // check for stacking...
         if(uroShownFID != uroMousedOverMarkerID)
         {
            uroCheckStacking(2,uroMousedOverMarkerID, unstackedX, unstackedY);
         }     
         hovered = true;         
      
         if(uroGetCBChecked('_cbInhibitMPPopup') === false)
         {
            if(uroMousedOverMapComment !== null) 
            {
               uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for MP highlight');
               uroMousedOverOtherObjectWithinMapComment = true;
            }
                  
            isProblem = true;
            newPopupType = uroMousedOverMarkerType;
            ureq = W.model.problems.objects[uroMousedOverMarkerID];
            if(ureq === undefined)
            {
               if(uroDOMHasTurnProblems)
               {
                  ureq = W.model.turnProblems.objects[uroMousedOverMarkerID];
                  if(ureq != null) isTurnProb = true;
               }
            }
            
            // 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);

            uroFID = uroMousedOverMarkerID;
            // same method of disabling the on-click auto-centre behaviour as for PURs above...
            uroRestoreCentering();
            W.map.problemLayer.markers[uroMousedOverMarkerID].model.origGetDisconnectBounds = W.map.problemLayer.markers[uroMousedOverMarkerID].model.getDisconnectBounds;
            W.map.problemLayer.markers[uroMousedOverMarkerID].model.getDisconnectBounds = uroKillCentering;
            uroAutoCentreDisabledOn.push('MP', uroMousedOverMarkerID); 

            uroAddLog('building popup for problem '+uroMousedOverMarkerID);
            if(isTurnProb) result = '<b>Turn Problem ('+uroMousedOverMarkerID+'): ' + I18n.lookup("problems.types.turn.title");
            else
            {
               result = '<b>Map Problem ('+uroMousedOverMarkerID+'): ';

               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>';
               }
            }
         }
      }

      if(hovered === false)
      {
         uroFID = -1;
         if(uroStackType !== null)
         {
            var tStackType = uroStackType;
            uroRestackMarkers();
            if(tStackType == 1) uroFilterURs();
            else if(tStackType == 2) uroFilterProblems();
            else if(tStackType == 3) uroFilterPlaces();
         }
      }
      else if(newPopupType !== null)
      {
         // 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='+...
         var urLayers = '';
         urLink = urLink.substr(0,urLink.indexOf('?zoom'));
         urLink += '?zoom=5&lat='+urPos.lat+'&lon='+urPos.lon+urLayers;

         if(isUR) urLink += '&mapUpdateRequest='+uroMousedOverMarkerID;
         else if(isTurnProb) urLink += '&showturn='+uroMousedOverMarkerID+'&endshow';
         else if(isProblem) urLink += '&mapProblem='+uroMousedOverMarkerID;
         else if(isPlaceUpdate) 
         {
            if(uroMousedOverMarkerType == 'pur')
            {
               urLink += '&showpur='+uroMousedOverMarkerID+'&endshow';
            }
            else
            {
               urLink += '&showppur='+uroMousedOverMarkerID+'&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 != 'mp') && (newPopupType != 'pur') && (newPopupType != 'ppur'))
   {
      uroRestoreCentering();
   }

   // look for cameras
   if((newPopupType === null) && (uroGetCBChecked('_cbInhibitCamPopup') === false))
   {
      for(var clFeature in W.map.camerasLayer._featureMap)
      {
         if(W.map.camerasLayer._featureMap[clFeature] !== undefined)
         {
            if(W.map.camerasLayer._featureMap[clFeature].renderIntent == 'highlight')
            {
               ureq = W.map.camerasLayer._featureMap[clFeature].model;
               ureqID = ureq.attributes.id;

               // test isSelected() so that we only do overview data on cameras that are being hovered over
               if(ureq.isSelected() === false)
               {
                  if(uroMousedOverMapComment !== null) 
                  {
                     uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for camera highlight');
                     uroMousedOverOtherObjectWithinMapComment = true;
                  }
                  uroPopupY -= 20;
                  newPopupType = 'camera';
                  uroFID = ureqID;
                  uroAddLog('building 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, true);
                  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)
            {
               if(newPopupType == 'pur')
               {
                  uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnPUR, true);
               }
               else
               {
                  uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnPPUR, 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';
         uroPopupAutoHideTimer = (uroGetElmValue('_inputPopupAutoHideTimeout') * 10);
      }
      uroPopupTimer = -1;
   }
   else if((newPopupType === null) && (uroPopupDwellTimer !== 0) && (uroPopupShown === true))
   {
      uroHidePopup('uroNewLookHighlightedItemsCheck');
   }
   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 = '';
   var tName = '';
   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();
      var linkColour = '#c0c0ff';
      if (uroGetCBChecked('_cbMoveAMList') === false)
      {
         linkColour = '#800000';
      }

      for(var amObj in W.model.managedAreas.objects)
      {
         if(W.model.managedAreas.objects[amObj].geometry.containsPoint(mousePoint))
         {
            if(amList !== '') amList += ', ';
            
            tName = uroGetUserNameAndRank(W.model.managedAreas.objects[amObj].userID);
            if(tName.indexOf('a href') !== -1)
            {
               tName = tName.replace('a href', 'a style="color:'+linkColour+';" href');
            }
            amList += tName;
         }
      }
      if(amList === '')
      {
         amList = 'none';
      }
      amList = "&nbsp;<b>Area Managers:</b> "+amList;
   }
   document.getElementById("uroAMList").innerHTML = amList;
}

function uroMouseDown()
{
   uroMouseIsDown = true;
}
function uroMouseUp()
{
   uroMouseIsDown = false;
}
function uroTestPointerOutsideMap(mX, mY)
{
   var bLeft = map.parentElement.offsetLeft;
   var bRight = (bLeft + map.offsetWidth);
   var bTop = (map.parentElement.offsetTop + document.getElementById("topbar-container").clientHeight);
   var bBottom = (map.parentElement.offsetTop + map.offsetHeight + document.getElementById("topbar-container").clientHeight - document.getElementsByClassName("WazeMapFooter")[0].clientHeight);   

   if
   (
      (mX < bLeft) ||
      (mX > bRight) ||
      (mY < bTop) ||
      (mY > bBottom)
   )
   {
      uroPointerWithinMap = false;
      return true;
   }
   else
   {
      uroPointerWithinMap = true;
      return false;
   }
}
function uroMouseOut(e)
{
   if(uroTestPointerOutsideMap(e.clientX, e.clientY))
   {
      uroHidePopup('uroMouseOut');
   }
}

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)
{
   e.stopPropagation();
   
   var doClick = true;
   if(document.getElementsByClassName('form-control new-comment-text').length > 0)
   {
      if(document.getElementsByClassName('form-control new-comment-text')[0].textLength > 0)
      {
         uroShowAlertBox("fa-warning", "URO+ Warning", "Comment not sent, close report panel anyway?", true, "Yes", "No", uroCloseReportPanel, null);
		 // set doClick to false here, as uroCloseReportPanel will be called by the alert box handler if required...
		 doClick = false;
      }
   }
   // no alert box has been generated, so close the panel
   if(doClick)
   {
	   uroCloseReportPanel();
   }
}   
function uroCloseReportPanel()
{
   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)
         {
            if(cityObj.attributes.isEmpty === true)
            {
               // The donor point place we create to take the cloned RPP properties may have been automatically given
               // a city name by WME, and thus the city name field will already be filled in and activated...  If our
               // RPP doesn't however have a city name, we need to deactivate the city name field again, so that WME
               // doesn't complain about the user trying to save the new RPP with an empty city name
               if(document.getElementsByClassName("empty-city")[0].checked === false)
               {
                  document.getElementsByClassName("empty-city")[0].click();
               }
            }
            document.getElementsByClassName('city-name')[0].value = cityObj.attributes.name;
            document.getElementsByClassName('city-name')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
            
            // county
            document.getElementsByClassName('state-id')[0].value = cityObj.attributes.stateID;
            document.getElementsByClassName('state-id')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
            
            // country
            document.getElementsByClassName('country-id')[0].value = cityObj.attributes.countryID;
            document.getElementsByClassName('country-id')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
         }
      }
      
      // house number
      document.getElementsByClassName('house-number')[0].value = 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 venues menu in order to generate a
            // new point venue object that we can manipulate...
            document.getElementsByClassName('toolbar-group-venues')[0].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;
   
   var uroCLocation;
   var uroCReason;
   var uroCEvent;
   var uroCDirection;
   var uroCHasStartDate;
   var uroCStartDate;
   var uroCStartTime;
   var uroCEndDate;
   var uroCEndTime;
   var uroCIgnoreTraffic;

   function uroCompleteClosureCloning()
   {
      if(document.getElementsByClassName('edit-closure').length === 0)
      {
         window.setTimeout(uroCompleteClosureCloning,100);
         return;
      }

      // 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(uroCLocation !== null)
      {
         document.getElementsByName('closure_location')[0].value = uroCLocation;
         document.getElementsByName('closure_location')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }
      if(uroCReason !== null) 
      {
         document.getElementsByName('closure_reason')[0].value = uroCReason;
         document.getElementsByName('closure_reason')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }
      if(uroCDirection !== null)
      {
         document.getElementsByName('closure_direction')[0].selectedIndex = uroCDirection;
         document.getElementsByName('closure_direction')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }
      if(uroCHasStartDate !== null)
      {
         document.getElementsByName('closure_hasStartDate')[0].checked = uroCHasStartDate;
         document.getElementsByName('closure_hasStartDate')[0].dispatchEvent(new Event('change', {'bubbles':true})); 
      }
      if(uroCStartDate !== null)
      {
         document.getElementsByName('closure_startDate')[0].value = uroCStartDate;
         document.getElementsByName('closure_startDate')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }
      if(uroCStartTime !== null)
      {
         document.getElementsByName('closure_startTime')[0].value = uroCStartTime;
         document.getElementsByName('closure_startTime')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }
      
      if(uroCIgnoreTraffic !== null)
      {
         document.getElementsByName('closure_permanent')[0].checked = uroCIgnoreTraffic;
         document.getElementsByName('closure_permanent')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }     
      if(uroCEndTime !== null)
      {
         document.getElementsByName('closure_endTime')[0].value = uroCEndTime;
         // the cloning process doesn't alter the end time, which seems to confuse WME when it then receives the
         // change event - the MTE dropdown ends up being reset to just the "Choose event" and "None" entries
         // regardless of how many MTE entries there ought to be.  The fix for this appears to simply be to then
         // submit a second change event...
         document.getElementsByName('closure_endTime')[0].dispatchEvent(new Event('change', {'bubbles':true}));
         document.getElementsByName('closure_endTime')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }       

      // the current version of WME wipes any existing end date as soon as the end time is altered, so we now need
      // to set the date after the time instead of before as in earlier versions of this function...
      if(uroCEndDate !== null)
      {
         document.getElementsByName('closure_endDate')[0].value = uroCEndDate;
         document.getElementsByName('closure_endDate')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }
      
      
      uroTempFixMTEDropDown();
      if(uroCEvent !== null)
      {
         if(document.getElementsByName('closure_eventId')[0].options.length > 1)
         {
            for(var loop=0; loop<document.getElementsByName('closure_eventId')[0].options.length; loop++)
            {
               if(document.getElementsByName('closure_eventId')[0].options[loop].value == uroCEvent)
               {
                  document.getElementsByName('closure_eventId')[0].selectedIndex = loop;
                  break;
               }
            }
         }
         else
         {
            document.getElementsByName('closure_eventId')[0].selectedIndex = 0;
         }
         document.getElementsByName('closure_eventId')[0].dispatchEvent(new Event('change', {'bubbles':true}));
      }
   }   
      
   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();
      uroCLocation = uroGetElementProperty('closure_location', 0, 'value');
      uroCReason = uroGetElementProperty('closure_reason', 0, 'value');
      uroCEvent = uroGetElementProperty('closure_eventId', 0, 'value');
      uroCDirection = uroGetElementProperty('closure_direction', 0, 'selectedIndex');
      uroCHasStartDate = uroGetElementProperty('closure_hasStartDate', 0, 'checked');
      uroCStartDate = uroGetElementProperty('closure_startDate', 0, 'value');
      uroCStartTime = uroGetElementProperty('closure_startTime', 0, 'value');
      uroCEndDate = uroGetElementProperty('closure_endDate', 0, 'value');
      uroCEndTime = uroGetElementProperty('closure_endTime', 0, 'value');
      uroCIgnoreTraffic = uroGetElementProperty('closure_permanent', 0, 'checked');  
      document.getElementsByClassName('closures')[0].getElementsByClassName('cancel-button')[0].click();
      
      // auto-increment the start and end dates...
      uroCStartDate = uroIncrementClosureDate(uroCStartDate,1);
      uroCEndDate = uroIncrementClosureDate(uroCEndDate,1);
      
      // generate a click event on the Add a closure button to open up the closure editing UI, then
      // wait for the UI to finish opening...
      document.getElementsByClassName('add-closure-button')[0].click();
      window.setTimeout(uroCompleteClosureCloning,100);
   }

   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;
      uroShowAlertBox("fa-warning", "URO+ Warning", I18n.lookup("closures.delete_confirm_no_reason")+' ('+I18n.lookup("closures.apply_to_all")+')', true, "Yes", "No", uroDeleteAllClosuresAction, null);      
   }
   function uroDeleteAllClosuresAction()
   {
      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 uroForceFeedRefresh()
   {
      uroDeleteAutoRepeat = false;
      uroFeedFilterReloads = 0;
   }
   
   function uroDeleteAllVisibleFeedEntries()
   {
      if(uroGetCBChecked('_cbEnableDeleteFeedEntries'))
      {
         uroDeleteAutoRepeat = uroGetCBChecked('_cbReloadFeedAfterDelete');
         var nEntries = document.getElementsByClassName('feed-item').length;
         for(var loop = nEntries-1; loop >= 0; loop--)
         {
            if(document.getElementsByClassName('feed-item')[loop].style.display == 'block')
            {
               document.getElementsByClassName('feed-item')[loop].getElementsByClassName('delete')[0].click(); 
               uroFeedEntriesDeleted++;
            }
         }
         document.getElementById('_statsDeleteFeed').innerHTML = uroFeedEntriesDeleted+' entries nuked so far...';
      }
   }
   function uroConfirmDeleteAllVisibleFeedEntries()
   {
      uroFeedEntriesDeleted = 0;
      document.getElementById('_statsDeleteFeed').innerHTML = '';
      if(uroGetCBChecked('_cbReloadFeedAfterDelete') === false)
      {
         uroShowAlertBox("fa-warning", "URO+ Warning", "Deleting the currently visible feed entries <b>cannot</b> be undone.<br>Are you <i>sure</i> you want to do this?", true, "Yes", "No", uroDeleteAllVisibleFeedEntries, null);
      }
      else
      {
         uroShowAlertBox("fa-warning", "URO+ Warning", "You are about to delete ALL unfiltered feed entries currently visible in WME and also those stored on the server.  This <b>cannot</b> be undone, and may take some time to complete.<br>Are you <i>sure</i> you want to do this?", true, "Yes", "No", uroDeleteAllVisibleFeedEntries, null);
      }
   }
   
   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 += '<div id="_uroDeleteFeedEntryControls">';
            iHTML += '<div class="delete-all-button btn btn-primary" id="_btnDeleteAllVisibleFeedEntries">';
            iHTML += '<i class="fa fa-trash"></i> Delete visible feed entries?</div>';
            iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbReloadFeedAfterDelete" /><i class="fa fa-refresh"></i><br>';
            iHTML += '<div id="_statsDeleteFeed"></div><p><p>';
            iHTML += '</div>';
            
            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_TypeMP" />MP<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>';
            for(var loop=0; loop < uroFeedFilterFilters.length; loop++)
            {
               iHTML += '<input type="checkbox" id="_cbFeedFilter_'+loop+'" />'+I18n.lookup(uroFeedFilterFilters[loop][0])+'<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="_textFeedFilter_Keyword" />';
            
            iHTML += '<br><b>Filter feed by age:</b><br>';
            iHTML += '<input type="checkbox" id="_cbFeedFilter_HideLessThan">Hide entries less than </input>';
            iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFeedFilterMinDays"> days old<br>';
            iHTML += '<input type="checkbox" id="_cbFeedFilter_HideMoreThan">Hide entries more than </input>';
            iHTML += '<input type="number" min="0" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFeedFilterMaxDays"> days old<br>';
            

            iHTML += '</div>';
            nDiv.innerHTML = iHTML;
            document.getElementById('sidepanel-feed').insertBefore(nDiv,document.getElementById('sidepanel-feed').childNodes[0]);
            uroAddEventListener('_uroFFCtrlVisibility','click',uroToggleFFCtrls, true);
            if(uroGetCBChecked('_cbEnableDeleteFeedEntries'))
            {
               document.getElementById('_uroDeleteFeedEntryControls').style.display = "block";
            }
            else
            {
               document.getElementById('_uroDeleteFeedEntryControls').style.display = "none";
            }
            
            uroShowFeedFilter = true;
            uroToggleFFCtrls();
            
            var nDiv2 = document.createElement('div');
            nDiv2.id = "uroFeedRefresher";
            nDiv2.style.display = 'none';
            nDiv2.innerHTML = '<br><div id="_btnFeedRefresh" class="btn btn-block btn-default" style="display: block;"><i class="fa fa-refresh"></div>';
            document.getElementById('sidepanel-feed').appendChild(nDiv2);
            uroAddEventListener('_btnFeedRefresh','click',uroForceFeedRefresh, true);
            uroAddEventListener('_btnDeleteAllVisibleFeedEntries',"click", uroConfirmDeleteAllVisibleFeedEntries, false);           
         }
      }
   }
   
   function uroTSTFeedFilter()
   {
      if(uroGetCBChecked('_cbEnableDeleteFeedEntries'))
      {
         document.getElementById('_uroDeleteFeedEntryControls').style.display = "block";
      }
      else
      {
         document.getElementById('_uroDeleteFeedEntryControls').style.display = "none";
      }
      
      var feedEntries = document.getElementsByClassName('feed-item');
      var feedLength = feedEntries.length;
      if(feedLength !== uroPreviousFeedLength) uroDoFeedFilter = true;
      uroPreviousFeedLength = feedLength;
      if(uroDoFeedFilter === false) return;      
      if(feedLength === 0) return;    
      uroDoFeedFilter = false;
      if(document.getElementById('uroFeedFilter') === null) return;
      
      var hideFI;
      var iHTML;
      var fClass;
      
      var ffKeyword = uroGetElmValue('_textFeedFilter_Keyword');
      var ffShowKW = uroGetCBChecked('_cbFeedFilter_ShowKeyword');
      var ffHideKW = uroGetCBChecked('_cbFeedFilter_HideKeyword');
      var kwPresent;
      
      var isChecked_cbFeedFilter_TypeUR = uroGetCBChecked('_cbFeedFilter_TypeUR');
      var isChecked_cbFeedFilter_TypeMP = uroGetCBChecked('_cbFeedFilter_TypeMP');
      var isChecked_cbFeedFilter_TypePUR = uroGetCBChecked('_cbFeedFilter_TypePUR');
      var isChecked_cbFeedFilter_TypePM = uroGetCBChecked('_cbFeedFilter_TypePM');
      var isChecked_cbFeedFilter_MotNone = uroGetCBChecked('_cbFeedFilter_MotNone');
      var isChecked_cbFeedFilter_Invert = uroGetCBChecked('_cbFeedFilter_Invert');

      var trans = [];
      var isChecked = [];
      var nFilters = uroFeedFilterFilters.length;
     
      var i, j;
      
      for(i=0; i < nFilters; i++)
      {
         trans.push(I18n.lookup(uroFeedFilterFilters[i][0]));
         isChecked.push(uroGetCBChecked('_cbFeedFilter_'+i));
      }      

      var isChecked_filterLessThan = uroGetCBChecked('_cbFeedFilter_HideLessThan');
      var isChecked_filterMoreThan = uroGetCBChecked('_cbFeedFilter_HideMoreThan');
      var value_filterLessThan = uroGetElmValue('_inputFeedFilterMinDays');
      var value_filterMoreThan = uroGetElmValue('_inputFeedFilterMaxDays');
      
      var timestamp;
      var dayCount = 0;
      var dayTrans = ['', '', '', ''];
      var i18nLookups = ["datetime.distance_in_words.x_days.one", "datetime.distance_in_words.x_days.few", "datetime.distance_in_words.x_days.many", "datetime.distance_in_words.x_days.other"];
      
      for(i=0; i<4; i++)
      {
         var tTrans = I18n.lookup(i18nLookups[i]);
         if(tTrans !== undefined)
         {
            tTrans = tTrans.split(" ");
            if(tTrans.indexOf("1") === 0) dayTrans[i] = tTrans[1];
            else if(tTrans.indexOf("%{count}") === 0) dayTrans[i] = tTrans[1];
            else dayTrans[i] = tTrans[0];            
         }
         else
         {
            dayTrans[i] = '';
         }
      }

      for(i=0; i<feedLength; i++)
      {
         timestamp = feedEntries[i].getElementsByClassName("timestamp")[0].innerHTML;
         dayCount = 0;
         for(j=0; j<4; j++)
         {
            if(dayTrans[j] !== '')
            {
               if(timestamp.indexOf(dayTrans[j]) !== -1)
               {
                  dayCount = timestamp.split(" ");
                  if(dayCount[0].indexOf(dayTrans[j]) !== -1) dayCount = dayCount[1];
                  else dayCount = dayCount[0];
                  dayCount = parseInt(dayCount);                  
                  break;
               }
            }
         }
         hideFI = false;
         
         if(isChecked_filterLessThan)
         {
            if(dayCount < value_filterLessThan)
            {
               hideFI = true;
            }
         }
         if(isChecked_filterMoreThan)
         {
            if(dayCount > value_filterMoreThan)
            {
               hideFI = true;
            }
         }
         
         iHTML = feedEntries[i].innerHTML;
         fClass = feedEntries[i].className;
         
         if(isChecked_cbFeedFilter_TypeUR)
         {
            hideFI = (hideFI || (fClass.indexOf('feed-issue-ur') != -1));
         }
         if(isChecked_cbFeedFilter_TypeMP)
         {
            hideFI = (hideFI || (fClass.indexOf('feed-issue-mp') != -1));
         }
         if(isChecked_cbFeedFilter_TypePUR)
         {
            hideFI = (hideFI || (fClass.indexOf('feed-issue-pu') != -1));
         }
         if(isChecked_cbFeedFilter_TypePM)
         {
            hideFI = (hideFI || (fClass.indexOf('feed-notification-pm') != -1));
         }
         
         for(var filters=0; filters < nFilters; filters++)
         {
            if(isChecked[filters])
            {
               hideFI = (hideFI || (iHTML.indexOf(trans[filters]) != -1));
            }
         }
               
         if(isChecked_cbFeedFilter_MotNone)
         {
            hideFI = (hideFI || (iHTML.indexOf('motivation') == -1));
         }
                  
         if(isChecked_cbFeedFilter_Invert)
         {
            hideFI = !hideFI;
         }
        
         if((ffShowKW) || (ffHideKW))
         {
            kwPresent = (iHTML.indexOf(ffKeyword) != -1);
            hideFI = (hideFI || (kwPresent & ffHideKW));
            hideFI = (hideFI || ((!kwPresent) & ffShowKW));
         }
        
         if(hideFI) feedEntries[i].style.display = 'none';
         else feedEntries[i].style.display = 'block';
      }
      
      var nFeedItems = document.getElementById('sidepanel-feed').getElementsByClassName('feed-item').length;
      var nVisibleItems = 0;
      for(j=0; j < nFeedItems; j++)
      {
         if(document.getElementById('sidepanel-feed').getElementsByClassName('feed-item')[j].style.display == 'block')
         {
            nVisibleItems++;
         }
      }
     
      if(uroDeleteAutoRepeat === true)
      {
         if(nVisibleItems > 0)
         {
            uroDeleteAllVisibleFeedEntries();
         }
         else if(uroFeedFilterReloads == 5)
         {
            uroDeleteAutoRepeat = false;
         }
      }
      
      if(nVisibleItems < 5)
      {
         if(document.getElementById('sidepanel-feed').getElementsByClassName('feed-loading-more')[0].style.display == 'none')
         {
            if(uroFeedFilterReloads < 5)
            {
               document.getElementById('sidepanel-feed').getElementsByClassName('feed-load-more')[0].click();
               uroFeedFilterReloads++;
               document.getElementById('uroFeedRefresher').style.display = 'none';
            }
            else
            {
               document.getElementById('uroFeedRefresher').style.display = 'block';
            }
         }
      }
      else
      {
         uroFeedFilterReloads = 0;
         document.getElementById('uroFeedRefresher').style.display = 'none';
      }
   }   
//}

function uroGetMarkerType(markerDiv)
{
   var markerType = null;
   if(markerDiv.className.indexOf('user-generated') !== -1) markerType = 'ur';
   else if(markerDiv.className.indexOf('map-problem') !== -1) markerType = 'mp';
   else if(markerDiv.className.indexOf('place-update') !== -1) 
   {
      if(markerDiv.parentNode.id === W.map.parkingPlaceUpdatesLayer.div.id)
	   {
         markerType = 'ppur';
	   }
	   else
	   {
         markerType = 'pur';
      }
   }
   return markerType;
}
function uroMarkerMouseOver(e)
{
   var markerType;
   markerType = uroGetMarkerType(this);
   if(markerType !== null)
   {
      var markerID = null;
      markerID = this.attributes["data-id"].value;
      uroAddLog('hover over '+markerType+' ID '+markerID);
      uroMousedOverMarkerID = markerID;
      uroMousedOverMarkerType = markerType;
      
      if(markerType == 'ur') uroHoveredURID = markerID;
      
      if((markerType == 'ur') || (markerType == 'mp'))
      {
         uroChangeCustomMarkers(markerID,true,markerType);
      }
   }
   else
   {
      uroAddLog('hover over unknown object...');
   }
}
function uroMarkerMouseOut(e)
{
   var markerType;
   markerType = uroGetMarkerType(this);
   if(markerType !== null)
   {
      var markerID = null;
      markerID = this.attributes["data-id"].value;
      uroAddLog('hover off '+markerType+' ID '+markerID);
      uroMousedOverMarkerID = null;
      uroMousedOverMarkerType = null;
      uroHoveredURID = null;
      
      if((markerType == 'ur') || (markerType == 'mp'))
      {
         uroChangeCustomMarkers(markerID,false,markerType);
      }      
   }
   else
   {
      uroAddLog('hover off unknown object...');
   }
}
function uroMarkerClick()
{
   var markerType = uroGetMarkerType(this);
   if(markerType !== null)
   {
      var markerID = this.attributes["data-id"].value;
      uroAddLog('clicked on '+markerType+' marker '+markerID);
      uroClickedOnMarkerID = markerID;
      uroClickedOnMarkerType = markerType;
   }
}

function uroBlobMouseOver(e)
{
   var blobType = this.attributes.uroBlobType;
   if(blobType !== undefined)
   {
      var blobID = this.attributes.uroBlobID;
      uroAddLog('hover over '+blobType+' ID '+blobID);
      //uroMousedOverMarkerID = markerID;
      //uroMousedOverMarkerType = markerType;
   }
   else
   {
      uroAddLog('hover over unknown blob...');
   }
}
function uroBlobMouseOut(e)
{
   var blobType = this.attributes.uroBlobType;
   if(blobType !== undefined)
   {
      var blobID = this.attributes.uroBlobID;
      uroAddLog('hover off '+blobType+' ID '+blobID);
      //uroMousedOverMarkerID = markerID;
      //uroMousedOverMarkerType = markerType;
   }
   else
   {
      uroAddLog('hover off unknown blob...');
   }
}
function uroBlobClick()
{
   var blobType = this.attributes.uroBlobType;
   if(blobType !== undefined)
   {
      var blobID = this.attributes.uroBlobID;
      uroAddLog('clicked on '+blobType+' blob '+blobID);
   }
}

function uroMCLayerChanged()
{
	if(uroMCLayerIdx != null)
	{
		uroAddLog('adding MC blob event handlers');
		for(var mObj=0; mObj<W.map.layers[uroMCLayerIdx].features.length; mObj++)
		{
			var mcBlobID = W.map.layers[uroMCLayerIdx].features[mObj].model.attributes.geometry.id;
			var mcID = W.map.layers[uroMCLayerIdx].features[mObj].model.attributes.id;
			var mcBlob = document.getElementById(mcBlobID);
			if(mcBlob !== null)
			{
				mcBlob.addEventListener("mouseover", uroBlobMouseOver, false);
				mcBlob.addEventListener("mouseout", uroBlobMouseOut, false);
				mcBlob.addEventListener("click", uroBlobClick, false);
				mcBlob.attributes.uroBlobID = mcID;
				mcBlob.attributes.uroBlobType = "map_comment";
			}
		}
	}
}
function uroPlaceLayerChanged()
{
	uroAddLog('adding place blob event handlers');
	for(var mObj=0; mObj<W.map.landmarkLayer.features.length; mObj++)
	{
		var mcBlobID = W.map.landmarkLayer.features[mObj].model.attributes.geometry.id;
		var mcID = W.map.landmarkLayer.features[mObj].model.attributes.id;
		var mcBlob = document.getElementById(mcBlobID);
		if(mcBlob !== null)
		{
			mcBlob.addEventListener("mouseover", uroBlobMouseOver, false);
			mcBlob.addEventListener("mouseout", uroBlobMouseOut, false);
			mcBlob.addEventListener("click", uroBlobClick, false);
			mcBlob.attributes.uroBlobID = mcID;
			mcBlob.attributes.uroBlobType = "place";
		}
	}
}
function uroURLayerChanged()
{
   uroAddLog('adding UR marker event handlers');
   for(var mObj in W.map.updateRequestLayer.markers)
   {
      var mIcon = W.map.updateRequestLayer.markers[mObj].icon.div;
      mIcon.addEventListener("mouseover",uroMarkerMouseOver, false);
      mIcon.addEventListener("mouseout",uroMarkerMouseOut, false);
      mIcon.addEventListener("click",uroMarkerClick, false);
   }
}
function uroMPLayerChanged()
{
   uroAddLog('adding MP marker event handlers');
   for(var mObj in W.map.problemLayer.markers)
   {
      var mIcon = W.map.problemLayer.markers[mObj].icon.div;
      mIcon.addEventListener("mouseover", uroMarkerMouseOver, false);
      mIcon.addEventListener("mouseout", uroMarkerMouseOut, false);
      mIcon.addEventListener("click", uroMarkerClick, false);
   }
}
function uroPURLayerChanged()
{
   uroAddLog('adding PUR marker event handlers');   
   for(var mObj in W.map.placeUpdatesLayer.markers)
   {
      var mIcon = W.map.placeUpdatesLayer.markers[mObj].icon.div;
      mIcon.addEventListener("mouseover", uroMarkerMouseOver, false);
      mIcon.addEventListener("mouseout", uroMarkerMouseOut, false);
      mIcon.addEventListener("click", uroMarkerClick, false);
   }
}
function uroPPULayerChanged()
{
   uroAddLog('adding PPU marker event handlers');   
   for(var mObj in W.map.parkingPlaceUpdatesLayer.markers)
   {
      var mIcon = W.map.parkingPlaceUpdatesLayer.markers[mObj].icon.div;
      mIcon.addEventListener("mouseover", uroMarkerMouseOver, false);
      mIcon.addEventListener("mouseout", uroMarkerMouseOut, false);
      mIcon.addEventListener("click", uroMarkerClick, false);
   }
}
function uroRefilterFeedItems()
{
   uroDoFeedFilter = true;
}
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 uroMO_MCLayer = new MutationObserver(uroMCLayerChanged);
   ////uroMO_MCLayer.observe(W.map.layers[uroMCLayerIdx].div,{childList: true, attributes : true, characterData: true, subtree: true});
   ////var uroMO_PlaceLayer = new MutationObserver(uroPlaceLayerChanged);
   ////uroMO_PlaceLayer.observe(W.map.landmarkLayer.div,{childList: true, attributes : true, characterData : true, subtree: true});
   var uroMO_URLayer = new MutationObserver(uroURLayerChanged);
   uroMO_URLayer.observe(W.map.updateRequestLayer.div,{childList : true});
   var uroMO_MPLayer = new MutationObserver(uroMPLayerChanged);
   uroMO_MPLayer.observe(W.map.problemLayer.div,{childList : true});
   var uroMO_PURLayer = new MutationObserver(uroPURLayerChanged);
   uroMO_PURLayer.observe(W.map.placeUpdatesLayer.div,{childList : true});
   var uroMO_PPULayer = new MutationObserver(uroPPULayerChanged);
   uroMO_PPULayer.observe(W.map.parkingPlaceUpdatesLayer.div,{childList : true});   

   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('_selectPlacesUserID',"change", uroPlacesEditorSelected, true);

   uroAddEventListener('uroAlertTickBtn','click',uroCloseAlertBoxWithTick,true);
   uroAddEventListener('uroAlertCrossBtn','click',uroCloseAlertBoxWithCross,true);

   uroSetOnClick("_linkSelectUserRequests",uroShowURTab);
   uroSetOnClick("_linkSelectMapProblems",uroShowMPTab);
   uroSetOnClick("_linkSelectMapComments",uroShowMCTab);
   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');

   uroNewLookCheckDetailsRequest();
   // filter markers as and when the map is moved
   W.map.events.register("moveend", null, uroFilterItems);
   W.map.events.register("moveend", null, uroGetAMs);
   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;
   
   document.getElementById('sidepanel-feed').addEventListener("click", uroRefilterFeedItems);

   W.map.events.register("mouseup", null, uroMouseUp);
   W.map.events.register("mouseout", null, uroMouseOut);

   uroSetStyles(uroCtrlURs);
   uroSetStyles(uroCtrlMPs);
   uroSetStyles(uroCtrlMCs);
   uroSetStyles(uroCtrlPlaces);
   uroSetStyles(uroCtrlCameras);
   uroSetStyles(uroCtrlMisc);
   uroSetStyles(uroOWL);

   uroAddFeedFilterControls();

   uroLoadSettings();

   if(uroGetCBChecked('_cbEnableDTE'))
   {
      if(dteControlsIdx != -1)
      {
         dteSetNewTabLength();
      }
      else
      {
         uroAddLog('ERROR - archive panel not found!');
         uroSetStyleDisplay(uroUserTabId,'');
      }
   }

   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);
      }
   }
   
   // manually call the layer-change handlers on startup, since there's a good chance WME will already have
   // completed its own startup layer changes before our handlers get registered, preventing the marker handlers
   // from being set up as expected on any markers which are visible in the startup map view before the user forces
   // a layer update by panning/zooming/etc...
   ////uroMCLayerChanged();
   ////uroPlaceLayerChanged();
   uroURLayerChanged();
   uroMPLayerChanged();
   uroPURLayerChanged();
   
   uroSetupListeners = false; 
   uroInitialised = true;
}

function uroTSTPopupHandler()
{
   if(document.getElementsByClassName('panel')[0] === undefined)
   {
      uroHidePopupOnPanelOpen = true;
   }

         
   if(uroPopupShown === true)
   {
      var hidePopup = false;
      
      if(document.getElementById('layer-switcher-pinned-input').checked === false)
      {
         hidePopup = Boolean(hidePopup || (document.getElementsByClassName('menu not-visible').length === 0));
      }
      hidePopup = Boolean(hidePopup || (document.getElementsByClassName('toolbar-group open').length > 0));
      
      if(document.getElementsByClassName('panel')[0] != null)
      {
         if(uroHidePopupOnPanelOpen === true)
         {
            hidePopup = true;
            uroHidePopupOnPanelOpen = false;
         }
      }

      if(hidePopup === true)
      {        
         uroHidePopup('uroTSTPopupHandler 1');
      }
   }

   if((uroAreaNameHoverObj !== null) && (uroAreaNameHoverTime != -1) && (uroAreaNameOverlayShown === false))
   {
      if(++uroAreaNameHoverTime > 5)
      {
         uroAreaNameOverlaySetup();
      }
   }
   uroReplaceAreaNames(false);
   
   if(uroPopupAutoHideTimer > 0)
   {
      if(--uroPopupAutoHideTimer === 0)
      {
         uroHidePopup('uroTSTPopupHandler 2');
      }
   }

   if(uroPopupTimer > 0)
   {
      if(uroMouseInPopup === false)
      {
         uroPopupTimer--;
      }
   }
   if(uroPopupTimer === 0)
   {         
      uroHidePopup('uroTSTPopupHandler 3');
   }

   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()
{
   // replaces the "next xxx" button on UR, MP and PUR editing UIs
   
   // Correctly determining what WME is displaying for the "next" button in the UR/MP/(P)PUR panel is not trivial due to
   // inconsistencies in the panel behaviour depending on whether it was opened by clicking directly on the relevant 
   // marker, or by clicking on the associated feed entry...  For PURs, there's also the added complication of multi-part
   // update requests, where the same marker/panel are used to access more than one request and where, therefore, we need
   // to enable access to all requests contained within the PUR, but still inhibit the "next" button once the last
   // request in the multi-part sequence has been viewed.
   //
   // For directly-accesed markers, the "next" button caption is:
   //
   //    URs   = "Next update request" (update_requests.panel.next)
   //    MPs   = "Next map problem" (problems.panel.next)
   //    PURs  = "Next place" for single-part PURs or for the last part of a multi-part PUR (venues.update_requests.panel.next_venue)
   //          = "Next" for all but the last part of a multi-part PUR (venues.update_requests.panel.next)
   //    PPURs = "Next place" (venues.update_requests.panel.next_venue)
   // 
   // For markers accessed via the feed, the "next" button caption always appears to be "Next issue" (feed.issues.next)

   
   
   if(W.map.panelRegion.hasView() === true)
   {
      var nurButton = W.map.panelRegion.$el[0].getElementsByClassName('next')[0];
      if(nurButton === undefined)
      {
         nurButton = W.map.panelRegion.$el[0].getElementsByClassName('next-venue')[0];
      }
      if(nurButton !== undefined)
      {
         var doneString = I18n.lookup('problems.panel.done');
         var btnCaptionIsNextPlace = (nurButton.innerHTML.indexOf(I18n.lookup('venues.update_requests.panel.next_venue')) !== -1);
         var btnCaptionIsDefaultUR = (nurButton.innerHTML.indexOf(I18n.lookup('update_requests.panel.next')) !== -1);
         var btnCaptionIsDefaultMP = (nurButton.innerHTML.indexOf(I18n.lookup('problems.panel.next')) !== -1);
         var btnCaptionIsNextIssue = (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);   
         
         if(isURorMPPanel === true)
         {
            // user has enabled UR button mod?
            if(uroGetCBChecked('_cbInhibitNURButton') === true)
            {
               // the native UR panel button will always either be "Next update request" or "Next issue"
               updateButton = ((btnCaptionIsDefaultUR) || (btnCaptionIsNextIssue));
            }
            
            // user has enabled MP button mod?
            if(uroGetCBChecked('_cbInhibitNMPButton') === true)
            {
               // there's no way to determine if the edit panel has been opened for a UR or a MP, however as MPs
               // don't currently appear in the feed, the native button only uses "Next map problem" as its caption
               updateButton = (updateButton || btnCaptionIsDefaultMP);
            }
         }
         else if(isPURPanel === true)
         {
            if(uroGetCBChecked('_cbInhibitNPURButton') === true)
            {
               // for a (P)PUR, only modify the button if it's showing the "Next place" or "Next issue" caption, to
               // avoid messing up the "Next" button used to move to the next part of a multi-part PUR...
               updateButton = ((btnCaptionIsNextPlace === true) || (btnCaptionIsNextIssue));
            }
         }

         if(updateButton === true)
         {
            uroAddLog('inhibit Next UR/MP/PUR button');
            
            // alter the button caption
            nurButton.innerHTML = doneString;
            // 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 = false;
   var panelOpen = (document.getElementById('panel-container').firstChild !== null);
   if(panelOpen)
   {
      URDialogIsOpen = (document.getElementById('panel-container').getElementsByClassName('conversation').length > 0);
   }
   
   if(URDialogIsOpen)
   {
      var thisSelectedURID = document.getElementsByClassName('permalink')[0].href.split('&mapUpdateRequest=');
      if(thisSelectedURID.length > 1)
      {
         thisSelectedURID = thisSelectedURID[1].split('&')[0];
      }
      else
      {
         thisSelectedURID = null;
      }

      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('send-button')[0].addEventListener("click", uroAddedComment, false);
         
         uroAddLog('user is editing UR '+uroSelectedURID);
         uroExpectedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
         
         ////if(uroShowDebugOutput === true)
         {
            if((uroHoveredURID !== null) && (uroSelectedURID !== null) && (parseInt(uroHoveredURID) !== parseInt(uroSelectedURID)))
            {
               if(uroURReclickAttempts === 0)
               {
                  uroAddLog('DANGER, WILL ROBINSON!  You clicked on UR ID '+uroHoveredURID+' but WME has loaded the details for UR ID '+uroSelectedURID+' instead, attempting to fix...');
               }
               if(++uroURReclickAttempts < 3)
               {
                  //uroRestackMarkers();
                  W.map.updateRequestLayer.markers[uroHoveredURID].model.attributes.geometry.x = W.map.updateRequestLayer.markers[uroHoveredURID].model.attributes.geometry.realX;
                  W.map.updateRequestLayer.markers[uroHoveredURID].model.attributes.geometry.y = W.map.updateRequestLayer.markers[uroHoveredURID].model.attributes.geometry.realY;
                  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)...
         if(document.getElementsByClassName('full-closures').length > 0)
         {
            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 = WazeMap.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.setProperty('background',customColour,'important');
         }
         else
         {
            mapviewport.style.setProperty('background',"#354148",'important');
         }
      }

      // 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;
      }
      
      {
         if(uroGetCBChecked('_cbHideSegmentsWhenRoadsHidden'))
         {
            W.map.segmentLayer.drawn = W.map.roadLayers[0].visibility;
            W.map.nodeLayer.drawn = W.map.roadLayers[0].visibility;
         }
         else
         {
            W.map.segmentLayer.drawn = true;
            W.map.nodeLayer.drawn = true;
         }
      }
   }
}

function uroGetDirectionString(isForward)
{
   if(isForward === true)
   {
      return 'A-B';
   }
   else
   {
      return 'B-A';
   }
}


function uroGetVehicleDescription(vehicleType)
{
   var retval = null;
   var i18nLookup = null;
   if(vehicleType === 0) i18nLookup = "TRUCK";
   else if(vehicleType === 256) i18nLookup = "PUBLIC_TRANSPORTATION";
   else if(vehicleType === 272) i18nLookup = "TAXI";
   else if(vehicleType === 288) i18nLookup = "BUS";
   else if(vehicleType === 512) i18nLookup = "RV";
   else if(vehicleType === 768) i18nLookup = "TOWING_VEHICLE";
   else if(vehicleType === 1024) i18nLookup = "MOTORCYCLE";
   else if(vehicleType === 1280) i18nLookup = "PRIVATE";
   else if(vehicleType === 1536) i18nLookup = "HAZARDOUS_MATERIALS";
   else if(vehicleType === 1792) i18nLookup = "CAV";
   else if(vehicleType === 1808) i18nLookup = "EV";
   else if(vehicleType === 1824) i18nLookup = "HYBRID";
   else if(vehicleType === 1840) i18nLookup = "CLEAN_FUEL";   
   if(i18nLookup !== null)
   {
      retval = I18n.lookup("restrictions.vehicle_types."+i18nLookup);
   }
   return retval;
}
function uroFormatTBRDetails(tbrObj)
{
   var retval = '';
   if(tbrObj.description !== null)
   {
      retval += '&nbsp;&nbsp;Reason: ' + tbrObj.description + '<br>';
   }
   
   if(tbrObj.timeFrames.length > 0)
   {
      retval += '&nbsp;&nbsp;Dates: ';
      if(tbrObj.timeFrames[0].startDate === null)
      {
         retval += 'all dates';
      }
      else
      {
         retval += tbrObj.timeFrames[0].startDate + ' to ' + tbrObj.timeFrames[0].endDate;
      }
      retval += '<br>';
      
      retval += '&nbsp;&nbsp;Days: ';
      if(tbrObj.timeFrames[0].weekdays & (1<<0)) retval += 'S';
      else retval += '-';
      if(tbrObj.timeFrames[0].weekdays & (1<<1)) retval += 'M';
      else retval += '-';
      if(tbrObj.timeFrames[0].weekdays & (1<<2)) retval += 'T';
      else retval += '-';
      if(tbrObj.timeFrames[0].weekdays & (1<<3)) retval += 'W';
      else retval += '-';
      if(tbrObj.timeFrames[0].weekdays & (1<<4)) retval += 'T';
      else retval += '-';
      if(tbrObj.timeFrames[0].weekdays & (1<<5)) retval += 'F';
      else retval += '-';
      if(tbrObj.timeFrames[0].weekdays & (1<<6)) retval += 'S';
      else retval += '-';
      retval += '<br>';
      
      retval += '&nbsp;Timespan: ';
      if(tbrObj.timeFrames[0].fromTime === null)
      {
         retval += 'all day';
      }
      else
      {
         retval += tbrObj.timeFrames[0].fromTime + ' to ' + tbrObj.timeFrames[0].toTime;
      }
      retval += '<br>';
   }  

   var vtLength = 0;
   var i;
   if(tbrObj.driveProfiles.BLOCKED !== undefined)
   {
      vtLength = tbrObj.driveProfiles.BLOCKED[0].vehicleTypes.length;
      if(vtLength > 0)
      {
         retval += '&nbsp;Vehicle types prohibited:<br>';
         for(i=0; i<vtLength; i++)
         {
            retval += '&nbsp;&nbsp;'+uroGetVehicleDescription(tbrObj.driveProfiles.BLOCKED[0].vehicleTypes[i])+'<br>';
         }
      }
   }
   else if(tbrObj.driveProfiles.FREE !== undefined)
   {      
      vtLength = tbrObj.driveProfiles.FREE[0].vehicleTypes.length;
      if(vtLength > 0)
      {
         retval += '&nbsp;Vehicle types allowed:<br>';
         for(i=0; i<vtLength; i++)
         {
            retval += '&nbsp;&nbsp;'+uroGetVehicleDescription(tbrObj.driveProfiles.FREE[0].vehicleTypes[i])+'<br>';
         }
      }
   }
   else if(tbrObj.defaultType === "BLOCKED")
   {
      retval += '&nbsp;Blocked for all vehicle types<br>';
   }
   
   if(tbrObj.defaultType === "DIFFICULT")
   {
      retval += '&nbsp;Difficult Turn<br>';
   }
   
   return retval;
}
function uroFormatClosureDetails(cObjA, cObjB)
{
   var retval = '';
   if(cObjB === null)
   {
      retval += 'Reason: ' + cObjA.reason + '<br>';
      retval += 'From: ' + cObjA.startDate + '<br>';
      retval += 'To: ' + cObjA.endDate + '<br>';
      retval += 'Direction: ' + uroGetDirectionString(cObjA.forward) + '<br>';
      retval += 'Ignore traffic: ' + cObjA.permanent;
   }
   else
   {
      if(cObjA.reason !== cObjB.reason)
      {
         retval += 'Reason: ' + cObjA.reason;
         retval += ' <i>\>\>\> ' + cObjB.reason + '</i><br>';
      }
      if(cObjA.startDate !== cObjB.startDate)
      {
         retval += 'From: ' + cObjA.startDate;
         retval += ' <i>\>\>\> ' + cObjB.startDate + '</i><br>';
      }
      if(cObjA.endDate !== cObjB.endDate)
      {
         retval += 'To: ' + cObjA.endDate;
         retval += ' <i>\>\>\> ' + cObjB.endDate + '<i><br>';
      }
      if(cObjA.forward !== cObjB.forward)
      {
         retval += 'Direction: ' + uroGetDirectionString(cObjA.forward);
         retval += ' <i>\>\>\> ' + uroGetDirectionString(cObjB.forward) + '</i><br>';
      }
      if(cObjA.permanent !== cObjB.permanent)
      {
         retval += 'Ignore traffic: ' + cObjA.permanent;
         retval += ' <i>\>\>\> ' + cObjB.permanent + '</i><br>';
      }
   }
   return retval;
}
function uroGetTIOString(tioValue)
{
   var retval = I18n.lookup("instruction_override.no_opcode");
   if(tioValue !== null)
   {
      retval = I18n.lookup("instruction_override.opcodes." + tioValue);
   }
   return retval;
}
function uroSegmentHistoryNameString(segID)
{
   var retval = '';
   if(W.model.segments.objects[segID] !== undefined)
   {
      if(W.model.segments.objects[segID].attributes.primaryStreetID !== undefined)
      {
         var sID = W.model.segments.objects[segID].attributes.primaryStreetID;
         if(W.model.streets.objects[sID].name === null)
         {
            retval += 'unnamed segment';
         }
         else
         {
            retval += W.model.streets.objects[sID].name; 
         }
      }
      else
      {
         retval += 'unnamed segment';
      }
   }
   else
   {
      retval += 'unknown segment';
   }
   retval += ' (ID ' + segID + ')';
   return retval;
}   
function uroTSTEnhanceSegmentHistory(nextTransactionID)
{
   if(W.selectionManager.selectedItems.length === 1) 
   {
      if(W.selectionManager.selectedItems[0].model.type === "segment")
      {
         if((W.selectionManager.selectedItems[0].model.attributes.id !== uroEnhanceHistorySegID) || (nextTransactionID !== null))
         {
            uroEnhanceHistorySegID = W.selectionManager.selectedItems[0].model.attributes.id;
            var historyURL = 'https://' + document.location.host;
            historyURL += Waze.Config.api_base;
            historyURL += '/ElementHistory?objectID=' + uroEnhanceHistorySegID + '&objectType=segment';
            if(nextTransactionID !== null)
            {
               historyURL += '&till=' + nextTransactionID;
            }
            uroAddLog('requesting edit history for segment '+uroEnhanceHistorySegID);
            if(nextTransactionID === null)
            {
               uroSegHistoryDetails = null;
               uroSegHistoryEntries = 0;
            }
   
            var historyReq = new XMLHttpRequest();
            historyReq.onreadystatechange = function()
            {
               if(historyReq.readyState == 4)
               {
                  uroAddLog('segment history data request, response '+historyReq.status+' received');
                  if(historyReq.status == 200)
                  {
                     if(historyReq.responseText !== "")
                     {
                        var nextTransaction = null;
                        if(uroSegHistoryDetails === null)
                        {
                           uroSegHistoryDetails = JSON.parse(historyReq.responseText);
                           nextTransaction = uroSegHistoryDetails.transactions.nextTransaction;
                        }
                        else
                        {
                           var nextDetails = JSON.parse(historyReq.responseText);
                           uroSegHistoryDetails.users.objects = uroSegHistoryDetails.users.objects.concat(nextDetails.users.objects);
                           uroSegHistoryDetails.transactions.objects = uroSegHistoryDetails.transactions.objects.concat(nextDetails.transactions.objects);
                           nextTransaction = nextDetails.transactions.nextTransaction;
                        }
                        
                        if(nextTransaction !== null)
                        {
                           uroAddLog('segment has more history details available, getting those now...');
                           uroTSTEnhanceSegmentHistory(nextTransaction);
                        }
                        else
                        {
                           uroAddLog('segment history request complete');
                           uroSegHistoryLoaded = true;
                        }
                     }
                     else
                     {
                     }            
                  }
               }
            };
            historyReq.open('GET',historyURL,true);
            historyReq.send(); 
            uroSegHistoryLoaded = false;
         }
         
         if((document.getElementsByClassName('historyContent').length === 1) && (uroSegHistoryLoaded === true))
         {
            if(document.getElementsByClassName('historyContent')[0].style.display === "")
            {
               var historyLength = document.getElementsByClassName('tx-header').length;
               if((historyLength !== uroSegHistoryEntries) && (uroSegHistoryDetails.transactions.objects.length >= historyLength))
               {
                  if(uroSegHistoryEntries === 0)
                  {
                     uroAddLog('segment history list is open, do something fun...');
                  }
                  else
                  {
                     uroAddLog('more history loaded for segment...');
                  }
                  
                  var tHTML;
                  for(var i=0; i<historyLength; i++)
                  {                
                     var historyEntry = document.getElementsByClassName('element-history-item')[i];
                     var elmOffset = 0;
                     for(var j=0; j<uroSegHistoryDetails.transactions.objects[i].objects.length; j++)
                     {                      
                        tHTML = '';
                        var tObj = uroSegHistoryDetails.transactions.objects[i].objects[j];
                        var aType = tObj.actionType;
                        var oType = tObj.objectType;
                        if(oType === "nodeConnection")
                        {
                           var outboundTR = (tObj.objectID.fromSegID === uroEnhanceHistorySegID);
                           if(outboundTR === true)
                           {
                              tHTML += '<b>Outbound turn:</b> ';
                           }
                           else
                           {
                              tHTML += '<b>Inbound turn:</b> ';
                           }
                           if(aType == "DELETE")
                           {
                              tHTML += 'disabled from ';
                           }
                           else if(aType == "ADD")
                           {
                              tHTML += 'enabled from ';
                           }
                           if(outboundTR === true)
                           {
                              tHTML += 'node ';
                              if(tObj.objectID.fromSegFwd === true)
                              {
                                 tHTML += 'B';
                              }
                              else
                              {
                                 tHTML += 'A';
                              }
                              tHTML += ' to ';
                              tHTML += uroSegmentHistoryNameString(tObj.objectID.toSegID);
                           }
                           else
                           {
                              tHTML += uroSegmentHistoryNameString(tObj.objectID.fromSegID);
                              tHTML += ' to node ';
                              if(tObj.objectID.toSegFwd === true)
                              {
                                 tHTML += 'A';
                              }
                              else
                              {
                                 tHTML += 'B';
                              }
                           }
                           if(aType === "UPDATE")
                           {
                              if((tObj.oldValue !== undefined) && (tObj.newValue !== undefined))
                              {
                                 if(tObj.oldValue.instructionOpCode !== tObj.newValue.instructionOpCode)
                                 {
                                    tHTML += '<br><i>Instruction Override changed from '+uroGetTIOString(tObj.oldValue.instructionOpCode)+' to '+uroGetTIOString(tObj.newValue.instructionOpCode)+'</i>';
                                 }
                              }
                           }
                           else if(aType === "ADD")
                           {
                              if(tObj.newValue !== undefined)
                              {
                                 if(tObj.newValue.instructionOpCode !== null)
                                 {
                                    tHTML += '<br><i>Instruction Override set to ' + uroGetTIOString(tObj.newValue.instructionOpCode)+'</i>';
                                 }
                              }
                           }
                           else if(aType === "DELETE")
                           {
                              if(tObj.oldValue !== undefined)
                              {
                                 if(tObj.oldValue.instructionOpCode !== null)
                                 {
                                 }
                              }
                           }
                           if(aType === "UPDATE")
                           {
                              var nv;
                              var ov;
                              var hasChanged;
                              if((tObj.oldValue.restrictions !== null) && (tObj.oldValue.restrictions !== undefined))
                              {
                                 for(ov=0; ov<tObj.oldValue.restrictions.length; ov++)
                                 {
                                    hasChanged = true;
                                    if((tObj.newValue.restrictions !== null) && (tObj.newValue.restrictions.length > 0))
                                    {
                                       for(nv=0; nv<tObj.newValue.restrictions.length; nv++)
                                       {
                                          if(JSON.stringify(tObj.oldValue.restrictions[ov]) === JSON.stringify(tObj.newValue.restrictions[nv]))
                                          {
                                             hasChanged = false;
                                             break;
                                          }
                                       }
                                    }
                                    if(hasChanged === true)
                                    {
                                       tHTML += '<br><i>TBR deleted/changed:<br>';
                                       tHTML += uroFormatTBRDetails(tObj.oldValue.restrictions[ov]);
                                       tHTML += '</i>';
                                    }
                                 }
                              }
                              if((tObj.newValue.restrictions !== null) &&(tObj.newValue.restrictions !== undefined))
                              {
                                 for(nv=0; nv<tObj.newValue.restrictions.length; nv++)
                                 {
                                    hasChanged = true;
                                    if((tObj.oldValue.restrictions !== null) && (tObj.oldValue.restrictions.length > 0))
                                    {
                                       for(ov=0; ov<tObj.oldValue.restrictions.length; ov++)
                                       {
                                          if(JSON.stringify(tObj.oldValue.restrictions[ov]) === JSON.stringify(tObj.newValue.restrictions[nv]))
                                          {
                                             hasChanged = false;
                                             break;
                                          }
                                       }
                                    }
                                    if(hasChanged === true)
                                    {
                                       tHTML += '<br><i>TBR added/changed:<br>';
                                       tHTML += uroFormatTBRDetails(tObj.newValue.restrictions[nv]);
                                       tHTML += '</i>';
                                    }
                                 }
                              }
                           }
                           else if(aType === "ADD")
                           {
                              if((tObj.newValue.restrictions !== null) && (tObj.newValue.restrictions.length > 0))
                              {
                                 for(var l=0; l<tObj.newValue.restrictions.length; l++)
                                 {
                                    tHTML += '<br><i>TBR set to:<br>';
                                    tHTML += uroFormatTBRDetails(tObj.newValue.restrictions[l]);
                                    tHTML += '</i>';
                                 }
                              }
                           }
                        }
                        else if(oType === "roadClosure")
                        {
                           tHTML += '<b>Road closure:</b> ';
                           var cObjA = null;
                           var cObjB = null;
                           if(aType === "ADD")
                           {
                              tHTML += 'added<br>';
                              cObjA = tObj.newValue;
                           }
                           else if(aType === "DELETE")
                           {
                              tHTML += 'deleted<br>';
                              cObjA = tObj.oldValue;
                           }
                           else if(aType === "UPDATE")
                           {
                              tHTML += 'edited<br>';
                              cObjA = tObj.oldValue;
                              cObjB = tObj.newValue;
                           }
                           tHTML += 'ID: ' + tObj.objectID + '<br>';
                           tHTML += uroFormatClosureDetails(cObjA, cObjB);
                        }
                        
                        if(tHTML !== '')
                        {
                           historyEntry.getElementsByClassName('tx-changed-ro')[elmOffset++].innerHTML = tHTML + '<br>';
                        }
                     }
                  }
                  uroSegHistoryEntries = historyLength;
               }
            }
         }
      }
      else
      {
         uroEnhanceHistorySegID = null;
         uroSegHistoryDetails = null;
         uroSegHistoryEntries = 0;
      }
   }
   else
   {
      uroEnhanceHistorySegID = null;
      uroSegHistoryDetails = null;
      uroSegHistoryEntries = 0;
   }
}
function uroTSTHidePopup()
{
   var hideUI = false;   
   if(document.getElementById('layer-switcher-pinned-input').checked === false)
   {
      hideUI = Boolean(hideUI || (document.getElementsByClassName('menu not-visible').length === 0));
   }
   hideUI = Boolean(hideUI || (document.getElementsByClassName('toolbar-group open').length > 0));

   if(hideUI === true)
   {
      uroHidePopup('uroTSTHidePopup');
   }
}

function uroTSTFixVenueHistoryStyling()
{
   while(document.getElementsByClassName("ca-name ca-preview").length > 0)
   {
      document.getElementsByClassName("ca-name ca-preview")[0].className = "ca-description ca-preview";
   }
}

function uroTenthSecondTick()
{
   if(uroMTEMode) return;
   if(uroSetupListeners)
   {
      if(uroFinalisingListenerSetup === false)
      {
         if(W.loginManager.isLoggedIn())
         {
            uroFinalizeListenerSetup();
            
            if(uroGetCBChecked('_cbMoveAMList') === false)
            {
               uroControls.appendChild(uroAMList);
            }
            else
            {
               document.getElementsByClassName('area-managers-region')[0].innerHTML = '<div id="uroAMList" style="opacity:1;color:#c0c0ff;"></div>';
               document.getElementsByClassName('topbar')[0].style.backgroundColor="#000000";
            }            
         }
      }
   }
   else
   {
      uroTSTPopupHandler();
      uroTSTDTEHandler();
      uroTSTNextBtnHandler();
      uroTSTCommentAddedHandler();
      uroTSTOWLHandler();
      uroTSTClosureCloningHandler();
      uroTSTFeedFilter();
      uroMiscUITweaksHandler();
      uroTSTEnhanceSegmentHistory(null);
      uroTSTFixVenueHistoryStyling();
      uroTempFixMTEDropDown();
      uroTSTHidePopup();
   }
}

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("_tabSelectMapComments");
   uroInactiveTab("_tabSelectMisc");
   uroInactiveTab("_tabSelectUserRequests");
   uroInactiveTab("_tabSelectCWL");
   uroInactiveTab("_tabSelectPlaces");

   if(!uroCtrlsHidden)
   {
      uroSetStyleDisplay('uroCtrlURs','none');
      uroSetStyleDisplay('uroCtrlMPs','none');
      uroSetStyleDisplay('uroCtrlMCs','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 uroShowMCTab()
{
   uroInactiveAllTabs();
   uroActiveTab("_tabSelectMapComments");
   uroCurrentTab = 3;
   if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMCs','block');
   return false;
}

function uroShowPlacesTab()
{
   uroInactiveAllTabs();
   uroActiveTab("_tabSelectPlaces");
   uroCurrentTab = 4;
   if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlPlaces','block');
   for(var idx=0;idx<uroPlacesGroupsCollapsed.length;idx++)
   {
      uroPlacesGroupCEHandler(idx);
   }
   return false;
}

function uroShowCameraTab()
{
   uroInactiveAllTabs();
   uroActiveTab("_tabSelectCameras");
   uroCurrentTab = 5;
   if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlCameras','block');
   return false;
}

function uroShowOWLTab()
{
   uroInactiveAllTabs();
   uroActiveTab("_tabSelectCWL");
   uroCurrentTab = 6;
   if(!uroCtrlsHidden) uroSetStyleDisplay('uroOWL','block');
   uroOWLUpdateHTML();
   return false;
}

function uroShowMiscTab()
{
   uroInactiveAllTabs();
   uroActiveTab("_tabSelectMisc");
   uroCurrentTab = 7;
   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);
      }
      
      else
      {
         showmarkerpos = thisurl.indexOf('&showppur=');
         if((endmarkerpos != -1) && (showmarkerpos != -1))
         {
            showmarkerpos += 10;
            uroAddLog('showPPUR tab opened');
            urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
            uroAddLog(' PPUR ID = '+urID);

            try
            {
               W.map.parkingPlaceUpdatesLayer.markers[urID].icon.imageDiv.click();
               doRetry = false;
            }
            catch(err)
            {
               uroAddLog('PPURs not fully loaded, retrying...');
            }

            if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
         }         
      }
   }

}

function uroUpdateVenueEditorList()
{
   if(Object.keys(W.model.venues.objects).length === 0) return;

   var selector = document.getElementById('_selectPlacesUserID');
   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 = [];
   var idx;
   for(idx in W.model.venues.objects)
   {
      if(W.model.venues.objects.hasOwnProperty(idx))
      {
         var obj = W.model.venues.objects[idx].attributes;
         var cbID = obj.createdBy;
         var ubID = obj.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(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 uroPlacesEditorSelected()
{
   var selector = document.getElementById('_selectPlacesUserID');
   if(selector.selectedIndex > 0)
   {
      document.getElementById('_textPlacesEditor').value = document.getElementById('_selectPlacesUserID').selectedOptions[0].innerHTML;
   }
}

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 uroPopulateProblemsTab()
{
   var tHTML = '';
   tHTML += '<input type="checkbox" id="_cbMPFilterOutsideArea">Hide MPs outside my editable area</input><br><br>';
   tHTML += '<b>Filter MPs by type:</b><br>';
   var i;
   for(i=0; i<uroKnownProblemTypeNames.length; i++)
   {
      tHTML += '<input type="checkbox" id="_cbMPFilter_T'+uroKnownProblemTypeIDs[i]+'">'+uroKnownProblemTypeNames[i]+'</input><br>';
   }      
   tHTML += '<br><input type="checkbox" id="_cbMPFilterUnknownProblem">Unknown problem type</input><br><br>';

   tHTML += '&nbsp;&nbsp;<i>Specially tagged types</i><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterElgin">[Elgin]</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterTrafficCast">[TrafficCast]</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterTrafficMaster">[TM]</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterCaltrans">[Caltrans]</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterTFL">TfL</input><br>';
   
   tHTML += '<input type="checkbox" id="_cbMPFilterReopenedProblem">Reopened Problems</input><br><br>';

   tHTML += '<input type="checkbox" id="_cbInvertMPFilter">Invert operation of type filters?</input><br>';

   

   tHTML += '<br><b>Hide closed/solved/unidentified Problems:</b><br>';
   tHTML += '<input type="checkbox" id="_cbMPFilterClosed">Closed</input><br>';
   tHTML += '<input type="checkbox" id="_cbMPFilterSolved">Solved</input><br>';
   tHTML += '<input type="checkbox" id="_cbMPFilterUnidentified">Not identified</input><br><br>';

   tHTML += '<input type="checkbox" id="_cbMPClosedUserIDFilter" pairedWith="_cbMPNotClosedUserIDFilter">Closed</input> or ';
   tHTML += '<input type="checkbox" id="_cbMPNotClosedUserIDFilter" pairedWith="_cbMPClosedUserIDFilter">Not Closed</input> by user';
   tHTML += '<select id="_selectMPUserID" style="width:80%; height:22px;"></select><br>';

   tHTML += '<br><b>Hide problems (not turn) by severity:</b><br>';
   tHTML += '<input type="checkbox" id="_cbMPFilterLowSeverity">Low</input>&nbsp;&nbsp;';
   tHTML += '<input type="checkbox" id="_cbMPFilterMediumSeverity">Medium</input>&nbsp;&nbsp;';
   tHTML += '<input type="checkbox" id="_cbMPFilterHighSeverity">High</input><br>';
   
   uroCtrlMPs.innerHTML = tHTML;
}
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>&nbsp;&nbsp;';
   tHTML += '<input type="checkbox" id="_cbPURFilterMediumSeverity">Medium</input>&nbsp;&nbsp;';
   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 += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL0">1</input>';
   tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL1">2</input>';
   tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL2">3</input>';
   tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL3">4</input>';
   tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL4">5</input>';
   tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL5">6</input>'; 
   tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesStaff">Staff</input>';
   tHTML += '<br>&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesAdLocked">AdLocked</input><br>';
   
   tHTML += '<br>Hide by geometry:<br>';
   tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideAreaPlaces">Areas</input>';
   tHTML += '&nbsp;&nbsp;&nbsp;<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="_cbHideDescribedPlaces" pairedWith="_cbHideNonDescribedPlaces">Hide or </input>';
   tHTML += '<input type="checkbox" id="_cbHideNonDescribedPlaces" pairedWith="_cbHideDescribedPlaces">show ones with descriptive text</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><b>Show Places touched by a specific editor:</b><br>';
   tHTML += '<input type="checkbox" id="_cbShowOnlyPlacesCreatedBy">Created by</input>&nbsp;/&nbsp;';
   tHTML += '<input type="checkbox" id="_cbShowOnlyPlacesEditedBy">edited by</input><br>'; 
   tHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textPlacesEditor"><br>';   
   tHTML += '<select id="_selectPlacesUserID" style="width:80%; height:22px;"></select><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 += '&nbsp;&nbsp;<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 += '&nbsp;&nbsp;<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;
      
   uroTurnsLayerIdx = null;
   uroMCLayerIdx = null;
   for(i=0;i<W.map.layers.length;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 !== undefined)
      {
         if(W.map.layers[i].name.indexOf('marker_drawing_context') != -1) uroTurnsLayerIdx = i;
      }
	   if(W.map.layers[i].featureType == 'mapComment') uroMCLayerIdx = 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('Turns layer at idx '+uroTurnsLayerIdx);
   uroAddLog('MC layer at idx '+uroMCLayerIdx);
   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.prefs.on('change:isImperial', uroInitialise);
      W.app.modeController.model.bind("change:mode", uroInitialise);
      if(W.app.modeController.mode.mteModeState !== undefined)
      {
         uroMTEMode = true;
         uroHidePopup('uroRealWazeBits');
         uroAddLog('MTE mode, sleeping until normal service is resumed...');
         return;
      }
      uroMTEMode = false;

      uroSetupUI();
      uroDOMHasTurnProblems = (W.model.turnProblems != null);
      uroGetProblemTypes();
      uroPopulateProblemsTab();
      uroPopulatePlacesTab();

      uroControls.appendChild(uroCtrlURs);
      uroControls.appendChild(uroCtrlMPs);
      uroControls.appendChild(uroCtrlMCs);
      uroControls.appendChild(uroCtrlPlaces);
      uroControls.appendChild(uroCtrlCameras);
      uroControls.appendChild(uroOWL);
      uroControls.appendChild(uroCtrlMisc);
      uroControls.appendChild(uroCtrlHides);

      uroCtrlURs.onclick = uroFilterItems_URTabClick;
      uroCtrlMPs.onclick = uroFilterItems_MPTabClick;
      uroCtrlMCs.onclick = uroFilterItems_MCTabClick;
      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;
   uroSetupListeners = true;
   uroFinalisingListenerSetup = 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 = 10000;
   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 = 10000;
   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; overflow:auto;max-height:500px">Alert content goes here...</div>';
   alertsHTML += '<div id="controls" align="center" style="padding: 4px;">';   
   alertsHTML += '<span id="uroAlertTickBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px 10px 2px 10px;">';
   alertsHTML += '<i class="fa fa-check"> </i>';
   alertsHTML += '<span id="uroAlertTickBtnCaption" style="font-weight: bold;"></span>';
   alertsHTML += '</span>';
   alertsHTML += '&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;';
   alertsHTML += '<span id="uroAlertCrossBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px 10px 2px 10px;">';
   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://greasyfork.org/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 += '&nbsp;<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="_tabSelectMapComments"><a href="#" id="_linkSelectMapComments" style="text-decoration:none;font-size:12px">MCs</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;

   // tab elements
   uroCtrlURs = document.createElement('p');
   uroCtrlMPs = document.createElement('p');
   uroCtrlMCs = document.createElement('p');
   uroCtrlPlaces = document.createElement('p');
   uroCtrlCameras = document.createElement('p');
   uroOWL = document.createElement('p');
   uroCtrlMisc = document.createElement('p');
   
   // other sidebar elements
   uroAMList = document.createElement('div');
   uroCtrlHides = document.createElement('div');
   

   // 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 += '&nbsp;&nbsp;<i>Specially tagged types</i><br>';
      uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterRoadworks">[ROADWORKS]</input><br>';
      uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterConstruction">[CONSTRUCTION]</input><br>';
      uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterClosure">[CLOSURE]</input><br>';
      uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterEvent">[EVENT]</input><br>';
      uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterNote">[NOTE]</input><br>';
      uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterBOG">[BOG]</input><br>';
      uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterDifficult">[DIFFICULT]</input><br>';
      uroCtrlURs.innerHTML += '&nbsp;&nbsp;<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 += '&nbsp;&nbsp;<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 += '&nbsp;&nbsp;<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>';
   }

   // MC controls tab
   {
      uroCtrlMCs.id = "uroCtrlMCs";
      uroCtrlMCs.innerHTML = '<br>';

      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<i>Specially tagged types</i><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCFilterRoadworks">[ROADWORKS]</input><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCFilterConstruction">[CONSTRUCTION]</input><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCFilterClosure">[CLOSURE]</input><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCFilterEvent">[EVENT]</input><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCFilterNote">[NOTE]</input><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCFilterBOG">[BOG]</input><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCFilterDifficult">[DIFFICULT]</input><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCFilterWSLM">[WSLM]</input><br><br>';
      uroCtrlMCs.innerHTML += '<input type="checkbox" id="_cbInvertMCFilter">Invert operation of type filters?</input><br>';

      uroCtrlMCs.innerHTML += '<hr>';

      uroCtrlMCs.innerHTML += '<br><b>Filter by description/comments/following:</b><br>';
      uroCtrlMCs.innerHTML += '<input type="checkbox" id="_cbMCHideMyFollowed" pairedWith="_cbMCHideMyUnfollowed">Ones I am or </input>';
      uroCtrlMCs.innerHTML += '<input type="checkbox" id="_cbMCHideMyUnfollowed" pairedWith="_cbMCHideMyFollowed">am not following</input><br><br>';

      uroCtrlMCs.innerHTML += '<input type="checkbox" id="_cbMCDescriptionMustBePresent" pairedWith="_cbMCDescriptionMustBeAbsent">Hide</input> or ';
      uroCtrlMCs.innerHTML += '<input type="checkbox" id="_cbMCDescriptionMustBeAbsent" pairedWith="_cbMCDescriptionMustBePresent">show</input> MCs with no description<br>';
      uroCtrlMCs.innerHTML += '<input type="checkbox" id="_cbMCEnableKeywordMustBePresent">Hide MCs not including </input>';
      uroCtrlMCs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textMCKeywordPresent"><br>';
      uroCtrlMCs.innerHTML += '<input type="checkbox" id="_cbMCEnableKeywordMustBeAbsent">Hide MCs including </input>';
      uroCtrlMCs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textMCKeywordAbsent"><br>';
      uroCtrlMCs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCCaseInsensitive"><i>Case-insensitive matches?</i></input><br><br>';
   }
   
   // Map problems controls tab
   {
      uroCtrlMPs.id = "uroCtrlMPs";
      uroCtrlMPs.innerHTML = 'MP filter list being populated, please wait...';
   }

   // 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>&nbsp;/&nbsp;';
      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 += '&nbsp;&nbsp;&nbsp;<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 += '&nbsp;&nbsp;&nbsp;<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 += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowIfSpeedSet" checked> with speed data</input><br>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<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 += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowRLCIfZeroSpeedSet" checked> with speed limit = 0</input><br>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowRLCIfNonZeroSpeedSet" checked> with speed limit > 0</input><br>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowRLCIfNoSpeedSet" checked> with no speed data</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 += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByMe">me</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank0">L1</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank1">L2</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank2">L3</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank3">L4</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank4">L5</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank5">L6</input>';

      uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by updater:</b><br>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByMe">me</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank0">L1</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank1">L2</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank2">L3</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank3">L4</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank4">L5</input>';
      uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<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>Hide Road Closures:</b><br>';
      uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbHideUserRTCs">added from the app</input><br>';
      uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbHideEditorRTCs">added from WME</input><br>';
      
      uroCtrlMisc.innerHTML += '<br><b><input type="checkbox" id="_cbHideSegmentsWhenRoadsHidden" />Hide segment layer when road layer is hidden</b><br>';
      
      uroCtrlMisc.innerHTML += '<br><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="_cbEnableDeleteFeedEntries" />Enable delete feed entries button</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 += 'Auto-hide after <input type="number" min="0" max="10" value="0" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupAutoHideTimeout" /> seconds<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 += '&nbsp;&nbsp;<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 += '<input type="checkbox" id="_cbInhibitMapCommentPopup" />Map Comments<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="_cbMoveAMList" />Replace native topbar AM list</b><br>';
      uroCtrlMisc.innerHTML += '<br><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" />&nbsp;&nbsp;&nbsp;';
      uroCtrlMisc.innerHTML += '<input type="button" id="_btnTextToSettings" value="Restore" />&nbsp;&nbsp;|&nbsp;&nbsp;';
      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" />&nbsp;&nbsp;&nbsp;';
   uroCtrlHides.innerHTML += '<input type="button" id="_btnClearSessionHides" value="Undo all hides" /><p>';

   // footer for AM list
   uroAMList.id = 'uroAMList';
   
   window.addEventListener("beforeunload", uroSaveSettings, false);
}

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();