UROverview Plus (URO+)

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

当前为 2024-02-20 提交的版本,查看 最新版本

// ==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*
// @exclude             https://editor-beta.waze.com/*
// @exclude             https://beta.waze.com/*
// @exclude             https://www.waze.com/user/*editor/*
// @exclude             https://www.waze.com/*/user/*editor/*
// @grant               none
// @version             3.231
// ==/UserScript==

/*
=======================================================================================================================
Bug fixes - MUST BE CLEARED BEFORE RELEASE
=======================================================================================================================


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

*/

/* JSHint Directives */
/* globals $: */
/* globals W: true */
/* globals I18n: */
/* globals OpenLayers: true */
/* globals require: */
/* globals _: */
/* globals trustedTypes: */
/* jshint bitwise: false */
/* jshint eqnull: true */
/* jshint esversion: 11 */


const uroVersion = "3.231";
const uroReleaseDate = "20240220";

// list of changes affecting all users
const uroChanges =
[
   "RTC filtering now correctly identifies active closures if they started earlier in the same day",
   "RTC filtering now hides associated junction node closure markers",
   "Enables WME to display the arrow for a lower priority closure in one direction if a higher priority closure is set in the other direction",
   "Segment popup now shows correct last edited by/on details for segments which have never been altered since their creation"
];
// list of changes affecting only WME Beta users (at least until the next production release including these parts of the beta code...)
const uroBetaChanges =
[
];

// true enables debug output during script startup
let uroShowDebugOutput = true;
// true keeps debug output enabled after script startup
const uroPersistentDebugOutput = false;
// true enables performance monitoring debug output
const uroPerformanceMonitoringOutput = false;

let uroRecentDebug = [];
let uroURDupes = [];
let uroCtrlsHidden = false;
let uroFID = -1;
let uroShownFID = -1;
let uroShownPopupType = null;
let uroInhibitSave = true;
let uroPopupTimer = -2;
let uroPopupDwellTimer = -1;
let uroPopupAutoHideTimer = 0;
let uroPopupShown = false;
let uroPopupSuppressed = false;
let uroSetupListeners = true;
let uroMouseInPopup = false;
let uroConfirmIntercepted = false;

let uroRootContainer = null;
let uroPlacesRoot = null;
let uroMCLayer = null;
let uroVenueLayer = null;

let uroMO_PlaceLayer = new MutationObserver(uroPlaceLayerChanged);
let uroMO_RPULayer = new MutationObserver(uroRPULayerChanged);
let uroMO_PPULayer = new MutationObserver(uroPPULayerChanged);
let uroMO_PURLayer = new MutationObserver(uroPURLayerChanged);
let uroMO_MPLayer = new MutationObserver(uroMPLayerChanged);
let uroMO_URLayer = new MutationObserver(uroURLayerChanged);
let uroMO_ClosuresLayer = new MutationObserver(uroClosuresLayerChanged);
let uroMO_NodeLayer = new MutationObserver(uroNodeLayerChanged);
let uroMO_SidePanel = new MutationObserver(uroSidePanelChanged);

const URO_UFP_OPTS = 
{
   filterUneditable: 0,
   filterInsideManagedAreas: 1,
   excludeMyAreas: 2,
   filterLockRanked: 3,
   filterFlagged: 4,
   filterNewPlace: 5,
   filterUpdatedDetails: 6,
   filterNewPhoto: 7,
   filterMinPURAge: 8,
   filterMaxPURAge: 9,
   invertPURFilters: 10,
   filterHighSeverity: 11,
   filterMedSeverity: 12,
   filterLowSeverity: 13,
   leavePURGeos: 14,
   filterCFPhone: 15,
   filterCFName: 16,
   filterCFEntryExitPoints: 17,
   filterCFOpeningHours: 18,
   filterCFAliases: 19,
   filterCFServices: 20,
   filterCFGeometry: 21,
   filterCFHouseNumber: 22,
   filterCFCategories: 23,
   filterCFDescription: 24,
   filterOnCFs: 25,
   thresholdMinPURDays: 26,
   thresholdMaxPURDays: 27,
   isLoggedIn: 28,
   userRank: 29,
   N_OPTS: 30
};
let uro_uFP = [];

const URO_TMARKER =
{
   UR: 0,
   MP: 1,
   PUR: 2,
   PPUR: 3,
   RPUR: 4,
   RTC: 5
};
let uroMarkerLayers = [];

const URO_TRTC =
{
   UNKNOWN: 0,
   WME: 1,
   WAZEFEED: 2,
   WAZEOTHER: 3
};

const URO_SRTC = 
{
   UNKNOWN: 0,
   EXPIRED: 1,
   ACTIVE: 2,
   FUTURE: 3
};

let uroCustomMarkerList = [];
let uroPendingURSessionIDs = [];
let uroRequestedURSessionIDs = [];
let uroPlacesGroupsCollapsed = [];
let uroKnownProblemTypeIDs = [];
let uroKnownProblemTypeNames = [];
let uroSelectedItems = [];
let uroPURsToHide = [];

let uroNullOpenLayers = false;
let uroNullURLayer = false;
let uroNullProblemLayer = false;
let uroNullMapViewport = false;

let uroURDialogIsOpen = false;
let uroHoveredURID = null;
let uroSelectedURID = null;
let uroURReclickAttempts = 0;
let uroPendingCommentDataRefresh = false;
let uroWaitingCommentDataRefresh = false;
let uroExpectedCommentCount = null;
let uroCachedLastCommentID = null;

let uroMCSelected = false;
let uroMouseIsDown = false;
let uroBackfilling = false;
let uroPopulatingRequestSessions = false;
let uroHidePopupOnPanelOpen = false;
let uroPointerWithinMap = false;

let uroUserID = -1;

let uroDOMHasTurnProblems = false;
let uroBetaEditor = false;
let uroMTEMode = false;
let uroFinalisingListenerSetup = false;
let uroInitialised = false;

let uroDiv = null;
let uroAlerts = null;
let uroControls = null;
let uroCtrlHides = null;
let uroAMList = [];
let uroManagedAreas = [];
let uroIgnoreAreasUserID = null;

let uroCWLGroups = [];
let uroCamWatchObjects = [];

let uroFriendlyAreaNames = [];
let uroAreaNameHoverTime = -1;
let uroAreaNameHoverObj = null;
let uroAreaNameOverlayShown = false;
let uroANEditHovered = false;
let uroANEditBox = null;

let uroPrevMouseX = -1;
let uroPrevMouseY = -1;
let uroMousedOverMapComment = null;
let uroMousedOverOtherObjectWithinMapComment = false;
let uroLastZoom = -1;

let uroTBRObj = null;
let uroRTCObjs = null;

let uroBackfillQueue = [];

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

let uroMousedOverMarkerID = null;
let uroMousedOverMarkerType = null;
let uroClickedOnMarkerID = null;
let uroClickedOnMarkerType = null;
let uroClickedOnMarkerForCenterInterceptor = null;
let uroClickedOnMarkerMapCenter = null;

let uroNodeLayerScanAttempts = 0;

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

let uroMainTickHandlerID = null;
let uroMainTickStage = 0;

let uroSettingsApplied = false;

let uroInhibitURFiltering = false;

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

const 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
   [
      '',
      '',
      '',
      ''
   ]
];


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

const uroHighlightedCameraImages =
[
   // speed
   [""],
   // dummy
   [""],
   // rlc
   [""]
];

const URO_MAX_TABS_PER_ROW = 6;
const URO_TABS_ID =
{
   URS: 0,
   MPS: 1,
   MCS: 2,
   RTCS: 3,
   RAS: 4,
   PLACES: 5,
   CAMS: 6,
   OWL: 7,
   MISC: 8
};
const URO_TABS_FIELD =
{
   TABHEADER: 0,
   TABBODY: 1,
   LINKID: 2,
   TABTITLE: 3,
   SHOWFN: 4,
   CLICKFN: 5,
   STORAGE: 6,
   POPULATEFN: 7
};
let uroCtrlTabs =
[
   ['_tabURs',    null, '_linkURs',    'URs',      uroShowURsTab,    uroFilterItems_URsTabClick,      'UROverviewUROptions',     uroPopulateURTab],
   ['_tabMPs',    null, '_linkMPs',    'MPs',      uroShowMPsTab,    uroFilterItems_MPsTabClick,      'UROverviewMPOptions',     uroPopulateMPTab],
   ['_tabMCs',    null, '_linkMCs',    'MCs',      uroShowMCsTab,    uroFilterItems_MCsTabClick,      'UROverviewMCOptions',     uroPopulateMCTab],
   ['_tabRTCs',   null, '_linkRTCs',   'RTCs',     uroShowRTCsTab,   uroFilterItems_RTCsTabClick,     'UROverviewRTCOptions',    uroPopulateRTCTab],
   ['_tabRAs',    null, '_linkRAs',    'RAs',      uroShowRAsTab,    uroFilterItems_RAsTabClick,      'UROverviewRAOptions',     uroPopulateRATab],
   ['_tabPlaces', null, '_linkPlaces', 'Places',   uroShowPlacesTab, uroFilterItems_PlacesTabClick,   'UROverviewPlacesOptions', uroPopulatePlacesTab],
   ['_tabCams',   null, '_linkCams',   'Cams',     uroShowCamsTab,   uroFilterItems_CamsTabClick,     'UROverviewCameraOptions', uroPopulateCamsTab],
   ['_tabOWL',    null, '_linkOWL',    'OWL',      uroShowOWLTab,    null,                            null,                      null],
   ['_tabMisc',   null, '_linkMisc',   'Misc',     uroShowMiscTab,   uroFilterItems_MiscTabClick,     'UROverviewMiscOptions',   uroPopulateMiscTab]
];

function uroModifyHTML(htmlIn)
{
	if(typeof trustedTypes === "undefined")
	{
		return htmlIn;
	}
	else
	{
		const escapeHTMLPolicy = trustedTypes.createPolicy("forceInner", {createHTML: (to_escape) => to_escape});
		return escapeHTMLPolicy.createHTML(htmlIn);
	}
}
function uroPopulateMPTab()
{
   let tHTML = '';
   tHTML += '<input type="checkbox" id="_cbMPFilterOutsideArea">Hide MPs outside my editable area</input><br><br>';
   tHTML += '<b>Filter MPs by type:</b><br>';
   let 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>';

   tHTML += '<br><b>Show MPs based on start/end dates:</b><br>';
   tHTML += '<input type="checkbox" id="_cbMPFilterStartDate">Start</input>&nbsp;&nbsp;';
   tHTML += '<input type="number" min="1" max="31" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterStartDay"> / ';
   tHTML += '<input type="number" min="1" max="12" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterStartMonth"> / ';
   tHTML += '<input type="number" min="2010" max="2100" size="4" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterStartYear"><br>';
   tHTML += '<input type="checkbox" id="_cbMPFilterEndDate">End</input>&nbsp;&nbsp;';
   tHTML += '<input type="number" min="1" max="31" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterEndDay"> / ';
   tHTML += '<input type="number" min="1" max="12" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterEndMonth"> / ';
   tHTML += '<input type="number" min="2010" max="2100" size="4" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterEndYear"><br>';
   tHTML += '<input type="checkbox" id="_cbMPFilterEndDatePassed">End date in the past</input>';

   return tHTML;
}
function uroPopulatePlacesTab()
{
   let 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="_cbPURFilterInsideManagedAreas">Ones within AM areas</input>';
   tHTML += '&nbsp;(<input type="checkbox" id="_cbPURExcludeUserArea">except my area)</input><br>';
   tHTML += '<i>Requires Area Manager layer to be enabled</i><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 += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFPhone">Phone number</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFName">Name</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFEntryExitPoints">Entry//exit points</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFOpeningHours">Opening hours</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFAliases">Aliases</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFServices">Services</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFGeometry">Geometry</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFHouseNumber">House number</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFCategories">Categories</input><br>';
   tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPURFilterCFDescription">Description</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="_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><b>Hide Places touched by a specific editor:</b><br>';
   tHTML += '<input type="checkbox" id="_cbHideOnlyPlacesCreatedBy">Created by</input>&nbsp;/&nbsp;';
   tHTML += '<input type="checkbox" id="_cbHideOnlyPlacesEditedBy">edited by</input><br>';
   tHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textHidePlacesEditor"><br>';
   tHTML += '<select id="_selectHidePlacesUserID" style="width:80%; height:22px;"></select><br>';

   tHTML += '<br><br><b>Filter Places by category:</b><br>';
   tHTML += '<input type="checkbox" id="_cbLeavePURGeos" pairedWith="_cbHidePURsForFilteredPlaces">Keep place visible if linked PUR is hidden, or</input><br>';
   tHTML += '<input type="checkbox" id="_cbHidePURsForFilteredPlaces" pairedWith="_cbLeavePURGeos">Hide PURs linked to hidden places</input><br><br>';

   let nCategories = W.Config.venues.categories.length;
   let i;
   if(uroPlacesGroupsCollapsed.length != nCategories)
   {
      for(i=0; i<nCategories; i++)
      {
         uroPlacesGroupsCollapsed.push(false);
      }
   }

   for(i=0; i<nCategories; i++)
   {
      let parentCategory = W.Config.venues.categories[i];
      let 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;<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(let ii=0; ii<W.Config.venues.subcategories[parentCategory].length; ii++)
      {
         let subCategory = W.Config.venues.subcategories[parentCategory][ii];
         localisedName = I18n.lookup("venues.categories." + subCategory);
         tHTML += '&nbsp;&nbsp;&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>';

   return tHTML;   
}
function uroPopulateURTab()
{
   let iHTML = '';
   iHTML = '<br>';

   iHTML += '<input type="checkbox" id="_cbURFilterOutsideArea">Hide URs outside my editable area</input><br>';
   iHTML += '<input type="checkbox" id="_cbURFilterInsideManagedAreas">Hide URs within AM areas</input>';
   iHTML += '&nbsp;(<input type="checkbox" id="_cbURExcludeUserArea">except my area)</input><br>';
   iHTML += '&nbsp;<i>Requires Area Manager layer to be enabled</i><br>';
   iHTML += '<input type="checkbox" id="_cbNoFilterForURInURL">Don\'t filter selected UR</input><br><br>';
   iHTML += '<input type="checkbox" id="_cbURFilterDupes">Show only duplicate URs</input><br><br>';

   iHTML += '<b>Filter by type:</b><br>';
   iHTML += '<input type="checkbox" id="_cbFilterWazeAuto">Waze Automatic</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterIncorrectTurn">Incorrect turn</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterIncorrectAddress">Incorrect address</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterIncorrectRoute">Incorrect route</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterMissingRoundabout">Missing roundabout</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterGeneralError">General error</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterTurnNotAllowed">Turn not allowed</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterIncorrectJunction">Incorrect junction</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterMissingBridgeOverpass">Missing bridge overpass</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterWrongDrivingDirection">Wrong driving direction</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterMissingExit">Missing exit</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterMissingRoad">Missing road</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterBlockedRoad">Blocked road</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterMissingLandmark">Missing Landmark</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterSpeedLimits">Missing or Invalid Speed limit</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterUndefined">Undefined</input><br>';

   iHTML += '&nbsp;&nbsp;<i>Specially tagged types</i><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterRoadworks">[ROADWORKS]</input><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterConstruction">[CONSTRUCTION]</input><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterClosure">[CLOSURE]</input><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterEvent">[EVENT]</input><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterNote">[NOTE]</input><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterBOG">[BOG]</input><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterDifficult">[DIFFICULT]</input><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterWSLM">[WSLM]</input><br><br>';
   iHTML += '<input type="checkbox" id="_cbInvertURFilter">Invert operation of type filters?</input><br>';

   iHTML += '<hr>';

   iHTML += '<br><b>Hide by state:</b><br>';
   iHTML += '<input type="checkbox" id="_cbFilterOpenUR">Open</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterClosedUR">Closed</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterSolved">Solved</input><br>';
   iHTML += '<input type="checkbox" id="_cbFilterUnidentified">Not identified</input><br><br>';


   iHTML += '<br><b>Filter by age of submission:</b><br>';
   iHTML += '<input type="checkbox" id="_cbEnableMinAgeFilter">Hide URs less than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinDays"> days old<br>';
   iHTML += '<input type="checkbox" id="_cbEnableMaxAgeFilter">Hide URs more than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxDays"> days old<br>';

   iHTML += '<br><b>Filter by other details:</b><br>';
   iHTML += '<input type="checkbox" id="_cbHideMyFollowed" pairedWith="_cbHideMyUnfollowed">Hide</input> or ';
   iHTML += '<input type="checkbox" id="_cbHideMyUnfollowed" pairedWith="_cbHideMyFollowed">show</input> URs I\'m following<br><br>';

   iHTML += '<input type="checkbox" id="_cbURDescriptionMustBePresent" pairedWith="_cbURDescriptionMustBeAbsent">Hide</input> or ';
   iHTML += '<input type="checkbox" id="_cbURDescriptionMustBeAbsent" pairedWith="_cbURDescriptionMustBePresent">show</input> URs with no description<br>';
   iHTML += '<input type="checkbox" id="_cbEnableKeywordMustBePresent">Hide URs not including </input>';
   iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPresent"><br>';
   iHTML += '<input type="checkbox" id="_cbEnableKeywordMustBeAbsent">Hide URs including </input>';
   iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordAbsent"><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbCaseInsensitive"><i>Case-insensitive matches?</i></input><br><br>';

   iHTML += 'With comments from me?<br>';
   iHTML += '<input type="checkbox" id="_cbHideMyComments" pairedWith="_cbHideAnyComments">Yes </input>';
   iHTML += '<input type="checkbox" id="_cbHideAnyComments" pairedWith="_cbHideMyComments">No</input><br>';
   iHTML += 'If last comment made by me?<br>';
   iHTML += '<input type="checkbox" id="_cbHideIfLastCommenter" pairedWith="_cbHideIfNotLastCommenter">Yes </input>';
   iHTML += '<input type="checkbox" id="_cbHideIfNotLastCommenter" pairedWith="_cbHideIfLastCommenter">No </input><br>';
   iHTML += 'If last comment made by UR reporter?<br>';
   iHTML += '<input type="checkbox" id="_cbHideIfReporterLastCommenter" pairedWith="_cbHideIfReporterNotLastCommenter">Yes </input>';
   iHTML += '<input type="checkbox" id="_cbHideIfReporterNotLastCommenter" pairedWith="_cbHideIfReporterLastCommenter">No</input><br>';

   iHTML += '<input type="checkbox" id="_cbEnableMinCommentsFilter">With less than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinComments"> comments<br>';
   iHTML += '<input type="checkbox" id="_cbEnableMaxCommentsFilter">With more than </input>';
   iHTML += '<input type="number" min="0" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxComments"> comments<br><br>';

   iHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter2">Last comment less than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays2"> days ago<br>';
   iHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter">Last comment more than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays"> days ago<br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbIgnoreOtherEditorComments"><i>Ignore other editor comments?</i></input><br><br>';

   iHTML += '<input type="checkbox" id="_cbURUserIDFilter">Without comments from user</input>';
   iHTML += '<select id="_selectURUserID" style="width:80%; height:22px;"></select><br>';
   iHTML += '<input type="checkbox" id="_cbURResolverIDFilter">Not resolved by user</input>';
   iHTML += '<select id="_selectURResolverID" style="width:80%; height:22px;"></select>';

   iHTML += '<br><br><input type="checkbox" id="_cbInvertURStateFilter">Invert operation of state/age filters?</input><br>';
   iHTML += '<input type="checkbox" id="_cbNoFilterForTaggedURs">Don\'t apply state/age filters to tagged URs</input><br>';

   return iHTML;   
}
function uroPopulateMCTab()
{
   let iHTML = '';
   
   iHTML = '<br>';

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

   iHTML += '<hr>';

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

   iHTML += '<input type="checkbox" id="_cbMCDescriptionMustBePresent" pairedWith="_cbMCDescriptionMustBeAbsent">Hide</input> or ';
   iHTML += '<input type="checkbox" id="_cbMCDescriptionMustBeAbsent" pairedWith="_cbMCDescriptionMustBePresent">show</input> MCs with no description<br>';
   iHTML += '<input type="checkbox" id="_cbMCCommentsMustBePresent" pairedWith="_cbMCCommentsMustBeAbsent">Hide</input> or ';
   iHTML += '<input type="checkbox" id="_cbMCCommentsMustBeAbsent" pairedWith="_cbMCCommentsMustBePresent">show</input> MCs with no comments<br>';
   iHTML += '<input type="checkbox" id="_cbMCExpiryMustBePresent" pairedWith="_cbMCExpiryMustBeAbsent">Hide</input> or ';
   iHTML += '<input type="checkbox" id="_cbMCExpiryMustBeAbsent" pairedWith="_cbMCExpiryMustBePresent">show</input> MCs with no expiry date<br>';
   iHTML += '<input type="checkbox" id="_cbMCEnableKeywordMustBePresent">Hide MCs not including </input>';
   iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textMCKeywordPresent"><br>';
   iHTML += '<input type="checkbox" id="_cbMCEnableKeywordMustBeAbsent">Hide MCs including </input>';
   iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textMCKeywordAbsent"><br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbMCCaseInsensitive"><i>Case-insensitive matches?</i></input><br>';
   iHTML += '<input type="checkbox" id="_cbMCCreatorIDFilter">Show MCs created by user</input>';
   iHTML += '<select id="_selectMCCreatorID" style="width:80%; height:22px;"></select><br>';

   iHTML += '<br><input type="checkbox" id="_cbHideWRCMCs"><b>Hide Waze_roadclosures MCs</b></input><br>';

   iHTML += '<br><b>Hide MCs with lock level:</b><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideMCRank0">L1</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideMCRank1">L2</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideMCRank2">L3</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideMCRank3">L4</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideMCRank4">L5</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideMCRank5">L6</input>';
   iHTML += '<hr>';
   iHTML += '<input type="checkbox" id="_cbMCEnhancePointMCVisibility">Enhance visibility of point MCs</input>';

   return iHTML;
}
function uroPopulateCamsTab()
{
   let iHTML = '';
   
   iHTML = '<br><b>Show Cameras created by:</b><br>';
   iHTML += '<input type="checkbox" id="_cbShowWorldCams" checked>world_* users</input><br>';
   iHTML += '<input type="checkbox" id="_cbShowUSACams" checked>usa_* users</input><br>';
   iHTML += '<input type="checkbox" id="_cbShowNonWorldCams" checked>other users</input><br>';

   iHTML += '<br><b>Show Cameras touched by a specific editor:</b><br>';
   iHTML += '<input type="checkbox" id="_cbShowOnlyCamsCreatedBy">Created by</input>&nbsp;/&nbsp;';
   iHTML += '<input type="checkbox" id="_cbShowOnlyCamsEditedBy">edited by</input><br>';
   iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textCameraEditor"><br>';
   iHTML += '<select id="_selectCameraUserID" style="width:80%; height:22px;"></select><br>';
   iHTML += '<br><input type="checkbox" id="_cbShowOnlyMyCams">Show ONLY cameras created/edited by me</input><br>';

   iHTML += '<br><b>Show Cameras by type:</b><br>';
   iHTML += '<input type="checkbox" id="_cbShowSpeedCams" checked>Speed</input><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowIfSpeedSet" checked> with speed data</input><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowIfNoSpeedSet" checked> with no speed data</input><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowIfInvalidSpeedSet" checked> with invalid speed data (zoom 16+)</input><br>';
   iHTML += '<input type="checkbox" id="_cbShowRedLightCams" checked>Red Light</input><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowRLCIfZeroSpeedSet" checked> with speed limit = 0</input><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowRLCIfNonZeroSpeedSet" checked> with speed limit > 0</input><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowRLCIfNoSpeedSet" checked> with no speed data</input><br>';
   iHTML += '<input type="checkbox" id="_cbShowDummyCams" checked>Dummy</input><br>';

   iHTML += '<br><b>Hide Cameras by creator:</b><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByMe">me</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank0">L1</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank1">L2</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank2">L3</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank3">L4</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank4">L5</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank5">L6</input>';

   iHTML += '<br><b>Hide Cameras by updater:</b><br>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByMe">me</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank0">L1</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank1">L2</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank2">L3</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank3">L4</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank4">L5</input>';
   iHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank5">L6</input>';

   iHTML += '<br><br><b><input type="checkbox" id="_cbHideCWLCams">Hide cameras on watchlist</input></b><br>';

   iHTML += '<br><br><b><input type="checkbox" id="_cbInvertCamFilters">Invert operation of camera filters?</input></b><br>';

   iHTML += '<br><br><b><input type="checkbox" id="_cbHighlightInsteadOfHideCams">Highlight instead of hide</input></b><br>';
   
   return iHTML;
}
function uroPopulateRTCTab()
{
   let iHTML = '';
   
   iHTML = '<br><b>Hide Road Closures:</b><br>';
   // Hidden checkbox to avoid errors when applying settings from previous versions of the script where this was an active control...
   iHTML += '<input type="checkbox" id="_cbHideUserRTCs" style="display: none;" />';

   iHTML += '<table style="text-align:center;">';
   iHTML += '<tr><td/><td><div class="map-marker road-closure status-finished" style="margin-left:0px;margin-top:0px;" /></td><td><div class="map-marker road-closure status-active" style="margin-left:0px;margin-top:0px;" /></td><td><div class="map-marker road-closure status-not-started" style="margin-left:0px;margin-top:0px;" /></td></tr>';
   iHTML += '<tr><td>From WME</td><td><input type="checkbox" id="_cbHideExpiredEditorRTCs" /></td><td><input type="checkbox" id="_cbHideEditorRTCs" /></td><td><input type="checkbox" id="_cbHideFutureEditorRTCs" /></td></tr>';
   iHTML += '<tr><td>From WazeFeed</td><td><input type="checkbox" id="_cbHideExpiredWazeFeedRTCs" /></td><td><input type="checkbox" id="_cbHideWazeFeedRTCs" /></td><td><input type="checkbox" id="_cbHideFutureWazeFeedRTCs" /></td></tr>';
   iHTML += '<tr><td>From Staff</td><td><input type="checkbox" id="_cbHideExpiredWazeRTCs" /></td><td><input type="checkbox" id="_cbHideWazeRTCs" /></td><td><input type="checkbox" id="_cbHideFutureWazeRTCs" /></td></tr>';
   iHTML += '</table><br>';

   iHTML += '<input type="checkbox" id="_cbShowMTERTCs" pairedWith="_cbHideMTERTCs">Show</input> or ';
   iHTML += '<input type="checkbox" id="_cbHideMTERTCs" pairedWith="_cbShowMTERTCs">hide RTCs associated with MTE: </input>';
   iHTML += '<select id="_selectRTCMTE" style="width:80%; height:22px;"></select><br>';
   iHTML += '<br>';
   iHTML += 'Hide if:<br>';
   iHTML += '<input type="checkbox" id="_cbEnableRTCDurationFilterLessThan">Duration less than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterRTCDurationLessThan"> days<br>';
   iHTML += '<input type="checkbox" id="_cbEnableRTCDurationFilterMoreThan">Duration more than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterRTCDurationMoreThan"> days<br>';
   
   iHTML += '<br><b>Filter by date/time:</b><br>'; 
   iHTML += '<input type="checkbox" id="_cbRTCFilterShowForTS" pairedWith="_cbRTCFilterHideForTS">Show</input> or ';
   iHTML += '<input type="checkbox" id="_cbRTCFilterHideForTS" pairedWith="_cbRTCFilterShowForTS">hide</input> RTCs active at<br>';
   iHTML += 'Date: <input type="number" min="1" max="31" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterDay"> / ';
   iHTML += '<input type="number" min="1" max="12" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterMonth"> / ';
   iHTML += '<input type="number" min="2010" max="2100" size="4" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterYear"><br>';
   iHTML += 'Time: <input type="number" min="0" max="23" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterHour">:';
   iHTML += '<input type="number" min="0" max="59" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterMin">';
      
   return iHTML;
}
function uroPopulateRATab()
{
   let iHTML = '';
   iHTML = '<br><b>Filter Restricted Areas:</b><br>';
   iHTML += '<input type="checkbox" id="_cbShowSpecificRA">Show a specific area: </input>';
   iHTML += '<select id="_selectRA" style="width:80%; height:22px;"></select><br><br>';

   iHTML += '<input type="checkbox" id="_cbRAEditorIDFilter">Show areas edited by user: </input>';
   iHTML += '<select id="_selectRAEditorID" style="width:80%; height:22px;"></select><br><br>';

   iHTML += 'Hide if:<br>';
   iHTML += '<input type="checkbox" id="_cbEnableRAAgeFilterLessThan">Last modified less than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterRAAgeLessThan"> days ago<br>';
   iHTML += '<input type="checkbox" id="_cbEnableRAAgeFilterMoreThan">Last modified more than </input>';
   iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterRAAgeMoreThan"> days ago<br>';
   
   return iHTML;
}
function uroPopulateMiscTab()
{
   let iHTML = '';
   iHTML += '<br><b><input type="checkbox" id="_cbHideSegmentsWhenRoadsHidden" />Hide segment layer when road layer is hidden</b><br>';
   iHTML += '<br><b><input type="checkbox" id="_cbKillInertialPanning" />Stop inertial panning when mouse moves out of map area</b><br>';

   iHTML += '<br><br><b>Use default conversation markers:</b><br>';
   iHTML += '<input type="checkbox" id="_cbNativeConvoMarkers" checked />in public WME<br>';
   iHTML += '<input type="checkbox" id="_cbNativeBetaConvoMarkers" checked />in beta WME<br>';

   iHTML += '<br><br><b><input type="checkbox" id="_cbCommentCount" />Show comment count on UR markers</b><br>';

   // Hidden checkbox to avoid errors when applying settings from previous versions of the script where this was an active control...
   iHTML += '<input type="checkbox" style="display: none;" id="_cbEnableDeleteFeedEntries" />';

   iHTML += '<br><br><b><input type="checkbox" id="_cbAutoApplyClonedClosure" />Auto-apply cloned closures</b><br>';
   iHTML += '<b><input type="checkbox" id="_cbAutoScrollClosureList" />Auto-scroll to end of closures</b><br>';

   iHTML += '<br><br><b><input type="checkbox" id="_cbURBackfill" />Backfill UR data</b><br>';

   iHTML += '<br><br><b>Disable filtering above zoom level </b>';
   iHTML += '<input type="number" min="12" max="22" value="22" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinZoomLevel" /><br>';

   iHTML += '<br><br><b>Marker Unstacking:</b><br>';
   iHTML += '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>';
   iHTML += 'Disable below zoom: <input type="number" min="12" max="22" value="15" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackZoomLevel" /><br>';

   iHTML += '<br><br><b>Use custom marker for URs tagged as:</b><br>';
   iHTML += '<input type="checkbox" id="_cbCustomRoadworksMarkers" />[ROADWORKS]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomConstructionMarkers" />[CONSTRUCTION]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomClosuresMarkers" />[CLOSURE]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomEventsMarkers" />[EVENT]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomNotesMarkers" />[NOTE]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomBOGMarkers" />[BOG]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomDifficultMarkers" />[DIFFICULT]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomWSLMMarkers" />[WSLM]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomNativeSLMarkers" />Native speed limit reports<br>';
   iHTML += '<input type="checkbox" id="_cbCustomKeywordMarkers" />';
   iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textCustomKeyword" /><br>';

   iHTML += '<br><br><b>Use custom marker for MPs tagged as:</b><br>';
   iHTML += '<input type="checkbox" id="_cbCustomElginMarkers" />[Elgin]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomTrafficMasterMarkers" />[TM]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomTrafficCastMarkers" />[TrafficCast]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomCaltransMarkers" />[Caltrans]<br>';
   iHTML += '<input type="checkbox" id="_cbCustomTFLMarkers" />[TfL Open Data]<br>';

   iHTML += '<br><br><b>Popup mouse behaviour:</b><br>';
   iHTML += '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>';
   iHTML += '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>';
   iHTML += '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>';
   iHTML += '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>';

   iHTML += '<br><br><b>Disable popup for:</b><br>';
   iHTML += '<input type="checkbox" id="_cbInhibitURPopup" />URs<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitMPPopup" />MPs<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitCamPopup" />Cameras<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitSegPopup" />Segments<br>';
   iHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbInhibitSegGenericPopup" />Speed limit info<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitTurnsPopup" />Restricted Turns<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitLandmarkPopup" />Places<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitPUPopup" />Place Updates<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitMapCommentPopup" />Map Comments<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitNodesPopup" />Junction Nodes<br>';

   iHTML += '<br><br><b>Date/Time formatting for popups:</b><br>';
   iHTML += '<input type="checkbox" id="_cbDateFmtDDMMYY" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtYYMMDD" checked />day/month/year<br>';
   iHTML += '<input type="checkbox" id="_cbDateFmtMMDDYY" pairedWith="_cbDateFmtDDMMYY,_cbDateFmtYYMMDD" />month/day/year<br>';
   iHTML += '<input type="checkbox" id="_cbDateFmtYYMMDD" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtDDMMYY" />year/month/day<br><br>';
   iHTML += '<input type="checkbox" id="_cbTimeFmt24H" pairedWith="_cbTimeFmt12H" checked />24 hour<br>';
   iHTML += '<input type="checkbox" id="_cbTimeFmt12H" pairedWith="_cbTimeFmt24H" />12 hour<br><br>';
   iHTML += '<i>Unticked uses browser default setting</i>';

   iHTML += '<br><br><b><input type="checkbox" id="_cbWhiteBackground" />Use custom background colour</b><br>';
   iHTML += '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" />';
   iHTML += '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" />';
   iHTML += '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>';

   iHTML += '<br><br><b>Replace "Next ..." button with "Done" for:</b><br>';
   iHTML += '<input type="checkbox" id="_cbInhibitNURButton" />URs<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitNMPButton" />MPs<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitNPURButton" />PURs<br>';

   iHTML += '<br><br><b>Disable on-click recentering for:</b><br>';
   iHTML += '<input type="checkbox" id="_cbInhibitURCentering" />URs<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitMPCentering" />MPs<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitPURCentering" />PURs<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitPPURCentering" />PPURs<br>';
   iHTML += '<input type="checkbox" id="_cbInhibitRPURCentering" />RPURs<br>';

   iHTML += '<br><br><b><input type="checkbox" id="_cbHideAMLayer" />Hide Area Manager polygons</b><br>';
   iHTML += '<b><input type="checkbox" id="_cbMoveAMList" />Show AMs in topbar when AM layer is active</b><br>';
   iHTML += '<br><b><input type="checkbox" id="_cbDisablePlacesFiltering" />Disable Places filtering</b><br>';

   // Hidden checkbox to avoid errors when applying settings from previous versions of the script where this was an active control...
   iHTML += '<b><input type="checkbox" style="display: none;" id="_cbDisableTabStyling" />Use default tab styling</b><br>';
   if(document.getElementById("user-details") !== null)
   {
      iHTML += '<b><input type="checkbox" id="_cbHideEditorInfo" />Hide sidebar editor info</b><br>';
   }

   iHTML += '<br><br><b>Settings backup/restore/reset:</b><br>';
   iHTML += '<input type="button" id="_btnSettingsToText" value="Backup" />&nbsp;&nbsp;&nbsp;';
   iHTML += '<input type="button" id="_btnTextToSettings" value="Restore" />&nbsp;&nbsp;|&nbsp;&nbsp;';
   iHTML += '<input type="button" id="_btnResetSettings" value="Reset" /><br><br>';
   iHTML += '<textarea id="_txtSettings" value=""></textarea><br>';
   iHTML += '<input type="button" id="_btnClearSettingsText" value="Clear" /><br>';

   /*
   iHTML += '<br><br><b>Debug:</b><br>';
   iHTML += '<input type="button" id="_btnDebugToScreen" value="Show debug data" />';
   */   
   
   return iHTML;
}

function uroAddDebug(debugtext)
{
   let 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 = uroModifyHTML('<textarea id="uroDbgOutput" style="width:100%;height:100%">');
      let dbgOutput = '';
      for(let i=0; i<uroRecentDebug.length; i++)
      {
         dbgOutput += uroRecentDebug[i]+'\n';
      }
      document.getElementById('uroDbgOutput').textContent = dbgOutput;
   }
}

function uroTempFixMTEDropDown()
{
   // Auto-selects the "None" event in the closures tab to avoid the user having to manually select it for almost
   // every single closure they'll ever add - the only time you'd need to select anything other than None is when
   // adding a closure related to a MTE...

   let retval = false;

   let tObj = document.getElementById('closure_eventId');
   // Make sure the closure event list is available, and that we haven't already messed with it.
   if((tObj !== null) && (tObj.tag != "touchedByURO"))
   {   
      // The event dropdown is now some byzantine piece of DOM manipulation to generate something which looks like a
      // regular select list, but which can't be manipulated like one...  The first gotcha is that the selected item
      // exists only within a shadow DOM section within the dropdown rather than simply being part of the list from
      // which we'd be able to read off its selected index.  So to check whether or not the user has selected an
      // event already, we need to drill down into this shadow DOM to get its text contents, and compare those against
      // the I18n translation for the choose event text.  What a palaver...
      let shadowElm = tObj.shadowRoot.querySelectorAll('.selected-value')[0];
      if(shadowElm !== undefined)
      {
         let eventText = tObj.shadowRoot.querySelectorAll('.selected-value')[0].innerText;
         // Sometimes we get here before WME has finished rendering, so if the event text hasn't been set yet then we
         // need to return false and let the caller deal with it...
         if(eventText !== '')
         {
            if(eventText == I18n.lookup('closures.choose_event'))
            {
               // Having now established that, yes, the closure hasn't yet been associated with any event, it's surprisingly
               // easy to change it to "None" - we just generate a click event on the first child element in the main DOM (not
               // the shadow DOM this time), which replicates what the user would do to select None manually.
               tObj.children[0].click();
            }
            // Tag the event list to prevent further processing attempts whilst this closure remains open.
            tObj.tag = "touchedByURO";
            retval = true;
         }
      }
   }

   return retval;
}
function uroPerformanceMonitoring(source, ts)
{
   if(uroPerformanceMonitoringOutput === true)
   {
      console.log(source+': '+(performance.now() - ts));
   }
}
function uroAddLog(logtext)
{
   if(uroShowDebugOutput) console.log('URO+: '+Date()+' '+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
   {
      if(typeof elm == 'object')
      {
         elm.style.display = style;
      }
      else
      {
         document.getElementById(elm).style.display = style;
      }
   }
   catch(err)
   {
      uroAddLog('uroSetStyleDisplay() - '+elm+' not found!');
   }
}
function uroSetOnClick(elm,fn)
{
   try
   {
      if(typeof elm == 'object')
      {
         elm.onclick = fn;
      }
      else
      {
         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 = uroModifyHTML('');
   document.getElementById('uroAlerts').childNodes[1].innerHTML = uroModifyHTML('');
   document.getElementById('uroAlertTickBtnCaption').innerHTML = uroModifyHTML('');
   document.getElementById('uroAlertCrossBtnCaption').innerHTML = uroModifyHTML('');
   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;
   let 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 = uroModifyHTML(titleContent);
   document.getElementById('uroAlerts').childNodes[1].innerHTML = uroModifyHTML(uroAlertBoxStack[0].content);
   document.getElementById('uroAlertTickBtnCaption').innerHTML = uroModifyHTML(uroAlertBoxStack[0].tickText);
   if(uroAlertBoxStack[0].hasCross)
   {
      document.getElementById('uroAlertCrossBtnCaption').innerHTML = uroModifyHTML(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');

   let releaseNotes = '';
   releaseNotes += '<p>Thanks for upgrading to URO+ '+uroVersion+' ('+uroReleaseDate+').  What\'s changed?</p>';

   let loop;
   if(uroChanges.length > 0)
   {
      releaseNotes += '<ul>';
      for(loop=0; loop < uroChanges.length; loop++)
      {
         releaseNotes += '<li>'+uroChanges[loop];
      }
      releaseNotes += '</ul>';
   }
   else if(uroBetaChanges.length > 0)
   {
      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...');

   let 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)
{
   let options = '';
   if(typeof(container) == 'string')
   {
      container = document.getElementById(container);
   }
   let urOptions = container.getElementsByTagName('input');
   for (let optIdx=0;optIdx<urOptions.length;optIdx++)
   {
      // Don't save settings for any of the legacy input elements we've now hidden...
      if(urOptions[optIdx].style.display != "none")
      {
         let 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()
{
   let liststr = '';
   for(let loop=0;loop<uroCamWatchObjects.length;loop++)
   {
      let 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.groupID+',';
         liststr += camObj.server;
      }
   }
   return liststr;
}
function uroGatherCWLGroups()
{
   let liststr = '';
   for(let loop=0;loop<uroCWLGroups.length;loop++)
   {
      let groupObj = uroCWLGroups[loop];
      if(groupObj.groupID != -1)
      {
         if(loop > 0) liststr += ':';

         liststr += groupObj.groupID+',';
         liststr += groupObj.groupName+',';
         liststr += groupObj.groupCollapsed;
      }
   }
   return liststr;
}
function uroGatherPlacesGroups()
{
   let liststr = '';
   for(let loop=0;loop<uroPlacesGroupsCollapsed.length;loop++)
   {
      if(loop > 0) liststr += ':';
      liststr += uroPlacesGroupsCollapsed[loop];
   }
   return liststr;
}
function uroGatherFriendlyAreaNames()
{
   let liststr = '';
   for(let loop=0;loop<uroFriendlyAreaNames.length;loop++)
   {
      let fnObj = uroFriendlyAreaNames[loop];
      if(loop > 0) liststr += ':';

      liststr += fnObj.fName+',';
      liststr += fnObj.area+',';
      liststr += fnObj.server;
   }
   return liststr;
}
function uroSaveSettings()
{
   if((uroInhibitSave) || (uroMTEMode === true) || (uroSettingsApplied === false))
   {
      uroAddLog('save inhibited');
      return;
   }

   if (localStorage)
   {
      try
      {
         for(let i = 0; i < uroCtrlTabs.length; ++i)
         {
            localStorage[uroCtrlTabs[i][URO_TABS_FIELD.STORAGE]] = uroGatherSettings(uroCtrlTabs[i][URO_TABS_FIELD.TABBODY]);
         }

         localStorage.UROverviewCamWatchList = uroGatherCamWatchList();
         localStorage.UROverviewCWLGroups = uroGatherCWLGroups();
         localStorage.UROverviewFriendlyAreaNames = uroGatherFriendlyAreaNames();
         localStorage.UROverviewPlacesGroups = uroGatherPlacesGroups();

         localStorage.UROverviewMasterEnable = uroGetCBChecked('_cbMasterEnable');
         localStorage.UROverviewCurrentVersion = uroVersion;

         uroAddLog('save complete');
      }
      catch(err)
      {
         uroAddLog('exception thrown during save - probably script reload whilst in MTE mode...');
      }
   }
   else
   {
      uroAddLog('no localStorage, save blocked');
   }
}
function uroApplySettings(settings)
{
   uroSettingsApplied = true;
   if(settings != undefined)
   {
      if(document.querySelector('#_cbMasterEnable') === null)
      {
         uroSettingsApplied = false;
      }
      else
      {
         let options = settings.split(':');
         for(let optIdx=0;optIdx<options.length;optIdx++)
         {
            let 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()
{
   let objects = localStorage.UROverviewCamWatchList.split(':');
   uroCamWatchObjects = [];
   if(objects.length > 0)
   {
      for(let objIdx=0;objIdx<objects.length;objIdx++)
      {
         let fields = objects[objIdx].split(',');
         if(fields.length == 9)
         {
            // CWL entries with 9 fields include the validated property which is now redundant, so we need to strip this property before adding
            // the camera to the object collection.  Whilst WME no longer displays unapproved cameras, it's preferable at this stage to leave
            // any watched unapproved cameras in the object collection, just in case any of them were approved (and will therefore still be
            // present in WME) inbetween the last time the user ran URO and now.  For those unapproved cameras which were still unapproved when
            // removed from WME, URO will then list them as deleted and the user can then perform a single manual tidy-up of their watchlist to
            // remove them there as well.
            uroCamWatchObjects.push(new uroCamWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[7],fields[8]));
         }
         else if(fields.length == 8)
         {
            uroCamWatchObjects.push(new uroCamWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7]));
         }
      }
   }
}
function uroApplyCWLGroups()
{
   let objects = localStorage.UROverviewCWLGroups.split(':');
   uroCWLGroups = [];

   if(objects.length === 0)
   {
      uroCWLGroups.push(new uroOWLGroupObj(0,'No group',false));
   }
   else
   {
      for(let objIdx=0;objIdx<objects.length;objIdx++)
      {
         let fields = objects[objIdx].split(',');
         if(fields.length < 2)
         {
            fields.push(false);
         }
         uroCWLGroups.push(new uroOWLGroupObj(fields[0],fields[1],(fields[2] == 'true')));
      }
   }
}
function uroApplyFriendlyAreaNames()
{
   let objects = localStorage.UROverviewFriendlyAreaNames.split(':');
   uroFriendlyAreaNames = [];

   for(let objIdx=0;objIdx<objects.length;objIdx++)
   {
      let fields = objects[objIdx].split(',');
      uroFriendlyAreaNames.push(new uroAFNObj(fields[0],parseFloat(fields[1]),fields[2]));
   }

   uroReplaceAreaNames(true);
}
function uroTranslateLegacyMPTabSettings()
{
   let options = localStorage.UROverviewMPOptions.split(':');
   for(let optIdx=0;optIdx<options.length;optIdx++)
   {
      let 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 uroTranslateLegacyZoomLevels()
{
   let tZoom = parseInt(document.getElementById("_inputFilterMinZoomLevel").value);
   if(tZoom < 12)
   {
      tZoom += 12;
      document.getElementById("_inputFilterMinZoomLevel").value = tZoom;
   }
   tZoom = parseInt(document.getElementById("_inputUnstackZoomLevel").value);
   if(tZoom < 12)
   {
      tZoom += 12;
      document.getElementById("_inputUnstackZoomLevel").value = tZoom;
   }
}
function uroLoadSettings()
{
   let isNewInstall = true;
   let isUpgradeInstall = true;
   let notifyAboutCustomIcons = true;

   uroAddLog('loadSettings()');
   
   for(let i = 0; i < uroCtrlTabs.length; ++i)
   {
      if (uroCtrlTabs[i][URO_TABS_FIELD.STORAGE] != null)
      {
         uroAddLog('recover '+uroCtrlTabs[i][URO_TABS_FIELD.TABTITLE]+' tab settings');
         uroApplySettings(localStorage[uroCtrlTabs[i][URO_TABS_FIELD.STORAGE]]);
         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.UROverviewMPOptions != null)
   {
      uroTranslateLegacyMPTabSettings();
   }   
   if (localStorage.UROverviewMiscOptions != null)
   {
      uroTranslateLegacyZoomLevels();
   }   

   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");
      uroAddLog('enable checkbox state set...');
   }

   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()
{
   let defaultSettings = '';

   defaultSettings += '[UROverviewCWLGroups][len=16]0,No group,false[END]';
   defaultSettings += '[UROverviewCamWatchList][len=0][END]';
   defaultSettings += '[UROverviewCameraOptions][len=852]:_cbShowWorldCams,true:_cbShowUSACams,true:_cbShowNonWorldCams,true:_cbShowOnlyCamsCreatedBy,false:_cbShowOnlyCamsEditedBy,false:_textCameraEditor,:_cbShowOnlyMyCams,false:_cbShowSpeedCams,true:_cbShowIfSpeedSet,true:_cbShowIfNoSpeedSet,true:_cbShowIfInvalidSpeedSet,true:_cbShowRedLightCams,true:_cbShowRLCIfZeroSpeedSet,true:_cbShowRLCIfNonZeroSpeedSet,true:_cbShowRLCIfNoSpeedSet,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:_cbHighlightInsteadOfHideCams,false[END]';
   defaultSettings += '[UROverviewCurrentVersion][len=5]3.146[END]';
   defaultSettings += '[UROverviewFriendlyAreaNames][len=0][END]';
   defaultSettings += '[UROverviewMCOptions][len=529]:_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:_cbMCEnhancePointMCVisibility,false[END]';
   defaultSettings += '[UROverviewMPOptions][len=1219]:_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_T23,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 += '[UROverviewMasterEnable][len=4]true[END]';
   defaultSettings += '[UROverviewMiscOptions][len=1698]:_cbHideUserRTCs,false:_cbHideEditorRTCs,false:_cbHideFutureEditorRTCs,false:_cbHideWazeRTCs,false:_cbHideFutureWazeRTCs,false:_cbHideSegmentsWhenRoadsHidden,false:_cbKillInertialPanning,false:_cbNativeConvoMarkers,true:_cbNativeBetaConvoMarkers,true:_cbCommentCount,false:_cbEnableDeleteFeedEntries,false:_cbURBackfill,false:_inputFilterMinZoomLevel,10:_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:_cbInhibitNodesPopup,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:_cbMoveAMList,false:_cbDisablePlacesFiltering,false:_cbDisableTabStyling,false:_cbHideEditorInfo,false:_cbEnableDTE,false[END]';
   defaultSettings += '[UROverviewPlaceWatchList][len=0][END]';
   defaultSettings += '[UROverviewPlacesGroups][len=65]false:false:false:false:false:false:false:false:false:false:false[END]';
   defaultSettings += '[UROverviewPlacesOptions][len=6062]:_cbFilterUneditablePlaceUpdates,false:_cbFilterLockRankedPlaceUpdates,false:_cbFilterNewPlacePUR,false:_cbFilterUpdatedDetailsPUR,false:_cbPURFilterCFPhone,false:_cbPURFilterCFName,false:_cbPURFilterCFEntryExitPoints,false:_cbPURFilterCFOpeningHours,false:_cbPURFilterCFAliases,false:_cbPURFilterCFServices,false:_cbPURFilterCFGeometry,false:_cbPURFilterCFHouseNumber,false:_cbPURFilterCFCategories,false:_cbPURFilterCFDescription,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,:_cbHideOnlyPlacesCreatedBy,false:_cbHideOnlyPlacesEditedBy,false:_textHidePlacesEditor,:_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-REST_AREAS,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-TRASH_AND_RECYCLING_FACILITIES,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-TELECOM,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 += '[UROverviewSegWatchList][len=0][END]';
   defaultSettings += '[UROverviewUROptions][len=1779]:_cbURFilterDupes,false:_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]';

   document.getElementById('_txtSettings').value = defaultSettings;
   uroTextToSettings();
   document.getElementById('_txtSettings').value = '';
}
function uroSettingsToText()
{
   let txtSettings = '';

   uroSaveSettings();

   for(let 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()
{
   let txtSettings = '';
   txtSettings = uroGetElmValue('_txtSettings');
   if(txtSettings.indexOf('[END]') == -1) return;

   let subText = txtSettings.split('[END]');
   for(let i=0;i<subText.length;i++)
   {
      let aPos = subText[i].indexOf('[');
      let bPos = subText[i].indexOf(']');
      if((aPos != -1) && (bPos != -1))
      {
         let settingID = subText[i].substr(aPos+1,bPos-1-aPos);
         subText[i] = subText[i].substr(bPos+1);
         bPos = subText[i].indexOf(']');
         if(bPos != -1)
         {
            let 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)
{
   let dateNow = new Date();

   let elapsedSinceEpoch = dateNow.getTime();
   let elapsedSinceEvent = elapsedSinceEpoch - dateToConvert;

   dateNow.setHours(0);
   dateNow.setMinutes(0);
   dateNow.setSeconds(0);
   dateNow.setMilliseconds(0);
   let elapsedSinceMidnight = elapsedSinceEpoch - dateNow.getTime();
   dateNow.setHours(24);
   let pendingUntilMidnight = elapsedSinceEpoch - dateNow.getTime();

   if((elapsedSinceEvent < elapsedSinceMidnight) && (elapsedSinceEvent > pendingUntilMidnight))
   {
      // event occurred today...
      return 0;
   }
   else if(elapsedSinceEvent < 0)
   {
      // event occurrs at some point in the future after midnight today, so return a minimum value of -1...
      return -1 - Math.floor((pendingUntilMidnight - elapsedSinceEvent) / 86400000);
   }
   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(mcAttrs, ageType, getRaw)
{
   if(ageType === 0)
   {
      if((mcAttrs.createdOn === null)||(mcAttrs.createdOn === 0)) return -1;
      if(getRaw) return mcAttrs.createdOn;
      else return uroDateToDays(mcAttrs.createdOn);
   }
   else if(ageType === 1)
   {
      if((mcAttrs.updatedOn === null)||(mcAttrs.updatedOn === 0)) return -1;
      if(getRaw) return mcAttrs.updatedOn;
      else return uroDateToDays(mcAttrs.updatedOn);
   }
   else if(ageType === 2)
   {
      if((mcAttrs.endDate === null)||(mcAttrs.endDate === 0)) return -1;
      let tDate = new Date(mcAttrs.endDate);
      if(getRaw) return tDate;
      else return uroDateToDays(tDate);
   }
   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 uroParseDaysToGo(days)
{
  days = 0 - days;
  if(days === 0) return 'today';
  else if(days === 1) return 'in 1 day';
  else return 'in '+days+' days';
}
function uroGetLocalisedSpeedString(thisSpeed)
{
   if(thisSpeed !== null)
   {
      let conversionFactor = 1;  // default to metric
      let multipleFactor = 10;   // default to limits being set in multiples of 10

      let country = null;

      if((W.model.getTopCountry()) && (W.model.getTopCountry().attributes.name !== undefined))
      {
         country = W.model.getTopCountry().attributes.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;
         }
      }

      let speed = Math.round(thisSpeed / conversionFactor);
      let retval = speed;
      if(conversionFactor == 1) retval += "KM/H";
      else retval += "MPH";

      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)
{
   let foundExisting = false;
   for(let 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()
{
   let newName = document.getElementById('_textAreaName').value;
   // sanitise name to avoid conflicts with config storage delimiters...
   newName = newName.replace(',','');
   newName = newName.replace(':','');
   let server = W.app.getAppRegionCode();
   let area = uroGetAreaArea(uroAreaNameHoverObj.parentNode.children[1].innerText);
   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(area)
{
   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();
   let existingName = uroAreaNameHoverObj.innerHTML;
   let italicTagPos = existingName.indexOf(' <i>');
   if(italicTagPos == -1)
   {
      existingName = "";
   }
   else
   {
      existingName = existingName.substr(0,italicTagPos);
   }
   uroANEditBox.innerHTML = uroModifyHTML('<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(document.getElementById('sidepanel-areas').getElementsByClassName('result-list').length === 0)
   {
      return;
   }

   if(replaceAfterNameChange === false)
   {
      if(document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id == "friendlyNamed")
      {
         return;
      }
   }

   let 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;
   }
   let areaNameObjs = panelRootObj.getElementsByClassName('list-item-card-info');
   if(areaNameObjs.length === 0)
   {
      return;
   }

   let localisedManagedArea = I18n.lookup("user.areas.managed_area");
   for(let loop=0; loop < areaNameObjs.length; loop++)
   {
      if(areaNameObjs[loop].children.length === 2)
      {
         let title = areaNameObjs[loop].children[0].innerText;
         if(title.indexOf(localisedManagedArea) > -1)
         {
            let area = uroGetAreaArea(areaNameObjs[loop].children[1].innerText);
            areaNameObjs[loop].children[0].innerHTML = uroModifyHTML(localisedManagedArea);

            for(let fnIdx=0; fnIdx < uroFriendlyAreaNames.length; fnIdx++)
            {
               let fnObj = uroFriendlyAreaNames[fnIdx];
               if((fnObj.area == area) && (fnObj.server == W.app.getAppRegionCode()))
               {
                  areaNameObjs[loop].children[0].innerHTML = uroModifyHTML(fnObj.fName +' <i>('+localisedManagedArea+')</i>');
                  break;
               }
            }
            let titleObj = areaNameObjs[loop].getElementsByClassName('list-item-card-title')[0];
            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, lat, lon)
{
   if(type !== null) type = uroTypeCast(type);
   if(azymuth !== null) azymuth = uroTruncate(uroTypeCast(azymuth)%360);
   if(speed !== null) speed = uroTruncate(uroTypeCast(speed));
   if(lat !== null) lat = uroTruncate(uroTypeCast(lat));
   if(lon !== null) lon = uroTruncate(uroTypeCast(lon));

   this.type = type;
   this.azymuth = azymuth;
   this.speed = speed;
   this.lat = lat;
   this.lon = lon;
}
function uroCamWatchObj(persistent, fid, lon, lat, type, azymuth, speed, groupID, server)
{
   fid = uroTypeCast(fid);
   groupID = uroTypeCast(groupID);
   if(typeof persistent == "string") persistent = (persistent == "true");
   if(server === "undefined") server = "??";

   this.fid = fid;
   this.persistent = persistent;
   this.loaded = false;
   this.server = server;
   this.groupID = groupID;
   this.watch = new uroCamWatchObjCheckProps(type, azymuth, speed, lat, lon);
   this.current = new uroCamWatchObjCheckProps(null, null, null, null, null);
}
function uroCamDataChanged(idx)
{
   let 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.lat != camObj.watch.lat) return true;
   if(camObj.current.lon != camObj.watch.lon) return true;
   return false;
}
function uroFindCWLGroupByIdx(groupIdx)
{
   let groupName = '';
   for(let loop=0;loop<uroCWLGroups.length;loop++)
   {
      if(uroCWLGroups[loop].groupID == groupIdx)
      {
         groupName = uroCWLGroups[loop].groupName;
         break;
      }
   }
   return groupName;
}
function uroIsCamOnWatchList(fid)
{
   for(let loop=0;loop<uroCamWatchObjects.length;loop++)
   {
      if(uroCamWatchObjects[loop].fid == fid) return loop;
   }
   return -1;
}
function uroAddCurrentCamWatchData(idx, lat, lon, type, azymuth, speed, server)
{
   let camObj = uroCamWatchObjects[idx];
   camObj.loaded = true;
   camObj.server = server;
   camObj.current = new uroCamWatchObjCheckProps(type, azymuth, speed, lat, lon);
   return(uroCamDataChanged(idx));
}
function uroAddCamToWatchList()
{
   if(uroIsCamOnWatchList(uroShownFID) == -1)
   {
      let 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, 0, W.app.getAppRegionCode()));
      uroAddCurrentCamWatchData(uroCamWatchObjects.length-1, camObj.geometry.y, camObj.geometry.x, camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, W.app.getAppRegionCode());
      uroAddLog('added camera '+uroShownFID+' to watchlist');
      uroOWLUpdateHTML();
   }
}
function uroRemoveCamFromWatchList()
{
   let camidx = uroIsCamOnWatchList(uroShownFID);
   if(camidx != -1)
   {
      uroCamWatchObjects.splice(camidx,1);
      uroAddLog('removed camera '+uroShownFID+' from watchlist');
      uroOWLUpdateHTML();
   }
}
function uroUpdateCamWatchList()
{
   let camIdx = uroIsCamOnWatchList(uroShownFID);
   if(camIdx != -1)
   {
      let camObj = W.model.cameras.objects[uroShownFID];
      uroCamWatchObjects[camIdx].watch = new uroCamWatchObjCheckProps(camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, 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)
{
   let camPos = new OpenLayers.LonLat();
   let camChanged = false;

   camPos.lon = lon;
   camPos.lat = lat;
   camPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));

   let camURL = 'https://' + document.location.host;
   camURL += W.Config.api_base;
   camURL += '/Features?language=en&cameras=true&bbox=';
   let latl = camPos.lat - 0.25;
   let latu = camPos.lat + 0.25;
   let lonl = camPos.lon - 0.25;
   let lonr = camPos.lon + 0.25;
   camURL += lonl+','+latl+','+lonr+','+latu;
   uroAddLog('retrieving camera data around '+camPos.lon+','+camPos.lat);

   let camReq = new XMLHttpRequest();
   camReq.open('GET',camURL,false);
   try
   {
      camReq.send();
      uroAddLog('response '+camReq.status+' received for camera data request');
      if (camReq.status === 200)
      {
         let camData = JSON.parse(camReq.responseText);
         for(let camIdx = 0; camIdx < camData.cameras.objects.length; camIdx++)
         {
            let camObj = camData.cameras.objects[camIdx];
            let 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, W.app.getAppRegionCode()) || camChanged);
            }
         }
      }
      else
      {
         uroAddLog('camera data request failed (status != 200)');
      }
   }
   catch(err)
   {
      uroAddLog('camera data request failed (exception '+err+' caught)');
   }
   return camChanged;
}
function uroGetCurrentCamWatchListObjects()
{
   let camChanged = false;
   let camsChanged = [];
   let camsDeleted = [];
   let camidx;
   let camObj;
   for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
   {
      camObj = uroCamWatchObjects[camidx];
      if((camObj.loaded === false) && ((camObj.server == W.app.getAppRegionCode()) || (camObj.server == '??')))
      {
         if(typeof W.model.cameras.objects[camObj.fid] == 'object')
         {
            if(W.model.cameras.objects[camObj.fid].state != "Delete")
            {
               let 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, W.app.getAppRegionCode()) || 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.app.getAppRegionCode()))
      {
         camsDeleted.push(camObj);
      }
   }
   if((camsChanged.length > 0) || (camsDeleted.length > 0))
   {
      let 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(let camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
   {
      if(uroCamWatchObjects[camidx].loaded === false)
      {
         uroShownFID = uroCamWatchObjects[camidx].fid;
         uroRemoveCamFromWatchList();
      }
   }
}
function uroAcceptCameraChanges()
{
   for(let 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.lat = uroCamWatchObjects[camidx].current.lat;
         uroCamWatchObjects[camidx].watch.lon = uroCamWatchObjects[camidx].current.lon;
      }
   }
   uroOWLUpdateHTML();
}
function uroClearUnknownServerCameras()
{
   let 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(let camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
   {
      if(uroCamWatchObjects[camidx].server == '??')
      {
         uroShownFID = uroCamWatchObjects[camidx].fid;
         uroRemoveCamFromWatchList();
      }
   }
}
function uroRescanCamWatchList()
{
   for(let camidx=0;camidx<uroCamWatchObjects.length;camidx++)
   {
      uroCamWatchObjects[camidx].loaded = false;
   }
   uroGetCurrentCamWatchListObjects();
   uroOWLUpdateHTML();
}
function uroGotoCam()
{
   let camidx = this.id.substr(13);
   let camPos = new OpenLayers.LonLat();
   camPos.lon = uroCamWatchObjects[camidx].watch.lon;
   camPos.lat = uroCamWatchObjects[camidx].watch.lat;
   W.map.setCenter(camPos,16);
   W.map.camerasLayer.setVisibility(true);
   return false;
}

function uroHighlightCWLEntry()
{
   this.style.backgroundColor = '#FFFFAA';
   return false;
}
function uroUnhighlightCWLEntry()
{
   let camidx = this.id.substr(8);
   let changed = uroCamDataChanged(camidx);
   let deleted = (uroCamWatchObjects[camidx].loaded === false);

   if(uroCamWatchObjects[camidx].server != W.app.getAppRegionCode())
   {
      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()
{
   let selector = document.getElementById('_uroCWLGroupSelect');
   while(selector.options.length > 0)
   {
      selector.options.remove(0);
   }
   for(let loop=0;loop<uroCWLGroups.length;loop++)
   {
      let groupObj = uroCWLGroups[loop];
      if(groupObj.groupID != -1)
      {
         selector.options.add(new Option(groupObj.groupName,groupObj.groupID));
      }
   }
}
function uroGetNextCWLGroupID()
{
   let nextID = 1;
   for(let loop=0;loop<uroCWLGroups.length;loop++)
   {
      if(uroCWLGroups[loop].groupID >= nextID)
      {
         nextID = uroCWLGroups[loop].groupID + 1;
      }
   }
   return nextID;
}
function uroFindCWLGroupByName(groupName)
{
   let groupID = -1;
   for(let loop=0;loop<uroCWLGroups.length;loop++)
   {
      if((uroCWLGroups[loop].groupName == groupName) && (uroCWLGroups[loop].groupID != -1))
      {
         groupID = uroCWLGroups[loop].groupID;
         break;
      }
   }
   return groupID;
}
function uroAddCWLGroup()
{
   let groupID = uroGetNextCWLGroupID();
   let groupName = uroGetElmValue('_uroCWLGroupEntry');
   if(uroFindCWLGroupByName(groupName) == -1)
   {
      uroCWLGroups.push(new uroOWLGroupObj(groupID,groupName,false));
      uroPopulateCWLGroupSelect();
   }
}
function uroRemoveCWLGroup()
{
   let loop;
   let selector = document.getElementById('_uroCWLGroupSelect');
   let groupID = parseInt(selector.selectedOptions[0].value);
   if(groupID === 0) return false;   // prevent deletion of the default group

   for(loop=0;loop<uroCamWatchObjects.length;loop++)
   {
      let cwObj = uroCamWatchObjects[loop];
      if(cwObj.groupID == groupID)
      {
         cwObj.groupID = 0;
      }
   }
   for(loop=0;loop<uroCWLGroups.length;loop++)
   {
      let groupObj = uroCWLGroups[loop];
      if(groupObj.groupID == groupID)
      {
         groupObj.groupID = -1;
      }
   }
   uroOWLUpdateHTML();
}
function uroAssignCameraToGroup()
{
   let camidx = this.id.substr(13);
   let selector = document.getElementById('_uroCWLGroupSelect');
   uroCamWatchObjects[camidx].groupID = parseInt(selector.selectedOptions[0].value);
   uroOWLUpdateHTML();
   return false;
}
function uroAddBtnEvl(btnID, evlType, evlFunction)
{
   let btnObj = document.getElementById(btnID);
   if(btnObj !== null)
   {
      btnObj.addEventListener(evlType, evlFunction, true);
   }
}
function uroCWLGroupCollapseExpand()
{
   let groupidx = this.id.substr(18);
   if(uroCWLGroups[groupidx].groupCollapsed === true) uroCWLGroups[groupidx].groupCollapsed = false;
   else uroCWLGroups[groupidx].groupCollapsed = true;
   uroOWLUpdateHTML();
   return false;
}

let uroSelectedOWLGroup = null;
function uroOWLUpdateHTML(doFullUpdate)
{
   let camTypes = new Array("","","Speed", "Dummy", "Red Light");
   let 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)
   {
      let camidx;
      for(let groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
      {
         let 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++)
            {
               let camObj = uroCamWatchObjects[camidx];
               if(camObj.groupID == groupObj.groupID)
               {
                  groupObj.groupCount++;
                  let changed = uroCamDataChanged(camidx);
                  let 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.app.getAppRegionCode())
                  {
                     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.app.getAppRegionCode())
                  {
                     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.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.app.getAppRegionCode())
                  {
                     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>';
   uroCtrlTabs[URO_TABS_ID.OWL][URO_TABS_FIELD.TABBODY].innerHTML = uroModifyHTML(iHTML);

   uroFinaliseOWLHTMLUpdate();
}
function uroFinaliseOWLHTMLUpdate()
{
   if(uroCamWatchObjects.length > 0)
   {
      if(document.getElementById("_uroCWL-0") == null)
      {
         window.setTimeout(uroFinaliseOWLHTMLUpdate,100);
         return;
      }

      for(let camidx=0;camidx<uroCamWatchObjects.length;camidx++)
      {
         document.getElementById("_uroCWL-"+camidx).onmouseover = uroHighlightCWLEntry;
         document.getElementById("_uroCWL-"+camidx).onmouseleave = uroUnhighlightCWLEntry;

         if(uroCamWatchObjects[camidx].server == W.app.getAppRegionCode())
         {
            let icon1 = document.getElementById("_uroCWLIcon1-"+camidx);
            let 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)
   {
      window.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();
      let selector = document.getElementById('_uroCWLGroupSelect');
      if(uroSelectedOWLGroup >= selector.length)
      {
         uroSelectedOWLGroup = 0;
      }
      selector.selectedIndex = uroSelectedOWLGroup;
   }

   if(uroCWLGroups.length > 0)
   {
      for(let 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()
{
   let btnState = "visible";
   if(sessionStorage.UROverview_FID_IgnoreList === '')
   {
      btnState = "hidden";
   }
   try
   {
      document.getElementById('_btnUndoLastHide').style.visibility = btnState;
      document.getElementById('_btnClearSessionHides').style.visibility = btnState;
      uroFilterItems();
   }
   catch(err)
   {
      uroAddLog('exception thrown in uroEnableIgnoreListControls()');
   }
}
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()
{
   let ignorelist = sessionStorage.UROverview_FID_IgnoreList;
   let 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)
{
   let 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)
{
   // The terminators array consists of pairs of characters which may be found at either
   // end of a URL.  The first entry in each pair indicates which character needs to be
   // found immediately prior to the URL ('' indicates any character) in order for the
   // second entry to be considered as a potential end of URL marker
   let terminators = [
                        ['',  ' '],
                        ['',  ','],
                        ['(', ')'],
                        ['[', ']'],
                        ['',  '\r'],
                        ['',  '\n']
                     ];

   if(desc === null) return '';
   if(desc === undefined) return '';
   if(desc === '') return '';

   if(desc.indexOf("https:  one.network") == 0)
   {
      desc = desc.replaceAll(' ','/');
   }

   desc = desc.replace(/<\/?[^>]+(>|$)/g, "");
   if(desc !== "null")
   {
      // At the moment we can only clickify links that start with http or https...
      if(desc.indexOf('http') != -1)
      {
         let links = desc.split("http");
         desc = '';
         let i, j, linkEndPos, descPostLink;
         for(i=0; i<links.length; i++)
         {
            if(links[i].indexOf('://') != -1)
            {
               let prefix = links[i - 1][links[i - 1].length - 1];
               links[i] = "http" + links[i];
               linkEndPos = links[i].length + 1;

               // work out where the end of the URL is, based on what the character immediately
               // preceding the "http" is
               for(j=0; j<terminators.length; j++)
               {
                  if(links[i].indexOf(terminators[j][1]) !== -1)
                  {
                     if((prefix === terminators[j][0]) || (terminators[j][0] === ''))
                     {
                        linkEndPos = Math.min(linkEndPos, links[i].indexOf(terminators[j][1]));
                     }
                  }
               }

               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];
            }
         }
      }
      desc = desc.replace(/\n/g,"<br>");
      return desc + suffix;
   }
   else
   {
      return '';
   }
}
let uroPendingURSessionsTotal;
let uroFinalizeTimeoutHandle = null;
function uroFinalizeURSessionsGet()
{
   if(uroPendingURSessionsTotal != uroPendingURSessionIDs.length)
   {
      uroPendingURSessionsTotal = uroPendingURSessionIDs.length;
      if(uroFinalizeTimeoutHandle !== null)
      {
         window.clearTimeout(uroFinalizeTimeoutHandle);
         uroFinalizeTimeoutHandle = null;
      }
      uroFinalizeTimeoutHandle = window.setTimeout(uroFinalizeURSessionsGet, 500);
      return;
   }

   let idList = [];

   while((idList.length < 50) && (uroPendingURSessionIDs.length))
   {
      let id = uroPendingURSessionIDs.shift();
      idList.push(id);
   }

   if(idList.length > 0)
   {
      uroAddLog('grabbing '+idList.length+' updateRequestSessions, IDs: '+idList);
      W.model.updateRequestSessions.getAsync(idList);
   }

   if((uroPendingURSessionIDs.length) || (uroRequestedURSessionIDs.length))
   {
      window.setTimeout(uroGetUpdateRequestSessions,1000);
   }
   else
   {
      uroPopulatingRequestSessions = false;
   }
}
function uroGetUpdateRequestSessions()
{
   uroPendingURSessionsTotal = uroPendingURSessionIDs.length;
   if(uroFinalizeTimeoutHandle !== null)
   {
      window.clearTimeout(uroFinalizeTimeoutHandle);
      uroFinalizeTimeoutHandle = null;
   }
   uroFinalizeTimeoutHandle = window.setTimeout(uroFinalizeURSessionsGet,500);
}
function uroRefreshUpdateRequestSessions()
{
   let urcount = 0;
   uroPendingURSessionIDs = [];
   uroRequestedURSessionIDs = [];
   uroPopulatingRequestSessions = true;
   for (let 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;
   }
   let nComments = W.model.updateRequestSessions.objects[fid].attributes.comments.length;
   if(nComments === 0)
   {
      return false;
   }

   for(let cidx=0; cidx<nComments; cidx++)
   {
      if(W.model.updateRequestSessions.objects[fid].attributes.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)
{
   let 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)
   {
      let customType = null;
      let customVariant = 0;
      let marker = uroGetMarker(markerType, urID);
      if(marker !== null)
      {
         customType = marker.uroCustomType;
         if(markerType == URO_TMARKER.UR)
         {
            if(W.model.mapUpdateRequests.objects[urID].attributes.open === false) customVariant = 2;
         }
         else if(markerType == URO_TMARKER.MP)
         {
            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 = uroModifyHTML('<img src="'+uroAltMarkers[customType][customVariant]+'">');
      }
   }
}
function uroRenderCustomMarkers(markerType)
{
   let urID;
   let elmID;
   let newSpan;
   let divElem;
   let objIdx;
   let customType;
   let customVariant;
   let cmlObj;
   let customMarker;
   let touchedByURO = false;

   if(markerType == URO_TMARKER.UR)
   {
      let useDefaultConvoMarker = false;
      let 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;
      }

      let uRCM_masterEnable = uroGetCBChecked('_cbMasterEnable');

      divElem = document.getElementById(W.map.getLayerByName("update_requests").id);

      if(divElem.childNodes.length > 0)
      {
         for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
         {
            customType = -1;
            cmlObj = uroCustomMarkerList[objIdx];
            if(cmlObj.markerType == URO_TMARKER.UR)
            {
               if(uRCM_masterEnable === true)
               {
                  customType = cmlObj.customType;
               }
               if(customType < 100)
               {
                  urID = cmlObj.urID;
                  let nComments = cmlObj.nComments;

                  let marker = uroGetMarker(URO_TMARKER.UR, urID);
                  if(marker !== null)
                  {                  
                     let iconObj = marker.element;
                     let classList = iconObj.classList;
                     newSpan = '';

                     if(nComments !== 0)
                     {
                        elmID = "commentCount_"+urID;

                        if((addCommentCount) && (nComments > 0))
                        {
                           // add a new comment count bubble
                           newSpan += '<span id="'+elmID+'" style="position:absolute;top:-12px;left:-16px;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:3px;left:0px;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();
                           }
                        }

                        if(nComments == -1)
                        {
                           // if we've set nComments to -1 to force a marker refresh when filtering is disabled,
                           // we now need to locally reset it to 0 to prevent the remainder of this code assuming
                           // the UR has a non-zero comment count...
                           nComments = 0;
                        }
                     }

                     if(nComments !== 0)
                     {
                        elmID = "convoMarker_"+urID;
                        if(useDefaultConvoMarker === false)
                        {
                           let 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 = marker.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);
                        marker.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.innerHTML = newSpan;
                     }

                     if((customMarker !== '') && (document.getElementById(elmID) !== null))
                     {
                        document.getElementById(elmID).innerHTML = uroModifyHTML(customMarker);
                     }

                     if(addCommentCount)
                     {
                        elmID = "commentCount_"+urID+"_inner";
                        if(document.getElementById(elmID) !== null)
                        {
                           let styleLeft;
                           if(nComments < 10) styleLeft = '11px';
                           else if(nComments < 100) styleLeft = '8px';
                           else styleLeft = '5px';
                           document.getElementById(elmID).innerHTML = uroModifyHTML(nComments);
                           document.getElementById(elmID).style.left = styleLeft;
                        }
                     }
                  }
               }
            }
         }
      }
   }

   else if(markerType == URO_TMARKER.MP)
   {
      divElem = document.getElementById(W.map.getLayerByName("mapProblems").id);
      if(divElem.childNodes.length > 0)
      {
         for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
         {
            cmlObj = uroCustomMarkerList[objIdx];
            if(cmlObj.markerType == URO_TMARKER.MP)
            {
               customType = cmlObj.customType;
               if((customType >= 100) || (customType == -1))
               {
                  urID = cmlObj.urID;
                  let markerObj = uroGetMarker(URO_TMARKER.MP, urID);

                  // change main marker if required
                  if(markerObj !== null) 
                  {
                     touchedByURO = markerObj.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(markerObj !== null)
                        {
                           markerObj.icon.$div.prepend(newSpan);
                        }
                     }
                     if(document.getElementById(elmID) !== null)
                     {
                        customType = uroGetCustomMarkerIdx(customType);
                        if(markerObj !== null)
                        {
                           markerObj.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 = uroModifyHTML(customMarker);
                     }
                  }
                  else
                  {
                     if(document.getElementById(elmID) !== null)
                     {
                        document.getElementById(elmID).remove();
                     }
                  }
               }
            }
         }
      }
   }
}
function uroIsFilteringEnabled(ignoreZoom)
{
   let retval = false;
   if
   (
      (uroGetCBChecked('_cbMasterEnable') === true) && 
      (
         (ignoreZoom === true) || 
         (W.map.getZoom() <= uroGetElmValue('_inputFilterMinZoomLevel'))
      )
   )
   {
      retval = true;
   }
   return retval;
}
function uroUpdateMTEList()
{
   if(Object.keys(W.model.majorTrafficEvents.objects).length === 0) return;

   let selectedIdx = null;
   let idx;
   let mteNames = [];
   let mteIDs = [];
   for(idx in W.model.majorTrafficEvents.objects)
   {
      if(W.model.majorTrafficEvents.objects.hasOwnProperty(idx))
      {
         let name = W.model.majorTrafficEvents.objects[idx].attributes.names[0]?.value;
         if(mteNames.indexOf(name) == -1)
         {
            mteNames.push(name);
            mteIDs.push(idx);
         }
      }
   }
   // check for any previously selected ID in the list, then clear it and repopulate
   // using the newly gathered ID collection from above, and finally reselect the
   // previously selected MTE if its still present in the new list...
   let selector;
   let selectedID;
   let selectorEntry;

   selector = document.getElementById('_selectRTCMTE');
   selectedID = null;
   if(selector.selectedOptions[0] != null)
   {
      selectedID = selector.selectedOptions[0].value;
   }
   while(selector.options.length > 0)
   {
      selector.options.remove(0);
   }
   selector.options.add(new Option('<select a MTE>', null));
   if(mteNames.length > 0)
   {
      selectorEntry = '';
      for(idx=0; idx<mteNames.length; idx++)
      {
         selectorEntry = mteNames[idx];
         selector.options.add(new Option(selectorEntry, mteIDs[idx]));
         if(mteIDs[idx] == selectedID)
         {
            selectedIdx = idx+1;
         }
      }
   }

   if(selectedIdx !== null)
   {
      selector.selectedIndex = selectedIdx;
   }
}
function uroRTCMarkerInfo(mIdx, isVisible)
{
   let mObj = W.map.getLayerByName("closures").markers[mIdx];
   let pri = null;
   let status = null;
   let dir = null;

   // Store the marker status - if we've already made a note of the original
   // status by adding an _orig-suffixed classname, extract the status from
   // that rather than from the class currently being used to display the
   // marker, to preserve the original status regardless of what we may have
   // done to it subsequently as a result of our RTC filtering setup...
   let cList = mObj.element.classList;
   for(let i = 0; i < cList.length; ++i)
   {
      if(cList[i].indexOf('_orig') != -1)
      {
         status = cList[i].replace('_orig', '');
      }
   }
   if(status === null)
   {
      for(let i = 0; i < cList.length; ++i)
      {
         if(cList[i].indexOf('status-') != -1)
         {
            status = cList[i];
         }
      }
   }

   // Assign a priority level to the marker so we know whether to show or hide
   // it if it's stacked up with others on the same segment - the level set here
   // follows the priority used by WME to decide which closures to show normally.
   if(status == "status-active")
   {
      pri = 2;
   }
   else if(status == "status-not-started")
   {
      pri = 1;
   }
   else
   {
      pri = 0;
   }

   // To avoid the need to cross-reference with the closure model object, use the
   // classname of the arrow attached to this marker to determine if the closure is
   // in the forward or reverse direction.
   if(mObj.element.childNodes.length == 1)
   {
      if(mObj.element.childNodes[0].className.indexOf('forward') != -1)
      {
         dir = "fwd";
      }
      else if(mObj.element.childNodes[0].className.indexOf('backward') != -1)
      {
         dir = "rev";
      }
   }

   this.isVisible = isVisible;
   this.pos = mObj.px;
   this.pri = pri;
   this.mID = mObj.element.dataset.id;
   this.status = status;
   this.dir = dir;
}

function uroFilterRTCs()
{
   let pmTStart = performance.now();
   let pmFunction = "uroFilterRTCs";

   if(uroFilterPreamble() === false) return;

   let uFR_filterActiveFromWME = uroGetCBChecked('_cbHideEditorRTCs');
   let uFR_filterActiveFromWazeFeed = uroGetCBChecked('_cbHideWazeFeedRTCs');
   let uFR_filterActiveFromWazeOther = uroGetCBChecked('_cbHideWazeRTCs');
   let uFR_filterFutureFromWME = uroGetCBChecked('_cbHideFutureEditorRTCs');
   let uFR_filterFutureFromWazeFeed = uroGetCBChecked('_cbHideFutureWazeFeedRTCs');
   let uFR_filterFutureFromWazeOther = uroGetCBChecked('_cbHideFutureWazeRTCs');
   let uFR_filterExpiredFromWME = uroGetCBChecked('_cbHideExpiredEditorRTCs');
   let uFR_filterExpiredFromWazeFeed = uroGetCBChecked('_cbHideExpiredWazeFeedRTCs');
   let uFR_filterExpiredFromWazeOther = uroGetCBChecked('_cbHideExpiredWazeRTCs');
   let uFR_filterShowForMTE = uroGetCBChecked('_cbShowMTERTCs');
   let uFR_filterHideForMTE = uroGetCBChecked('_cbHideMTERTCs');
   let uFR_filterHideDurationLessThan = uroGetCBChecked('_cbEnableRTCDurationFilterLessThan');
   let uFR_filterHideDurationMoreThan = uroGetCBChecked('_cbEnableRTCDurationFilterMoreThan');
   let uFR_thresholdDurationLessThan = uroGetElmValue('_inputFilterRTCDurationLessThan');
   let uFR_thresholdDurationMoreThan = uroGetElmValue('_inputFilterRTCDurationMoreThan');

   let uFR_filterShowForTS = uroGetCBChecked('_cbRTCFilterShowForTS');
   let uFR_filterHideForTS = uroGetCBChecked('_cbRTCFilterHideForTS');
   
   let tsD = uroGetElmValue('_inputRTCFilterDay');
   let tsMo = uroGetElmValue('_inputRTCFilterMonth');
   let tsY = uroGetElmValue('_inputRTCFilterYear');
   let tsH = uroGetElmValue('_inputRTCFilterHour');
   let tsMi = uroGetElmValue('_inputRTCFilterMin');
   let filterTS = uroGetTS(tsD, tsMo, tsY, tsH, tsMi);

   uroUpdateMTEList();
   let mteID = null;
   let selectorMTE = document.getElementById('_selectRTCMTE');
   if(selectorMTE.selectedOptions[0] != null)
   {
      mteID = selectorMTE.selectedOptions[0].value;
   }

   let uFR_masterEnable = uroIsFilteringEnabled(false);

   let markerInfo = [];

   // Pass 1 - determine which filtering to apply to each of the RTC markers
   let markerIdx = 0;
   for (let rtcObj in W.model.roadClosures.objects)
   {
      let isVisible = true;

      if(uFR_masterEnable === true)
      {   
         let rtcModel = W.model.roadClosures.objects[rtcObj];

         if(mteID !== null)
         {
            if((uFR_filterShowForMTE === true) && (rtcModel.attributes.eventId !== mteID))
            {
               isVisible = false;
            }
            if((uFR_filterHideForMTE === true) && (rtcModel.attributes.eventId === mteID))
            {
               isVisible = false;
            }
         }

         let rtcType = uroGetRTCOrigin(rtcModel);
         let rtcState = uroGetRTCState(rtcModel);

         if(rtcType == URO_TRTC.WAZEFEED)
         {
            if
            (
               ((rtcState === URO_SRTC.ACTIVE) && (uFR_filterActiveFromWazeFeed === true)) ||
               ((rtcState === URO_SRTC.FUTURE) && (uFR_filterFutureFromWazeFeed === true)) ||
               ((rtcState === URO_SRTC.EXPIRED) && (uFR_filterExpiredFromWazeFeed === true))
            )
            {
               isVisible = false;
            }
         }
         else if(rtcType == URO_TRTC.WAZEOTHER)
         {
            if
            (
               ((rtcState === URO_SRTC.ACTIVE) && (uFR_filterActiveFromWazeOther === true)) ||
               ((rtcState === URO_SRTC.FUTURE) && (uFR_filterFutureFromWazeOther === true)) ||
               ((rtcState === URO_SRTC.EXPIRED) && (uFR_filterExpiredFromWazeOther === true))
            )
            {
               isVisible = false;
            }
         }
         else
         {
            if
            (
               ((rtcState === URO_SRTC.ACTIVE) && (uFR_filterActiveFromWME === true)) ||
               ((rtcState === URO_SRTC.FUTURE) && (uFR_filterFutureFromWME === true)) ||
               ((rtcState === URO_SRTC.EXPIRED) && (uFR_filterExpiredFromWME === true))
            )
            {
               isVisible = false;
            }
         }
         
         let rtcDuration = uroGetRTCDuration(rtcModel);
         if(uFR_filterHideDurationLessThan === true)
         {
            if(rtcDuration < uFR_thresholdDurationLessThan)
            {
               isVisible = false;
            }
         }
         if(uFR_filterHideDurationMoreThan === true)
         {
            if(rtcDuration > uFR_thresholdDurationMoreThan)
            {
               isVisible = false;
            }
         }
         
         if((uFR_filterShowForTS === true) || (uFR_filterHideForTS === true))
         {
            let startTS = new Date(rtcModel.attributes.startDate).getTime();
            let endTS = new Date(rtcModel.attributes.endDate).getTime();
            
            if(uFR_filterShowForTS === true)
            {
               if((filterTS < startTS) || (filterTS > endTS))
               {
                  isVisible = false;
               }
            }
            if (uFR_filterHideForTS === true)
            {
               if((filterTS >= startTS) && (filterTS <= endTS))
               {
                  isVisible = false;
               }            
            }
         }
      }

      markerInfo.push(new uroRTCMarkerInfo(markerIdx, isVisible));
      ++markerIdx;
   }

   // Pass 2 - based on the initial filtering results, determine which markers *actually* should be
   // made visible or hidden according both to our filtering settings AND any masking of this marker
   // due to the presence of other higher priority markers at the same position which have also been
   // marked as visible following the filtering pass...
   //
   // For added merriment, we also deal with the WME limitation that prevents it displaying two 
   // different types of closure arrow if a segment has a one-way closure which is a higher priority
   // than a closure in the opposite direction.
   for (let i = 0; i < markerIdx; ++i)
   {
      let status = markerInfo[i].status;

      // Only apply this pass to closures which haven't already been hidden in pass 1, AND which
      // have a valid marker position
      if((markerInfo[i].isVisible === true) && (markerInfo[i].pos !== null))
      {
         // Iterate through all the other markers, looking for any which are also still visible
         // and have the same marker position as the one we're currently processing
         for (let j = 0; j < markerIdx; ++j)
         {
            if(j != i)
            {
               if((markerInfo[j].isVisible === true) && (markerInfo[j].pos !== null))
               {
                  if((markerInfo[i].pos.x == markerInfo[j].pos.x) && (markerInfo[i].pos.y == markerInfo[j].pos.y))
                  {            
                     if(markerInfo[j].pri > markerInfo[i].pri)
                     {
                        if(markerInfo[j].dir == markerInfo[i].dir)
                        {
                           // Mark the currently processed marker to be hidden only if this higher priority
                           // marker is for a closure in the same direction - if it's not, then we need to
                           // leave the current marker visible so that its arrow remains visible...
                           markerInfo[i].isVisible = false;
                        }
                        else
                        {
                           // Otherwise, if we're leaving the current marker visible then we'll need to 
                           // alter its status class to match this higher-priority marker, so that the
                           // marker which is shown to the user remains correct regardless of what the
                           // relative stacking order of the markers is on this segment.
                           status = markerInfo[j].status;
                        }
                        break;
                     }
                  }
               }
            }
         }
      }
      
      let markerClass = W.map.getLayerByName("closures").markers[i].element.className;

      // Remove the hidden class if present - this allows markers natively hidden by WME due to being masked
      // by higher priority markers to become visible again if our own filtering settings have hidden those
      // higher priority marker...
      markerClass = markerClass.replace(" road-closure-hidden", "");
      if(markerInfo[i].isVisible == false)
      {
         // Apply the hidden class for any markers WE'VE decided need to be hidden
         markerClass += " road-closure-hidden";
      }
      else
      {
         // For any markers which we're leaving visible, first check to see if it's a marker we haven't yet
         // seen.  If so, then we need to store its original status class for future reference.
         if(markerClass.indexOf('_orig') == -1)
         {
            markerClass += (' '+markerInfo[i].status + '_orig');
         }
         
         // Now, to ensure this segment displays the appropriate marker for the highest priority closure 
         // still visible on it, we remove the existing status class and replace it with the one we
         // chose above
         markerClass = markerClass.replace(" status-finished "," ");
         markerClass = markerClass.replace(" status-active "," ");
         markerClass = markerClass.replace(" status-not-started "," ");
         markerClass += ' '+status;
      }
      
      let toHide = document.querySelectorAll("[data-id='"+markerInfo[i].mID+"']");
      for(let j = 0; j < toHide.length; ++j)
      {
         if(toHide[j].className.indexOf("road-closure") !== -1)
         {
            toHide[j].className = markerClass;
         }
         else if(toHide[j].className.indexOf("closure-node") !== -1)
         {
            if(markerInfo[i].isVisible === false)
            {
               toHide[j].style.visibility = "hidden";
            }
            else
            {
               toHide[j].style.visibility = "";
            }
         }
      }
   }

   uroPerformanceMonitoring(pmFunction, pmTStart);
}
function uroUpdateRAList()
{
   if(Object.keys(W.model.restrictedDrivingAreas.objects).length === 0) return;

   let selectedIdx = null;
   let idx;
   let raNames = [];
   for(idx in W.model.restrictedDrivingAreas.objects)
   {
      if(W.model.restrictedDrivingAreas.objects.hasOwnProperty(idx))
      {
         let name = W.model.restrictedDrivingAreas.objects[idx].attributes.name;
         if(raNames.indexOf(name) == -1)
         {
            raNames.push(name);
         }
      }
   }
   // check for any previously selected name in the list, then clear it and repopulate
   // using the newly gathered collection from above, and finally reselect the
   // previously selected MTE if its still present in the new list...
   let selector;
   let selectedName;
   let selectorEntry;

   selector = document.getElementById('_selectRA');
   selectedName = null;
   if(selector.selectedOptions[0] != null)
   {
      selectedName = selector.selectedOptions[0].value;
   }
   while(selector.options.length > 0)
   {
      selector.options.remove(0);
   }
   selector.options.add(new Option('<select a RA>', null));
   if(raNames.length > 0)
   {
      selectorEntry = '';
      for(idx=0; idx<raNames.length; idx++)
      {
         selectorEntry = raNames[idx];
         selector.options.add(new Option(selectorEntry, selectorEntry));
         if(selectorEntry == selectedName)
         {
            selectedIdx = idx+1;
         }
      }
   }

   if(selectedIdx !== null)
   {
      selector.selectedIndex = selectedIdx;
   }
}
function uroUpdateEditorList(modelObj, listElement, useCreated, useUpdated, useResolved, useCommenter)
{
   if(Object.keys(modelObj).length === 0) return;

   let selector = document.getElementById(listElement);

   let selectedUser = null;
   if(selector.selectedOptions[0] != null)
   {
      selectedUser = parseInt(selector.selectedOptions[0].value);
   }

   while(selector.options.length > 0)
   {
      selector.options.remove(0);
   }

   let selectedIdx = null;
   let listedIDs = [];
   let idx;
   for(idx in modelObj)
   {
      if(modelObj.hasOwnProperty(idx))
      {
         let obj;
         if(useCommenter == true)
         {
            obj = modelObj[idx];
            if(obj.attributes.comments.length > 0)
            {
               for(let cidx=0; cidx < obj.attributes.comments.length; cidx++)
               {
                  let userID = obj.attributes.comments[cidx].userID;                  
                  if((listedIDs.indexOf(userID) == -1) && (userID != -1))
                  {
                     listedIDs.push(userID);                     
                  }
               }
            }
         }
         else
         {
            obj = modelObj[idx].attributes;
            let cbID = null;
            let ubID = null;
            let rbID = null;
            if(useCreated == true) cbID = obj.createdBy;
            if(useUpdated == true) ubID = obj.updatedBy;
            if(useResolved == true) ubID = obj.resolvedBy;
            
            if((cbID !== null) && (listedIDs.indexOf(cbID) == -1))
            {
               listedIDs.push(cbID);
            }
            if((ubID !== null) && (listedIDs.indexOf(ubID) == -1))
            {
               listedIDs.push(ubID);
            }
            if((rbID !== null) && (listedIDs.indexOf(rbID) == -1))
            {
               listedIDs.push(rbID);
            }
         }
      }
   }

   selector.options.add(new Option('<select a user>', null));
   if(listedIDs.length > 0)
   {
      let users = W.model.users.getByIds(listedIDs);
      let selectorEntry = '';
      for(idx=0; idx<users.length; idx++)
      {
         if(listedIDs.indexOf(users[idx].id) != -1)
         {
            listedIDs.splice(listedIDs.indexOf(users[idx]), 1);
         }
         
         if(users[idx].attributes.userName === undefined)
         {
            selectorEntry = users[idx].attributes.id;
         }
         else
         {
            selectorEntry = users[idx].attributes.userName;
         }
         selector.options.add(new Option(selectorEntry, users[idx].id));
         if(users[idx].attributes.id == selectedUser)
         {
            selectedIdx = idx+1;
         }
      }
   }

   if(selectedIdx !== null)
   {
      selector.selectedIndex = selectedIdx;
   }
}
function uroGetUserID(filterNameID, tbUserName)
{
   if(filterNameID === null)
   {
      for(let idx in W.model.users.objects)
      {
         if(W.model.users.objects.hasOwnProperty(idx))
         {
            if(W.model.users.objects[idx].attributes.userName == tbUserName)
            {
               filterNameID = W.model.users.objects[idx].attributes.id;
               break;
            }
         }
      }      
   }
   return filterNameID;
}  
function uroFilterRAs()
{
   let pmTStart = performance.now();
   let pmFunction = "uroFilterRAs";

   if(uroFilterPreamble() === false) return;
   let uFURs_masterEnable = uroIsFilteringEnabled(false);
   let filterByArea = uroGetCBChecked('_cbShowSpecificRA');
   let filterByLastEditor = uroGetCBChecked('_cbRAEditorIDFilter');
   let filterByMinAge = uroGetCBChecked('_cbEnableRAAgeFilterLessThan');
   let filterByMaxAge = uroGetCBChecked('_cbEnableRAAgeFilterMoreThan');
   let thresholdMinAge = uroGetElmValue('_inputFilterRAAgeLessThan');
   let thresholdMaxAge = uroGetElmValue('_inputFilterRAAgeMoreThan');
   
   let selectorRA = document.getElementById('_selectRA');
   if(filterByArea === false)
   {
      while(selectorRA.options.length > 0)
      {
         selectorRA.options.remove(0);
      }
   }
   let shownRA = null;
   if(filterByArea === true)
   {
      if(selectorRA.options.length === 0)
      {
         uroUpdateRAList();
      }
      if(selectorRA.selectedOptions[0] != null)
      {
         shownRA = selectorRA.selectedOptions[0].value;
      }
   }

   let filterNameID = null;
   if(filterByLastEditor == true)
   {
      uroUpdateEditorList(W.model.restrictedDrivingAreas.objects, '_selectRAEditorID', true, true, false, false);
      let selector = document.getElementById('_selectRAEditorID');
      if(selector.selectedIndex > 0)
      {
         filterNameID = document.getElementById('_selectRAEditorID').selectedOptions[0].value;
      }
   }
   
   let nRANames = document.querySelectorAll('.restricted-driving-area-name-marker').length;
   for (let raIdx = 0; raIdx < W.map.restrictedDrivingAreaLayer.features.length; raIdx++)
   {
      let raObj = W.map.restrictedDrivingAreaLayer.features[raIdx].attributes.wazeFeature;
      if(raObj !== undefined)
      {
         let raStyle = 'visible';
         if(uFURs_masterEnable === true)
         {
            if(shownRA !== null)
            {
               if(raObj._wmeObject.attributes.name != shownRA) raStyle = 'hidden';
            }
            
            if(filterNameID !== null)
            {
               if((raObj._wmeObject.attributes.createdBy != filterNameID) && (raObj._wmeObject.attributes.updatedBy != filterNameID))
               {
                  raStyle = 'hidden';
               }
            }
         
            let raAge = uroDateToDays(raObj._wmeObject.attributes.updatedOn);
            if(filterByMinAge == true)
            {
               if(raAge < thresholdMinAge) raStyle = 'hidden';
            }
            if(filterByMaxAge == true)
            {
               if(raAge > thresholdMaxAge) raStyle = 'hidden';
            }
         }
         
         let geoID = W.map.restrictedDrivingAreaLayer.features[raIdx].geometry.id;
         if(document.getElementById(geoID) !== null)
         {
            document.getElementById(geoID).style.visibility = raStyle;
         }

         // This doesn't always work, as the order in which the markers are listed on their layer isn't guaranteed
         // to match the order in which the corresponding RA polys are listed on theirs...  
         if(raIdx < nRANames)
         {
            document.querySelectorAll('.restricted-driving-area-name-marker')[raIdx].style.visibility = raStyle;
         }
      }
   }      
   uroPerformanceMonitoring(pmFunction, pmTStart);
}

function uroFilterPlaceMarker(mObj, vObj, uFP_masterEnable)
{
   if((mObj === undefined) || (vObj === undefined))
   {
      return;
   }

   let purAge = null;
   let placeStyle = 'visible';

   if(uFP_masterEnable === true)
   {
      if(uro_uFP[URO_UFP_OPTS.filterInsideManagedAreas] === true)
      {
         let tPt = new OpenLayers.Geometry.Point();
         tPt.x = mObj.lonlat.lon;
         tPt.y = mObj.lonlat.lat;
         if(uroCheckGeometryWithinManagedAreas(tPt) === true) placeStyle = 'hidden';
      }

      if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterUneditable] === true))
      {
         if(vObj.attributes.permissions === 0)
         {
            placeStyle = 'hidden';
         }
         if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.isLoggedIn]))
         {
            if(uro_uFP[URO_UFP_OPTS.userRank] < vObj.attributes.lockRank)
            {
               placeStyle = 'hidden';
            }
         }
         if((placeStyle == 'visible') && (vObj.attributes.adLocked))
         {
            placeStyle = 'hidden';
         }
      }

      if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterLockRanked] === true))
      {
         if(vObj.attributes.lockRank !== 0)
         {
            placeStyle = 'hidden';
         }
      }

      if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterFlagged] === true))
      {
         if(mObj.element.className.indexOf('flag') != -1)
         {
            placeStyle = 'hidden';
         }
      }

      if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterNewPlace] === true))
      {
         if(mObj.element.className.indexOf('add_venue') != -1)
         {
            placeStyle = 'hidden';
         }
      }
      if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterUpdatedDetails] === true))
      {
         if((mObj.element.className.indexOf('update_venue') != -1) || (mObj.element.className.indexOf('multiple') != -1))
         {
            placeStyle = 'hidden';
         }
      }
      if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterOnCFs] === true))
      {
         let nVUR = vObj.attributes.venueUpdateRequests.length;
         while(nVUR > 0)
         {
            nVUR--;
            let tCF = vObj.attributes.venueUpdateRequests[nVUR].attributes.changedFields;
            if(tCF !== undefined)
            {
               if(tCF.length > 0)
               {
                  let tFN = tCF[0].attributes.fieldName;
                  if((tFN == "phone") && (uro_uFP[URO_UFP_OPTS.filterCFPhone] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "name") && (uro_uFP[URO_UFP_OPTS.filterCFName] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "entryExitPoints") && (uro_uFP[URO_UFP_OPTS.filterCFEntryExitPoints] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "openingHours") && (uro_uFP[URO_UFP_OPTS.filterCFOpeningHours] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "aliases") && (uro_uFP[URO_UFP_OPTS.filterCFAliases] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "services") && (uro_uFP[URO_UFP_OPTS.filterCFServices] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "geometry") && (uro_uFP[URO_UFP_OPTS.filterCFGeometry] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "houseNumber") && (uro_uFP[URO_UFP_OPTS.filterCFHouseNumber] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "categories") && (uro_uFP[URO_UFP_OPTS.filterCFCategories] === true))
                  {
                     placeStyle = 'hidden';
                  }
                  if((tFN == "description") && (uro_uFP[URO_UFP_OPTS.filterCFDescription] === true))
                  {
                     placeStyle = 'hidden';
                  }
               }
            }
         }
      }
      if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterNewPhoto] === true))
      {
         if(mObj.element.className.indexOf('add_image') != -1)
         {
            placeStyle = 'hidden';
         }
      }

      if(uro_uFP[URO_UFP_OPTS.invertPURFilters] === true)
      {
         if(placeStyle == 'hidden') placeStyle = 'visible';
         else placeStyle = 'hidden';
      }

      if(uro_uFP[URO_UFP_OPTS.filterMinPURAge] || uro_uFP[URO_UFP_OPTS.filterMaxPURAge])
      {
         purAge = uroGetPURAge(vObj);
         if(uro_uFP[URO_UFP_OPTS.filterMinPURAge] === true)
         {
            if(purAge < uro_uFP[URO_UFP_OPTS.thresholdMinPURDays]) placeStyle = 'hidden';
         }
         if(uro_uFP[URO_UFP_OPTS.filterMaxPURAge] === true)
         {
            if(purAge > uro_uFP[URO_UFP_OPTS.thresholdMaxPURDays]) placeStyle = 'hidden';
         }
      }

      if(placeStyle == 'visible')
      {
         if((uro_uFP[URO_UFP_OPTS.filterHighSeverity]) && (mObj.element.className.indexOf("high") != -1)) placeStyle = 'hidden';
         if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterMedSeverity]) && (mObj.element.className.indexOf("medium") != -1)) placeStyle = 'hidden';
         if((placeStyle == 'visible') && (uro_uFP[URO_UFP_OPTS.filterLowSeverity]) && (mObj.element.className.indexOf("low") != -1)) placeStyle = 'hidden';
      }

      if(uroPURsToHide.indexOf(vObj.attributes.id) !== -1)
      {
         placeStyle = 'hidden';
      }
   }

   mObj.element.style.visibility = placeStyle;

   if((uro_uFP[URO_UFP_OPTS.leavePURGeos] === false) && (placeStyle === 'hidden'))
   {
      if(vObj.model != null)
      {
         if(vObj.attributes.geometry != null)
         {
            let puGeo = document.getElementById(vObj.attributes.geometry.id);
            if(puGeo !== null)
            {
               puGeo.style.visibility = 'hidden';
            }
         }
      }
   }
}
function uroFilterPlaces()
{
   let pmTStart = performance.now();
   let pmFunction = "uroFilterPlaces";

   if(uroFilterPreamble() === false) return;

   let moObj = uroGetHighlightedMapFeature();
   let renderIntent = uroGetFeatureRenderIntent(moObj);
   if(moObj != null)
   {
      if(moObj.featureType === 'venue')
      {
         if((renderIntent == 'select') || (renderIntent == 'highlightselected'))
         {
            return;
         }
      }
   }

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

   uroUpdateVenueEditorLists();

   let filterNameID = null;
   let tbUserName = uroGetElmValue('_textPlacesEditor');
   let selector = document.getElementById('_selectPlacesUserID');
   if(selector.selectedIndex > 0)
   {
      let selUserName = document.getElementById('_selectPlacesUserID').selectedOptions[0].innerHTML;
      if(selUserName == tbUserName)
      {
         filterNameID = document.getElementById('_selectPlacesUserID').selectedOptions[0].value;
      }
   }
   filterNameID = uroGetUserID(filterNameID, tbUserName);

   let filterHideNameID = null;
   let tbHideUserName = uroGetElmValue('_textHidePlacesEditor');
   let selectorHide = document.getElementById('_selectHidePlacesUserID');
   if(selectorHide.selectedIndex > 0)
   {
      let selHideUserName = document.getElementById('_selectHidePlacesUserID').selectedOptions[0].innerHTML;
      if(selHideUserName == tbHideUserName)
      {
         filterHideNameID = document.getElementById('_selectHidePlacesUserID').selectedOptions[0].value;
      }
   }
   filterHideNameID = uroGetUserID(filterHideNameID, tbHideUserName);

   let filterCats = [];
   for(let i=0; i<W.Config.venues.categories.length; i++)
   {
      let parentCategory = W.Config.venues.categories[i];
      let subCategory;

      if(uroGetCBChecked('_cbPlacesFilter-'+parentCategory) === true)
      {
         filterCats.push(parentCategory);
         for(let i1=0; i1<W.Config.venues.subcategories[parentCategory].length; i1++)
         {
            subCategory = W.Config.venues.subcategories[parentCategory][i1];
            filterCats.push(subCategory);
         }
      }
      else
      {
         for(let 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);
            }
         }
      }
   }

   let placeStyle;

   let uFP_filterEditedLessThan = uroGetCBChecked('_cbPlaceFilterEditedLessThan');
   let uFP_filterEditedMoreThan = uroGetCBChecked('_cbPlaceFilterEditedMoreThan');
   let uFP_filterL0 = uroGetCBChecked('_cbHidePlacesL0');
   let uFP_filterL1 = uroGetCBChecked('_cbHidePlacesL1');
   let uFP_filterL2 = uroGetCBChecked('_cbHidePlacesL2');
   let uFP_filterL3 = uroGetCBChecked('_cbHidePlacesL3');
   let uFP_filterL4 = uroGetCBChecked('_cbHidePlacesL4');
   let uFP_filterL5 = uroGetCBChecked('_cbHidePlacesL5');
   let uFP_filterStaff = uroGetCBChecked('_cbHidePlacesStaff');
   let uFP_filterAL = uroGetCBChecked('_cbHidePlacesAdLocked');
   let uFP_filterOnLockLevel = (uFP_filterL0 || uFP_filterL1 || uFP_filterL2 || uFP_filterL3 || uFP_filterL4 || uFP_filterL5 || uFP_filterStaff);
   let uFP_filterNoPhotos = uroGetCBChecked('_cbHideNoPhotoPlaces');
   let uFP_filterWithPhotos = uroGetCBChecked('_cbHidePhotoPlaces');
   let uFP_filterNoLinks = uroGetCBChecked('_cbHideNoLinkedPlaces');
   let uFP_filterWithLinks = uroGetCBChecked('_cbHideLinkedPlaces');
   let uFP_filterNoDescription = uroGetCBChecked('_cbHideNonDescribedPlaces');
   let uFP_filterWithDescription = uroGetCBChecked('_cbHideDescribedPlaces');
   let uFP_filterNoKeyword = uroGetCBChecked('_cbHideKeywordPlaces');
   let uFP_filterKeyword = uroGetCBChecked('_cbHideNoKeywordPlaces');
   let uFP_filterPrivate = uroGetCBChecked('_cbFilterPrivatePlaces');
   let uFP_invertFilters = uroGetCBChecked('_cbInvertPlacesFilter');
   let uFP_masterEnable = uroIsFilteringEnabled(false);
   let uFP_filterAreaPlaces = uroGetCBChecked('_cbHideAreaPlaces');
   let uFP_filterPointPlaces = uroGetCBChecked('_cbHidePointPlaces');
   let uFP_filterCreatedBy = uroGetCBChecked('_cbShowOnlyPlacesCreatedBy');
   let uFP_filterEditedBy = uroGetCBChecked('_cbShowOnlyPlacesEditedBy');
   let uFP_filterHideCreatedBy = uroGetCBChecked('_cbHideOnlyPlacesCreatedBy');
   let uFP_filterHideEditedBy = uroGetCBChecked('_cbHideOnlyPlacesEditedBy');

   let uFP_hidePURsForFilteredPlaces = uroGetCBChecked('_cbHidePURsForFilteredPlaces');

   let uFP_NameKeyword = document.getElementById('_textKeywordPlace').value.toLowerCase();
   let uFP_thresholdMinDays = document.getElementById('_inputFilterPlaceEditMinDays').value;
   let uFP_thresholdMaxDays = document.getElementById('_inputFilterPlaceEditMaxDays').value;

   uroPURsToHide = [];

   for(let v=0; v<uroVenueLayer.features.length; v++)
   {
      placeStyle = 'visible';
      if(uFP_masterEnable === true)
      {
         let lmObj = uroVenueLayer.features[v];

         // when an area place is selected, the drag points for editing the place outline now get added as objects into uroVenueLayer.features,
         // however none of these objects had the .attributes.repositoryObject property - whilst the devs have now replaced this with the almost
         // identical .wazeFeature._wmeObject property, it's unclear if drag points still need to be excluded from this scan, so the check
         // remains in place as a "let's just make sure it has it before trying to use it"...
         if(lmObj?.attributes?.wazeFeature?._wmeObject != null)
         {
            lmObj = lmObj.attributes.wazeFeature._wmeObject.attributes;
            if(lmObj.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.geometry.id.indexOf('Polygon') !== -1)
               {
                  placeStyle = 'hidden';
               }
            }
            if(uFP_filterPointPlaces)
            {
               if(lmObj.geometry.id.indexOf('Point') !== -1)
               {
                  placeStyle = 'hidden';
               }
            }


            if(placeStyle == 'visible')
            {
               if((uFP_filterEditedLessThan) || (uFP_filterEditedMoreThan))
               {
                  let editDate = lmObj.updatedOn;
                  if(editDate === undefined)
                  {
                     // where a place has never been edited since its creation, use the creation date instead...
                     editDate = lmObj.createdOn;
                  }
                  if(editDate != null)
                  {
                     let 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)
               {
                  let lockLevel = lmObj.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.adLocked) placeStyle = 'hidden';
               }
            }

            if(placeStyle == 'visible')
            {
               if(uFP_filterNoPhotos || uFP_filterWithPhotos)
               {
                  let nPhotos = 0;
                  for(let loop=0; loop<lmObj.images.length; loop++)
                  {
                     if(lmObj.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)
               {
                  let nLinks = lmObj.externalProviderIDs.length;
                  if((uFP_filterNoLinks) && (nLinks === 0)) placeStyle = 'hidden';
                  if((uFP_filterWithLinks) && (nLinks !== 0)) placeStyle = 'hidden';
               }
            }

            if(placeStyle == 'visible')
            {
              if(uFP_filterNoDescription || uFP_filterWithDescription)
              {
                let lDesc = lmObj.description.length;
                if((uFP_filterNoDescription) && (lDesc === 0)) placeStyle = 'hidden';
                if((uFP_filterWithDescription) && (lDesc !== 0)) placeStyle = 'hidden';
              }
            }

            if(placeStyle == 'visible')
            {
               if((uFP_filterPrivate === true) && (lmObj.residential === true))
               {
                  placeStyle = 'hidden';
               }
               else
               {
                  for(let cat=0; cat<filterCats.length; cat++)
                  {
                     if(_.includes(lmObj.categories, filterCats[cat]))
                     {
                        placeStyle = 'hidden';
                        break;
                     }
                  }
               }
            }

            if(placeStyle == 'visible')
            {
               if(uFP_filterNoKeyword || uFP_filterKeyword)
               {
                  let venueName = lmObj.name.toLowerCase();
                  let 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.createdBy) placeStyle = 'hidden';
                  }
                  if(uFP_filterEditedBy === true)
                  {
                     if(filterNameID != lmObj.updatedBy) placeStyle = 'hidden';
                  }
               }
            }
            if(placeStyle == 'visible')
            {
               if(filterHideNameID != null)
               {
                  if(uFP_filterHideCreatedBy === true)
                  {
                     if(filterHideNameID == lmObj.createdBy) placeStyle = 'hidden';
                  }
                  if(uFP_filterHideEditedBy === true)
                  {
                     if(filterHideNameID == lmObj.updatedBy) placeStyle = 'hidden';
                  }
               }
            }

            if(uFP_invertFilters === true)
            {
               if(placeStyle == 'hidden') placeStyle = 'visible';
               else placeStyle = 'hidden';
            }
         }

         if((placeStyle == 'hidden') && (uFP_hidePURsForFilteredPlaces === true))
         {
            uroPURsToHide.push(lmObj.id);
         }
      }

      let geoID = uroVenueLayer.features[v].geometry.id;
      if(document.getElementById(geoID) !== null)
      {
         document.getElementById(geoID).style.visibility = placeStyle;
      }
   }

   uro_uFP[URO_UFP_OPTS.filterUneditable] = uroGetCBChecked('_cbFilterUneditablePlaceUpdates');
   uro_uFP[URO_UFP_OPTS.filterInsideManagedAreas] = uroGetCBChecked('_cbPURFilterInsideManagedAreas');
   uro_uFP[URO_UFP_OPTS.excludeMyAreas] = uroGetCBChecked('_cbPURExcludeUserArea');
   uro_uFP[URO_UFP_OPTS.filterLockRanked] = uroGetCBChecked('_cbFilterLockRankedPlaceUpdates');
   uro_uFP[URO_UFP_OPTS.filterFlagged] = uroGetCBChecked("_cbFilterFlaggedPUR");
   uro_uFP[URO_UFP_OPTS.filterNewPlace] = uroGetCBChecked("_cbFilterNewPlacePUR");
   uro_uFP[URO_UFP_OPTS.filterUpdatedDetails] = uroGetCBChecked("_cbFilterUpdatedDetailsPUR");
   uro_uFP[URO_UFP_OPTS.filterNewPhoto] = uroGetCBChecked("_cbFilterNewPhotoPUR");
   uro_uFP[URO_UFP_OPTS.filterMinPURAge] = uroGetCBChecked('_cbEnablePURMinAgeFilter');
   uro_uFP[URO_UFP_OPTS.filterMaxPURAge] = uroGetCBChecked('_cbEnablePURMaxAgeFilter');
   uro_uFP[URO_UFP_OPTS.invertPURFilters] = uroGetCBChecked('_cbInvertPURFilters');
   uro_uFP[URO_UFP_OPTS.filterHighSeverity] = uroGetCBChecked('_cbPURFilterHighSeverity');
   uro_uFP[URO_UFP_OPTS.filterMedSeverity] = uroGetCBChecked('_cbPURFilterMediumSeverity');
   uro_uFP[URO_UFP_OPTS.filterLowSeverity] = uroGetCBChecked('_cbPURFilterLowSeverity');
   uro_uFP[URO_UFP_OPTS.leavePURGeos] = uroGetCBChecked('_cbLeavePURGeos');
   uro_uFP[URO_UFP_OPTS.filterCFPhone] = uroGetCBChecked('_cbPURFilterCFPhone');
   uro_uFP[URO_UFP_OPTS.filterCFName] = uroGetCBChecked('_cbPURFilterCFName');
   uro_uFP[URO_UFP_OPTS.filterCFEntryExitPoints] = uroGetCBChecked('_cbPURFilterCFEntryExitPoints');
   uro_uFP[URO_UFP_OPTS.filterCFOpeningHours] = uroGetCBChecked('_cbPURFilterCFOpeningHours');
   uro_uFP[URO_UFP_OPTS.filterCFAliases] = uroGetCBChecked('_cbPURFilterCFAliases');
   uro_uFP[URO_UFP_OPTS.filterCFServices] = uroGetCBChecked('_cbPURFilterCFServices');
   uro_uFP[URO_UFP_OPTS.filterCFGeometry] = uroGetCBChecked('_cbPURFilterCFGeometry');
   uro_uFP[URO_UFP_OPTS.filterCFHouseNumber] = uroGetCBChecked('_cbPURFilterCFHouseNumber');
   uro_uFP[URO_UFP_OPTS.filterCFCategories] = uroGetCBChecked('_cbPURFilterCFCategories');
   uro_uFP[URO_UFP_OPTS.filterCFDescription] = uroGetCBChecked('_cbPURFilterCFDescription');

   uro_uFP[URO_UFP_OPTS.filterOnCFs] = (uro_uFP[URO_UFP_OPTS.filterCFPhone] || uro_uFP[URO_UFP_OPTS.filterCFName] || uro_uFP[URO_UFP_OPTS.filterCFEntryExitPoints] || uro_uFP[URO_UFP_OPTS.filterCFOpeningHours]);
   uro_uFP[URO_UFP_OPTS.filterOnCFs] = (uro_uFP[URO_UFP_OPTS.filterOnCFs] || uro_uFP[URO_UFP_OPTS.filterCFAliases] || uro_uFP[URO_UFP_OPTS.filterCFServices] || uro_uFP[URO_UFP_OPTS.filterCFGeometry]);
   uro_uFP[URO_UFP_OPTS.filterOnCFs] = (uro_uFP[URO_UFP_OPTS.filterOnCFs] || uro_uFP[URO_UFP_OPTS.filterCFHouseNumber] || uro_uFP[URO_UFP_OPTS.filterCFCategories] || uro_uFP[URO_UFP_OPTS.filterCFDescription]);

   uro_uFP[URO_UFP_OPTS.thresholdMinPURDays] = uroGetElmValue('_inputPURFilterMinDays');
   uro_uFP[URO_UFP_OPTS.thresholdMaxPURDays] = uroGetElmValue('_inputPURFilterMaxDays');
   uro_uFP[URO_UFP_OPTS.isLoggedIn] = W.loginManager.isLoggedIn();
   uro_uFP[URO_UFP_OPTS.userRank] = W.loginManager.user.attributes.rank;

   uro_uFP[URO_UFP_OPTS.filterInsideManagedAreas] = uro_uFP[URO_UFP_OPTS.filterInsideManagedAreas] && (uroGetManagedAreas() !== 0);
   if(uroGetCBChecked('_cbPURExcludeUserArea') == true)
   {
      uroIgnoreAreasUserID = W.loginManager.user.attributes.id;
   }

   uroPrepForFilterPlaceMarker(URO_TMARKER.PUR, uFP_masterEnable);
   uroPrepForFilterPlaceMarker(URO_TMARKER.PPUR, uFP_masterEnable);
   uroPrepForFilterPlaceMarker(URO_TMARKER.RPUR, uFP_masterEnable);

   uroPerformanceMonitoring(pmFunction, pmTStart);
}
function uroPrepForFilterPlaceMarker(markerType, masterEnable)
{
   let markerLayer;
   if(markerType == URO_TMARKER.PUR)
   {
      markerLayer = W.map.getLayerByName("place_updates");
   }
   else if(markerType == URO_TMARKER.PPUR)
   {
      markerLayer = W.map.getLayerByName("PARKING_PLACE_UPDATES");
   }
   else if(markerType == URO_TMARKER.RPUR)
   {
      markerLayer = W.map.getLayerByName("RESIDENTIAL_PLACE_UPDATES");
   }
   else
   {
      markerLayer = null;
   }

   if((markerLayer !== null) && (markerLayer.getVisibility() === true))
   {
      let pu;
      let mObj;
      let vObj;   
      let idList = uroGetMarkerIDs(markerType);
      for(pu of idList)
      {
         mObj = uroGetMarker(markerType, pu);
         if(mObj !== null)
         {
            vObj = W.model.venues.objects[pu];
            uroFilterPlaceMarker(mObj, vObj, masterEnable);
         }
      }
   }
}
function uroGetClosestSegmentToPoint(p)
{
   let retval = null;
   if(W.map.getZoom() >= 16)
   {
      let minDist = 99999999;
      for(let s in W.model.segments.objects)
      {
         if(W.model.segments.objects.hasOwnProperty(s))
         {
            let dist = W.model.segments.getObjectById(s).attributes.geometry.distanceTo(p);
            if(dist < minDist)
            {
               minDist = dist;
               retval = s;
            }
         }
      }
   }

   return retval;
}
function uroIsCamSpeedValid(camObj)
{
   let retval = true;

   let cPoint = camObj.attributes.geometry.getCentroid();
   let nSeg = uroGetClosestSegmentToPoint(cPoint);
   if(nSeg !== null)
   {
      let fwdSpeed = W.model.segments.getObjectById(nSeg).attributes.fwdMaxSpeed;
      let revSpeed = W.model.segments.getObjectById(nSeg).attributes.revMaxSpeed;
      let camSpeed = camObj.attributes.speed;
      if(W.model.isImperial == true)
      {
         fwdSpeed = Math.round(fwdSpeed / 1.609);
         revSpeed = Math.round(revSpeed / 1.609);
         camSpeed = Math.round(camSpeed / 1.609);
      }
      if((camSpeed !== fwdSpeed) && (camSpeed !== revSpeed))
      {
         retval = false;
      }
   }

   return retval;
}
function uroFilterCameras()
{
   let pmTStart = performance.now();
   let pmFunction = "uroFilterCameras";

   if(uroFilterPreamble() === false)
   {
      return;
   }

   if(uroMouseIsDown === false) W.map.camerasLayer.redraw();
   if(uroIsFilteringEnabled(false) === true)
   {
      uroUpdateEditorList(W.model.cameras.objects, '_selectCameraUserID', true, true, false, false);
      let tbUserName = uroGetElmValue('_textCameraEditor');
      let selector = document.getElementById('_selectCameraUserID');
      let filterNameID = null;
      if(selector.selectedIndex > 0)
      {
         let selUserName = document.getElementById('_selectCameraUserID').selectedOptions[0].innerHTML;
         if(selUserName == tbUserName)
         {
            filterNameID = document.getElementById('_selectCameraUserID').selectedOptions[0].value;
         }
      }
      filterNameID = uroGetUserID(filterNameID, tbUserName);

      let isChecked_cbShowOnlyCamsCreatedBy = uroGetCBChecked('_cbShowOnlyCamsCreatedBy');
      let isChecked_cbShowOnlyCamsEditedBy = uroGetCBChecked('_cbShowOnlyCamsEditedBy');
      let isChecked_cbShowOnlyMyCams = uroGetCBChecked('_cbShowOnlyMyCams');
      let isChecked_cbShowWorldCams = uroGetCBChecked('_cbShowWorldCams');
      let isChecked_cbShowUSACams = uroGetCBChecked('_cbShowUSACams');
      let isChecked_cbShowNonWorldCams = uroGetCBChecked('_cbShowNonWorldCams');
      let isChecked_cbShowSpeedCams = uroGetCBChecked('_cbShowSpeedCams');
      let isChecked_cbShowRedLightCams = uroGetCBChecked('_cbShowRedLightCams');
      let isChecked_cbShowDummyCams = uroGetCBChecked('_cbShowDummyCams');
      let isChecked_cbShowIfNoSpeedSet = uroGetCBChecked('_cbShowIfNoSpeedSet');
      let isChecked_cbShowIfSpeedSet = uroGetCBChecked('_cbShowIfSpeedSet');
      let isChecked_cbShowIfInvalidSpeedSet = uroGetCBChecked('_cbShowIfInvalidSpeedSet');
      let isChecked_cbShowRLCIfNoSpeedSet = uroGetCBChecked('_cbShowRLCIfNoSpeedSet');
      let isChecked_cbShowRLCIfNonZeroSpeedSet = uroGetCBChecked('_cbShowRLCIfNonZeroSpeedSet');
      let isChecked_cbShowRLCIfZeroSpeedSet = uroGetCBChecked('_cbShowRLCIfZeroSpeedSet');
      let isChecked_cbHideCreatedByMe = uroGetCBChecked('_cbHideCreatedByMe');
      let isChecked_cbHideCreatedByRank0 = uroGetCBChecked('_cbHideCreatedByRank0');
      let isChecked_cbHideCreatedByRank1 = uroGetCBChecked('_cbHideCreatedByRank1');
      let isChecked_cbHideCreatedByRank2 = uroGetCBChecked('_cbHideCreatedByRank2');
      let isChecked_cbHideCreatedByRank3 = uroGetCBChecked('_cbHideCreatedByRank3');
      let isChecked_cbHideCreatedByRank4 = uroGetCBChecked('_cbHideCreatedByRank4');
      let isChecked_cbHideCreatedByRank5 = uroGetCBChecked('_cbHideCreatedByRank5');
      let isChecked_cbHideUpdatedByMe = uroGetCBChecked('_cbHideUpdatedByMe');
      let isChecked_cbHideUpdatedByRank0 = uroGetCBChecked('_cbHideUpdatedByRank0');
      let isChecked_cbHideUpdatedByRank1 = uroGetCBChecked('_cbHideUpdatedByRank1');
      let isChecked_cbHideUpdatedByRank2 = uroGetCBChecked('_cbHideUpdatedByRank2');
      let isChecked_cbHideUpdatedByRank3 = uroGetCBChecked('_cbHideUpdatedByRank3');
      let isChecked_cbHideUpdatedByRank4 = uroGetCBChecked('_cbHideUpdatedByRank4');
      let isChecked_cbHideUpdatedByRank5 = uroGetCBChecked('_cbHideUpdatedByRank5');
      let isChecked_cbHideCWLCams = uroGetCBChecked('_cbHideCWLCams');
      let isChecked_cbHighlightInsteadOfHideCams = uroGetCBChecked('_cbHighlightInsteadOfHideCams');
      let isChecked_InvertFiltere = uroGetCBChecked('_cbInvertCamFilters');

      let nCameras = W.map.getLayerByUniqueName('speed_cameras').features.length;
      for (let i = 0; i < nCameras; ++i)
      {
         let uroCamUpdater = '';
         let uroCamUpdaterRank = -1;
         let uroCamCreator = '';
         let uroCamCreatorRank = -1;
         let wf = W.map.getLayerByUniqueName('speed_cameras').features[i].attributes.wazeFeature;
         // When a camera is selected, the alignment/positioning UI elements get added to features[].
         // As these elements aren't camera markers and therefore have no attributes, we need to
         // ignore them to prevent errors in the filtering code below...
         if(wf !== undefined)
         {
            let uroCam = wf._wmeObject;
            let uroCamStyle = 'visible';

            if(uroCam.attributes.createdBy !== null)
            {
               if(W.model.users.objects[uroCam.attributes.createdBy] != null)
               {
                  uroCamCreator = W.model.users.objects[uroCam.attributes.createdBy].attributes.userName;
                  uroCamCreatorRank = W.model.users.objects[uroCam.attributes.createdBy].attributes.rank;
               }
            }

            if(uroCam.attributes.updatedBy !== null)
            {
               if(W.model.users.objects[uroCam.attributes.updatedBy] != null)
               {
                  uroCamUpdater = W.model.users.objects[uroCam.attributes.updatedBy].attributes.userName;
                  uroCamUpdaterRank = W.model.users.objects[uroCam.attributes.updatedBy].attributes.rank;
               }
            }

            let 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))
            {
               let posWorld = uroCamCreator.indexOf('world_');
               let 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_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_cbShowIfInvalidSpeedSet === false)
               {
                  if(uroIsCamSpeedValid(uroCam) === false)
                  {
                     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';

            if(isChecked_InvertFiltere === true)
            {
               if(uroCamStyle == "hidden")
               {
                  uroCamStyle = "";
               }
               else
               {
                  uroCamStyle = "hidden";
               }
            }

            let uroCamGeometryID =  W.map.getLayerByUniqueName('speed_cameras').features[i].geometry.id;
            let svgElm = document.getElementById(uroCamGeometryID);

            if(svgElm !== null)
            {
               let origImage;
               if(uroCamStyle == "hidden")
               {
                  if(isChecked_cbHighlightInsteadOfHideCams === true)
                  {
                     // set the "highlight" camera image here...
                     let hrefImage = svgElm.getAttribute("xlink:href");
                     origImage = svgElm.getAttribute("origImage");
                     if((hrefImage === origImage) || (origImage === null))
                     {
                        svgElm.setAttribute("origImage", hrefImage);
                        svgElm.setAttribute("xlink:href", uroHighlightedCameraImages[(uroCamType-2)]);

                        svgElm.addEventListener("mouseover", uroMarkerMouseOver, false);
                        svgElm.addEventListener("mouseout", uroMarkerMouseOut, false);
                     }
                  }
                  else
                  {
                     svgElm.remove();
                  }
               }
               else
               {
                  // restore the original camera image here...
                  if(svgElm.getAttribute("origImage") !== null)
                  {
                     origImage = svgElm.getAttribute("origImage");
                     svgElm.setAttribute("xlink:href", origImage);
                     svgElm.removeAttribute("origImage");
                  }
               }
            }
         }
      }
      uroCamLayerChanged();
   }
   uroPerformanceMonitoring(pmFunction, pmTStart);
}
function uroFilterMapComments()
{
   let pmTStart = performance.now();
   let pmFunction = "uroFilterMapComments";

   if(uroFilterPreamble() === false) return;

   let uFURs_masterEnable = uroIsFilteringEnabled(false);
   let filterDescMustBePresent = uroGetCBChecked('_cbMCDescriptionMustBePresent');
   let filterDescMustBeAbsent = uroGetCBChecked('_cbMCDescriptionMustBeAbsent');
   let filterKeywordMustBePresent = uroGetCBChecked('_cbMCEnableKeywordMustBePresent');
   let filterKeywordMustBeAbsent = uroGetCBChecked('_cbMCEnableKeywordMustBeAbsent');
   let filterMyFollowed = uroGetCBChecked('_cbMCHideMyFollowed');
   let filterMyUnfollowed = uroGetCBChecked('_cbMCHideMyUnfollowed');
   let filterRoadworks = uroGetCBChecked('_cbMCFilterRoadworks');
   let filterConstruction = uroGetCBChecked('_cbMCFilterConstruction');
   let filterClosure = uroGetCBChecked('_cbMCFilterClosure');
   let filterEvent = uroGetCBChecked('_cbMCFilterEvent');
   let filterNote = uroGetCBChecked('_cbMCFilterNote');
   let filterWSLM = uroGetCBChecked('_cbMCFilterWSLM');
   let filterBOG = uroGetCBChecked('_cbMCFilterBOG');
   let filterDifficult = uroGetCBChecked('_cbMCFilterDifficult');
   let invertFilters = uroGetCBChecked('_cbInvertMCFilter');
   let keywordPresent = uroGetElmValue('_textMCKeywordPresent');
   let keywordAbsent = uroGetElmValue('_textMCKeywordAbsent');
   let caseInsensitive = uroGetCBChecked('_cbMCCaseInsensitive');
   let filterCommentsMustBePresent = uroGetCBChecked('_cbMCCommentsMustBePresent');
   let filterCommentsMustBeAbsent = uroGetCBChecked('_cbMCCommentsMustBeAbsent');

   let filterExpiryMustBePresent = uroGetCBChecked('_cbMCExpiryMustBePresent');
   let filterExpiryMustBeAbsent = uroGetCBChecked('_cbMCExpiryMustBeAbsent');
   let filterByCreatorEnable = uroGetCBChecked('_cbMCCreatorIDFilter');
   let filterL1 = uroGetCBChecked('_cbHideMCRank0');
   let filterL2 = uroGetCBChecked('_cbHideMCRank1');
   let filterL3 = uroGetCBChecked('_cbHideMCRank2');
   let filterL4 = uroGetCBChecked('_cbHideMCRank3');
   let filterL5 = uroGetCBChecked('_cbHideMCRank4');
   let filterL6 = uroGetCBChecked('_cbHideMCRank5');
   
   let filterWRCMC = uroGetCBChecked('_cbHideWRCMCs');
   
   let selectorCreator = document.getElementById('_selectMCCreatorID');

   if(filterByCreatorEnable === false)
   {
      while(selectorCreator.options.length > 0)
      {
         selectorCreator.options.remove(0);
      }
   }
   let creatorUser = null;
   if(filterByCreatorEnable === true)
   {
      if(selectorCreator.options.length === 0)
      {
         uroUpdateEditorList(W.model.mapComments.objects, '_selectMCCreatorID', true, false, false, false);
      }
      if(selectorCreator.selectedOptions[0] != null)
      {
         creatorUser = parseInt(selectorCreator.selectedOptions[0].value);
      }
   }

   for (let mcIdx = 0; mcIdx < uroMCLayer.features.length; mcIdx++)
   {
      {
         let mcObj = uroMCLayer?.features[mcIdx]?.attributes?.wazeFeature?._wmeObject;
         if(mcObj !== undefined)
         {
            let desc = '';
            if(mcObj.attributes.subject !== null) desc += mcObj.attributes.subject.replace(/<\/?[^>]+(>|$)/g, "");
            if(mcObj.attributes.body !== null) desc += mcObj.attributes.body.replace(/<\/?[^>]+(>|$)/g, "");
            let nComments = mcObj.attributes.conversation.length;
            if(nComments > 0)
            {
               for(let cIdx=0; cIdx < nComments; cIdx++)
               {
                  desc += mcObj.attributes.conversation[cIdx].text.replace(/<\/?[^>]+(>|$)/g, "");
               }
            }

            let mcStyle = 'visible';
            if(uroIsOnIgnoreList(mcObj.attributes.id)) mcStyle = 'hidden';
            
            if(uFURs_masterEnable === true)
            {
               let ukroadworks_ur = false;
               let construction_ur = false;
               let closure_ur = false;
               let event_ur = false;
               let note_ur = false;
               let wslm_ur = false;
               let bog_ur = false;
               let difficult_ur = false;

               let filterByNotIncludedKeyword = false;
               let filterByIncludedKeyword = true;

               let customType = uroGetCustomType(null, "mc", 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;

               let rank = mcObj.attributes.lockRank;
               let expiry = mcObj.attributes.endDate;                  

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

                  if(filterCommentsMustBePresent === true)
                  {
                     if(nComments === 0) mcStyle = 'hidden';
                  }
                  if(filterCommentsMustBeAbsent === true)
                  {
                     if(nComments > 0) mcStyle = 'hidden';
                  }

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

                  filterByNotIncludedKeyword = (filterByNotIncludedKeyword && filterKeywordMustBeAbsent);
                  filterByIncludedKeyword = (filterByIncludedKeyword && filterKeywordMustBePresent);
                  if(filterByNotIncludedKeyword || filterByIncludedKeyword)
                  {
                     mcStyle = 'hidden';
                  }

               }

               //lock rank
               if(mcStyle == 'visible')
               {
                  if((filterL1 === true) && (rank == 0)) mcStyle = 'hidden';
                  if((filterL2 === true) && (rank == 1)) mcStyle = 'hidden';
                  if((filterL3 === true) && (rank == 2)) mcStyle = 'hidden';
                  if((filterL4 === true) && (rank == 3)) mcStyle = 'hidden';
                  if((filterL5 === true) && (rank == 4)) mcStyle = 'hidden';
                  if((filterL6 === true) && (rank == 5)) mcStyle = 'hidden';
               }

               // expiry
               if(mcStyle == 'visible')
               {
                  if((filterExpiryMustBePresent === true) && (expiry === null)) mcStyle = 'hidden';
                  if((filterExpiryMustBeAbsent === true) && (expiry != null)) mcStyle = 'hidden';
               }

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

               if(mcStyle == 'visible')
               {
                  if(creatorUser !== null)
                  {
                     if(mcObj.attributes.createdBy != creatorUser) mcStyle = 'hidden';
                  }
                  
                  if(filterWRCMC === true)
                  {
                     if(mcObj.attributes.createdBy == 304740435) 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';
                  }
               }
            }

            let geoID = uroMCLayer.features[mcIdx].geometry.id;
            if(document.getElementById(geoID) !== null)
            {
               document.getElementById(geoID).style.visibility = mcStyle;
            }
         }
      }
   }
   uroPerformanceMonitoring(pmFunction, pmTStart);
}
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;
   }

   let nextBFQueueObj = uroBackfillQueue.shift();

   let lon = parseFloat(nextBFQueueObj.lon);
   let lat = parseFloat(nextBFQueueObj.lat);
   let blockSize = parseFloat(nextBFQueueObj.blockSize);
   uroAddLog('Backfill square '+lon+','+lat);
   let backfillReq = new XMLHttpRequest();
   backfillReq.onreadystatechange = function ()
   {
      if (backfillReq.readyState == 4)
      {
         uroAddLog('backfill data request, response '+backfillReq.status+' received');
         if (backfillReq.status == 200)
         {
            let tResp = JSON.parse(backfillReq.responseText);
            let 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');
            }

            let backfilled = 0;
            for(let i=0; i<urCount; i++)
            {
               let urID = tResp.mapUpdateRequests.objects[i].id;
               if(W.model.mapUpdateRequests.objects[urID] === undefined)
               {
                  let newUR = require('Waze/Feature/Vector/UpdateRequest');
                  let tUR = new newUR(tResp.mapUpdateRequests.objects[i]);
                  let 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;
                  let tReqBounds = new OpenLayers.Geometry.Polygon();
                  let 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();
      }
   };
   let tURL = 'https://' + document.location.host;
   tURL += W.Config.api_base;
   tURL += '/Features?language=en&mapUpdateRequestFilter=3';
   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;
   }

   let 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...');

   let subSize = 0.1;
   let vpWidth = W.map.getExtent().getWidth();
   let vpHeight = W.map.getExtent().getHeight();
   let vpCentre = W.map.getCenter();
   let vpLL = new OpenLayers.LonLat();
   let 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(let bfLat = vpLL.lat; bfLat <= vpUR.lat; bfLat += subSize)
   {
      for(let bfLon = vpLL.lon; bfLon <= vpUR.lon; bfLon += subSize)
      {
         uroBackfillQueue.push(new uroBackfillQueueObj(bfLon, bfLat, subSize));
      }
   }
   uroURBackfill_GetData();
}
function uroGetManagedAreas()
{
   uroManagedAreas = [];
   uroIgnoreAreasUserID = null;

   for(let maObj in W.model.managedAreas.objects)
   {
      if(W.model.managedAreas.objects.hasOwnProperty(maObj))
      {
         uroManagedAreas.push(W.model.managedAreas.objects[maObj]);
      }
   }
   return uroManagedAreas.length;
}
function uroCheckGeometryWithinManagedAreas(geo)
{
   let retval = false;
   let ignoreUserMA = false;

   // If we're ignoring the user's managed area, then we first check to see if
   // the geopoint lies within that - if so then we can skip checking all the
   // other areas in the list...
   if(uroIgnoreAreasUserID !== null)
   {
      for(let uma = 0; uma < uroManagedAreas.length; ++uma)
      {
         if(uroIgnoreAreasUserID == uroManagedAreas[uma].userID)
         {
            if(uroManagedAreas[uma].geometry.containsPoint !== undefined)
            {
               if(uroManagedAreas[uma].geometry.containsPoint(geo) == true)
               {
                  ignoreUserMA = true;
               }
            }
            break;
         }
      }
   }

   // Point either isn't within the user's area, or we're not ignoring it, so
   // check the rest of the areas in the list
   if(ignoreUserMA == false)
   {
      for(let ma = 0; ma < uroManagedAreas.length; ++ma)
      {
         if(uroIgnoreAreasUserID != uroManagedAreas[ma].userID)
         {
            if(uroManagedAreas[ma].geometry.containsPoint !== undefined)
            {
               if(uroManagedAreas[ma].geometry.containsPoint(geo) == true)
               {
                  retval = true;
                  break;
               }
            }
         }
      }
   }

   return retval;
}

function uroGetURDriveGeoms()
{
   let retval = [];

   for (let urobj in W.model.mapUpdateRequests.objects)
   {
      if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
      {
         let ureq = W.model.mapUpdateRequests.objects[urobj];
         let ureqID = ureq.attributes.id;

         let hasGeo = false;
         let thisRet = [];
         thisRet.push(ureqID);
         thisRet.push(null);
         thisRet.push([]);

         let latMin = 9999;
         let latMax = -9999;
         let lonMin = 9999;
         let lonMax = -9999;

         let urs = W.model.updateRequestSessions.objects[ureqID];
         if((urs !== undefined) && (urs.attributes.driveGeometry !== undefined))
         {
            let cPairs = [];
            for(let i = 0; i < urs.attributes.driveGeometry.coordinates.length; ++i)
            {
               for(let j = 0; j < urs.attributes.driveGeometry.coordinates[i].length; ++j)
               {
                  if((i === 0) || (j > 0))
                  {
                     let coords = urs.attributes.driveGeometry.coordinates[i][j];
                     cPairs.push(coords);

                     if(coords[0] > lonMax)
                     {
                        lonMax = coords[0];
                     }
                     if(coords[0] < lonMin)
                     {
                        lonMin = coords[0];
                     }
                     if(coords[1] > latMax)
                     {
                        latMax = coords[1];
                     }
                     if(coords[1] < latMin)
                     {
                        latMin = coords[1];
                     }

                     hasGeo = true;
                  }
               }
            }
            let bbox = [];
            bbox.push(lonMin);
            bbox.push(lonMax);
            bbox.push(latMin);
            bbox.push(latMax);
            thisRet.push(bbox);
            thisRet.push(cPairs);
         }

         if(hasGeo === true)
         {
            retval.push(thisRet);
         }
      }
   }
   return retval;
}
function uroCompareDriveGeos(geoA, geoB)
{
   const matchLength = 5;
   let retval = false;

   if((geoA.length >= matchLength) && (geoB.length >= matchLength))
   {
      for(let i = 0; i < (geoA.length - matchLength); ++i)
      {
         for(let j = 0; j < (geoB.length - matchLength); ++j)
         {
            if((geoA[i][0] == geoB[j][0]) && (geoA[i][1] == geoB[j][1]))
            {
               retval = true;
               for(let k = 1; k < matchLength; ++k)
               {
                  if((geoA[i+k][0] != geoB[j+k][0]) || (geoA[i+k][1] != geoB[j+k][1]))
                  {
                     retval = false;
                     break;
                  }
               }
            }
            if(retval === true)
            {
               break;
            }
         }

         if(retval === true)
         {
            break;
         }
      }
   }
   return retval;
}
function uroCompareDriveBBoxes(bbA, bbB)
{
   let retval = true;

   if
   (
      (bbA[0] > bbB[1]) ||
      (bbA[1] < bbB[0]) ||
      (bbA[2] > bbB[3]) ||
      (bbA[3] < bbB[2])
   )
   {
      retval = false;
   }

   return retval;
}
function uroGetURDupes()
{
   uroURDupes = [];

   // To determine which URs are duplicates of one another (i.e. have been raised by the same user
   // during the same section of a journey), we first compare the geometries of the drive tracks
   // attached to any URs which have them - as this is based on the users GPS position rather than
   // the WME map data, it makes it vanishingly unlikely that any two users would have identical
   // GPS positions (especially given the level of accuracy to which the track points are stored)
   // even if they were driving exactly the same route at the same speed, in the same lane etc.
   //
   // To accelerate this geometry comparison, we start by performing a simple bounding box overlap
   // check for the two geometries under consideration - if there's no overlap then there can't be
   // any geometry match, so no need to continue onto the more detailed comparision of the GPS
   // tracks themselves...
   let driveGeos = uroGetURDriveGeoms();
   if(driveGeos.length > 1)
   {
      for(let i = 0; i < (driveGeos.length - 1); ++i)
      {
         if(driveGeos[i].length !== 5)
         {
            driveGeos[i][1] = false;
         }
         else
         {
            for(let j = (i + 1); j < driveGeos.length; ++j)
            {
               if(driveGeos[j].length === 5)
               {
                  let geoMatch = uroCompareDriveBBoxes(driveGeos[i][3], driveGeos[j][3]);
                  if(geoMatch === true)
                  {
                     geoMatch = uroCompareDriveGeos(driveGeos[i][4], driveGeos[j][4]);
                     if(geoMatch === true)
                     {
                        driveGeos[i][2].push(driveGeos[j][0]);
                        driveGeos[j][2].push(driveGeos[i][0]);
                     }
                  }
               }
            }
         }
      }

      for(let i = 0; i < driveGeos.length; ++i)
      {
         if(driveGeos[i][2].length > 0)
         {
            let res = [];
            res.push(driveGeos[i][0]);
            res.push(driveGeos[i][2]);
            uroURDupes.push(res);
         }
      }
   }

   // Once we've done the initial drive track comparision, uroURDupes will contain a list of
   // all the URs which were matched up based on that.  However, as the track comparision is
   // inherently limited by the amount of track data included with each UR, this initial
   // comparison may mean some more widely spaced URs aren't flagged as duplicates simply
   // because there's insufficient overlap between their tracks, even though there may have
   // been further URs dropped inbetween to which they were matched.
   //
   // e.g. if a user drops 4 URs along a section of their journey, spaced such that each of
   // the GPS tracks generates a comparison match with the UR either side of it, but no
   // further than that, we would get:
   //
   // UR A......UR B.......UR C......UR D
   //
   // Matches: A to B, B to A & C, C to B & D, D to C
   // 
   // Note how, although we know all of these URs are in fact duplicates, and can infer this
   // from seeing that e.g. A is flagged only as a duplicate of B, however as B is flagged as
   // a duplicate of A & C, and C is flagged as a duplicate of B & D, A MUST therefore be a
   // duplicate of B, C & D...
   //
   // To fix this, we now run a merging pass over each of the entries in uroURDupes - for
   // each entry we see if its ID appears in any of the other entries as a duplicate, and
   // if so we merge the duplicates for both entries.

   for(let i = 0; i < uroURDupes.length; ++i)
   {
      let urID = uroURDupes[i][0];
      for(let j = 0; j < uroURDupes.length; ++j)
      {
         if(i != j)
         {
            if(uroURDupes[j][1].includes(urID) === true)
            {
               // https://stackoverflow.com/questions/1584370/how-to-merge-two-arrays-in-javascript-and-de-duplicate-items
               let mergedDupes = [...new Set([...uroURDupes[i][1], ...uroURDupes[j][1]])];
               uroURDupes[i][1] = mergedDupes;
               uroURDupes[j][1] = mergedDupes;
            }
         }
      }
   }
}
function uroFilterURs()
{
   let pmTStart = performance.now();
   let pmFunction = "uroFilterURs";

   if(uroUserID === -1) 
   {
      return;
   }

   if(uroInhibitURFiltering === true)
   {
      return;
   }


   // compatibility fix for URComments - based on code supplied by RickZabel
   let hasActiveURFilters = false;
   if(uroIsFilteringEnabled(false) === true)
   {
      let urTabInputs = uroCtrlTabs[URO_TABS_ID.URS][URO_TABS_FIELD.TABBODY].getElementsByTagName('input');
      for(let loop = 0; loop < urTabInputs.length; loop++)
      {
         if(urTabInputs[loop].type == 'checkbox')
         {
            let 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();
   let selectorResolver = document.getElementById('_selectURResolverID');
   let 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;
   }
   let commenterUser = null;
   if(uroGetCBChecked('_cbURUserIDFilter') === true)
   {
      if(selectorCommentUser.options.length === 0)
      {
         uroUpdateEditorList(W.model.updateRequestSessions.objects, '_selectURUserID', false, false, false, true);
      }
      if(selectorCommentUser.selectedOptions[0] != null)
      {
         commenterUser = parseInt(selectorCommentUser.selectedOptions[0].value);
      }
   }
   let resolverUser = null;
   if(uroGetCBChecked('_cbURResolverIDFilter') === true)
   {
      if(selectorResolver.options.length === 0)
      {
         uroUpdateEditorList(W.model.mapUpdateRequests.objects, '_selectURResolverID', false, false, true, false);
      }
      if(selectorResolver.selectedOptions[0] != null)
      {
         resolverUser = parseInt(selectorResolver.selectedOptions[0].value);
      }
   }
   uroCustomMarkerList = [];
   uroGetURDupes();

   let uFURs_masterEnable = uroIsFilteringEnabled(false);
   let filterOutsideEditableArea = uroGetCBChecked('_cbURFilterOutsideArea');
   let filterInsideManagedAreas = uroGetCBChecked('_cbURFilterInsideManagedAreas');
   let filterSolved = uroGetCBChecked('_cbFilterSolved');
   let filterUnidentified = uroGetCBChecked('_cbFilterUnidentified');
   let filterClosed = uroGetCBChecked('_cbFilterClosedUR');
   let filterOpen = uroGetCBChecked('_cbFilterOpenUR');
   let filterDescMustBePresent = uroGetCBChecked('_cbURDescriptionMustBePresent');
   let filterDescMustBeAbsent = uroGetCBChecked('_cbURDescriptionMustBeAbsent');
   let filterKeywordMustBePresent = uroGetCBChecked('_cbEnableKeywordMustBePresent');
   let filterKeywordMustBeAbsent = uroGetCBChecked('_cbEnableKeywordMustBeAbsent');
   let filterMinURAge = uroGetCBChecked('_cbEnableMinAgeFilter');
   let filterMaxURAge = uroGetCBChecked('_cbEnableMaxAgeFilter');
   let filterMinComments = uroGetCBChecked('_cbEnableMinCommentsFilter');
   let filterMaxComments = uroGetCBChecked('_cbEnableMaxCommentsFilter');
   let filterReporterLastCommenter = uroGetCBChecked('_cbHideIfReporterLastCommenter');
   let filterReporterNotLastCommenter = uroGetCBChecked('_cbHideIfReporterNotLastCommenter');
   let filterHideAnyComments = uroGetCBChecked('_cbHideAnyComments');
   let filterHideNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
   let filterHideMyComments = uroGetCBChecked('_cbHideMyComments');
   let filterIfLastCommenter = uroGetCBChecked('_cbHideIfLastCommenter');
   let filterIfNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
   let filterCommentMinAge = uroGetCBChecked('_cbEnableCommentAgeFilter2');
   let filterCommentMaxAge = uroGetCBChecked('_cbEnableCommentAgeFilter');
   let filterUserID = uroGetCBChecked('_cbURUserIDFilter');
   let filterMyFollowed = uroGetCBChecked('_cbHideMyFollowed');
   let filterMyUnfollowed = uroGetCBChecked('_cbHideMyUnfollowed');

   let filterWazeAuto = uroGetCBChecked('_cbFilterWazeAuto');
   let filterRoadworks = uroGetCBChecked('_cbFilterRoadworks');
   let filterConstruction = uroGetCBChecked('_cbFilterConstruction');
   let filterClosure = uroGetCBChecked('_cbFilterClosure');
   let filterEvent = uroGetCBChecked('_cbFilterEvent');
   let filterNote = uroGetCBChecked('_cbFilterNote');
   let filterWSLM = uroGetCBChecked('_cbFilterWSLM');
   let filterBOG = uroGetCBChecked('_cbFilterBOG');
   let filterDifficult = uroGetCBChecked('_cbFilterDifficult');

   let filterIncorrectTurn = uroGetCBChecked('_cbFilterIncorrectTurn');
   let filterIncorrectAddress = uroGetCBChecked('_cbFilterIncorrectAddress');
   let filterIncorrectRoute = uroGetCBChecked('_cbFilterIncorrectRoute');
   let filterMissingRoundabout = uroGetCBChecked('_cbFilterMissingRoundabout');
   let filterGeneralError = uroGetCBChecked('_cbFilterGeneralError');
   let filterTurnNotAllowed = uroGetCBChecked('_cbFilterTurnNotAllowed');
   let filterIncorrectJunction = uroGetCBChecked('_cbFilterIncorrectJunction');
   let filterMissingBridgeOverpass = uroGetCBChecked('_cbFilterMissingBridgeOverpass');
   let filterWrongDrivingDirection = uroGetCBChecked('_cbFilterWrongDrivingDirection');
   let filterMissingExit = uroGetCBChecked('_cbFilterMissingExit');
   let filterMissingRoad = uroGetCBChecked('_cbFilterMissingRoad');
   let filterMissingLandmark = uroGetCBChecked('_cbFilterMissingLandmark');
   let filterNativeSpeedLimit = uroGetCBChecked('_cbFilterSpeedLimits');
   let filterBlockedRoad = uroGetCBChecked('_cbFilterBlockedRoad');
   let filterUndefined = uroGetCBChecked('_cbFilterUndefined');

   let invertURFilters = uroGetCBChecked('_cbInvertURFilter');
   let invertURStateFilters = uroGetCBChecked('_cbInvertURStateFilter');
   let noFilterTaggedURs = uroGetCBChecked('_cbNoFilterForTaggedURs');
   let noFilterURInURL = uroGetCBChecked('_cbNoFilterForURInURL');
   let showOnlyDupes = uroGetCBChecked('_cbURFilterDupes');

   let keywordPresent = uroGetElmValue('_textKeywordPresent');
   let keywordAbsent = uroGetElmValue('_textKeywordAbsent');
   let caseInsensitive = uroGetCBChecked('_cbCaseInsensitive');
   let thresholdMinAge = uroGetElmValue('_inputFilterMinDays');
   let thresholdMaxAge = uroGetElmValue('_inputFilterMaxDays');
   let thresholdMinComments = uroGetElmValue('_inputFilterMinComments');
   let thresholdMaxComments = uroGetElmValue('_inputFilterMaxComments');
   let thresholdMaxCommentAge = uroGetElmValue('_inputFilterCommentDays');
   let thresholdMinCommentAge = uroGetElmValue('_inputFilterCommentDays2');
   let ignoreOtherEditorComments = uroGetCBChecked('_cbIgnoreOtherEditorComments');
   let urcFilteringIsActive = false;
   let 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;
      }
   }

   filterInsideManagedAreas = filterInsideManagedAreas && (uroGetManagedAreas() !== 0);
   if(uroGetCBChecked('_cbURExcludeUserArea') == true)
   {
      uroIgnoreAreasUserID = W.loginManager.user.attributes.id;
   }

   for (let urobj in W.model.mapUpdateRequests.objects)
   {
      if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
      {
         let ureq = W.model.mapUpdateRequests.objects[urobj];
         let ureqID = ureq.attributes.id;

         let urStyle = 'visible';
         let inhibitFiltering = ((ureqID == uroSelectedURID) && (noFilterURInURL));

         let hasMyComments = false;
         let nComments = 0;
         let desc = ureq.attributes.description;
         let customType = uroGetCustomType(ureqID, URO_TMARKER.UR, desc);
         if(W.model.updateRequestSessions.objects[ureqID] != null)
         {
            nComments = W.model.updateRequestSessions.objects[ureqID].attributes.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;
            }
         }

         // check UR against current session ignore list...
         if(uroIsOnIgnoreList(ureqID)) urStyle = 'hidden';

         if((uFURs_masterEnable === true) && (inhibitFiltering === false))
         {
            let wazeauto_ur = false;
            let ukroadworks_ur = false;
            let construction_ur = false;
            let closure_ur = false;
            let event_ur = false;
            let note_ur = false;
            let wslm_ur = false;
            let bog_ur = false;
            let difficult_ur = false;

            let filterByNotIncludedKeyword = false;
            let 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';
            }

            if(filterInsideManagedAreas === true)
            {
               if(uroCheckGeometryWithinManagedAreas(ureq.attributes.geometry) === true) urStyle = 'hidden';
            }

            if(showOnlyDupes === true)
            {
               let isDupe = false;
               for(let i = 0; i < uroURDupes.length; ++i)
               {
                  if(uroURDupes[i][0] === ureqID)
                  {
                     isDupe = true;
                     break;
                  }
               }
               if(isDupe === false) 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)
                  {
                     let keywordIsPresentInDesc = uroKeywordPresent(desc,keywordPresent,caseInsensitive);
                     filterByIncludedKeyword = (filterByIncludedKeyword && (!keywordIsPresentInDesc));
                  }
                  if(filterKeywordMustBeAbsent === true)
                  {
                     let 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].attributes.comments.length;
                     let commentDaysOld = -1;


                     if(filterMinComments === true)
                     {
                        if(nComments < thresholdMinComments) urStyle = 'hidden';
                     }
                     if(filterMaxComments === true)
                     {
                        if(nComments > thresholdMaxComments) urStyle = 'hidden';
                     }


                     if(nComments > 0)
                     {
                        let reporterIsLastCommenter = false;
                        if(W.model.updateRequestSessions.objects[ureqID].attributes.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';

                           let userIsLastCommenter = false;
                           if(W.model.updateRequestSessions.objects[ureqID].attributes.comments[nComments-1].userID == uroUserID) userIsLastCommenter = true;

                           if(filterIfLastCommenter === true)
                           {
                              if(userIsLastCommenter === true) urStyle = 'hidden';
                           }
                           else if(filterIfNotLastCommenter === true)
                           {
                              if(userIsLastCommenter === false) urStyle = 'hidden';
                           }
                        }

                        let cidx;

                        if(ignoreOtherEditorComments === false)
                        {
                           commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].attributes.comments[nComments-1]);
                        }
                        else
                        {
                           for(cidx=0; cidx<nComments; cidx++)
                           {
                              let cObj = W.model.updateRequestSessions.objects[ureqID].attributes.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].attributes.comments[cidx].userID == commenterUser)
                              {
                                 urStyle = 'visible';
                                 break;
                              }
                           }
                        }

                        let commentText = '';
                        for(cidx=0; cidx<nComments; cidx++)
                        {
                           commentText += W.model.updateRequestSessions.objects[ureqID].attributes.comments[cidx].text;
                        }

                        if(filterKeywordMustBePresent === true)
                        {
                           let keywordIsPresentInComments = uroKeywordPresent(commentText,keywordPresent,caseInsensitive);
                           filterByIncludedKeyword = (filterByIncludedKeyword && (!keywordIsPresentInComments));
                        }
                        if(filterKeywordMustBeAbsent === true)
                        {
                           let 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].attributes.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))
         {
            let urMarker = uroGetMarker(URO_TMARKER.UR,urobj);
            if(urMarker !== null)
            {
               urMarker.element.style.visibility = urStyle;
            }
         }
         if(urStyle != 'hidden')
         {
            uroAddCustomMarkers(ureqID,URO_TMARKER.UR,customType, hasMyComments, nComments);
         }
      }
   }

   uroRenderCustomMarkers(URO_TMARKER.UR);
   uroPerformanceMonitoring(pmFunction, pmTStart);
}
function uroGetProblemTypes()
{
   uroKnownProblemTypeIDs = [];
   uroKnownProblemTypeNames = [];
   let tProblemList = I18n.lookup("problems.types");
   for(let tObj in tProblemList)
   {
      if(tObj !== undefined)
      {
         uroKnownProblemTypeIDs.push(parseInt(tObj));
         uroKnownProblemTypeNames.push(tProblemList[tObj].title);
      }
   }
}
function uroGetTS(day, month, year, hours, mins)
{
   let retval = new Date(0);
   retval.setDate(day);
   retval.setMonth(month - 1);
   retval.setYear(year);
   retval.setHours(hours);
   retval.setMinutes(mins);
   return retval.getTime();
}
function uroFilterProblems()
{
   let pmTStart = performance.now();
   let pmFunction = "uroFilterProblems";

   if(uroFilterPreamble() === false) return;
   let selector;

   if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === false) && (uroGetCBChecked('_cbMPClosedUserIDFilter') === false))
   {
      selector = document.getElementById('_selectMPUserID');
      while(selector.options.length > 0)
      {
         selector.options.remove(0);
      }
   }

   let solverUser = null;
   if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === true) || (uroGetCBChecked('_cbMPClosedUserIDFilter') === true))
   {
      selector = document.getElementById('_selectMPUserID');
      if(selector.options.length === 0)
      {
         uroUpdateEditorList(W.model.problems.objects, '_selectMPUserID', false, false, true, false);
      }
      if(selector.selectedOptions[0] != null)
      {
         solverUser = parseInt(selector.selectedOptions[0].value);
      }
   }

   let uFP_masterEnable = uroIsFilteringEnabled(false);
   let filter_OutsideEditableArea = uroGetCBChecked('_cbMPFilterOutsideArea');
   let filter_Solved = uroGetCBChecked('_cbMPFilterSolved');
   let filter_Unidentified = uroGetCBChecked('_cbMPFilterUnidentified');
   let filter_Closed = uroGetCBChecked('_cbMPFilterClosed');
   let filter_NotClosedUserID = uroGetCBChecked('_cbMPNotClosedUserIDFilter');
   let filter_ClosedUserID = uroGetCBChecked('_cbMPClosedUserIDFilter');
   let filter_Reopened = uroGetCBChecked('_cbMPFilterReopenedProblem');

   let filter_LowSeverity = uroGetCBChecked('_cbMPFilterLowSeverity');
   let filter_MediumSeverity = uroGetCBChecked('_cbMPFilterMediumSeverity');
   let filter_HighSeverity = uroGetCBChecked('_cbMPFilterHighSeverity');

   let filter_TurnProblems = uroGetCBChecked('_cbMPFilter_T200');

   let filterTypes = [];
   let i;
   for(i=0; i<uroKnownProblemTypeIDs.length; i++)
   {
      if(uroGetCBChecked('_cbMPFilter_T'+uroKnownProblemTypeIDs[i])) filterTypes.push(uroKnownProblemTypeIDs[i]);
   }
   let filter_TypeUnknown = uroGetCBChecked('_cbMPFilterUnknownProblem');

   let filter_TaggedElgin = uroGetCBChecked('_cbFilterElgin');
   let filter_TaggedTrafficCast = uroGetCBChecked('_cbFilterTrafficCast');
   let filter_TaggedTrafficMaster = uroGetCBChecked('_cbFilterTrafficMaster');
   let filter_TaggedCaltrans = uroGetCBChecked('_cbFilterCaltrans');
   let filter_TaggedTFL = uroGetCBChecked('_cbFilterTFL');

   let filter_Invert = uroGetCBChecked('_cbInvertMPFilter');

   let filter_StartDateEnabled = uroGetCBChecked('_cbMPFilterStartDate');
   let filter_EndDateEnabled = uroGetCBChecked('_cbMPFilterEndDate');
   let filter_EndDatePassed = uroGetCBChecked('_cbMPFilterEndDatePassed');

   let tsD = uroGetElmValue('_inputMPFilterStartDay');
   let tsM = uroGetElmValue('_inputMPFilterStartMonth');
   let tsY = uroGetElmValue('_inputMPFilterStartYear');
   let startDate = uroGetTS(tsD, tsM, tsY, 0, 0);

   tsD = uroGetElmValue('_inputMPFilterEndDay');
   tsM = uroGetElmValue('_inputMPFilterEndMonth');
   tsY = uroGetElmValue('_inputMPFilterEndYear');
   let endDate = uroGetTS(tsD, tsM, tsY, 0, 0);
   
   let nowTime = (new Date()).getTime();
   let addCustomMarkers = uroIsFilteringEnabled(true);

   for (let urobj in W.model.problems.objects)
   {
      if(W.model.problems.objects.hasOwnProperty(urobj))
      {
         let problem = W.model.problems.objects[urobj];
         let ureqID = problem.attributes.id;
         let desc = '';

         let problemStyle = 'visible';
         // check problem against current session ignore list...
         if(uroIsOnIgnoreList(ureqID)) problemStyle = 'hidden';

         if(problem.attributes.description != null)
         {
            desc = problem.attributes.description;
         }

         let customType = null;
         let elgin_mp = false;
         let trafficcast_mp = false;
         let trafficmaster_mp = false;
         let caltrans_mp = false;
         let tfl_mp = false;         
         if(addCustomMarkers === true)
         {
            customType = uroGetCustomType(ureqID, URO_TMARKER.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;
         }

         if(uFP_masterEnable === true)
         {
            if(filter_OutsideEditableArea === true)
            {
               if(problem.canEdit() === false)
               {
                  problemStyle = 'hidden';
               }
            }

            if(filter_EndDatePassed == true)
            {
               if(problem.attributes.endTime > nowTime)
               {
                  problemStyle = 'hidden';
               }
            }
            if(filter_StartDateEnabled == true)
            {
               let tStart = new Date(problem.attributes.startTime);
               tStart.setHours(0);
               tStart.setMinutes(0);
               tStart.setSeconds(0);
               tStart = tStart.getTime();
               if(tStart != startDate)
               {
                  problemStyle = 'hidden';
               }
            }
            if(filter_EndDateEnabled == true)
            {
               let tEnd = new Date(problem.attributes.endTime);
               tEnd.setHours(0);
               tEnd.setMinutes(0);
               tEnd.setSeconds(0);
               tEnd = tEnd.getTime();
               if(tEnd != endDate)
               {
                  problemStyle = 'hidden';
               }
            }

            // check against closed/not identified filtering if enabled...
            if(problem.geometry.id !== null)
            {
               if(document.getElementById(problem.geometry.id) !== null)
               {
                  let 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')
            {
               let 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';
            }
         }

         let marker = uroGetMarker(URO_TMARKER.MP, urobj);
         if(marker !== null)
         {
            marker.element.style.visibility = problemStyle;
         }

         if((problemStyle != 'hidden') && (ureqID !== null) && (customType !== null))
         {
            uroAddCustomMarkers(ureqID,URO_TMARKER.MP,customType, false, 0);
         }
      }
   }

   if(uroDOMHasTurnProblems)
   {
      for (let urobj in W.model.turnProblems.objects)
      {
         if(W.model.turnProblems.objects.hasOwnProperty(urobj))
         {
            let problem = W.model.turnProblems.objects[urobj];
            let problemStyle = 'visible';
            // check problem against current session ignore list...
            if(uroIsOnIgnoreList(problem.attributes.id)) problemStyle = 'hidden';

            if(uFP_masterEnable === true)
            {
               // check against closed/not identified filtering if enabled...
               if(problem.geometry.id !== null)
               {
                  if(document.getElementById(problem.geometry.id) !== null)
                  {
                     let 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';
                  }
               }
            }

            let marker = uroGetMarker(URO_TMARKER.MP, urobj);
            if(marker !== null)
            {            
               marker.element.style.visibility = problemStyle;
            }
         }
      }
   }
   uroRenderCustomMarkers(URO_TMARKER.MP);
   uroPerformanceMonitoring(pmFunction, pmTStart);
}
function uroToHex(decValue,digits)
{
   let modifier = 1;
   for(let i=0; i<digits; i++)
   {
      modifier *= 16;
   }
   decValue = parseInt(decValue);
   decValue += modifier;
   let retval = decValue.toString(16);
   retval = retval.substr(-digits);
   retval = retval.toUpperCase();
   return retval;
}
function uroFilterPreamble()
{
   let mapviewport = document.getElementsByClassName("olMapViewport")[0];
   if(mapviewport === null)
   {
      if(uroNullMapViewport === false)
      {
         uroAddLog('caught null mapviewport');
         uroNullMapViewport = true;
      }
      return false;
   }

   let uiElms = uroCtrlTabs[URO_TABS_ID.MISC][URO_TABS_FIELD.TABBODY];
   if(uiElms == null)
   {
      uroAddLog('caught missing UI');
      return false;
   }
   if(uiElms.innerHTML.length === 0)
   {
      uroAddLog('caught empty UI');
      return false;
   }

   if(uroSettingsApplied === false)
   {
      return false;
   }

   uroNullMapViewport = false;

   return true;
}
function uroFilterItems_URsTabClick()
{
   uroFilterURs();
}
function uroFilterItems_MPsTabClick()
{
   uroFilterProblems();
}
function uroFilterItems_MCsTabClick()
{
   uroFilterMapComments();
   uroMCLayerChanged();
}
function uroFilterItems_PlacesTabClick()
{
   uroFilterPlaces();
}
function uroFilterItems_CamsTabClick()
{
   uroFilterCameras();
}
function uroFilterItems_MiscTabClick()
{
   uroFilterItems();
}
function uroFilterItems_RTCsTabClick()
{
   uroFilterRTCs();
}
function uroFilterItems_RAsTabClick()
{
   uroFilterRAs();
}
function uroFilterItems_MasterEnableClick()
{
   if(uroGetCBChecked('_cbMasterEnable') === false)
   {
      uroHidePopup('uroFilterItems_MasterEnableClick');
   }
   uroFilterItems();
}
function uroFilterItems()
{
   uroFilterProblems();
   uroFilterPlaces();
   uroFilterCameras();
   uroFilterURs();
   uroFilterRTCs();
   uroFilterRAs();
   uroFilterMapComments();
}
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();
   let actionObj = require('Waze/Action/DeleteObject');
   let deleteAction = new actionObj(W.model.cameras.objects[uroShownFID], null);
   W.model.actionManager.add(deleteAction);
   uroExitPopup();
   uroHidePopup('uroDeleteObject');
   return false;
}
function uroGetUserNameFromID(userID)
{
   let userName;
   if(W.model.users.objects[userID] != null)
   {
      userName = W.model.users.objects[userID].attributes.userName;
      if(userName === undefined)
      {
         userName = userID;
      }
   }
   else
   {
      userName = userID;
   }
   return userName;
}
function uroGetUserNameAndRank(userID)
{
   let userName;
   let userLevel;
   if(W.model.users.objects[userID] != null)
   {
      userName = W.model.users.objects[userID].attributes.userName;
      if(userName === undefined)
      {
         userName = userID;
      }
      else
      {
         userName = '<a href="' + (W.Config.user_profile.url + userName) + '" target="_blank">' + userName + '</a>';
      }
      userLevel = W.model.users.objects[userID].attributes.rank + 1;
   }
   else
   {
      userName = userID;
      userLevel = '?';
   }
   return userName + ' (' + userLevel + ')';
}
function uroCheckCommentsForTag(idSrc, customText)
{
   let ursObj = W.model.updateRequestSessions.objects[idSrc];
   if(typeof(ursObj) == 'undefined') return -1;
   if(ursObj.attributes.comments.length === 0) return -1;

   for(let idx=ursObj.attributes.comments.length-1; idx>=0; idx--)
   {
      for(let tag=0; tag<uroCustomURTags.length; tag++)
      {
         let keyword = uroCustomURTags[tag];
         if(ursObj.attributes.comments[idx].text.indexOf(keyword) != -1)
         {
            return tag;
         }
      }

      if(customText !== '')
      {
         if(ursObj.attributes.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)
{
   let provider = '';
   let customText = '';
   if(uroGetCBChecked('_cbCustomKeywordMarkers')) customText = document.getElementById('_textCustomKeyword').value.toLowerCase();
   if(desc === null) desc = '';
   if(markerType == URO_TMARKER.UR)
   {
      let ureq = W.model.mapUpdateRequests.objects[idSrc];
      // early test for native speed limit URs
      if(ureq.attributes.type == 23) return 98;
   }
   else if(markerType == URO_TMARKER.MP)
   {
      let mp = W.model.problems.objects[idSrc];
      if(mp.attributes.provider != null)
      {
         provider = mp.attributes.provider;
      }
   }

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

      if(markerType == URO_TMARKER.UR)
      {
         if((uroGetCBChecked('_cbCustomKeywordMarkers')) && (customText !== ''))
         {
            if(desc.toLowerCase().indexOf(customText) != -1) return 99;
         }
      }

      if(markerType == URO_TMARKER.MP)
      {
         if(desc.indexOf('[Elgin]') != -1) return 100;
         if(desc.indexOf('[ELGIN]') != -1) return 100;
         if(desc.indexOf('[elginroadworks]') != -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 == URO_TMARKER.UR)
   {
      return uroCheckCommentsForTag(idSrc, customText);
   }

   return -1;
}
function uroGetRestrictionLanes(disposition)
{
   let retval = '';
   if(disposition == 1) retval += 'All lanes';
   else if(disposition == 2) retval += 'Left lane';
   else if(disposition == 3) retval += 'Middle lane';
   else if(disposition == 4) retval += 'Right lane';
   else retval += ' - ';
   return retval;
}
function uroGetRestrictionLaneType(laneType)
{
   let retval = '';
   if(laneType === null) retval += ' - ';
   else
   {
      if(laneType == 1) retval += 'HOV';
      else if(laneType == 2) retval += 'HOT';
      else if(laneType == 3) retval += 'Express';
      else if(laneType == 4) retval += 'Bus lane';
      else if(laneType == 5) retval += 'Fast lane';
      else retval += ' - ';
   }
   return retval;
}
let uroVehicleTypes =
[
   [1280, 'fa-car'],
   [1024, 'fa-motorcycle'],
   [272,  'fa-taxi'],
   [1808, 'fa-bolt']
];
function uroGetRestrictionVehicleTypes(restObj, allowInit, profileKey)
{
   let i;
   let j;
   let k;
   let tVT;
   let retval = [];
   for(i = 0; i < uroVehicleTypes.length; ++i)
   {
      retval.push(allowInit);
   }
   let tRest = restObj._driveProfiles.get(profileKey);
   if(tRest !== undefined)
   {
      for(i = 0; i < tRest._driveProfiles.length; ++i)
      {
         tVT = tRest._driveProfiles[i].getVehicleTypes();
         {
            if(tVT.length > 0)
            {
               for(j = 0; j < tVT.length; ++j)
               {
                  for(k = 0; k < uroVehicleTypes.length; ++k)
                  {
                     if(tVT[j] == uroVehicleTypes[k][0])
                     {
                        retval[k] = !allowInit;
                     }
                  }
               }
            }
         }
      }
   }
   return retval;
}
function uroFormatRestriction(restObj)
{
   let retval = '';

   if(restObj._defaultType == "DIFFICULT")
   {
      retval = '<tr><td colspan=13>Difficult Turn';
   }
   else
   {
      let roDays = null;
      let roFromDate = null;
      let roToDate = null;
      let roFromTime = null;
      let roToTime = null;
      let roRepeats = false;
      let roAllDay = false;
      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;
            roRepeats = restObj._timeFrames[0]._repeatYearly;
         }
      }

      if((roFromTime === null) && (roToTime === null))
      {
         roFromTime = "0:00";
         roToTime = "23:59";
         roAllDay = true;
      }

      let hasExpired = false;
      let isFuture = false;
      let tNow = Date.now();
      let tFrom = null;
      let tTo = null;
      
      if(roFromDate !== null)
      {
         tFrom = new Date(roFromDate + " " + roFromTime);
         isFuture = (tFrom.getTime() > tNow);
      }
      if(roToDate !== null)
      {
         tTo = new Date(roToDate + " " + roToTime);
         hasExpired = (tTo.getTime() < tNow);
      }
      if((hasExpired === true) && (roRepeats === true))
      {
         while(tTo.getTime() < tNow)
         {
            tFrom.setFullYear(tFrom.getFullYear() + 1);
            tTo.setFullYear(tTo.getFullYear() + 1);
         }
         isFuture = (tFrom.getTime() > tNow);
         hasExpired = false;
      }


      if(isFuture === true)
      {
         retval = '<tr bgcolor="#8080FF">';
      }
      else if(hasExpired === true)
      {
         retval = '<tr bgcolor="#FFFFC0">';
      }
      else
      {
         retval = '<tr>';
      }

      if(roDays === null)
      {
         roDays = 127;
      }

      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><td nowrap style="text-align:center;">';

      if(roFromDate === null) retval += 'All dates';
      else retval += tFrom.toISOString().slice(0,10) + ' to ' + tTo.toISOString().slice(0,10);
      if(roRepeats === true)
      {
         retval += '&nbsp;<i class="fa fa-repeat"> </i>';
      }

      retval += '</td><td nowrap style="text-align:center;">';

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

      retval += '</td><td nowrap style="text-align:center;">';

      retval += uroGetRestrictionLanes(restObj._disposition);

      retval += '</td><td nowrap style="text-align:center;">';

      retval += uroGetRestrictionLaneType(restObj._laneType);

      retval += '</td><td nowrap style="text-align:center;">';

      // for brevity, the popup only displays the allowed/prohibited restriction for the driveable vehicle types in the app...
      let typesAllowed = [];
      if((restObj._defaultType == "BLOCKED") || (restObj._defaultType == "TOLL"))
      {
         if(restObj._defaultType == "TOLL")
         {
            retval += I18n.lookup('restrictions.editing.segment.toll_road');
         }
         typesAllowed = uroGetRestrictionVehicleTypes(restObj, false, "FREE");
      }
      else
      {
         typesAllowed = uroGetRestrictionVehicleTypes(restObj, true, "BLOCKED");
      }

      let i;
      for(i = 0; i < uroVehicleTypes.length; ++i)
      {
         if(typesAllowed[i] === true)
         {
            retval += '<i class="fa '+uroVehicleTypes[i][1]+'" style="color:#000000;"> </i>&nbsp;';
         }
         else
         {
            retval += '<i class="fa '+uroVehicleTypes[i][1]+'" style="color:#d0d0d0;"> </i>&nbsp;';
         }
      }

      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 uroOpenURDialog(urID)
{
   let t = {showNext: false, nextButtonString: I18n.lookup('problems.panel.done')};
   let urObj = W.model.mapUpdateRequests.objects[urID];
   W.reqres.request("problems:browse", _.extend(t, {problem: urObj}));
}
function uroRecentreSessionOnUR()
{
   //uroGetMarker(URO_TMARKER.UR, uroShownFID).element.click();
   uroOpenURDialog(uroShownFID);
   W.map.moveTo(uroGetMarker(URO_TMARKER.UR, uroShownFID).lonlat, 17);
   uroHidePopup('uroRecentreSessionOnUR');
   return false;
}
function uroRecentreSessionOnMP()
{
   uroGetMarker(URO_TMARKER.MP, uroShownFID).element.click();
   W.map.moveTo(uroGetMarker(URO_TMARKER.MP, uroShownFID).lonlat, 17);
   uroHidePopup('uroRecentreSessionOnMP');
   return false;
}
function uroRecentreSessionOnPUR()
{
   uroGetMarker(URO_TMARKER.PUR, uroShownFID).element.click();
   W.map.moveTo(uroGetMarker(URO_TMARKER.PUR, uroShownFID).lonlat, 17);
   uroHidePopup('uroRecentreSessionOnPUR');
   return false;
}
function uroRecentreSessionOnPPUR()
{
   uroGetMarker(URO_TMARKER.PPUR, uroShownFID).element.click();
   W.map.moveTo(uroGetMarker(URO_TMARKER.PPUR, uroShownFID).lonlat, 17);
   uroHidePopup('uroRecentreSessionOnPPUR');
   return false;
}
function uroRecentreSessionOnVenueNavPoint()
{
   W.map.moveTo(uroGetVenueNavPoint(uroShownFID), 17);
   uroHidePopup('uroRecentreSessionOnVenueNavPoint');
   return false;
}
function uroGetDateTimeString(ts)
{
   let tDateObj = new Date(ts);
   let dateLocale;
   let 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;

   if(uroMarkerLayers[uroStackType] !== 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...
      let idList = uroGetMarkerIDs(uroStackType);
      for(let marker of idList)
      {
         let testMarkerAttributes = uroGetAttributes(uroStackType, marker);
         if(testMarkerAttributes.geometry.realX != null)
         {
            testMarkerAttributes.geometry.x = testMarkerAttributes.geometry.realX;
            testMarkerAttributes.geometry.y = testMarkerAttributes.geometry.realY;
            delete(testMarkerAttributes.geometry.realX);
            delete(testMarkerAttributes.geometry.realY);
         }
      }
      // now restack any markers that were repositioned...
      for(let idx=0; idx<uroStackList.length; idx++)
      {
         let orig_x = uroStackList[idx].x + 'px';
         let orig_y = uroStackList[idx].y + 'px';
         let fid = uroStackList[idx].fid;

         if(uroGetMarker(uroStackType, fid) != null)
         {
            uroGetMarker(uroStackType, fid).element.style.left = orig_x;
            uroGetMarker(uroStackType, fid).element.style.top = orig_y;
         }
      }
      uroStackList = [];
      uroUnstackedMasterID = null;
      uroStackType = null;
      uroAddLog('...stacked!');
   }
}
function uroIsIDAlreadyUnstacked(idSrc)
{
   if(uroStackList.length === 0) return false;
   for(let idx=0; idx<uroStackList.length; idx++)
   {
      if(uroStackList[idx].fid == idSrc) return true;
   }
   return false;
}
function uroCheckStacking(stackType, masterID, unstackedX, unstackedY)
{
   if(typeof(masterID) === 'number')
   {
      masterID = masterID.toString();
   }

   if(uroIsIDAlreadyUnstacked(masterID) === true) return;
   if(uroStackType !== null) return;
   if(uroPopupDwellTimer > 0) return;

   uroAddLog('checking for marker stack, masterID: '+masterID+', stackType: '+stackType);
   let stackList = [];
   stackList.push(masterID);
   let threshSquared = uroGetElmValue('_inputUnstackSensitivity');
   threshSquared *= threshSquared;

   let marker;

   let offset = 0.000000001;
   if(uroMarkerLayers[stackType] !== null)
   {
      let idList = uroGetMarkerIDs(stackType);
      let showOpen = true;
      let showClosed = false;
      let showTypes = null;
      if(stackType === URO_TMARKER.UR)
      {
         showTypes = W?.issueTrackerController?.app?.attributes?.issueTrackerFilter?.attributes?.mapUpdateRequestsFilter?.attributes?.status;
      }
      else if(stackType === URO_TMARKER.MP)
      {
         showTypes = W?.issueTrackerController?.app?.attributes?.issueTrackerFilter?.attributes?.mapProblemsFilter?.attributes?.status;
      }
      if(showTypes !== null)
      {
         showOpen = ((showTypes == 'OPEN') || (showTypes == 'BOTH'));
         showClosed = ((showTypes == 'CLOSED') || (showTypes == 'BOTH'));
      }

      for(marker of idList)
      {
         let testMarkerObj = uroGetMarker(stackType, marker);
         let testMarkerAttributes = uroGetAttributes(stackType, marker);
         if((testMarkerAttributes !== null) && (testMarkerObj !== null))
         {
            // 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.
            //
            // As of 3.169, the offset added to create the false coordinates has now been changed to a fractional value as opposed to the large integer
            // it previously was, as the latter method has now been seen to cause problems with displaying the "A" marker when a road closure request MP
            // is being viewed.  By using a small fractional offset added to each stacked marker, the risk of accidentally setting the offset coords to
            // those of another marker is low
            if(testMarkerAttributes.geometry.realX === undefined)
            {
               testMarkerAttributes.geometry.realX = testMarkerAttributes.geometry.x;
               testMarkerAttributes.geometry.x += offset;
               testMarkerAttributes.geometry.realY = testMarkerAttributes.geometry.y;
               testMarkerAttributes.geometry.y += offset;
               offset += 0.000000001;
            }

            let includeInStack = (testMarkerObj.element.style.visibility != 'hidden');
            let isClosed = testMarkerObj.element.classList.contains("recently-closed");
            includeInStack = includeInStack && ((isClosed && showClosed) || (!isClosed && showOpen));
            if(includeInStack)
            {
               if(testMarkerAttributes.id != masterID)
               {
                  let xdiff = unstackedX - uroParsePxString(testMarkerObj.element.style.left);
                  let ydiff = unstackedY - uroParsePxString(testMarkerObj.element.style.top);
                  let distSquared = ((xdiff * xdiff) + (ydiff * ydiff));
                  if(distSquared < threshSquared)
                  {
                     stackList.push(testMarkerAttributes.id);
                  }
               }
            }
         }
      }
   }

   let inhibitUnstacking = (W.map.getZoom() < uroGetElmValue('_inputUnstackZoomLevel'));
   inhibitUnstacking = inhibitUnstacking || (stackList.length < 2);

   if(inhibitUnstacking == false)
   {
      uroStackType = stackType;
      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(let shoveIdx=0; shoveIdx < stackList.length; shoveIdx++)
         {
            let fid = stackList[shoveIdx];
            let stackMarker = uroGetMarker(stackType, fid);
            if(stackMarker !== null)
            {
               let x = uroParsePxString(stackMarker.element.style.left);
               let y = uroParsePxString(stackMarker.element.style.top);
               // store the unstacked marker positions so they can be reinstated later
               uroStackList.push(new uroStackListObj(fid,x,y));
               stackMarker.element.style.left = unstackedX + 'px';
               stackMarker.element.style.top = unstackedY + 'px';
               unstackedX += 5;
               unstackedY -= 20;
            }
         }


         // hide other markers to prevent confusion with the unstacked markers
         let listIDs = uroGetMarkerIDs(stackType);
         for(marker in listIDs)
         {
            if(listIDs.hasOwnProperty(marker))
            {
               let toHideMarker = uroGetMarker(stackType, marker);
               if(toHideMarker !== null)
               {
                  let toHideID = toHideMarker.id;
                  if(uroIsIDAlreadyUnstacked(toHideID) === false)
                  {
                     toHideMarker = uroGetMarker(stackType, toHideID);
                     if(toHideMarker !== null)
                     {
                        toHideMarker.element.style.visibility = 'hidden';
                     }
                  }
               }
            }
         }
      }
   }
   else
   {
      uroRestackMarkers();
   }
}
function uroGetVenueNavPoint(uroFID)
{
   let retval = W.map.getCenter();   // allow the function to return a safe value in case we can't find the requested venue object...

   let vObj = W.model.venues.objects[uroFID];
   if(vObj !== undefined)
   {
      if(vObj.attributes.entryExitPoints.length > 0)
      {
         // if the venue has any navpoints defined, use the position of the first one
         let tPoint = vObj.attributes.entryExitPoints[0].getPoint();
         retval.lon = tPoint.coordinates[0];
         retval.lat = tPoint.coordinates[1];
      }
      else
      {
         // otherwise use the centrepoint of the venue point or polygon
         let tPoint = vObj.attributes.geometry.getCentroid();
         let tLL = new OpenLayers.LonLat();
         tLL.lon = tPoint.x;
         tLL.lat = tPoint.y;
         tLL.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
         retval.lon = tLL.lon;
         retval.lat = tLL.lat;
      }
   }
   return retval;
}
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 uroGetRTCDuration(rcObj)
{
   let duration = new Date(rcObj.attributes.endDate) - new Date(rcObj.attributes.startDate);
   return Math.floor(duration / 86400000);
}
function uroGetRTCOffset(rcDate)
{
   let dateObj = new Date(rcDate);
   return (0 - uroDateToDays(dateObj));
}
function uroGetRTCOrigin(rcObj)
{
   let retval = URO_TRTC.UNKNOWN;

   if(rcObj !== undefined)
   {
      if(rcObj.attributes.createdBy == -5)
      {
         retval = URO_TRTC.WAZEFEED;
      }
      else if((W.model.users.objects[rcObj.attributes.createdBy] !== undefined) && (W.model.users.objects[rcObj.attributes.createdBy].attributes.rank == 6))
      {
         retval = URO_TRTC.WAZEOTHER;
      }
      else
      {
         retval = URO_TRTC.WME;
      }
   }

   return retval;
}
function uroGetRTCState(rcObj)
{
   let retval = URO_SRTC.UNKNOWN;

   let nowDate = new Date();
   let startInPast = ((new Date(rcObj.attributes.startDate) - nowDate) < 0);
   let endInPast = ((new Date(rcObj.attributes.endDate) - nowDate) < 0);

   if(endInPast === true)
   {
      retval = URO_SRTC.EXPIRED;
   }
   else if(startInPast === true)
   {
      retval = URO_SRTC.ACTIVE;
   }
   else
   {
      retval = URO_SRTC.FUTURE;
   }

   return retval;
}
function uroAddClosureRowToTable(rcObj)
{
   let result = '';

   if(rcObj.attributes.active === true)
   {
      result += '<tr>';
   }
   else
   {
      result += '<tr bgcolor="#C0C0C0">';
   }

   let startDate = rcObj.attributes.startDate;
   let endDate = "unknown";
   if(rcObj.attributes.endDate !== null)
   {
      endDate = rcObj.attributes.endDate;
   }
   let provider = "---";
   if(rcObj.attributes.provider !== null)
   {
      provider = rcObj.attributes.provider;
   }
   else if(rcObj.attributes.createdBy !== null)
   {
      provider = uroGetUserNameAndRank(rcObj.attributes.createdBy);
   }
   let reason = "---";
   if(rcObj.attributes.reason !== null)
   {
      reason = rcObj.attributes.reason;
   }
   let mte = "---";
   if(rcObj.attributes.eventId !== null)
   {
      try
      {
         mte = W.model.majorTrafficEvents.objects[rcObj.attributes.eventId].attributes.names[0].value;
      }
      catch(err)
      {
      }
   }

   let startOffset = uroGetRTCOffset(rcObj.attributes.startDate);
   let duration = uroGetRTCDuration(rcObj);
   if(duration === 0) duration = '<1 day';
   else if(duration === 1) duration = '1 day';
   else duration = duration + ' days';

   let state = uroGetRTCState(rcObj);
   let status = '';
   if(state === URO_SRTC.EXPIRED)
   {
      status = "Expired";
   }
   else if(state === URO_SRTC.ACTIVE)
   {
      status = 'Active';
   }
   else if(state === URO_SRTC.FUTURE)
   {
      if(startOffset == 0)
      {
         status = 'Today';
      }
      else if(startOffset == 1)
      {
         status = 'In 1 day';
      }
      else
      {
         status = 'In ' + startOffset + ' days';
      }
   }
   result += '<td nowrap>' + status + '</td>';
   result += '<td nowrap>' + startDate + ' to ' + endDate + ' (' + duration + ')</td>';
   result += '<td nowrap>' + provider + '</td>';
   result += '<td nowrap>' + reason + '</td>';
   result += '<td nowrap>' + mte + '</td>';
   result += '</td></tr>';
   return result;
}
function uroGetAddress(streetID, houseNumber, formatForSegmentPopup, formatForNodePopup, showAsToll)
{
   let result = '';
   if((houseNumber !== undefined) && (houseNumber !== null))
   {
      result += houseNumber + ' ';
   }

   if(streetID != null)
   {
      let streetName = I18n.lookup('edit.address.no_street');
      let doesStreetIDExist = true;
      if(W.model.streets.objects[streetID] === undefined)
      {
         streetName = 'non-existent streetID';
         doesStreetIDExist = false;
      }
      else
      {
         if((streetName !== null) && (W.model.streets.objects[streetID].attributes.isEmpty === false))
         {
            streetName = W.model.streets.objects[streetID].attributes.name;
         }
      }
      if(formatForSegmentPopup === true)
      {
         if(showAsToll == true)
         {
            result += '<i class="fa fa-credit-card"></i> ';
         }
         result += '<b>'+streetName+'</b><br>';
      }
      else
      {
         result += streetName + ', ';
      }

      if(doesStreetIDExist === true)
      {
         let cityName = I18n.lookup('edit.address.no_city');
         let doesCityIDExist = true;
         let cityID = W.model.streets.objects[streetID].attributes.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)
         {
            let 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].attributes.name;
            }
         }
      }
   }
   result += '<br>';

   return result;
}
function uroGetSelectedSegmentRTCs(segID)
{
   let closureTypes = 0;
   uroRTCObjs = [];
   let selectedSegs = [];

   if(segID === null)
   {
      // segID should always be set to a valid segment ID if we're being called from the segment mouseover
      // handler, so if it's null it implies we've instead been called from the closure panel handler where
      // we might therefore be dealing with a multi-segment selection...
      selectedSegs = W.selectionManager.getSegmentSelection().segments;
   }

   if((selectedSegs.length > 0) || (segID !== null))
   {
      let i;
      for(let roadClosure in W.model.roadClosures.objects)
      {
         if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
         {
            let rcObj = W.model.roadClosures.objects[roadClosure];
            rcObj.segIDs = [rcObj.attributes.segID]; // copy the segID property into an array so we can push extra segIDs into it later...

            // set a direction value corresponding to the A-B or B-A setting - if we later end up combining an A-B and B-A closure
            // into a two-way closure, we can then change the direction value to indicate this as well
            if(rcObj.attributes.forward === true)
            {
               rcObj.direction = 1;
            }
            else
            {
               rcObj.direction = 2;
            }

            // for each of the selected or moused-over segments, find all the closures which have matching segIDs
            if(segID !== null)
            {
               if(rcObj.attributes.segID == segID)
               {
                  uroRTCObjs.push(rcObj);
               }
            }
            else
            {
               for(i = 0; i < selectedSegs.length; ++i)
               {
                  if(rcObj.attributes.segID == selectedSegs[i].attributes.id)
                  {
                     uroRTCObjs.push(rcObj);
                     break;
                  }
               }
            }
         }
      }

      // uroRTCObjs now contains all of the closures relating to all of the segments of interest, so we can
      // begin to organise them such that by the time we exit this function, the array will then contain an
      // optimised list of closures that matches up to the list shown in the closure sidepanel, taking into
      // account closures applying to all segments vs some, closures that can be merged into two-ways etc.

      // first sort the closure by their start date, with a secondary sort by direction for those closures
      // that have the same start date
      uroRTCObjs = uroRTCObjs.sort(function(a,b)
      {
         if(a.attributes.startDate === b.attributes.startDate)
         {
            if(a.direction == 1) return -1;
            return 1;
         }
         if(a.attributes.startDate > b.attributes.startDate) return 1;
         return -1;
      });

      // if we've got at least two closures in the sorted list, we then test adjacent list entries
      // to see if they contain closure details which are identical except for their segment IDs, and
      // combine them if so
      if(uroRTCObjs.length > 1)
      {
         i = 0;
         while(i < (uroRTCObjs.length - 1))
         {
            if(
               (uroRTCObjs[i].attributes.createdBy == uroRTCObjs[i+1].attributes.createdBy) &&
               (uroRTCObjs[i].attributes.endDate == uroRTCObjs[i+1].attributes.endDate) &&
               (uroRTCObjs[i].attributes.eventId == uroRTCObjs[i+1].attributes.eventId) &&
               (uroRTCObjs[i].attributes.location == uroRTCObjs[i+1].attributes.location) &&
               (uroRTCObjs[i].attributes.reason == uroRTCObjs[i+1].attributes.reason) &&
               (uroRTCObjs[i].attributes.startDate == uroRTCObjs[i+1].attributes.startDate) &&
               (uroRTCObjs[i].direction == uroRTCObjs[i+1].direction)
            )
            {
               uroRTCObjs[i].segIDs.push(uroRTCObjs[i+1].attributes.segID);
               uroRTCObjs.splice(i+1, 1);
            }
            else
            {
               ++i;
            }
         }
      }

      // after that first trimming of the list, if there are still two or more entries then
      // we perform a second pass, this time merging any adjacent entries which have the same
      // segment IDs in their segIDs arrays - these are two-way closures applying to all those
      // segments, and so we also change the direction value to indicate two-way vs A-B or B-A
      if(uroRTCObjs.length > 1)
      {
         i = 0;
         while(i < (uroRTCObjs.length - 1))
         {
            if
            (
               (uroRTCObjs[i].segIDs.sort().join(',') == uroRTCObjs[i+1].segIDs.sort().join(',')) &&
               (uroRTCObjs[i].attributes.createdBy == uroRTCObjs[i+1].attributes.createdBy) &&
               (uroRTCObjs[i].attributes.endDate == uroRTCObjs[i+1].attributes.endDate) &&
               (uroRTCObjs[i].attributes.eventId == uroRTCObjs[i+1].attributes.eventId) &&
               (uroRTCObjs[i].attributes.location == uroRTCObjs[i+1].attributes.location) &&
               (uroRTCObjs[i].attributes.reason == uroRTCObjs[i+1].attributes.reason) &&
               (uroRTCObjs[i].attributes.startDate == uroRTCObjs[i+1].attributes.startDate)
            )
            {
               uroRTCObjs[i].direction = 4;
               uroRTCObjs.splice(i+1, 1);
            }
            ++i;
         }
      }

      for(i = 0; i < uroRTCObjs.length; ++i)
      {
         closureTypes |= uroRTCObjs[i].direction;
      }
   }

   // the closure list ordering at this point doesn't always match up to the order used by the closures panel when
   // a mixture of "all segment" and "some segment" closures are present - need to work out what ordering rules
   // WME is using here...
   return closureTypes;
}
function uroGetLengthString(length)
{
   let retval = '';
   if(length == null)
   {
      retval = "Default";
   }
   else if(W.model.isImperial == true)
   {
      retval = (length / (12 * 2.54)).toFixed(1) + "ft";
   }
   else
   {
      retval = (length / 100).toFixed(1) + "m";
   }

   return retval;
}
function uroGetHighlightedMapFeature()
{
   let featureID = W.selectionManager.getWebMapSelectionManager().mouseInFeature;
   let retval = null;

   if(featureID !== null)
   {
      let isSelected = W.selectionManager.isSelected(featureID);
      if(isSelected === false)
      {
         retval = W.selectionManager.getWebMapSelectionManager().getFeature(featureID);
      }
   }
   
   return retval;
}
function uroGetFeatureRenderIntent(moObj)
{
   let retval = "unknown";

   if(moObj !== null)
   {
      let isSelected = moObj._wmeObject.selected;

      if(isSelected === true)
      {
         retval = "highlightselected";
      }
      else
      {
         retval = "highlight";
      }
   }

   return retval;
}
function uroNewLookHighlightedItemsCheck(e)
{
   if
   (
      (W.selectionManager._mouseInFeature === null) &&
      (uroMousedOverMarkerType === null) &&
      (uroPopupTimer === 0)
   )
   {
      if(uroPopupShown === true)
      {
         uroHidePopup('uroNewLookHighlightedItemsCheck');
      }
      uroMousedOverMapComment = null;
   
      return;
   }

   let moObj = uroGetHighlightedMapFeature();
   let renderIntent = uroGetFeatureRenderIntent(moObj);

   let result = '';
   let rw;
   let rh;
   let objHasIgnoreLink = false;
   let objHasDeleteLink = false;
   let objHasAddWatchLink = false;
   let objHasRemoveWatchLink = false;
   let objHasUpdateWatchLink = false;
   let objHasRecentreSessionLink = false;
   let objHasOpenInNewTabLink = false;
   let objHasCloneLink = false;
   let isVenue = false;
   let isMapComment = false;
   let newPopupType = null;
   let ureq = null;
   let idx;
   let hovered = false;
   let targetTab = '';
   let unstackedX;
   let unstackedY;
   let ureqID = null;
   let isUR = false;
   let isProblem = false;
   let isTurnProb = false;
   let isPlaceUpdate = false;
   let mouseX;
   let mouseY;
   let uroDaysResolved;
   let deltaX;
   let deltaY;
   let userLock;

   // 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.getLayerByName("update_requests") === null)
      {
         if(uroNullURLayer === false)
         {
            uroAddLog('caught null UR layer');
            uroNullURLayer = true;
         }
         return;
      }
      uroNullURLayer = false;

      if(W.map.getLayerByName("mapProblems") === 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;
         deltaX = 0;
         deltaY = 0;

         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;

         let maxJitter = uroGetElmValue('_inputMaxJitter');
         if((Math.abs(uroPrevMouseX - mouseX) > maxJitter) || (Math.abs(uroPrevMouseY - mouseY) > maxJitter))
         {
            uroPopupDwellTimer = uroGetElmValue('_inputPopupDwellTimeout');
         }
         deltaX = mouseX - uroPrevMouseX;
         deltaY = mouseY - uroPrevMouseY;
         uroPrevMouseX = mouseX;
         uroPrevMouseY = mouseY;
      }
   }
   uroWazeBits();

   let mouseLonLat = W.map.getLonLatFromViewPortPx(new OpenLayers.Pixel(mouseX,mouseY));
   let 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;
      }
   }

   let popupXOffset = document.getElementById('editor-container').getBoundingClientRect().x;
   let popupYOffset = $(document.getElementById("WazeMap")).offset().top;
   let uroPopupX = mouseX + popupXOffset + 10;
   let uroPopupY = mouseY + popupYOffset;

   // popup for segments
   if((uroMousedOverMarkerType === null) && (uroGetCBChecked('_cbInhibitSegPopup') === false))
   {
      if((moObj !== null) && (moObj.featureType === 'segment'))
      {
         if(W.map.getExtent().intersectsBounds(moObj._wmeObject.attributes.geometry.getBounds()))
         {
            let doPopUp = false;
            let restObj;

            if(uroMousedOverMapComment !== null)
            {
               uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for segment highlight');
               uroMousedOverOtherObjectWithinMapComment = true;
            }

            let streetID = moObj._wmeObject.attributes.primaryStreetID;
            if(streetID !== null)
            {
               // generic segment data
               if(uroGetCBChecked('_cbInhibitSegGenericPopup') === false)
               {
                  doPopUp = true;
                  uroAddLog('building popup for segment '+streetID);

                  let isToll = ((moObj._wmeObject.attributes.fwdToll == true) || (moObj._wmeObject.attributes.revToll == true));
                  result += uroGetAddress(streetID, null, true, false, isToll);

                  if(moObj._wmeObject.attributes.streetIDs.length > 0)
                  {
                     // list any alternate names
                     result += '<br>Alternate names:<br>';
                     for(let i = 0; i < moObj._wmeObject.attributes.streetIDs.length; ++i)
                     {
                        result += '&nbsp;<i>' + W.model.streets.objects[moObj._wmeObject.attributes.streetIDs[i]].attributes.name + ', ';
                        let cityName = "";
                        if(W.model.cities.objects[W.model.streets.objects[moObj._wmeObject.attributes.streetIDs[i]].attributes.cityID] != undefined)
                        {
                           cityName = W.model.cities.objects[W.model.streets.objects[moObj._wmeObject.attributes.streetIDs[i]].attributes.cityID].attributes.name;
                        }
                        if(cityName != "")
                        {
                           result += cityName;
                        }
                        else
                        {
                           result += ' no city';
                        }
                        result += '</i><br>';
                     }
                     result += '<br>';
                  }

                  result += '<b>ID: </b>'+moObj._wmeObject.attributes.id+' - ';
                  let autoLock = moObj._wmeObject.attributes.rank;
                  userLock = moObj._wmeObject.attributes.lockRank;
                  result += '<b>' + I18n.lookup("edit.segment.fields.lock") + ': </b>';
                  if(userLock !== null)
                  {
                     result += 'M' + (userLock+1) + ' / ';
                  }
                  result += 'A' + (autoLock+1) + ' - ';

                  let level = moObj._wmeObject.attributes.level;
                  result += '<b>' + I18n.lookup("edit.segment.fields.level") +': </b>';
                  if(level == 0)
                  {
                     result += I18n.lookup("edit.segment.levels")[0];
                  }
                  else
                  {
                     result += level;
                  }
                  result += '<br>';

                  let leBy = moObj._wmeObject.attributes.updatedBy;
                  let leOn = moObj._wmeObject.attributes.updatedOn;
                  if(leOn == null)
                  {
                     leBy = moObj._wmeObject.attributes.createdBy;
                     leOn = moObj._wmeObject.attributes.createdOn;
                  }
                  result += "<b>Last edit by</b> " + uroGetUserNameAndRank(leBy) + " <b>on</b> " + uroGetDateTimeString(leOn) + "<br><br>";

                  let fwdSpeed = moObj._wmeObject.attributes.fwdMaxSpeed;
                  let revSpeed = moObj._wmeObject.attributes.revMaxSpeed;
                  let fwdLanes = moObj._wmeObject.attributes.fwdLaneCount;
                  let revLanes = moObj._wmeObject.attributes.revLaneCount;
                  let fwdWidth = 'Not set';
                  let revWidth = 'Not set';
                  if(moObj._wmeObject.attributes.fromLanesInfo != null)
                  {                     
                     fwdWidth = uroGetLengthString(moObj._wmeObject.attributes.fromLanesInfo.laneWidth);
                  }
                  if(moObj._wmeObject.attributes.toLanesInfo != null)
                  {
                     revWidth = uroGetLengthString(moObj._wmeObject.attributes.toLanesInfo.laneWidth);
                  }
                  let fwdUnverified = moObj._wmeObject.attributes.fwdMaxSpeedUnverified;
                  let revUnverified = moObj._wmeObject.attributes.revMaxSpeedUnverified;
                  let fwdASC = ((moObj._wmeObject.attributes.fwdFlags & 1) === 1);
                  let revASC = ((moObj._wmeObject.attributes.revFlags & 1) === 1);
                  let roadType = moObj._wmeObject.attributes.roadType;
                  let verifyLimits = true;
                  if((roadType === 17) || (roadType === 20))
                  {
                     verifyLimits = false;
                  }

                  result += '<table border=1><tr><th>Dir</th><th>Speed</th><th>ASC</th><th>Lanes</th><th>Width</th></tr>';
                  if(moObj._wmeObject.attributes.fwdDirection)
                  {
                     result += '<tr><td><b>A-B</b></td><td>'+uroGetLocalisedSpeedString(fwdSpeed)+'</td><td>';
                     if(fwdASC == true)
                     {
                        result += 'Yes';
                     }
                     else
                     {
                        result += 'No';
                     }
                     result += '</td><td>';
                     if(fwdLanes > 0)
                     {
                        result += fwdLanes;
                     }
                     else
                     {
                        result += 'Not set';
                     }
                     result += '</td><td>'+fwdWidth+'</td></tr>';
                  }
                  if(moObj._wmeObject.attributes.revDirection)
                  {
                     result += '<tr><td><b>B-A</b></td><td>'+uroGetLocalisedSpeedString(revSpeed)+'</td><td>';
                     if(revASC == true)
                     {
                        result += 'Yes';
                     }
                     else
                     {
                        result += 'No';
                     }
                     result += '</td><td>';
                     if(revLanes > 0)
                     {
                        result += revLanes;
                     }
                     else
                     {
                        result += 'Not set';
                     }
                     result += '</td><td>'+revWidth+'</td></tr>';
                  }
                  result += '</table>';

                  if((moObj._wmeObject.attributes.fwdDirection) && (moObj._wmeObject.attributes.revDirection) && (fwdSpeed != revSpeed) && (!fwdUnverified) && (!revUnverified))
                  {
                     result += '<br>Two-way segment has different verified speed limits...';
                  }
               }

               // segment restrictions
               if(moObj._wmeObject.attributes.restrictions.length > 0)
               {
                  result += '<br><table border=1>';
                  doPopUp = true;
                  let fwdResult = '<tr><td colspan=13><b>A-B restrictions:</b></td></tr>';
                  let revResult = '<tr><td colspan=13><b>B-A restrictions:</b></td></tr>';
                  let bothResult = '<tr><td colspan=13><b>Two-way restrictions:</b></td></tr>';

                  let nABRestrictions = 0;
                  let nBARestrictions = 0;
                  let nBothRestrictions = 0;
                  for(idx = 0; idx < moObj._wmeObject.attributes.restrictions.length; idx++)
                  {
                     restObj = moObj._wmeObject.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.getLayerByName("closures").getVisibility() === true)
               {
                  let closureTypes = uroGetSelectedSegmentRTCs(moObj._wmeObject.attributes.id);
                  if(closureTypes !== 0)
                  {
                     result += '<br><table border=1 width="100%">';
                     let rcObj;
                     let roadClosure;

                     if((closureTypes & 1) === 1)
                     {
                        result += '<tr><td colspan=4><b>A-B closures:</b></td></tr>';
                        for(roadClosure in uroRTCObjs)
                        {
                           if(uroRTCObjs.hasOwnProperty(roadClosure))
                           {
                              rcObj = uroRTCObjs[roadClosure];
                              if(rcObj.direction === 1)
                              {
                                 result += uroAddClosureRowToTable(rcObj);
                              }
                           }
                        }
                     }
                     if((closureTypes & 2) === 2)
                     {
                        result += '<tr><td colspan=4><b>B-A closures:</b></td></tr>';
                        for(roadClosure in uroRTCObjs)
                        {
                           if(uroRTCObjs.hasOwnProperty(roadClosure))
                           {
                              rcObj = uroRTCObjs[roadClosure];
                              if(rcObj.direction === 2)
                              {
                                 result += uroAddClosureRowToTable(rcObj);
                              }
                           }
                        }
                     }
                     if((closureTypes & 4) === 4)
                     {
                        result += '<tr><td colspan=4><b>Two-way closures:</b></td></tr>';
                        for(roadClosure in uroRTCObjs)
                        {
                           if(uroRTCObjs.hasOwnProperty(roadClosure))
                           {
                              rcObj = uroRTCObjs[roadClosure];
                              if(rcObj.direction === 4)
                              {
                                 result += uroAddClosureRowToTable(rcObj);
                              }
                           }
                        }
                     }
                     if(closureTypes > 0)
                     {
                        doPopUp = true;
                     }

                     result += '</table>';
                  }
               }

               if(doPopUp === true)
               {
                  uroFID = moObj._wmeObject.attributes.id;
                  newPopupType = 'segment_restriction';
               }
            }
         }
         else
         {
            uroAddLog('segment '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup');
         }
      }
   }
   // popup for landmarks
   if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitLandmarkPopup') === false))
   {
      let navpointPos=new OpenLayers.LonLat();
      if((moObj !== null) && (moObj.featureType === 'venue'))
      {
         if(renderIntent === 'highlight')
         {
            if(W.map.getExtent().intersectsBounds(moObj._wmeObject.attributes.geometry.getBounds()))
            {
               if(uroMousedOverMapComment !== null)
               {
                  uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for place highlight');
                  uroMousedOverOtherObjectWithinMapComment = true;
               }

               if(newPopupType === null)
               {
                  uroFID = moObj._wmeObject.attributes.id;
                  uroAddLog('building popup for place '+uroFID);

                  navpointPos = uroGetVenueNavPoint(uroFID);
                  
                  result += '<b>';
                  if(moObj._wmeObject.attributes.name === '')
                  {
                     if(moObj._wmeObject.attributes.residential === true) result += '<i>Residential</i>';
                     else result += '<i>Unnamed</i>';
                  }
                  else result += uroClickify(moObj._wmeObject.attributes.name, '');
                  if(moObj._wmeObject.attributes.externalProviderIDs.length > 0)
                  {
                     result += ' <i>(linked)</i>';
                  }
                  if(moObj._wmeObject.attributes.adLocked)
                  {
                     result += ' <i>(AdLocked)</i>';
                  }
                  result += '</b><br>';
                  if(moObj._wmeObject.attributes.brand !== null)
                  {
                     result += '<i>Brand: ' + moObj._wmeObject.attributes.brand + '</i><br>';
                  }
                  let vDesc = moObj._wmeObject.attributes.description;
                  if(vDesc !== '')
                  {
                     result += '"<i>' + uroClickify(vDesc, '') + '</i>"<br>';
                  }

                  userLock = moObj._wmeObject.attributes.lockRank;
                  result += '<b>Lock: </b>' + (userLock+1);

                  result += '<hr>';
                  result += uroGetAddress(moObj._wmeObject.attributes.streetID, moObj._wmeObject.attributes.houseNumber, false, false, false);
                  result += '<ul>';
                  for(idx = 0; idx < moObj._wmeObject.attributes.categories.length; idx++)
                  {
                     result += '<li>' + I18n.lookup("venues.categories." + moObj._wmeObject.attributes.categories[idx]);
                  }
                  result += '</ul>';
                  if(moObj._wmeObject.attributes.residential === true)
                  {
                     if(moObj.geometry.type === "Point")
                     {
                        result += '<a href="#" id="_cloneRP">Clone place</a>';
                        objHasCloneLink = true;
                     }
                  }
                  let npLink = document.location.href;
                  let npLayers = '';
                  npLink = npLink.substr(0,npLink.indexOf('?zoomLevel'));
                  npLink += '?zoomLevel=17&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;
               }
               else
               {
                  let otherID = moObj._wmeObject.attributes.id;
                  uroAddLog('venue '+otherID+' is also highlighted');
               }
            }
            else
            {
               uroAddLog('landmark '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup');
            }
         }
      }
   }
   // popup for map comments
   if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitMapCommentPopup') === false))
   {
      if(uroMCLayer.name !== 'mapComments')
      {
         uroWazeBits();
      }
      if(uroMCLayer !== null)
      {
         uroMCSelected = false;

         if((moObj !== null) && (moObj.featureType === 'comment'))
         {
            if(renderIntent == 'highlight')
            {
               let moAttrs = moObj._wmeObject.attributes;
               if(W.map.getExtent().intersectsBounds(moAttrs.geometry.getBounds()))
               {
                  if(newPopupType === null)
                  {
                     if((uroMousedOverMapComment === moAttrs.id) && (uroMousedOverOtherObjectWithinMapComment === true))
                     {
                        uroAddLog('inhibit popup for map comment '+uroMousedOverMapComment);
                     }
                     else
                     {
                        uroMousedOverOtherObjectWithinMapComment = false;
                        if(moAttrs.geometry.id.indexOf('Polygon') !== -1)
                        {
                           // only capture ID for area comments...
                           uroMousedOverMapComment = moAttrs.id;
                        }
                        uroFID = moAttrs.id;
                        uroAddLog('building popup for map comment '+uroFID);

                        result += '<b>';
                        if(moAttrs.subject === '')
                        {
                           result += '<i>No subject</i>';
                        }
                        else result += uroClickify(moAttrs.subject, '');
                        result += '</b><br>';
                        result += uroClickify(moAttrs.body, '<br>');

                        let mcDaysOld = uroGetMCAge(moAttrs, 0, false);
                        let mcSubmittedTS = uroGetMCAge(moAttrs, 0, true);
                        if(mcSubmittedTS != -1)
                        {
                           mcSubmittedTS = uroGetDateTimeString(mcSubmittedTS);
                        }
                        if(mcDaysOld != -1)
                        {
                           result += '<i>Submitted ' + uroParseDaysAgo(mcDaysOld) + ' ';
                           if(mcSubmittedTS != -1) result += '(' + mcSubmittedTS + ') ';
                           if(moAttrs.createdBy != null)
                           {
                              result += ' by '+uroGetUserNameAndRank(moAttrs.createdBy);
                           }
                           result += '</i><br>';
                        }
                        mcDaysOld = uroGetMCAge(moAttrs, 1, false);
                        mcSubmittedTS = uroGetMCAge(moAttrs, 1, true);
                        if(mcSubmittedTS != -1)
                        {
                           mcSubmittedTS = uroGetDateTimeString(mcSubmittedTS);
                        }
                        if(mcDaysOld != -1)
                        {
                           result += '<i>Updated ' + uroParseDaysAgo(mcDaysOld) + ' ';
                           if(mcSubmittedTS != -1) result += '(' + mcSubmittedTS + ') ';
                           if(moAttrs.createdBy != null)
                           {
                              result += ' by '+uroGetUserNameAndRank(moAttrs.updatedBy);
                           }
                           result += '</i><br>';
                        }

                        mcDaysOld = uroGetMCAge(moAttrs,2,false);
                        mcSubmittedTS = uroGetMCAge(moAttrs,2,true);
                        if(mcDaysOld != -1)
                        {
                           result += '<i>Expires ' + uroParseDaysToGo(mcDaysOld) + ' ';
                           result += '(' + uroGetDateTimeString(mcSubmittedTS) +')</i><br>';
                        }

                        let mcHasMyComments = false;
                        let mcNComments = moAttrs.conversation.length;
                        if(mcNComments > 0)
                        {
                           for(let j=0; j<mcNComments; j++)
                           {
                              if(moAttrs.conversation[j].userID == uroUserID)
                              {
                                 mcHasMyComments = true;
                                 break;
                              }
                           }
                        }
                        result += '<br>' + mcNComments +' comment';
                        if(mcNComments != 1) result += 's';
                        if((mcHasMyComments === false) && (mcNComments > 0)) result += ' (none by me)';

                        // add "ignore for this session" link
                        result += '<br><a href="#" id="_addtoignore">Hide for this session</a>';
                        objHasIgnoreLink = true;

                        newPopupType = 'map_comment';
                        isMapComment = true;
                     }
                  }
                  else
                  {
                     let mcOtherID = moAttrs.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 !== null) && (uroMousedOverMarkerID !== null))
      {
         let mousedMarker = uroGetMarker(uroMousedOverMarkerType, uroMousedOverMarkerID);
         if((uroMousedOverMarkerType == URO_TMARKER.UR) && (newPopupType === null) && (mousedMarker !== null))
         {
            unstackedX = uroParsePxString(mousedMarker.element.style.left);
            unstackedY = uroParsePxString(mousedMarker.element.style.top);
                       
            // check for stacking...
            if(uroShownFID != uroMousedOverMarkerID)
            {
               uroCheckStacking(URO_TMARKER.UR,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];

               let markerBCR = mousedMarker.element.getBoundingClientRect();
               uroPopupX = markerBCR.left + markerBCR.width;
               uroPopupY = markerBCR.top + (markerBCR.height / 2);

               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>');
               let uroDaysOld = uroGetURAge(ureq,0,false);
               let 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);
                  let 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)
               {
                  let hasMyComments = uroURHasMyComments(uroMousedOverMarkerID);
                  let nComments = W.model.updateRequestSessions.objects[uroMousedOverMarkerID].attributes.comments.length;
                  result += '<br>' + nComments + ' comment';
                  if(nComments != 1) result += 's';
                  if((hasMyComments === false) && (nComments > 0)) result += ' (none by me)';
                  if(nComments > 0)
                  {
                     let commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[uroMousedOverMarkerID].attributes.comments[nComments-1]);
                     if(commentDaysOld != -1)
                     {
                        result += ', last update '+uroParseDaysAgo(commentDaysOld);
                     }
                  }
               }
               if(uroURDupes.length > 0)
               {
                  let thisID = parseInt(uroMousedOverMarkerID);
                  for(let i = 0; i < uroURDupes.length; ++i)
                  {
                     if(uroURDupes[i][0] === thisID)
                     {
                        result += '<br><br>Duplicate of: ';
                        let dupes = 0;
                        for(let j = 0; j < uroURDupes[i][1].length; ++j)
                        {
                           if(uroURDupes[i][1][j] !== thisID)
                           {
                              if(dupes > 0)
                              {
                                 result += ', ';
                              }   
                              result += uroURDupes[i][1][j];
                              ++dupes;
                           }
                        }
                     }
                  }
               }
            }
         }

         if(((uroMousedOverMarkerType == URO_TMARKER.PUR) || (uroMousedOverMarkerType == URO_TMARKER.PPUR) || (uroMousedOverMarkerType == URO_TMARKER.RPUR)) && (newPopupType === null) && (mousedMarker !== null))
         {
            ureq = W.model.venues.objects[uroMousedOverMarkerID];
            unstackedX = uroParsePxString(mousedMarker.element.style.left);
            unstackedY = uroParsePxString(mousedMarker.element.style.top);

            if(uroShownFID != uroMousedOverMarkerID)
            {
               // check for stacking...
               uroCheckStacking(uroMousedOverMarkerType, uroMousedOverMarkerID, unstackedX, unstackedY);
            }
            hovered = true;

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

               isPlaceUpdate = true;

               let markerBCR = mousedMarker.element.getBoundingClientRect();
               uroPopupX = markerBCR.left + markerBCR.width;
               uroPopupY = markerBCR.top + (markerBCR.height / 2);

               uroFID = uroMousedOverMarkerID;

               if(uroMousedOverMarkerType == URO_TMARKER.PUR)
               {
                  uroAddLog('building popup for placeUpdate '+uroMousedOverMarkerID);
               }
               else if(uroMousedOverMarkerType == URO_TMARKER.RPUR)
               {
                  uroAddLog('building popup for residentialPlaceUpdate '+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>';
               }

               let daysOld = uroGetPURAge(ureq);
               if(daysOld != -1)
               {
                  result += '<br><i>Submitted '+uroParseDaysAgo(daysOld)+'</i>';
               }
            }
         }

         if((uroMousedOverMarkerType == URO_TMARKER.MP) && (newPopupType === null) && (mousedMarker !== null))
         {
            unstackedX = uroParsePxString(mousedMarker.element.style.left);
            unstackedY = uroParsePxString(mousedMarker.element.style.top);
            
            // check for stacking...
            if(uroShownFID != uroMousedOverMarkerID)
            {
               uroCheckStacking(URO_TMARKER.MP,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;
                  }
               }

               let markerBCR = mousedMarker.element.getBoundingClientRect();
               uroPopupX = markerBCR.left + markerBCR.width;
               uroPopupY = markerBCR.top + (markerBCR.height / 2);

               uroFID = 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+'): ';

                  let 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: ' + uroClickify(ureq.attributes.extraInfo, '<br>');
               }
               if(ureq.attributes.provider != null)
               {
                  result += 'Provider: ' + ureq.attributes.provider + '<br>';
               }
               if(ureq.attributes.startTime != null)
               {
                  result += 'From: ' + uroGetDateTimeString(ureq.attributes.startTime) + '<br>';
               }
               if(ureq.attributes.endTime != null)
               {
                  result += 'To: ' + uroGetDateTimeString(ureq.attributes.endTime) + '<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)
         {
            let 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
         let urPos=new OpenLayers.LonLat();
         if(isPlaceUpdate)
         {
            urPos.lon = ureq.getOLGeometry().getCentroid().x;
            urPos.lat = ureq.getOLGeometry().getCentroid().y;
         }
         else
         {
            if(ureq.attributes.geometry.realX === undefined)
            {
               urPos.lon = ureq.attributes.geometry.x;
               urPos.lat = ureq.attributes.geometry.y;
            }
            else
            {
               urPos.lon = ureq.attributes.geometry.realX;
               urPos.lat = ureq.attributes.geometry.realY;
            }
         }
         urPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
         let urLink = document.location.href;
         let urLayers = '';
         urLink = urLink.substr(0,urLink.indexOf('?zoomLevel'));
         urLink += '?zoomLevel=17&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 == URO_TMARKER.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
         let 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)
         {
            let 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;
         }
      }
   }
   // look for nodes
   if((newPopupType === null) && (uroGetCBChecked('_cbInhibitNodesPopup') === false))
   {
      if(uroMousedOverMarkerType == 'node')
      {
         ureq = W.model.nodes.objects[uroMousedOverMarkerID];
         if(ureq === undefined)
         {
            uroMousedOverMarkerID = null;
         }
         else
         {
            ureqID = uroMousedOverMarkerID;

            if(uroMousedOverMapComment !== null)
            {
               uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for node highlight');
               uroMousedOverOtherObjectWithinMapComment = true;
            }
            uroPopupY += 40;
            newPopupType = 'node';
            uroFID = ureqID;
            uroAddLog('building popup for node '+uroFID);
            result += '<b>Node: ' + uroFID + '</b><br>';
            for(let k=0; k<ureq.attributes.segIDs.length; k++)
            {
               let nodeSegID = ureq.attributes.segIDs[k];
               let nodeSegObj = W.model.segments.objects[nodeSegID];
               if(nodeSegObj !== undefined)
               {
                  let nodeStreetID = nodeSegObj.attributes.primaryStreetID;
                  result += uroGetAddress(nodeStreetID, null, false, true, false);
               }
            }
         }
      }
   }
   // look for cameras
   if((newPopupType === null) && (uroGetCBChecked('_cbInhibitCamPopup') === false))
   {
      if(uroMousedOverMarkerType == 'cam')
      {
         ureq = moObj?._wmeObject;
         ureqID = uroMousedOverMarkerID;

         if(renderIntent === "highlight")
         {
            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>';
            }
            else
            {
               result += '<b>Camera: ' + I18n.lookup("edit.camera.fields.type." + ureq.attributes.type) + '</b>';
            }
            result += '<br>';
            result += 'ID: '+uroFID+'<br>';
            result += 'Created by ';
            let userID;

            if(W.model.users.getByIds([ureq.attributes.createdBy])[0] != null)
            {
               userID = ureq.attributes.createdBy;
               result += uroGetUserNameAndRank(userID);
            }
            else result += 'unknown';
            result += ', ';
            let camAge = uroGetCameraAge(ureq,1);
            if(camAge != -1)
            {
               result += uroParseDaysAgo(camAge);
            }
            else result += 'unknown days ago';
            result += '<br>Updated by ';
            if(W.model.users.getByIds([ureq.attributes.updatedBy])[0] != null)
            {
               userID = ureq.attributes.updatedBy;
               result += uroGetUserNameAndRank(userID);
            }
            else result += 'unknown';
            result += ', ';
            camAge = uroGetCameraAge(ureq,0);
            if(camAge != -1)
            {
               result += uroParseDaysAgo(camAge);
            }
            else result += 'unknown days ago';

            if(ureq.attributes.type !== 4)
            {
               result += '<br>Speed data: ';
               result += uroGetLocalisedSpeedString(ureq.attributes.speed);
               if(uroIsCamSpeedValid(ureq) === false)
               {
                  result += ' (differs to nearest segment)';
               }
            }
            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>';
         }
      }
   }

   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.style.overflow = "auto";
         uroDiv.innerHTML = uroModifyHTML(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 == URO_TMARKER.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);
         uroAddLog('popup width is '+rw);
         if(rw > (window.innerWidth * 0.45))
         {
            rw = (window.innerWidth * 0.45);
            uroDiv.style.width = rw+'px';
            uroAddLog('restricted popup width to 45% of window width...');
         }
         // 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);
         uroAddLog('popup height is '+rh);

         // similarly restrict the popup height to avoid it dropping of the bottom of the screen if a
         // segment has a bunch of closures/restrictions
         rh = parseInt(uroDiv.clientHeight);
         uroAddLog('popup height is '+rh);
         if(rh > (window.innerHeight * 0.80))
         {
            rh = (window.innerHeight * 0.80);
            uroDiv.style.height = rh+'px';
            uroDiv.style.overflow = 'scroll';
            uroAddLog('restricted popup height to 80% of window height...');
         }

         let origPopupX = uroPopupX;
         let movedLeft = false;
         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;
            movedLeft = true;
            uroAddLog('popup would fall off RHS of screen, reposition to other side of mouse pointer...');
            uroAddLog('uroPopupX is now '+uroPopupX);
         }
         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);
            uroAddLog('popup would fall off bottom of screen, shift up to keep it all visible...');
         }
         if(uroPopupY < 0) uroPopupY = 0;
         uroDiv.style.top = uroPopupY+'px';
         uroDiv.style.left = uroPopupX+'px';

         if(movedLeft === true)
         {
            // after relocating the popup to the left of the pointer, it may end up resizing itself
            // which may then cause it to completely overlap the UR marker, so perform one more check
            // of the div width and nudge to the left if required...
            rw = parseInt(uroDiv.clientWidth);
            uroAddLog('popup width after moving to left is '+rw);
            if(rw > (window.innerWidth * 0.45))
            {
               rw = (window.innerWidth * 0.45);
               uroDiv.style.width = rw+'px';
               uroAddLog('restricted popup width to 45% of window width...');
            }
            let nudgeDist = parseInt(20 + (uroPopupX + rw) - origPopupX);
            if((uroPopupX + rw + 30) >= origPopupX)
            {
               uroDiv.style.left = parseInt(uroPopupX - nudgeDist)+'px';
            }
         }

         uroDiv.style.visibility = 'visible';
         uroPopupAutoHideTimer = (uroGetElmValue('_inputPopupAutoHideTimeout') * 10);
      }
      uroPopupTimer = -1;
   }
   else
   {
      if((uroPopupTimer == -1) && (uroFID != uroShownFID))
      {
         uroPopupTimer = uroGetElmValue('_inputPopupEntryTimeout');
      }
   }
}
function uroExclusiveCB()
{
   let cbChecked = uroGetCBChecked(this.id);

   if(cbChecked === true)
   {
      let pairedList = this.attributes.pairedWith.value.split(',');
      for(let i=0; i<pairedList.length; i++)
      {
         uroSetCBChecked(pairedList[i], false);
      }
   }
}
function uroGetAMs(e)
{
   if(uroMTEMode) return;
   if(!uroFilterPreamble) return;
   if(!uroInitialised) return;
   if(document.getElementById("uroAMList") == null) return;
   if(document.getElementsByClassName('topbar') == null) return;

   if(uroGetCBChecked("_cbMoveAMList") === false)
   {
      document.getElementsByClassName('area-managers-region')[0].style.display = "block";
      uroAMList.innerHTML = uroModifyHTML("");
      document.getElementsByClassName('topbar')[0].style.backgroundColor=null;
      return;
   }

   document.getElementsByClassName('topbar')[0].style.backgroundColor="#000000";
   document.getElementsByClassName('area-managers-region')[0].style.display = "none";

   let amList = '';
   let tName = '';
   if(W.map.managedAreasLayer.getVisibility() === true)
   {
      let mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
      let mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top;
      let mousePixel = W.map.getLonLatFromPixel(new OpenLayers.Pixel(mouseX, mouseY));
      let mousePoint = new OpenLayers.Geometry.Point(mousePixel.lon, mousePixel.lat);

      for(let amObj in W.model.managedAreas.objects)
      {
         if(W.model.managedAreas.objects[amObj].geometry.containsPoint !== undefined)
         {
            if(W.model.managedAreas.objects[amObj].geometry.containsPoint(mousePoint))
            {
               let amName = uroGetUserNameFromID(W.model.managedAreas.objects[amObj].attributes.userID);
               if(amList.indexOf(amName) === -1)
               {
                  if(amList !== '') amList += ', ';
                  tName = uroGetUserNameAndRank(W.model.managedAreas.objects[amObj].attributes.userID);
                  if(tName.indexOf('a href') !== -1)
                  {
                     tName = tName.replace('a href', 'a style="color:#c0c0ff;" href');
                  }
                  amList += tName;
               }
            }
         }
      }
      if(amList === '')
      {
         amList = 'none';
      }
      amList = "&nbsp;-&nbsp;<b>Area Managers:</b> "+amList;
   }
   document.getElementById("uroAMList").innerHTML = uroModifyHTML(amList);
}
function uroNewTabAtMouseLoc(x, y)
{
   let tPix = new OpenLayers.Pixel(x,y);
   let mPos = W.map.getLonLatFromPixel(tPix).transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
   let nZoom = W.map.getZoom();
   if(nZoom < 17) nZoom = 17;
   let nHref = window.location.origin + window.location.pathname;
   nHref += '?lon=' + mPos.lon;
   nHref += '&lat=' + mPos.lat;
   nHref += '&zoomLevel=' + nZoom;
   window.open(nHref);
}
function uroMouseDown(e)
{
   uroMouseIsDown = true;
   if((e.altKey === true) && (e.ctrlKey === true))
   {
      uroNewTabAtMouseLoc(e.offsetX, e.offsetY);
   }
}
function uroMouseUp()
{
   uroMouseIsDown = false;
}
function uroTestPointerOutsideMap(mX, mY)
{
   let mapElm = document.getElementById("map");
   if(mapElm === undefined) return;
   let mapBCR = mapElm.getBoundingClientRect();

   if
   (
      (mX < mapBCR.left) ||
      (mX > mapBCR.right) ||
      (mY < mapBCR.top) ||
      (mY > mapBCR.bottom)
   )
   {
      uroPointerWithinMap = false;
      if(uroGetCBChecked('_cbKillInertialPanning') === true)
      {
         let controller = null;
         if (W.map.navigationControl) 
         {
		      controller = W.map.navigationControl;
         } 
         else if(W.map.controls.find(control => control.CLASS_NAME == 'OpenLayers.Control.Navigation'))
         {
            controller = W.map.controls.find(control => control.CLASS_NAME == 'OpenLayers.Control.Navigation');
         }
         if (controller !== null)
         {
            controller.dragPan.panMapStart();
         }
      }
      return true;
   }
   else
   {
      uroPointerWithinMap = true;
      return false;
   }
}
function uroMouseOut(e)
{
   if(uroTestPointerOutsideMap(e.clientX, e.clientY))
   {
      uroHidePopup('uroMouseOut');
   }
}
function uroUREvent_onObjectsAdded()
{
   if(uroGetCBChecked('_cbURResolverIDFilter') === true)
   {
      uroUpdateEditorList(W.model.mapUpdateRequests.objects, '_selectURResolverID', false, false, true, false);
   }
   if((uroBackfilling === false) && (uroPopulatingRequestSessions === false))
   {
      uroFilterURs();
   }
}
function uroGetSelectedURCommentCount()
{
   if(W.model.updateRequestSessions.objects[uroSelectedURID] != null)
   {
      let cachedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].attributes.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].attributes.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');
            }
            let 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.getAsync(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)
            {
               let currentLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].attributes.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.model.updateRequestSessions.objects[uroSelectedURID].attributes.comments.length;
   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();

   let 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 uroIncrementClosureDate(oldDate, incByDays)
{
   // Thanks to WME no longer using consistent ISO8601 date formatting when displaying
   // closure details, parsing the date string is now somewhat more involved.  Thanks devs...  

   // Default to returning the existing date, just in case we can't increment it
   let retval = oldDate;
   let sepChar = null;

   // Search through oldDate for a non-digit character, so we know what's
   // being used to seperate the year, month and date values by this locale...
   for(let i = 0; i < oldDate.length; ++i)
   {
      if((oldDate[i] < '0') || (oldDate[i] > '9'))
      {
         sepChar = oldDate[i];
         break;
      }
   }

   if(sepChar != null)
   {
      // First build a "probe" date object we can use to determine what the user's WME locale
      // does with dates.
      let incDate = new Date();
      incDate.setFullYear(3000);
      incDate.setMonth(10);
      incDate.setDate(22);

      // With these three carefully chosen date elements set, we now generate a localised datestring
      // using the WME locale setting - the first character of which then tells us the date
      // format - 1 = MDY (ack ptui), 2 = DMY (good), 3 = YMD (sweet!)
      let localeDate = incDate.toLocaleDateString(I18n.locale);
      let dateFormat = localeDate[0];

      // Now we know the seperator character and the date format, so we can finally start to parse
      // the existing date string...
      let oldDateBits = oldDate.split(sepChar);
   
      let datePos;
      let monthPos;
      let yearPos;
      if(dateFormat == '1')
      {
         datePos = 1;
         monthPos = 0;
         yearPos = 2;
      }
      else if(dateFormat == '2')
      {
         datePos = 0;
         monthPos = 1;
         yearPos = 2;
      }
      else
      {
         datePos = 2;
         monthPos = 1;
         yearPos = 0;
      }
      
      incDate.setFullYear(parseInt(oldDateBits[yearPos]));
      incDate.setMonth(parseInt(oldDateBits[monthPos]) - 1);
      incDate.setDate(parseInt(oldDateBits[datePos]));
      let tDate = new Date(incDate.getTime());
      incDate.setDate(tDate.getDate() + incByDays);

      retval = incDate.toLocaleDateString(I18n.locale);

      // Except for those pesky locales where toLocaleDateString() doesn't *quite*
      // return what WME is expecting.  Because, you know, why make it easy for
      // scripters when you can chuck in a few subtle curveballs like this, eh...

      if(I18n.locale == 'bg')
      {
         // Remove the "r." suffix
         retval = retval.split(" ")[0] + " ";
      }
   }
   
   return retval;
}
function uroGetElementProperty(elmName, elmOffset, elmProperty)
{
   let retval = null;
   if(document.getElementsByName(elmName).length > elmOffset)
   {
      retval = document.getElementsByName(elmName)[elmOffset][elmProperty];
   }
   else if(document.getElementById(elmName) !== null)
   {
      retval = document.getElementById(elmName)[elmProperty];
   }
   return retval;
}
function uroGetShadowElementProperty(elmName, shadowElmType, property)
{
   let retval = null;
   let tObj = document.getElementById(elmName);
   if(tObj !== null)
   {
      let sObj = tObj.shadowRoot.querySelector(shadowElmType);
      if(sObj !== null)
      {
         retval = sObj[property];
      }
   }
   return retval;
}
// Residential Place Cloning
//{
   let uroCRPStreetID;
   let 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
      let streetObj = W.model.streets.getByIds([uroCRPStreetID])[0];
      if(streetObj !== undefined)
      {
         document.getElementsByClassName('street-name')[0].value = streetObj.name;
         document.getElementsByClassName('street-name')[0].dispatchEvent(new Event('change', { 'bubbles': true }));

         // city name
         let cityObj = W.model.cities.getByIds([streetObj.cityID])[0];
         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)
               {
                  let snelm = document.getElementsByClassName('empty-city')[0];
                  snelm.checked = true;
                  snelm.dispatchEvent(new Event('change', { 'bubbles': true }));
               }
            }
            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...
      window.setTimeout(uroFinaliseCloneRP, 100);
   }
   function uroFinaliseCloneRP()
   {
      if(document.getElementsByClassName('address-form')[0].style.display != 'none')
      {
         window.setTimeout(uroFinaliseCloneRP, 100);
         return;
      }
      // once the user has applied the address changes and closed the address edit panel, WME will then
      // allow the place to be converted to residential...
      document.getElementsByClassName("toggle-residential")[0].click();
   }
   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)
      {
         window.setTimeout(uroConvertToRP, 100);
         return;
      }

      // panel is open, so move to the next step of the cloning procedure by clicking on the address edit icon...
      document.getElementsByClassName('full-address-container')[0].children[0].click();

      // now click on the "none" checkbox for the street name edit field so we can enter the street name
      let snelm = document.getElementsByClassName('empty-street')[0];
      snelm.checked = false;
      snelm.dispatchEvent(new Event('change', { 'bubbles': true }));

      // 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
      window.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)
      {
         let 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
//{
   let uroConfirmClosureDelete = true;
   let uroClosuresToDelete = 0;

   let uroCReason;
   let uroCEvent;
   let uroCDirection;
   let uroCStartDate;
   let uroCStartTime;
   let uroCEndDate;
   let uroCEndTime;
   let uroCIgnoreTraffic;
   let uroCClosedNodes;
   function uroCompleteClosureCloning()
   {
      let loop;
      
      if(document.getElementsByClassName('edit-closure').length === 0)
      {
         window.setTimeout(uroCompleteClosureCloning,100);
         return;
      }

      if(uroTempFixMTEDropDown() == false)
      {
         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...
      let form = $('#edit-panel .closures .edit-closure form');

      if(uroCReason !== null)
      {
         let fObj = form.find('wz-text-input#closure_reason');
         fObj.val(uroCReason);
         fObj.change();
      }
      if(uroCDirection !== null)
      {
         let fObj = form.find('wz-select#closure_direction');
         fObj[0].value = uroCDirection;
         fObj.change();
      }
      if(uroCStartDate !== null)
      {
         let fObj = form.find('input#closure_startDate');
         fObj.val(uroCStartDate);
         fObj.change();
      }
      if(uroCStartTime !== null)
      {
         let fObj = form.find('div.form-group.start-date-form-group input.time-picker-input');
         fObj.focus();
         fObj.val(uroCStartTime);
         fObj.change();
      }

      if(uroCIgnoreTraffic !== null)
      {
         let fObj = form.find('wz-checkbox#closure_permanent');
         fObj.val(uroCIgnoreTraffic);
         fObj.change();
      }
      if(uroCEndTime !== null)
      {
         let fObj = form.find('div.form-group.end-date-form-group input.time-picker-input');
         fObj.focus();
         fObj.val(uroCEndTime);
         fObj.change();
      }

      // 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)
      {
         let fObj = form.find('input#closure_endDate');
         fObj.val(uroCEndDate);
         fObj.change();
      }

      // the old method of setting the MTE just by changing the value attribute on closure_eventId no longer
      // seems to work as expected (it runs OK from the dev console, but not within the scope of the userscript),
      // so just as we do for setting the event to None, we set the event to the desired value here by finding
      // the appropriate menu entry and clicking on it...
      let cEvents = document.getElementById('closure_eventId').getElementsByTagName('wz-option');
      for(let i of cEvents)
      {
         if(i.value == uroCEvent)
         {
            i.click();
            break;
         }
      }

      let nNodes = uroCClosedNodes.length;
      if(nNodes > 0)
      {
         let fObj = form.find('wz-toggle-switch');
         for(loop = 0; loop < nNodes; ++loop)
         {
            if(uroCClosedNodes[loop] === true)
            {
               fObj[loop].click();
            }
         }
      }      
      if(uroGetCBChecked('_cbAutoApplyClonedClosure') == true)
      {
         window.setTimeout(uroClickClosureSave,100);
      }

      uroPendingClosureClone = -1;
   }
   function uroClickClosureSave()
   {
      document.getElementsByClassName('closures')[0].getElementsByClassName('save-button')[0].click();
   }

   let uroPendingClosureClone = -1;
   let uroPendingClosureCloneIncrement = 0;
   function uroCopyClosure(closureOffset)
   {
      // grab the current closure details from the UI...
      uroCReason = uroGetShadowElementProperty('closure_reason', 'input', 'value');
      uroCDirection = uroGetElementProperty('closure_direction', 0, 'value');
      uroCStartDate = uroGetElementProperty('closure_startDate', 0, 'value');
      uroCStartTime = document.querySelector('.start-date-form-group').querySelector('.time-picker-input').value;
      uroCEndDate = uroGetElementProperty('closure_endDate', 0, 'value');
      uroCEndTime = document.querySelector('.end-date-form-group').querySelector('.time-picker-input').value;
      uroCEvent = uroGetElementProperty('closure_eventId', 0, 'value');
      uroCIgnoreTraffic = uroGetElementProperty('closure_permanent', 0, 'checked');
      uroCClosedNodes = [];
      let nNodes = document.getElementsByClassName('fromNodeClosed').length;
      if(nNodes > 0)
      {
         for(let loop = 0; loop < nNodes; ++loop)
         {
            uroCClosedNodes.push(document.getElementsByClassName('fromNodeClosed')[loop].checked);
         }
      }

      document.getElementsByClassName('closures')[0].getElementsByClassName('cancel-button')[0].click();

      // auto-increment the start and end dates
      uroCStartDate = uroIncrementClosureDate(uroCStartDate,uroPendingClosureCloneIncrement);
      uroCEndDate = uroIncrementClosureDate(uroCEndDate,uroPendingClosureCloneIncrement);

      uroPendingClosureClone = -2;
   }
   function uroCloneClosure()
   {
      uroPendingClosureCloneIncrement = parseInt(this.id.split('-')[1]);
      uroPendingClosureClone = parseInt(this.id.split('-')[2]);
   }
   function uroDeleteNextClosureOnList()
   {
      let nClosures = document.querySelectorAll('.closure-item.is-editable').length;
      if(nClosures > 0)
      {
         if (nClosures != uroClosuresToDelete)
         {
            uroClosuresToDelete = nClosures;
            let ctObj = document.querySelector('.closure-item.is-editable');
            let deleteMenuEntry = ctObj.querySelector('wz-menu-item.delete');
            if(deleteMenuEntry !== null)
            {
               deleteMenuEntry.click();
            }
         }
         window.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;
      let nClosures = document.getElementsByClassName('closure-item').length;
      if(nClosures > 0)
      {
         uroClosuresToDelete = -1;
         uroDeleteNextClosureOnList();
      }
      else
      {
         uroConfirmClosureDelete = true;
      }
   }
//}

function uroGetMarkerType(marker)
{
   let markerType = null;

   if(marker.tagName == "image")
   {
      markerType = 'cam';
   }
   else if(marker.tagName == "circle")
   {
      markerType = 'node';
   }
   else
   {
      if(marker.className.indexOf('user-generated') !== -1) markerType = URO_TMARKER.UR;
      else if(marker.className.indexOf('map-problem') !== -1) markerType = URO_TMARKER.MP;
      else if(marker.className.indexOf('place-update') !== -1)
      {
         if(marker.parentNode.id === W.map.getLayerByName("PARKING_PLACE_UPDATES").div.id)
         {
            markerType = URO_TMARKER.PPUR;
         }
         else if(marker.parentNode.id === W.map.getLayerByName("RESIDENTIAL_PLACE_UPDATES").div.id)
         {
            markerType = URO_TMARKER.RPUR;
         }
         else if(marker.parentNode.id === W.map.getLayerByName("place_updates").div.id)
         {
            markerType = URO_TMARKER.PUR;
         }
      }
   }
   return markerType;
}
function uroGetCameraIDFromGeoID(geoID)
{
   let camID = null;
   let i=W.map.camerasLayer.features.length;
   let camObj;
   while(i > 0)
   {
      camObj = W.map.camerasLayer.features[i-1];
      if(camObj.attributes?.wazeFeature?._wmeObject !== undefined)
      {
         if(camObj.attributes.wazeFeature._wmeObject.geometry.id === geoID)
         {
            camID = camObj.attributes.wazeFeature._wmeObject.attributes.id;
            break;
         }
      }
      i--;
   }
   return camID;
}
function uroGetNodeIDFromGeoID(geoID)
{
   let nodeID = null;
   let i=W.map.nodeLayer.features.length;
   let nodeObj;
   while(i > 0)
   {
      nodeObj = W.map.nodeLayer.features[i-1];
      if(nodeObj.geometry.id === geoID)
      {
         nodeID = nodeObj?.attributes?.wazeFeature?._wmeObject?.attributes?.id;
         break;
      }
      i--;
   }
   return nodeID;
}
let uroLastMarkerMousedOver = null;
function uroMarkerMouseOver(e)
{
   let markerType = uroGetMarkerType(this);
   if(markerType !== null)
   {
      let markerID = null;
      if(markerType === 'cam')
      {
         let camObj = W.userscripts.getDataModelByFeatureElement(e.currentTarget);
         if(camObj !== null)
         {
            markerID = camObj.attributes.id;
            if(uroGetCBChecked('_cbHighlightInsteadOfHideCams') === true)
            {
               if(uroLastMarkerMousedOver !== markerID)
               {
                  window.setTimeout(uroFilterCameras, 50);
               }
            }
         }
      }
      else if(markerType === 'node')
      {
         markerID = uroGetNodeIDFromGeoID(this.id);
      }
      else
      {
         markerID = this.attributes["data-id"].value;
      }
      uroAddLog('hover over marker (Type '+markerType+' / ID '+markerID+')');
      uroMousedOverMarkerID = markerID;
      uroMousedOverMarkerType = markerType;

      if(markerType == URO_TMARKER.UR) uroHoveredURID = markerID;

      if((markerType == URO_TMARKER.UR) || (markerType == URO_TMARKER.MP))
      {
         uroChangeCustomMarkers(markerID,true,markerType);
      }
      uroLastMarkerMousedOver = markerID;
   }
   else
   {
      uroAddLog('hover over unknown object...');
   }
}
function uroMarkerMouseOut(e)
{
   let markerType;
   markerType = uroGetMarkerType(this);
   if(markerType !== null)
   {
      let markerID = null;
      if(markerType === 'cam')
      {
         markerID = uroGetCameraIDFromGeoID(this.id);
         if(uroGetCBChecked('_cbHighlightInsteadOfHideCams') === true)
         {
            window.setTimeout(uroFilterCameras, 50);
         }
      }
      else if(markerType === 'node')
      {
         markerID = uroGetNodeIDFromGeoID(this.id);
      }
      else
      {
         markerID = this.attributes["data-id"].value;
      }
      uroAddLog('hover off '+markerType+' ID '+markerID);
      uroMousedOverMarkerID = null;
      uroMousedOverMarkerType = null;
      uroHoveredURID = null;

      if((markerType == URO_TMARKER.UR) || (markerType == URO_TMARKER.MP))
      {
         uroChangeCustomMarkers(markerID,false,markerType);
      }
   }
   else
   {
      uroAddLog('hover off unknown object...');
   }
   uroLastMarkerMousedOver = null;
}

let uroInhibitSetCenter = false;
function uroMarkerMouseDown()
{
   // Do this stuff in the mousedown event rather than the click event so we fire before any of the native
   // click events - we need to ensure this happens for inhibiting marker centering, as we need to capture
   // the markerType ahead of our interceptor function being called to deal with the centering...
   let markerType = uroGetMarkerType(this);
   if(markerType !== null)
   {
      let markerID = this.attributes["data-id"].value;
      uroAddLog('clicked on '+markerType+' marker '+markerID);
      uroClickedOnMarkerID = markerID;
      uroClickedOnMarkerType = markerType;
      uroClickedOnMarkerForCenterInterceptor = markerType;
      uroClickedOnMarkerMapCenter = W.map.getCenter();

      uroInhibitURFiltering = true;

      /*
      if(uroInhibitSetCenter === false)
      {
         if(uroDecentre() === true)
         {
            console.debug('Inhibiting setCenter...');
            uroInhibitSetCenter = true;
            window.setTimeout(uroReenableSetCenter, 2000);
         }
      }
      */
   }
}
function uroDecentre()
{
   let inhibit = false;

   inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.UR) && (uroGetCBChecked("_cbInhibitURCentering")));
   inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.MP) && (uroGetCBChecked("_cbInhibitMPCentering")));
   inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.PUR) && (uroGetCBChecked("_cbInhibitPURCentering")));
   inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.PPUR) && (uroGetCBChecked("_cbInhibitPPURCentering")));
   inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.RPUR) && (uroGetCBChecked("_cbInhibitRPURCentering")));

   return inhibit;
}
function uroReenableSetCenter()
{
   console.debug('Re-enabling setCenter...');
   uroInhibitSetCenter = false;
}

function uroBlobMouseOver(e)
{
   let blobType = this.attributes.uroBlobType;
   if(blobType !== undefined)
   {
      let blobID = this.attributes.uroBlobID;
      uroAddLog('hover over '+blobType+' ID '+blobID);
   }
   else
   {
      uroAddLog('hover over unknown blob...');
   }
}
function uroBlobMouseOut(e)
{
   let blobType = this.attributes.uroBlobType;
   if(blobType !== undefined)
   {
      let blobID = this.attributes.uroBlobID;
      uroAddLog('hover off '+blobType+' ID '+blobID);
      if(blobType == 'map_comment')
      {
         if(W.model.mapComments.objects[blobID] != undefined)
         {
            let geoID = W.model.mapComments.objects[blobID].attributes.geometry.id;
            if(geoID.indexOf('Point') != -1)
            {
               // reapply visibility mods
               let svgElm = document.getElementById(uroMCLayer.div.id+'_vroot');
               for(let svgIdx = 0; svgIdx < svgElm.children.length; svgIdx++)
               {
                  if(svgElm.children[svgIdx].id === geoID)
                  {
                     window.setTimeout(uroReapplyPointMCVisibilityMods,10);
                  }
               }
            }
         }
      }
   }
   else
   {
      uroAddLog('hover off unknown blob...');
   }
}
function uroBlobClick()
{
   let blobType = this.attributes.uroBlobType;
   if(blobType !== undefined)
   {
      let blobID = this.attributes.uroBlobID;
      uroAddLog('clicked on '+blobType+' blob '+blobID);
   }
}
function uroMCLayerChanged_changed()
{
   uroMCLayerChanged();
}
function uroMCLayerChanged_added()
{
   uroMCLayerChanged();
}
function uroMCLayerChanged_removed()
{
   uroMCLayerChanged();
}
function uroReapplyPointMCVisibilityMods()
{
   if(uroApplyPointMCVisibilityMods() === false)
   {
      window.setTimeout(uroReapplyPointMCVisibilityMods,100);
   }
}
function uroApplyPointMCVisibilityMods()
{
   let retval = true;
   if(uroHasSelectedMCs() === true)
   {
      retval = false;
   }
   else
   {
      let svgElm = document.getElementById(uroMCLayer.div.id+'_vroot');
      for(let svgIdx = 0; svgIdx < svgElm.children.length; svgIdx++)
      {
         let svgChild = svgElm.children[svgIdx];
         if(svgChild.id.indexOf('Point') != -1)
         {
            if(uroGetCBChecked('_cbMCEnhancePointMCVisibility') === true)
            {
               if(svgChild.getAttribute('r') == 6)
               {
                  svgChild.setAttribute('fill','#ffff00');
                  svgChild.setAttribute('fill-opacity',0.75);
                  svgChild.setAttribute('r',12);
                  svgChild.setAttribute('touchedByURO',true);
               }
               else if((svgChild.getAttribute('touchedByURO') === "true")&&(svgChild.getAttribute('fill') === '#ffff00'))
               {
                  // do nothing...
               }
               else
               {
                  retval = false;
                  break;
               }
            }
            else
            {
               if((svgChild.getAttribute('touchedByURO') === "true")&&(svgChild.getAttribute('fill') === '#ffff00'))
               {
                  svgChild.setAttribute('fill','#ffffff');
                  svgChild.setAttribute('fill-opacity',1);
                  svgChild.setAttribute('r',6);
                  svgChild.setAttribute('touchedByURO',false);
               }
            }
         }
      }
   }
   return retval;
}
function uroHasSelectedMCs()
{
   let retval = false;
   for(let mcObj in W.model.mapComments.objects)
   {
      if(W.model.mapComments.objects[mcObj].isSelected() === true)
      {
         retval = true;
         break;
      }
   }
   return retval;
}
function uroMCLayerChanged()
{   
   uroWazeBits();
   if(uroMCLayer != null)
   {
      if(uroHasSelectedMCs() === false)
      {
         uroAddLog('adding MC blob event handlers');
         let mcModel = null;
         for(let mObj=0; mObj<uroMCLayer.features.length; mObj++)
         {
            if(uroMCLayer.features[mObj].attributes.wazeFeature._wmeObject !== undefined)
            {
               mcModel = uroMCLayer.features[mObj].attributes.wazeFeature._wmeObject;
               {
                  if(mcModel.selected !== true)
                  {
                     let mcBlobID = mcModel.attributes.geometry.id;
                     let mcID = mcModel.attributes.id;
                     let 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";
                        uroAddLog('added handlers to MC '+mcID);
                     }
                  }
               }
            }
         }
         uroApplyPointMCVisibilityMods();
      }
      else
      {
         uroAddLog('MC selected, handlers not added yet...');
      }
      
      uroFilterMapComments();
   }
}
function uroPlaceLayerChanged()
{
   uroAddLog('adding place blob event handlers');
   for(let mObj=0; mObj<uroVenueLayer.features.length; mObj++)
   {
      // clicking on an area place now adds the polygon drag handles into the features[] array, so we need to test that
      // the current array entry isn't referring to one of these handles before trying to access the attributes...
      if(uroVenueLayer.features[mObj]?.attributes?.wazeFeature?._wmeObject !== undefined)
      {
         let mcBlobID = uroVenueLayer.features[mObj].attributes.wazeFeature._wmeObject.attributes.geometry.id;
         let mcID = uroVenueLayer.features[mObj].attributes.wazeFeature._wmeObject.attributes.id;
         let 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 uroAddMarkerEventHandlers(layerType)
{
   let idList = uroGetMarkerIDs(layerType);
   for(let mObj of idList)
   {
      let mMarker = uroGetMarker(layerType, mObj);
      if(mMarker !== null)
      {
         let mIcon = mMarker.element;
         if(mIcon !== undefined)
         {
            mIcon.addEventListener("mouseover",uroMarkerMouseOver, false);
            mIcon.addEventListener("mouseout",uroMarkerMouseOut, false);
            mIcon.addEventListener("mousedown",uroMarkerMouseDown, false);
            ////mIcon.addEventListener("click",uroMarkerClick, false);
         }
      }
   }
}
function uroObserve_URLayer()
{
   uroMO_URLayer.observe(W.map.getLayerByName("update_requests").div,{childList: true, attributes : true, characterData : true, subtree: true});
}
function uroURLayerChanged()
{
   uroAddLog('UR layer change detected');
   uroAddMarkerEventHandlers(URO_TMARKER.UR);
   uroMO_URLayer.disconnect();
   uroFilterURs();
   uroObserve_URLayer();
}
function uroMPLayerChanged()
{
   uroAddLog('adding MP marker event handlers');
   uroAddMarkerEventHandlers(URO_TMARKER.MP);
}
function uroPURLayerChanged()
{
   uroAddLog('adding PUR marker event handlers');
   uroAddMarkerEventHandlers(URO_TMARKER.PUR);
}
function uroPPULayerChanged()
{
   uroAddLog('adding PPU marker event handlers');
   uroAddMarkerEventHandlers(URO_TMARKER.PPUR);
}
function uroRPULayerChanged()
{
   uroAddLog('adding RPU marker event handlers');
   uroAddMarkerEventHandlers(URO_TMARKER.RPUR);
}
function uroCamLayerChanged()
{
   uroAddLog('adding camera marker event handlers');
   let i = W.map.camerasLayer.features.length;
   let svgElm = null;
   while(i > 0)
   {
      svgElm = document.getElementById(W.map.camerasLayer.features[i-1].geometry.id);
      if(svgElm !== null)
      {
         svgElm.addEventListener("mouseover", uroMarkerMouseOver, false);
         svgElm.addEventListener("mouseout", uroMarkerMouseOut, false);
      }
      i--;
   }
}
let uroDelayNodeLayerUpdate = true;
function uroNodeLayerChanged()
{
   if(uroDelayNodeLayerUpdate === true)
   {
      // When the layer change event fires, WME hasn't yet updated the SVG content and so the node markers visible at this point
      // in time will get nuked once the update takes place.  We must therefore wait a short time to allow the SVG update to occur
      // before attaching event listeners to the node markers...
      uroDelayNodeLayerUpdate = false;
      window.setTimeout(uroNodeLayerChanged, 1000);
   }
   else
   {
      uroDelayNodeLayerUpdate = true;
      uroAddLog('adding node event handlers');
      let i = W.map.nodeLayer.features.length;

      if(i > 0)
      {
         let svgElm = null;
         while(i > 0)
         {
            svgElm = document.getElementById(W.map.nodeLayer.features[i-1].geometry.id);
            if(svgElm !== null)
            {
               svgElm.addEventListener("mouseover", uroMarkerMouseOver, false);
               svgElm.addEventListener("mouseout", uroMarkerMouseOut, false);
            }
            i--;
         }
         uroNodeLayerScanAttempts = 0;
      }
   }
}
function uroObserve_ClosuresLayer()
{
   uroMO_ClosuresLayer.observe(W.map.getLayerByName("closures").div,{childList: true, attributes : true, characterData : true, subtree: true});
}
function uroClosuresLayerChanged()
{
   uroAddLog('reapplying closures filter');
   uroMO_ClosuresLayer.disconnect();
   uroFilterRTCs();
   uroObserve_ClosuresLayer();
}
function uroSidePanelChanged()
{
   if(document.querySelector('wz-tab.closures-tab') !== null)
   {
      let uroMO_ClosureUI = new MutationObserver(uroClosureEditUIChanged);
      uroMO_ClosureUI.disconnect();
      uroMO_ClosureUI.observe(document.querySelector('wz-tab.closures-tab'),{subtree: true, attributes: true});
      uroClosureEditUIChanged();
   }
}
function uroClosureEditUIChanged()
{
   if(document.getElementsByClassName('edit-closure').length === 1)
   {
      // note: this also fires when the UI is closed, due to the change events triggered as its elements are removed 
      // prior to the tab itself closing...

      let notReady = 0;
      if(document.getElementById('closure_reason').shadowRoot.querySelector('input') === null) notReady += 1;
      if(document.getElementById('closure_direction').shadowRoot.querySelector('.selected-value') === null) notReady += 2;
      else if(document.getElementById('closure_direction').shadowRoot.querySelector('.selected-value').innerText === "") notReady += 4;
      if(document.getElementById('closure_startDate') === null) notReady += 8;
      if(document.getElementById('closure_endDate') === null) notReady += 16;
      if(document.getElementById('closure_eventId').shadowRoot.querySelector('.selected-value') === null) notReady += 32;
      else if(document.getElementById('closure_eventId').shadowRoot.querySelector('.selected-value').innerText === "") notReady += 64;
      if(document.getElementById('closure_permanent').shadowRoot.querySelector('.wz-checkbox') === null) notReady += 128;
      if(notReady === 0)
      {
         if(uroPendingClosureClone === -3)
         {
            uroCompleteClosureCloning();
         }
         else if(uroPendingClosureClone !== -1)
         {
            uroCopyClosure(uroPendingClosureClone);
         }
         else
         {
            uroTempFixMTEDropDown();
         }
      }
   }
   else
   {
      if(uroPendingClosureClone === -2)
      {
         // 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();
         uroPendingClosureClone = -3;
      }
      uroTSTClosureCloningHandler();
   }
}
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("objectsadded", uroUREvent_onObjectsAdded);

   W.model.cameras.on("objectschanged", uroCamLayerChanged);
   W.model.cameras.on("objectsadded", uroCamLayerChanged);
   W.model.cameras.on("objectsremoved", uroCamLayerChanged);
   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);

   W.model.mapComments.on("objectschanged", uroMCLayerChanged_changed);
   W.model.mapComments.on("objectsadded", uroMCLayerChanged_added);
   W.model.mapComments.on("objectsremoved", uroMCLayerChanged_removed);

   uroMO_PlaceLayer.observe(uroVenueLayer.div,{childList: true, attributes : true, characterData : true, subtree: true});
   uroObserve_URLayer();
   uroMO_MPLayer.observe(W.map.getLayerByName("mapProblems").div,{childList: true, attributes : true, characterData : true, subtree: true});
   uroMO_PURLayer.observe(W.map.getLayerByName("place_updates").div,{childList: true, attributes : true, characterData : true, subtree: true});
   uroMO_PPULayer.observe(W.map.getLayerByName("PARKING_PLACE_UPDATES").div,{childList: true, attributes : true, characterData : true, subtree: true});
   uroMO_RPULayer.observe(W.map.getLayerByName("RESIDENTIAL_PLACE_UPDATES").div,{childList: true, attributes : true, characterData : true, subtree: true});
   uroObserve_ClosuresLayer();
   uroMO_NodeLayer.observe(W.map.nodeLayer.div,{childList: true, attributes : true, characterData : true, subtree: true});
   uroMO_SidePanel.observe(document.getElementById('edit-panel'),{childList: true, subtree: true});

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

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

   for(let i = 0; i < uroCtrlTabs.length; ++i)
   {
      uroSetOnClick(uroCtrlTabs[i][URO_TABS_FIELD.LINKID], uroCtrlTabs[i][URO_TABS_FIELD.SHOWFN]);
   }

   for(let 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
   if(document.getElementsByClassName('street-view-control').length === 0) return;
   document.getElementsByClassName('street-view-control')[0].onmousedown = uroMouseDown;

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

   uroSetSectionTabStyles();

   uroLoadSettings();

   uroAddLog('getting user ID...');
   uroUserID = W.loginManager.user.attributes.id;
   uroAddLog('...ID is '+uroUserID);
   uroAddLog('filtering...');
   uroFilterItems();
   uroAddLog('...done');

   uroShowDebugOutput = uroPersistentDebugOutput;
   let 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
   uroAddLog('adding exclusiveCB handlers...');
   let cbList = document.getElementsByTagName('input');
   for (let optIdx=0;optIdx<cbList.length;optIdx++)
   {
      if((cbList[optIdx].id.indexOf('_cb') === 0) && (cbList[optIdx].attributes.pairedWith != null))
      {
         uroSetOnClick(cbList[optIdx].id,uroExclusiveCB);
      }
   }
   uroAddLog('...done');

   // 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();
   uroCamLayerChanged();
   uroNodeLayerChanged();
   uroPPULayerChanged();
   uroRPULayerChanged();
   uroClosuresLayerChanged();

   uroSetupListeners = false;
   uroMainTickStage = 0;
   window.clearInterval(uroMainTickHandlerID);
   window.setInterval(uroMainTick, 10);

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

   if(uroPopupShown === true)
   {
      let hidePopup = false;

      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 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)
   {
      let 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)
      {
         let doneString = I18n.lookup('problems.panel.done');
         let btnCaptionIsNextPlace = (nurButton.innerHTML.indexOf(I18n.lookup('venues.update_requests.panel.next_venue')) !== -1);
         let btnCaptionIsDefaultUR = (nurButton.innerHTML.indexOf(I18n.lookup('update_requests.panel.next')) !== -1);
         let btnCaptionIsDefaultMP = (nurButton.innerHTML.indexOf(I18n.lookup('problems.panel.next')) !== -1);
         let btnCaptionIsNextIssue = (nurButton.innerHTML.indexOf(I18n.lookup('feed.issues.next')) !== -1);

         let updateButton = false;

         let panelClass = W.map.panelRegion.$el[0].childNodes[0].childNodes[0].className;
         let isURorMPPanel = (panelClass.indexOf('problem-edit') !== -1);
         let 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 = uroModifyHTML(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);
         }

         uroInhibitURFiltering = false;
      }
   }
}
function uroTSTCommentAddedHandler()
{
   // test for the opening or closing of the UR editing dialog so we can detect when a new comment is added
   let URDialogIsOpen = false;
   let panelOpen = (document.getElementById('panel-container').firstChild !== null);

   if(panelOpen)
   {
      URDialogIsOpen = (document.getElementById('panel-container').getElementsByClassName('conversation').length > 0);
   }

   if(URDialogIsOpen)
   {
      let thisSelectedURID = document.getElementsByClassName('permalink')[0].href.split('&mapUpdateRequest=');
      if(thisSelectedURID.length > 1)
      {
         thisSelectedURID = thisSelectedURID[1].split('&')[0];
      }
      else
      {
         thisSelectedURID = null;
      }

      if((thisSelectedURID != uroSelectedURID) || ((thisSelectedURID != uroClickedOnMarkerID) && (uroClickedOnMarkerID != null)))
      {
         // 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;
         uroSelectedURID = null;
      }

      if(((uroURDialogIsOpen === false) && (uroSelectedURID === null)) || (uroURReclickAttempts > 0))
      {
         // user is editing a new UR

         // add our own click event handler to the Send button, so we can do stuff whenever a new comment is added
         if(document.getElementsByClassName('new-comment-form').length > 0)
         {
            if(document.getElementsByClassName('new-comment-form')[0].getElementsByClassName('send-button').length > 0)
            {
               document.getElementsByClassName('new-comment-form')[0].getElementsByClassName('send-button')[0].addEventListener("click", uroAddedComment, false);

               uroSelectedURID = thisSelectedURID;
               uroAddLog('user is editing UR '+uroSelectedURID);
               uroExpectedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].attributes.comments.length;

               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();
                     let urMarker = uroGetMarker(URO_TMARKER.UR,uroHoveredURID);
                     if(urMarker !== null)
                     {
                        let urMarkerAttributes = uroGetAttributes(URO_TMARKER.UR, uroHoveredURID);
                        if(urMarkerAttributes !== null)
                        {
                           urMarkerAttributes.geometry.x = urMarkerAttributes.geometry.realX;
                           urMarkerAttributes.geometry.y = urMarkerAttributes.geometry.realY;
                           uroOpenURDialog(uroHoveredURID);
                        }
                     }
                     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;
               uroFilterURs();
            }
         }
      }
   }
   else if(uroURDialogIsOpen === true)
   {
      // dialog was open and has now been closed
      uroSelectedURID = null;
      uroClickedOnMarkerID = null;
      uroFilterURs();
   }
   uroURDialogIsOpen = URDialogIsOpen;

   if(((uroPendingCommentDataRefresh === true) || (uroWaitingCommentDataRefresh === true)) && (uroSelectedURID !== null))
   {
      uroAddLog('check completion of comment data refresh for UR '+uroSelectedURID+' ('+uroPendingCommentDataRefresh+','+uroWaitingCommentDataRefresh+')');
      uroGetSelectedURCommentCount();
   }

}
function uroTSTOWLHandler()
{
/*
   let selectedTotal = uroSelectedItems.length;
   if((selectedTotal > 0) && (document.getElementById('_uroDivOWLBtns') === null))
   {
      let selectedClass = uroSelectedItems[0].model.CLASS_NAME;
      let displayAddToOWLBtn = false;
      let displayUpdateOWLBtn = false;
      let displayRemoveFromOWLBtn = false;
      let selectedSegments = false;
      let selectedLandmarks = false;
      let fid;
      let 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.indexOf("Feature.Vector.Segment") != -1)
      {
         selectedSegments = true;
         for(loop=0; loop<selectedTotal; loop++)
         {
            fid = uroSelectedItems[loop].model.attributes.id;
            let segIdx = uroIsSegOnWatchList(fid);
            if(segIdx == -1)
            {
               displayAddToOWLBtn = true;
            }
            else
            {
               if(uroSegDataChanged(segIdx))
               {
                  displayUpdateOWLBtn = true;
               }
               displayRemoveFromOWLBtn = true;
            }
         }
      }

      else if(selectedClass.indexOf("Feature.Vector.Landmark") != -1)
      {
         selectedLandmarks = true;
         for(loop=0; loop<selectedTotal; loop++)
         {
            fid = uroSelectedItems[loop].model.attributes.id;
            let placeIdx = uroIsPlaceOnWatchList(fid);
            if(placeIdx == -1)
            {
               displayAddToOWLBtn = true;
            }
            else
            {
               if(uroPlaceDataChanged(placeIdx))
               {
                  displayUpdateOWLBtn = true;
               }
               displayRemoveFromOWLBtn = true;
            }
         }
      }

      let 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...
   //
   if(uroGetCBChecked('_cbMasterEnable') === false)
   {
      return;
   }

   if
   (
      (document.querySelectorAll('.closures-list').length > 0) && 
      (document.querySelector('.closures-list').getAttribute('touchedbyuro') === null)
   )
   {      
      let nClosures;
      let cLoop;
      let btnElm;

      // Cloning doesn't work with certain locales due to the way the date strings are formatted...
      if
      (
         (I18n.locale == "fa-IR") || 
         (I18n.locale == 'ar') ||
         (I18n.locale == 'zh') ||
         (I18n.locale == 'ko')
      )
      {
         // Sorry :-(
      }
      else
      {
         // for the others, are there any closures defined for all of the selected segment(s)...
         if(document.getElementsByClassName('full-closures').length > 0)
         {
            nClosures = document.getElementsByClassName('full-closures')[0].querySelectorAll('.closure-item.is-editable').length;
            if(nClosures > 0)
            {
               // Force a refresh of uroRTCObjs for the selected segment, as this is no longer guaranteed to have already occurred...
               let segID = W.selectionManager.getSelectedFeatureIds()[0];
               uroGetSelectedSegmentRTCs(segID);

               // and if so, have we already added the clone icon?
               for(cLoop = 0; cLoop < nClosures; cLoop++)
               {
                  btnElm = document.getElementsByClassName('full-closures')[0].querySelectorAll('.closure-item.is-editable')[cLoop].getElementsByClassName('closure-title')[0];
                  if((btnElm.innerHTML.indexOf('_uroCloneClosure-') == -1) && (uroGetRTCOrigin(uroRTCObjs[cLoop]) !== URO_TRTC.UNKNOWN))
                  {
                     let newNode = document.createElement("div");
                     let anchorID1 = '_uroCloneClosure-1-'+cLoop;
                     let anchorID2 = '_uroCloneClosure-7-'+cLoop;
                     let newAnchor = '<a id="'+anchorID1+'" href="#">';
                     newAnchor += "<i style='font-size: 150%; cursor: copy' class='fa fa-copy'></i>";
                     newAnchor += "</a><sup>+1</sup>&nbsp;";
                     newAnchor += '<a id="'+anchorID2+'" href="#">';
                     newAnchor += "<i style='font-size: 150%; cursor: copy' class='fa fa-copy'></i>";
                     newAnchor += "</a><sup>+7</sup>";
                     newNode.innerHTML = uroModifyHTML(newAnchor);
                     btnElm.prepend(newNode);
                     uroAddEventListener(anchorID1,"click",uroCloneClosure,false);
                     uroAddEventListener(anchorID2,"click",uroCloneClosure,false);

                     let chipElm = btnElm.querySelector("wz-image-chip");
                     chipElm.innerHTML = chipElm.innerHTML.split('>')[0] + '>';
                  }
               }
            }
         }
      }

      // if there's more than one closure (full or partial) listed, also add the delete all button if not already present
      nClosures = document.querySelectorAll('.closure-item.is-editable').length;
      if(nClosures > 0)
      {
         if(document.getElementById('_btnDeleteAllClosures') === null)
         {
            let daDiv = document.createElement('wz-button');
            daDiv.className = 'delete-all-button btn is-expanded'; //btn-primary';
            daDiv.id = '_btnDeleteAllClosures';

            let tHTML = '<i class="fa fa-trash"></i> '+I18n.lookup("closures.delete_confirm_no_reason");
            if(nClosures > 1)
            {
               tHTML += ' ('+I18n.lookup("closures.apply_to_all")+')';
            }
            daDiv.innerHTML = uroModifyHTML(tHTML);
            daDiv.style.width = '100%';
            daDiv.style.marginBottom = '10px';

            let acBtn = document.getElementsByClassName('add-closure-button')[0];
            if(acBtn !== undefined)
            {
               acBtn.parentNode.insertBefore(daDiv, acBtn.nextSibling);
               uroAddEventListener('_btnDeleteAllClosures',"click", uroDeleteAllClosures, false);
            }
         }
      }

      let deferTouching = false;
      if(uroGetCBChecked('_cbAutoScrollClosureList') == true)
      {
         // Scroll to the end of the closure tab, as that's where the closure you're most likely to be cloning
         // is located...
         let cItems = document.getElementsByClassName('closure-item');
         if(cItems.length > 0)
         {
            let lastOne = cItems[cItems.length - 1];
            if(lastOne.getBoundingClientRect().height === 0)
            {
               deferTouching = true;
            }
            else
            {
               lastOne.scrollIntoView();
            }
         }
      }

      if(deferTouching === false)
      {
         document.getElementsByClassName('closures-list')[0].setAttribute('touchedbyuro','true');
      }
   }
}
function uroMiscUITweaksHandler()
{
   if(uroFilterPreamble())
   {
      // give user the option of setting their own background colour...
      {
         let mapviewport = document.getElementById("WazeMap").getElementsByClassName("olMapViewport")[0];
         if((uroGetCBChecked('_cbWhiteBackground') === true) && (uroGetCBChecked('_cbMasterEnable') === true))
         {
            let 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 hiding the somewhat unnecessary editor info panel at the top of the sidebar - disabled
      // in current beta due to changes in sidebar layout, code can be removed entirely once this change hits production
      {
         if(document.getElementById("user-details") !== null)
         {
            let panelDisplay = '';
            if(uroGetCBChecked('_cbHideEditorInfo'))
            {
               panelDisplay = "none";
            }
            document.getElementById("user-details").style.display = panelDisplay;
         }
      }

      // gives user the option of hiding the vector segments when the raster segment layer is hidden
      {
         if(uroGetCBChecked('_cbHideSegmentsWhenRoadsHidden'))
         {
            W.map.segmentLayer.drawn = W.map.roadLayer.visibility;
            W.map.nodeLayer.drawn = W.map.roadLayer.visibility;
         }
         else
         {
            W.map.segmentLayer.drawn = true;
            W.map.nodeLayer.drawn = true;
         }
      }

      let tDesc;
      // reformats the comment body text whenever the map comments sidepanel is opened, so that any linebreaks
      // added to the body text when the comment was created, and which are available in the W.model. data for
      // WME itself to include if only it could be arsed to do so, are included in the sidepanel rendering of
      // the text...
      {
         if(document.getElementsByClassName('map-comment-feature-editor').length > 0)
         {
            if(document.getElementsByClassName('map-comment-feature-editor')[0].touchedByURO === undefined)
            {
               if(document.getElementsByClassName('body-preview').length > 0)
               {
                  let mcID = -1;
                  if(uroFID == -1)
                  {
                     // if uroFID is -1, this implies the map comment panel has been opened as the result of WME
                     // being started up with a map comment included in the URL...
                     mcID = document.location.href.split('&mapComments=')[1];
                     if(mcID === undefined)
                     {
                        mcID = -1;
                     }
                  }
                  else
                  {
                     mcID = uroFID;
                  }
                  if(mcID !== -1)
                  {
                     if(W.model.mapComments.objects[mcID] !== undefined)
                     {
                        if(W.model.mapComments.objects[mcID].attributes !== undefined)
                        {
                           tDesc = W.model.mapComments.objects[mcID].attributes.body;
                           tDesc = uroClickify(tDesc, '');
                           document.getElementsByClassName('body-preview')[0].innerHTML = uroModifyHTML(tDesc);
                           document.getElementsByClassName('map-comment-feature-editor')[0].touchedByURO = true;
                        }
                     }
                  }
               }
            }
         }
      }

      // clickifies the ExtraInfo URL present in some MPs
      {
         if(document.getElementById('panel-container').getElementsByClassName('extraInfo').length > 0)
         {
            if(document.getElementById('panel-container').getElementsByClassName('extraInfo')[0].touchedByURO === undefined)
            {
               tDesc = document.getElementById('panel-container').getElementsByClassName('extraInfo')[0].innerHTML;
               tDesc = uroClickify(tDesc, '');
               document.getElementById('panel-container').getElementsByClassName('extraInfo')[0].innerHTML = uroModifyHTML(tDesc);
               document.getElementById('panel-container').getElementsByClassName('extraInfo')[0].touchedByURO = true;
            }
         }
      }

      if(W.map.getZoom() != uroLastZoom)
      {
         uroLastZoom = W.map.getZoom();
         uroMCLayerChanged();
      }
   }
}


function uroMainTick()
{
   if(uroMTEMode) return;
   if(uroSetupListeners)
   {
      if(uroFinalisingListenerSetup === false)
      {
         if(W.loginManager.isLoggedIn())
         {
            if(document.getElementsByClassName('topbar').length === 0) return;
            uroFinalizeListenerSetup();
            document.getElementsByClassName('topbar')[0].appendChild(uroAMList);
         }
      }
   }
   else
   {
      if(uroGetCBChecked('_cbMasterEnable') === true)
      {
         // grab a copy of the currently selected items list here, so that we only need to handle the production/beta API
         // differences in one place...
         if(W.selectionManager.selectedItems === undefined)
         {
            uroSelectedItems = W.selectionManager._selectedFeatures;
         }
         else
         {
            uroSelectedItems = W.selectionManager.selectedItems;
         }

         // do one maintick handler call in each 10ms cycle to minimise the time stuck within the maintick handler without
         // unduly affecting the overall response time for each individual handler

         if(uroMainTickStage === 0) uroTSTPopupHandler();
         if(uroMainTickStage == 1) uroTSTNextBtnHandler();
         if(uroMainTickStage == 2) uroTSTCommentAddedHandler();
         if(uroMainTickStage == 3) uroTSTOWLHandler();
         if(uroMainTickStage == 4) uroMiscUITweaksHandler();

         if(++uroMainTickStage == 6) uroMainTickStage = 0;
      }
   }
}
function uroActiveTab(_id)
{
   let e = document.getElementById(_id);
   e.style.backgroundColor = "greenyellow";
   e.style.borderTop = "1px solid";
   e.style.borderLeft = "1px solid";
   e.style.borderRight = "1px solid";
   e.style.borderBottom = "0px solid";
}
function uroInactiveTab(_id)
{
   let e = document.getElementById(_id);
   e.style.backgroundColor = "gainsboro";
   e.style.borderTop = "0px solid";
   e.style.borderLeft = "0px solid";
   e.style.borderRight = "0px solid";
   e.style.borderBottom = "1px solid";
}
function uroInactiveAllTabs()
{
   for(let i = 0; i < uroCtrlTabs.length; ++i)
   {
      uroInactiveTab(uroCtrlTabs[i][URO_TABS_FIELD.TABHEADER]);
      if(!uroCtrlsHidden)
      {
         uroSetStyleDisplay(uroCtrlTabs[i][URO_TABS_FIELD.TABBODY], 'none');
      }
   }   
}
function uroShowTab(tabID)
{
   uroInactiveAllTabs();
   uroActiveTab(uroCtrlTabs[tabID][URO_TABS_FIELD.TABHEADER]);
   if(!uroCtrlsHidden) uroSetStyleDisplay(uroCtrlTabs[tabID][URO_TABS_FIELD.TABBODY], 'block');
   return false;   
}
function uroShowURsTab()
{
   uroShowTab(URO_TABS_ID.URS);
   return false;
}
function uroShowMPsTab()
{
   uroShowTab(URO_TABS_ID.MPS);
   return false;
}
function uroShowMCsTab()
{
   uroShowTab(URO_TABS_ID.MCS);
   return false;
}
function uroShowPlacesTab()
{
   uroShowTab(URO_TABS_ID.PLACES);
   for(let idx=0;idx<uroPlacesGroupsCollapsed.length;idx++)
   {
      uroPlacesGroupCEHandler(idx);
   }
   return false;
}
function uroShowCamsTab()
{
   uroShowTab(URO_TABS_ID.CAMS);
   return false;
}
function uroShowOWLTab()
{
   uroShowTab(URO_TABS_ID.OWL);
   uroOWLUpdateHTML();
   return false;
}
function uroShowMiscTab()
{
   uroShowTab(URO_TABS_ID.MISC);
   return false;
}
function uroShowRTCsTab()
{
   uroShowTab(URO_TABS_ID.RTCS);
   return false;
}
function uroShowRAsTab()
{
   uroShowTab(URO_TABS_ID.RAS);
   return false;
}
function uroNewLookCheckDetailsRequest()
{
   let thisurl = document.location.href;
   let doRetry = true;
   let urID;
   let endmarkerpos = thisurl.indexOf('&endshow');
   let 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
      {
         uroGetMarker(URO_TMARKER.MP,urID).element.click();
         doRetry = false;
      }
      catch(err)
      {
         uroAddLog('problems not fully loaded, retrying...');
      }

      if(doRetry) window.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
         {
            uroGetMarker(URO_TMARKER.PUR, urID).element.click();
            doRetry = false;
         }
         catch(err)
         {
            uroAddLog('PURs not fully loaded, retrying...');
         }

         if(doRetry) window.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
            {
               uroGetMarker(URO_TMARKER.PPUR, urID).element.click();
               doRetry = false;
            }
            catch(err)
            {
               uroAddLog('PPURs not fully loaded, retrying...');
            }

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

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

   // build the list of all userIDs contained in the currently loaded venue objects
   let selectedIdx = null;
   let listedIDs = [];
   let idx;
   for(idx in W.model.venues.objects)
   {
      if(W.model.venues.objects.hasOwnProperty(idx))
      {
         let obj = W.model.venues.objects[idx].attributes;
         let cbID = obj.createdBy;
         let 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);
         }
      }
   }

   // check for any previously selected userIDs in the two selector lists, then clear both lists
   // and repopulate using the newly gathered ID collection from above, and finally reselect the
   // previously selected user if they're still present in the new list...
   let selector;
   let selectedUser;
   let users = W.model.users.getByIds(listedIDs);
   let selectorEntry;

   for(let i=0; i<2; i++)
   {
      if(i === 0) selector = document.getElementById('_selectPlacesUserID');
      else selector = document.getElementById('_selectHidePlacesUserID');

      selectedUser = null;
      if(selector.selectedOptions[0] != null)
      {
         selectedUser = parseInt(selector.selectedOptions[0].value);
      }
      while(selector.options.length > 0)
      {
         selector.options.remove(0);
      }
      selector.options.add(new Option('<select a user>', null));
      if(listedIDs.length > 0)
      {
         selectorEntry = '';
         for(idx=0; idx<users.length; idx++)
         {
            if(users[idx].attributes.userName === undefined)
            {
               selectorEntry = users[idx].attributes.id;
            }
            else
            {
               selectorEntry = users[idx].attributes.userName;
            }
            selector.options.add(new Option(selectorEntry, users[idx].id));
            if(users[idx].attributes.id == selectedUser)
            {
               selectedIdx = idx+1;
            }
         }
      }

      if(selectedIdx !== null)
      {
         selector.selectedIndex = selectedIdx;
      }
   }
}
function uroPlacesEditorSelected()
{
   let selector = document.getElementById('_selectPlacesUserID');
   if(selector.selectedIndex > 0)
   {
      document.getElementById('_textPlacesEditor').value = document.getElementById('_selectPlacesUserID').selectedOptions[0].innerHTML;
   }
}
function uroHidePlacesEditorSelected()
{
   let selector = document.getElementById('_selectHidePlacesUserID');
   if(selector.selectedIndex > 0)
   {
      document.getElementById('_textHidePlacesEditor').value = document.getElementById('_selectHidePlacesUserID').selectedOptions[0].innerHTML;
   }
}
function uroCamEditorSelected()
{
   let 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.flex = '1';
   obj.style.overflowY = 'auto';
}
function uroSetSectionTabStyles()
{
   for(let i =0; i < uroCtrlTabs.length; ++i)
   {
      uroSetStyles(uroCtrlTabs[i][URO_TABS_FIELD.TABBODY]);
   }
}
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()
{
   let groupidx = this.id.substr(21);
   if(uroPlacesGroupsCollapsed[groupidx] === true) uroPlacesGroupsCollapsed[groupidx] = false;
   else uroPlacesGroupsCollapsed[groupidx] = true;
   uroPlacesGroupCEHandler(groupidx);
   return false;
}
function uroWazeBits()
{
   let i;

   uroMCLayer = null;
   
   for(i=0;i<W.map.layers.length;i++)
   {
      if(W.map.layers[i].CLASS_NAME.indexOf('Layer.Vector.RootContainer') !== -1) uroRootContainer = W.map.layers[i].div.id;
	   if(W.map.layers[i].name == 'mapComments') uroMCLayer = W.map.layers[i];
	   if(W.map.layers[i].name == 'venues') uroVenueLayer = W.map.layers[i];
   }
   uroPlacesRoot = uroVenueLayer.id + '_vroot';
}
function uroGetMarkerIDs(markerType)
{
   let idList = [];

   for(let i = 0; i < uroMarkerLayers[markerType].length; ++i)
   {
      let dID = uroMarkerLayers[markerType][i]?.element?.attributes['data-id']?.value;
      if(dID !== undefined)
      {
         idList.push(dID);
      }
   }
   return idList;
}
function uroGetAttributes(markerType, markerID)
{
   if(markerType == URO_TMARKER.UR) return W.model.mapUpdateRequests.objects[markerID].attributes;
   if(markerType == URO_TMARKER.MP) return W.model.problems.objects[markerID].attributes;
   return null;
}
function uroGetMarker(markerType, markerID)
{
   if(typeof(markerID) === 'number')
   {
      markerID = markerID.toString();
   }

   let retval = null;
   let validMarkerType = false;
   validMarkerType = validMarkerType || (markerType === URO_TMARKER.UR);
   validMarkerType = validMarkerType || (markerType === URO_TMARKER.MP);
   validMarkerType = validMarkerType || (markerType === URO_TMARKER.PUR);
   validMarkerType = validMarkerType || (markerType === URO_TMARKER.PPUR);
   validMarkerType = validMarkerType || (markerType === URO_TMARKER.RPUR);
   validMarkerType = validMarkerType || (markerType === URO_TMARKER.RTC);
   if((validMarkerType === true) && (markerID !== null))
   {
      for(let i = 0; i < uroMarkerLayers[markerType].length; ++i)
      {
         if(uroMarkerLayers[markerType][i]?.element?.attributes['data-id']?.value === markerID)
         {
            retval = uroMarkerLayers[markerType][i];
         }
      }
   }
   return retval;
}
function uroGetMarkerLayers()
{
   uroMarkerLayers = [];

   // do NOT alter the order in which the marker objects are pushed into uroMarkerLayers[], unless the corresponding
   // URO_TMARKER.x values are also changed to reflect the new order...

   // URO_TMARKER.UR
   uroMarkerLayers.push(W.map.getLayerByName("update_requests").markers);

   // URO_TMARKER.MP
   uroMarkerLayers.push(W.map.getLayerByName("mapProblems").markers);

   // URO_TMARKER.PUR
   uroMarkerLayers.push(W.map.getLayerByName("place_updates").markers);

   // URO_TMARKER.PPUR
   uroMarkerLayers.push(W.map.getLayerByName("PARKING_PLACE_UPDATES").markers);

   // URO_TMARKER.RPUR
   uroMarkerLayers.push(W.map.getLayerByName("RESIDENTIAL_PLACE_UPDATES").markers);

   // URO_TMARKER.RTC
   uroMarkerLayers.push(W.map.getLayerByName("closures").markers);
}
function uroWazeBitsAvailable()
{
   uroAddLog('All WazeBits present and correct...');
   W.prefs.on('change:isImperial', uroInitialise);

   if(W.map.layers === undefined)
   {
      W.map.layers = W.map.getLayers();
   }
   if(W.map.events === undefined)
   {
      W.map.events = W.map.getMapEventsListener();
   }
   if(W.model.problems === undefined)
   {
      W.model.problems = W.model.mapProblems;
   }
   
   uroGetMarkerLayers();

   // To avoid creating multiple URO tabs in the event that the initialise code
   // gets called more than once, we now test for the presence of our tab and skip
   // if it's already present
   if(document.getElementById('uroTabHeader') === null)
   {
      uroSetupUI();
   }
}
function uroWaitForW()
{
   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;
   }

   if(window.W === undefined)
   {
      window.setTimeout(uroWaitForW, 100);
      return;
   }

   if (W.userscripts?.state?.isReady)
   {
      uroWazeBitsAvailable();
   } 
   else 
   {
      document.addEventListener("wme-ready", uroWazeBitsAvailable, {once: true});
   }
}
function uroAddInterceptor()
{
   uroAddLog('Adding interceptor functions...');

   // add interceptor function for window.confirm(), to inhibit the closure deletion confirmation that would
   // pop up for each individual closure when we're using the delete all button - the user has already 
   // confirmed the delete action using our own requester
   let _confirm = window.confirm;
   window.confirm = function(msg)
   {
      let cm_delete_confirm = I18n.lookup("closures.delete_confirm").split('"')[0].trimRight(1);

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

   let _setCenter = W.map.setProjectedCenter;
   W.map.setProjectedCenter = function(e,t,n,s)
   {
      if(uroInhibitSetCenter === true)
      {
         console.debug('Inhibited setCenter');
         return true;
      }
      else
      {
         return _setCenter.call(W.map, e,t,n,s);
      }
   };
   
   uroConfirmIntercepted = true;
}
function uroEnterPopup()
{
   uroMouseInPopup = true;
}
function uroExitPopup()
{
   uroMouseInPopup = false;
}
function uroToggleDebug()
{
   uroShowDebugOutput = !uroShowDebugOutput;
   let 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;

   uroWaitForW();
}
function uroWaitForControlsContainer()
{
   if(document.getElementById('uroControlsContainer') === null)
   {
      window.setTimeout(uroWaitForControlsContainer,500);
   }
   else
   {
      let updateURL = 'https://greasyfork.org/scripts/1952-uroverview-plus-uro';

      uroAddLog('adding controls to sidebar container...');
      let tabbyHTML = '<div id="uroTabHeader"><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>';
      
      let i;
      let tabsTotal = uroCtrlTabs.length;
      let tabsPerRow = Math.ceil(tabsTotal / Math.ceil(tabsTotal / URO_MAX_TABS_PER_ROW));
      let tabCount = 0;
      for(i = 0; i < tabsTotal; ++i)
      {
         if(tabCount == 0)
         {
            tabbyHTML += '<table border=0 width="100%"><tr>';
         }
         tabbyHTML += '<td valign="center" align="center" id="'+uroCtrlTabs[i][URO_TABS_FIELD.TABHEADER]+'">';
         tabbyHTML += '<a href="#" id="'+uroCtrlTabs[i][URO_TABS_FIELD.LINKID]+'" style="text-decoration:none;font-size:12px">'+uroCtrlTabs[i][URO_TABS_FIELD.TABTITLE]+'</a></td>';
         if(((++tabCount == tabsPerRow) && (i < (tabsTotal - 1))) || (i == (tabsTotal - 1)))
         {
            tabbyHTML += '</tr></table>';
            tabCount = 0;
         }
      }
      tabbyHTML += '</div>';
      document.getElementById('uroControlsContainer').innerHTML = uroModifyHTML(tabbyHTML);

      // tab elements
      for(i = 0; i < uroCtrlTabs.length; ++i)
      {
         uroCtrlTabs[i][URO_TABS_FIELD.TABBODY] = document.createElement('div');
      }

      // other sidebar elements
      uroAMList = document.createElement('div');
      uroAMList.style.color = "#ffff00";
      uroCtrlHides = document.createElement('div');

      // Object watchlist tab
      {
         uroCWLGroups = [];
         uroOWLUpdateHTML();
      }

      // footer for tabs container
      uroCtrlHides.id = 'uroCtrlHides';
      let tHTML = '<input type="button" id="_btnUndoLastHide" value="Undo last hide" />&nbsp;&nbsp;&nbsp;';
      tHTML += '<input type="button" id="_btnClearSessionHides" value="Undo all hides" /><p>';
      uroCtrlHides.innerHTML = uroModifyHTML(tHTML);

      // footer for AM list
      uroAMList.id = 'uroAMList';
      
      uroCtrlTabs[0][URO_TABS_FIELD.SHOWFN]();

      window.addEventListener("beforeunload", uroSaveSettings, false);
   }
}
async function uroAddTab_API()
{
   let {tabLabel, tabPane} = W.userscripts.registerSidebarTab("URO+");
   tabLabel.innerText = "URO+";
   tabPane.innerHTML = uroModifyHTML(uroControls.innerHTML);
   await W.userscripts.waitForElementConnected(tabPane);
   uroCompleteUISetup();
}
function uroCompleteUISetup()
{
   uroAddLog('waiting for controls container...');
   uroWaitForControlsContainer();

   uroDOMHasTurnProblems = (W.model.turnProblems != null);
         
   uroGetProblemTypes();
   
   let i;
   for(i = 0; i < uroCtrlTabs.length; ++i)
   {
      if(uroCtrlTabs[i][URO_TABS_FIELD.POPULATEFN] != null)
      {
         uroCtrlTabs[i][URO_TABS_FIELD.TABBODY].innerHTML = uroCtrlTabs[i][URO_TABS_FIELD.POPULATEFN]();
      }
      document.getElementById('uroControlsContainer').appendChild(uroCtrlTabs[i][URO_TABS_FIELD.TABBODY]);
      uroCtrlTabs[i][URO_TABS_FIELD.TABBODY].onclick = uroCtrlTabs[i][URO_TABS_FIELD.CLICKFN];
   }
   document.getElementById('uroControlsContainer').appendChild(uroCtrlHides);

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

   uroAddStyle('urostyle_UnstackedMarkers', '.map-marker.marker-selected { transform: scale(1) !important; }');

   uroMainTickHandlerID = window.setInterval(uroMainTick,1000);
}
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%)";

   let 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 = uroModifyHTML(alertsHTML);
   document.body.appendChild(uroAlerts);

   uroControls = document.createElement('section');
   uroControls.style.fontSize = '12px';
   uroControls.style.height = '100%';
   uroControls.id = "sidepanel-uroverview";
   uroControls.className = "tab-pane";
   uroControls.innerHTML = uroModifyHTML('<div id="uroControlsContainer" style="display:flex;flex-direction:column;height:80vh;"></div>');

   uroAddTab_API();
}
function uroAddStyle(ID, css) 
{
	let head, style;
	head = document.getElementsByTagName('head')[0];
	if (!head) 
	{
		return;
	}
	
   uroRemoveStyle(ID); // in case it is already there
	style = document.createElement('style');
	style.type = 'text/css';
	style.innerHTML = uroModifyHTML(css);
	style.id = ID;
	head.appendChild(style);
}
function uroRemoveStyle(ID) 
{
	let style = document.getElementById(ID);
	if (style) 
	{
		style.parentNode.removeChild(style); 
	}
}

uroInitialise();