// ==UserScript==
// @name UROverview Plus (URO+)
// @namespace http://greasemonkey.chizzum.com
// @description Adds filtering and pop-up infobox for UR, MP and camera markers
// @include https://*.waze.com/*editor*
// @include https://editor-beta.waze.com/*
// @grant none
// @version 3.59
// ==/UserScript==
/*
TO-DO ITEMS
=======================================================================================================================
Bug fixes - MUST BE CLEARED BEFORE RELEASE
=======================================================================================================================
=======================================================================================================================
Things to be checked
=======================================================================================================================
=======================================================================================================================
Proposed functionality
=======================================================================================================================
Option to shuffle the UI elements around to restore sanity
- Street/City/County/Country fields in the segment edit panel
Flush settings to localStorage whenever a change is made, or at least before opening a new tab via a popup
User-defined setting presets
Extend unstacking to cameras
Place filtering
- by last user to edit
Localisation
First-run information
- show quickstart guide to URO features if no existing settings are present (i.e. new installation)
=======================================================================================================================
New functionality in progress
=======================================================================================================================
Addition of segment and place watchlist functionality
*/
/* JSHint Directives */
/* globals $: */
/* globals W: true */
/* globals I18n: */
/* globals loginManager: true */
/* globals selectionManager: true */
/* globals OL: true */
/* globals OpenLayers: true */
/* globals Waze: true */
/* globals unsafeWindow: true */
/* globals Components: true */
/* globals require: */
/* jshint bitwise: false */
var uroVersion = "3.59";
var uroReleaseDate = "20151108";
// list of changes affecting all users
var uroChanges =
[
'Clicking on a backfilled UR works again',
'Option to disable map auto-centering when clicking on URs',
'Option to disable filtering of any UR listed in the URL'
];
// list of changes affecting only WME Beta users (at least until the next production release including these parts of the beta code...)
var uroBetaChanges =
[
];
// true enables debug output during script startup
var uroShowDebugOutput = true;
// true keeps debug output enabled after script startup
var uroPersistentDebugOutput = false;
var uroCtrlsHidden = false;
var uroCurrentTab = 1;
var uroFID = -1;
var uroShownFID = -1;
var uroShownPopupType = null;
var uroInhibitSave = true;
var uroPopupTimer = -2;
var uroPopupDwellTimer = -1;
var uroPopupShown = false;
var uroSetupListeners = true;
var uroRootContainer = null;
var uroPlacesRoot = null;
var uroMaskLayer = null;
var uroCustomMarkerFID = null;
var uroCustomMarkerType = null;
var uroConfirmIntercepted = false;
var uroCustomMarkerList = [];
var uroPendingURSessionIDs = [];
var uroRequestedURSessionIDs = [];
var uroPlacesGroupsCollapsed = [];
var uroMouseInPopup = false;
var uroURControlsIdx = null;
var uroProblemControlsIdx = null;
var uroTurnsLayerIdx = null;
var uroNullCamLayer = false;
var uroNullOpenLayers = false;
var uroNullRootContainer = false;
var uroNullURLayer = false;
var uroNullProblemLayer = false;
var uroNullMapViewport = false;
var uroURDialogIsOpen = false;
var uroSelectedURID = null;
var uroPendingCommentDataRefresh = false;
var uroWaitingCommentDataRefresh = false;
var uroExpectedCommentCount = null;
var uroCachedLastCommentID = null;
var uroPlaceSelected = false;
var uroMouseIsDown = false;
var uroBackfilling = false;
var uroHidePopupOnPanelOpen = false;
var uroUserID = -1;
var uroURIDInURL = null;
var uroDOMHasTurnProblems = false;
var uroBetaEditor = false;
var uroPracticeMode = false;
var uroWazeBitsPresent = 0;
var uroOWL = null;
var uroDiv = null;
var uroControls = null;
var uroCtrlURs = null;
var uroCtrlMPs = null;
var uroCtrlPlaces = null;
var uroCtrlCameras = null;
var uroCtrlMisc = null;
var uroCtrlHides = null;
var uroAMList = [];
var uroCWLGroups = [];
var uroCamWatchObjects = [];
var uroSegWatchObjects = [];
var uroPlaceWatchObjects = [];
var uroFriendlyAreaNames = [];
var uroAreaNameHoverTime = -1;
var uroAreaNameHoverObj = null;
var uroAreaNameOverlayShown = false;
var uroANEditHovered = false;
var uroANEditBox = null;
var uroPrevMouseX = -1;
var uroPrevMouseY = -1;
var dteControlsIdx = -1;
var dteOldestFullDrive = new Date(0);
var dteEpoch = new Date(0);
var dteTopID = '';
var dteClearHighlightsOnPanelClose = false;
var dteArmClearHighlightsOnPanelClose = false;
var uroUserTabId = '';
var uroTBRObj = null;
var uroUnstackedMasterID = null;
var uroStackList = [];
var uroStackType = null;
var uroIcons =
[
// stuff used within the URO tabbed UI
// 0 = group collapse/group expand
[
"",
""
],
// 1 = addtogroup active/addtogroup idle
[
"",
""
],
// 2 = goto active/goto idle
[
"",
""
],
// replacements for native UR/problem markers
// 3 = normal/selected open roadworks UR
[
"",
""
],
// 4 = normal/selected closed roadworks UR
[
"",
""
],
// 5 = normal/selected open closure UR
[
"",
""
],
// 6 = normal/selected closed closure UR
[
"",
""
],
// 7 = normal/selected open event UR
[
"",
""
],
// 8 = normal/selected closed event UR
[
"",
""
],
// 9 = normal/selected open note UR
[
"",
""
],
// 10 = normal/selected closed note UR
[
"",
""
],
// 11 = normal/selected open Elgin MP
[
"",
""
],
// 12 - normal/selected closed Elgin MP
[
"",
""
],
// 13 - normal/selected open TrafficCast MP
[
"",
""
],
// 14 - normal/selected closed TrafficCast MP
[
"",
""
],
// 15 - normal/selected open TrafficMaster MP
[
"",
""
],
// 16 - normal/selected closed TrafficMaster MP
[
"",
""
]
];
var uroMarkers =
[
// 0 = comment count circle
[""],
// 1 = green comment marker
[""],
// 2 = yellow (own) comment marker
[""]
];
function uroAddLog(logtext)
{
if(uroShowDebugOutput) console.log('URO+: '+logtext);
}
function uroGetCBChecked(cbID)
{
return(document.getElementById(cbID).checked);
}
function uroSetCBChecked(cbID, state)
{
document.getElementById(cbID).checked = state;
}
function uroGetElmValue(elmID)
{
return(document.getElementById(elmID).value);
}
function uroSetStyleDisplay(elm,style)
{
document.getElementById(elm).style.display = style;
}
function uroSetOnClick(elm,fn)
{
document.getElementById(elm).onclick = fn;
}
function uroAddEventListener(elm,eventType,eventFn,eventBool)
{
document.getElementById(elm).addEventListener(eventType, eventFn, eventBool);
}
function uroFirstTimerWelcomePack()
{
uroAddLog('welcome new users to Club URO...');
// to be completed...
}
function uroShowUpdateNotes()
{
uroAddLog('let existing users know what\'s new in this release');
var alertMsg = 'URO+ Update Notes...\n\n';
alertMsg += 'Thanks for upgrading to URO+ '+uroVersion+' ('+uroReleaseDate+'). What\'s changed?\n\n';
var loop;
if(uroChanges.length > 0)
{
for(loop=0; loop < uroChanges.length; loop++)
{
alertMsg += '* '+uroChanges[loop]+'\n';
}
}
if((uroBetaEditor) && (uroBetaChanges.length > 0))
{
alertMsg += '\nFor WME Beta:\n';
for(loop=0; loop < uroBetaChanges.length; loop++)
{
alertMsg += '* '+uroBetaChanges[loop]+'\n';
}
}
alert(alertMsg);
}
function uroAdvertiseCustomIcons()
{
uroAddLog('advertise the benefits of custom UR icons...');
var confirmMsg = 'URO+ Installation/Upgrade Processing...\n\n';
confirmMsg += '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.\n\n';
confirmMsg += 'Markers are defined for [ROADWORKS], [CONSTRUCTION], [CLOSURE], [EVENT] and [NOTE] tags in URs, and [Elgin], [TM] and [TrafficCast] in MPs.\n\n';
confirmMsg += 'Would you like me to automatically enable these custom markers?\n\n';
confirmMsg += 'If you change your mind later on, they can be enabled/disabled via the Misc tab within the URO+ settings';
if(confirm(confirmMsg) === true)
{
uroSetCBChecked('_cbCustomRoadworksMarkers', true);
uroSetCBChecked('_cbCustomConstructionMarkers', true);
uroSetCBChecked('_cbCustomClosuresMarkers', true);
uroSetCBChecked('_cbCustomEventsMarkers', true);
uroSetCBChecked('_cbCustomNotesMarkers', true);
uroSetCBChecked('_cbCustomElginMarkers', true);
uroSetCBChecked('_cbCustomTrafficMasterMarkers', true);
uroSetCBChecked('_cbCustomTrafficCastMarkers', true);
}
}
function uroGatherSettings(container)
{
var options = '';
var urOptions = document.getElementById(container).getElementsByTagName('input');
for (var optIdx=0;optIdx<urOptions.length;optIdx++)
{
var id = urOptions[optIdx].id;
if((id.indexOf('_cb') === 0)||(id.indexOf('_text') === 0)||(id.indexOf('_input') === 0))
{
options += ':' + id;
if(urOptions[optIdx].type == 'checkbox') options += ',' + urOptions[optIdx].checked.toString();
else if((urOptions[optIdx].type == 'text')||(urOptions[optIdx].type == 'number')) options += ',' + urOptions[optIdx].value.toString();
}
}
return options;
}
function uroGatherCamWatchList()
{
var liststr = '';
for(var loop=0;loop<uroCamWatchObjects.length;loop++)
{
var camObj = uroCamWatchObjects[loop];
if((camObj.fid !== undefined) && (camObj.persistent === true))
{
if(loop > 0) liststr += ':';
liststr += camObj.fid+',';
liststr += camObj.watch.lon+',';
liststr += camObj.watch.lat+',';
liststr += camObj.watch.type+',';
liststr += camObj.watch.azymuth+',';
liststr += camObj.watch.speed+',';
liststr += camObj.watch.validated+',';
liststr += camObj.groupID+',';
liststr += camObj.server;
}
}
return liststr;
}
function uroGatherSegWatchList()
{
var liststr = '';
for(var loop=0;loop<uroSegWatchObjects.length;loop++)
{
var segObj = uroSegWatchObjects[loop];
if((segObj.fid !== undefined) && (segObj.persistent === true))
{
if(loop > 0) liststr += ':';
liststr += segObj.fid+',';
liststr += segObj.watch.left+',';
liststr += segObj.watch.right+',';
liststr += segObj.watch.bottom+',';
liststr += segObj.watch.top+',';
liststr += segObj.watch.fromNode+',';
liststr += segObj.watch.toNode+',';
liststr += segObj.watch.fwdDir+',';
liststr += segObj.watch.revDir+',';
liststr += segObj.watch.length+',';
liststr += segObj.watch.level+',';
liststr += segObj.watch.rank+',';
liststr += segObj.watch.roadType+',';
liststr += segObj.watch.updatedOn+',';
liststr += segObj.groupID+',';
liststr += segObj.server;
}
}
return liststr;
}
function uroGatherPlaceWatchList()
{
var liststr = '';
for(var loop=0;loop<uroPlaceWatchObjects.length;loop++)
{
var placeObj = uroPlaceWatchObjects[loop];
if((placeObj.fid !== undefined) && (placeObj.persistent === true))
{
if(loop > 0) liststr += ':';
liststr += placeObj.fid+',';
liststr += placeObj.watch.left+',';
liststr += placeObj.watch.right+',';
liststr += placeObj.watch.bottom+',';
liststr += placeObj.watch.top+',';
liststr += placeObj.watch.name+',';
liststr += placeObj.watch.imageCount+',';
liststr += placeObj.watch.residential+',';
liststr += placeObj.watch.updatedOn+',';
liststr += placeObj.groupID+',';
liststr += placeObj.server;
}
}
return liststr;
}
function uroGatherCWLGroups()
{
var liststr = '';
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
var groupObj = uroCWLGroups[loop];
if(groupObj.groupID != -1)
{
if(loop > 0) liststr += ':';
liststr += groupObj.groupID+',';
liststr += groupObj.groupName+',';
liststr += groupObj.groupCollapsed;
}
}
return liststr;
}
function uroGatherPlacesGroups()
{
var liststr = '';
for(var loop=0;loop<uroPlacesGroupsCollapsed.length;loop++)
{
if(loop > 0) liststr += ':';
liststr += uroPlacesGroupsCollapsed[loop];
}
return liststr;
}
function uroGatherFriendlyAreaNames()
{
var liststr = '';
for(var loop=0;loop<uroFriendlyAreaNames.length;loop++)
{
var fnObj = uroFriendlyAreaNames[loop];
if(loop > 0) liststr += ':';
liststr += fnObj.fName+',';
liststr += fnObj.area+',';
liststr += fnObj.server;
}
return liststr;
}
function uroSaveSettings()
{
if(uroInhibitSave)
{
uroAddLog('save inhibited');
return;
}
if (localStorage)
{
localStorage.UROverviewUROptions = uroGatherSettings('uroCtrlURs');
localStorage.UROverviewMPOptions = uroGatherSettings('uroCtrlMPs');
localStorage.UROverviewCameraOptions = uroGatherSettings('uroCtrlCameras');
localStorage.UROverviewMiscOptions = uroGatherSettings('uroCtrlMisc');
localStorage.UROverviewPlacesOptions = uroGatherSettings('uroCtrlPlaces');
localStorage.UROverviewCamWatchList = uroGatherCamWatchList();
localStorage.UROverviewSegWatchList = uroGatherSegWatchList();
localStorage.UROverviewPlaceWatchList = uroGatherPlaceWatchList();
localStorage.UROverviewCWLGroups = uroGatherCWLGroups();
localStorage.UROverviewFriendlyAreaNames = uroGatherFriendlyAreaNames();
localStorage.UROverviewPlacesGroups = uroGatherPlacesGroups();
localStorage.UROverviewMasterEnable = uroGetCBChecked('_cbMasterEnable');
localStorage.UROverviewCurrentVersion = uroVersion;
uroAddLog('save complete');
}
else
{
uroAddLog('no localStorage, save blocked');
}
}
function uroApplySettings(settings)
{
var options = settings.split(':');
for(var optIdx=0;optIdx<options.length;optIdx++)
{
var fields = options[optIdx].split(',');
if(fields[0].indexOf('_cb') === 0)
{
if(document.getElementById(fields[0]) !== null)
{
uroSetCBChecked(fields[0], (fields[1] == 'true'));
}
}
else if((fields[0].indexOf('_input') === 0)||(fields[0].indexOf('_text') === 0))
{
if(document.getElementById(fields[0]) !== null) document.getElementById(fields[0]).value = fields[1];
}
}
}
function uroApplyCamWatchList()
{
var objects = localStorage.UROverviewCamWatchList.split(':');
uroCamWatchObjects = [];
if(objects.length > 0)
{
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
if(fields.length >= 7)
{
// following two bits of code add in blank fields if the user has updated their copy of URO+ from an
// older version which didn't include support for either of these field types
// add default groupID field
if(fields.length == 7)
{
fields.push(0);
}
// set default groupID value to 0 (no group)
if(fields[7] == -1)
{
fields[7] = 0;
}
// add default server field
if(fields.length == 8)
{
fields.push('??');
}
// set default server value to unknown
if(fields[8] === 0)
{
fields[8] = '??';
}
uroCamWatchObjects.push(new uroCamWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7],fields[8]));
}
}
}
}
function uroApplySegWatchList()
{
var objects = localStorage.UROverviewSegWatchList.split(':');
uroSegWatchObjects = [];
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
uroSegWatchObjects.push(new uroSegWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7],fields[8],fields[9],fields[10],fields[11],fields[12],fields[13],fields[14],fields[15]));
}
}
function uroApplyPlaceWatchList()
{
var objects = localStorage.UROverviewPlaceWatchList.split(':');
uroPlaceWatchObjects = [];
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
uroPlaceWatchObjects.push(new uroPlaceWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7],fields[8],fields[9],fields[10]));
}
}
function uroApplyCWLGroups()
{
var objects = localStorage.UROverviewCWLGroups.split(':');
uroCWLGroups = [];
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
if(fields.length < 2)
{
fields.push(false);
}
uroCWLGroups.push(new uroOWLGroupObj(fields[0],fields[1],(fields[2] == 'true')));
}
}
function uroApplyPlacesGroups()
{
var t = localStorage.UROverviewPlacesGroups.split(':');
for(var i=0;i<t.length;i++)
{
uroPlacesGroupsCollapsed[i] = (t[i] == "true");
}
}
function uroApplyFriendlyAreaNames()
{
var objects = localStorage.UROverviewFriendlyAreaNames.split(':');
uroFriendlyAreaNames = [];
for(var objIdx=0;objIdx<objects.length;objIdx++)
{
var fields = objects[objIdx].split(',');
uroFriendlyAreaNames.push(new uroAFNObj(fields[0],parseFloat(fields[1]),fields[2]));
}
uroReplaceAreaNames(true);
}
function uroLoadSettings()
{
var isNewInstall = true;
var isUpgradeInstall = true;
var notifyAboutCustomIcons = true;
uroAddLog('loadSettings()');
if (localStorage.UROverviewUROptions !== undefined)
{
uroAddLog('recover UR tab settings');
uroApplySettings(localStorage.UROverviewUROptions);
isNewInstall = false;
}
if (localStorage.UROverviewCameraOptions !== undefined)
{
uroAddLog('recover camera tab settings');
uroApplySettings(localStorage.UROverviewCameraOptions);
isNewInstall = false;
}
if (localStorage.UROverviewMPOptions !== undefined)
{
uroAddLog('recover MP tab settings');
uroApplySettings(localStorage.UROverviewMPOptions);
isNewInstall = false;
}
if (localStorage.UROverviewPlacesOptions !== undefined)
{
uroAddLog('recover Places tab settings');
uroApplySettings(localStorage.UROverviewPlacesOptions);
isNewInstall = false;
}
if (localStorage.UROverviewMiscOptions !== undefined)
{
uroAddLog('recover misc tab settings');
uroApplySettings(localStorage.UROverviewMiscOptions);
isNewInstall = false;
if(localStorage.UROverviewCurrentVersion !== undefined)
{
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(localStorage.UROverviewCWLGroups !== undefined)
{
uroAddLog('recover CWL groups');
uroApplyCWLGroups();
isNewInstall = false;
}
else
{
uroAddLog('set default CWL group');
uroCWLGroups.push(new uroOWLGroupObj(0,'No group',false));
}
if(localStorage.UROverviewCamWatchList !== undefined)
{
uroAddLog('recover camera watchlist');
uroApplyCamWatchList();
uroGetCurrentCamWatchListObjects();
isNewInstall = false;
}
if(localStorage.UROverviewSegWatchList !== undefined)
{
uroAddLog('recover segment watchlist');
uroApplySegWatchList();
uroGetCurrentSegWatchListObjects();
isNewInstall = false;
}
if(localStorage.UROverviewPlaceWatchList !== undefined)
{
uroAddLog('recover places watchlist');
uroApplyPlaceWatchList();
//uroGetCurrentPlaceWatchListObjects();
isNewInstall = false;
}
if(localStorage.UROverviewPlacesGroups !== undefined)
{
uroAddLog('recover places groups');
uroApplyPlacesGroups();
isNewInstall = false;
}
if(localStorage.UROverviewCurrentVersion !== undefined)
{
uroAddLog('comparing install versions');
if(localStorage.UROverviewCurrentVersion == uroVersion)
{
isUpgradeInstall = false;
}
}
if(localStorage.UROverviewFriendlyAreaNames !== undefined)
{
uroAddLog('recover friendly area names');
uroApplyFriendlyAreaNames();
isNewInstall = false;
}
if(localStorage.UROverviewMasterEnable !== undefined)
{
uroAddLog('recover master enable state');
document.getElementById('_cbMasterEnable').checked = (localStorage.UROverviewMasterEnable == "true");
}
if(isNewInstall)
{
uroFirstTimerWelcomePack();
}
else if(isUpgradeInstall)
{
uroShowUpdateNotes();
}
if(notifyAboutCustomIcons)
{
uroAdvertiseCustomIcons();
}
uroInhibitSave = false;
}
function uroDefaultSettings()
{
if(confirm('Resetting URO+ settings cannot be undone\nAre you sure you want to do this?') === true)
{
var defaultSettings = '';
defaultSettings += '[UROverviewMPOptions][len=628]:_cbMPFilterMissingJunction,false:_cbMPFilterMissingRoad,false:_cbMPFilterCrossroadsJunctionMissing,false:_cbMPFilterDrivingDirectionMismatch,false:_cbMPFilterRoadTypeMismatch,false:_cbMPFilterRestrictedTurn,false:_cbMPFilterRoadClosureProblem,false:_cbMPFilterUnknownProblem,false:_cbMPFilterTurnProblem,false:_cbMPFilterReopenedProblem,false:_cbInvertMPFilter,false:_cbMPFilterOutsideArea,false:_cbMPFilterClosed,false:_cbMPFilterSolved,false:_cbMPFilterUnidentified,false:_cbMPClosedUserIDFilter,false:_cbMPNotClosedUserIDFilter,false:_cbMPFilterLowSeverity,false:_cbMPFilterMediumSeverity,false:_cbMPFilterHighSeverity,false[END]';
defaultSettings += '[UROverviewPlacesGroups][len=59]false:false:false:false:false:false:false:false:false:false[END]';
defaultSettings += '[UROverviewCamWatchList][len=0][END]';
defaultSettings += '[UROverviewFriendlyAreaNames][len=0][END]';
defaultSettings += '[UROverviewMiscOptions][len=1031]:_cbNativeConvoMarkers,true:_cbNativeBetaConvoMarkers,true:_cbCommentCount,false:_cbURBackfill,false:_inputUnstackSensitivity,15:_inputUnstackZoomLevel,3:_cbCustomRoadworksMarkers,false:_cbCustomConstructionMarkers,false:_cbCustomClosuresMarkers,false:_cbCustomEventsMarkers,false:_cbCustomNotesMarkers,false:_cbCustomElginMarkers,false:_cbCustomTrafficMasterMarkers,false:_cbCustomTrafficCastMarkers,false:_inputPopupDwellTimeout,2:_inputPopupEntryTimeout,2:_inputMaxJitter,2:_cbInhibitURPopup,false:_cbInhibitMPPopup,false:_cbInhibitCamPopup,false:_cbInhibitSegPopup,false:_cbInhibitTurnsPopup,false:_cbInhibitLandmarkPopup,false:_cbInhibitPUPopup,false:_cbDateFmtDDMMYY,true:_cbDateFmtMMDDYY,false:_cbDateFmtYYMMDD,false:_cbTimeFmt24H,false:_cbTimeFmt12H,false:_cbWhiteBackground,false:_inputCustomBackgroundRed,255:_inputCustomBackgroundGreen,255:_inputCustomBackgroundBlue,255:_cbHideAMLayer,false:_cbDisablePlacesFiltering,false:_cbInhibitNURButton,false:_cbInhibitURCentering,false:_cbHideEditorInfo,false:_cbEnableDTE,false[END]';
defaultSettings += '[UROverviewSegWatchList][len=0][END]';
defaultSettings += '[UROverviewPlacesOptions][len=5106]:_cbFilterUneditablePlaceUpdates,false:_cbFilterLockRankedPlaceUpdates,false:_cbFilterNewPlacePUR,false:_cbFilterUpdatedDetailsPUR,false:_cbFilterNewPhotoPUR,false:_cbFilterFlaggedPUR,false:_cbLeavePURGeos,false:_cbInvertPURFilters,false:_cbPURFilterLowSeverity,false:_cbPURFilterMediumSeverity,false:_cbPURFilterHighSeverity,false:_cbEnablePURMinAgeFilter,false:_inputPURFilterMinDays,:_cbEnablePURMaxAgeFilter,false:_inputPURFilterMaxDays,:_cbPlaceFilterEditedLessThan,false:_inputFilterPlaceEditMinDays,:_cbPlaceFilterEditedMoreThan,false:_inputFilterPlaceEditMaxDays,:_cbHidePlacesL0,false:_cbHidePlacesL1,false:_cbHidePlacesL2,false:_cbHidePlacesL3,false:_cbHidePlacesL4,false:_cbHidePlacesL5,false:_cbHidePhotoPlaces,false:_cbHideNoPhotoPlaces,false:_cbPlacesFilter-CAR_SERVICES,false:_cbPlacesFilter-GAS_STATION,false:_cbPlacesFilter-PARKING_LOT,false:_cbPlacesFilter-GARAGE_AUTOMOTIVE_SHOP,false:_cbPlacesFilter-CAR_WASH,false:_cbPlacesFilter-CHARGING_STATION,false:_cbPlacesFilter-TRANSPORTATION,false:_cbPlacesFilter-AIRPORT,false:_cbPlacesFilter-BUS_STATION,false:_cbPlacesFilter-FERRY_PIER,false:_cbPlacesFilter-SEAPORT_MARINA_HARBOR,false:_cbPlacesFilter-SUBWAY_STATION,false:_cbPlacesFilter-TRAIN_STATION,false:_cbPlacesFilter-BRIDGE,false:_cbPlacesFilter-TUNNEL,false:_cbPlacesFilter-TAXI_STATION,false:_cbPlacesFilter-JUNCTION_INTERCHANGE,false:_cbPlacesFilter-PROFESSIONAL_AND_PUBLIC,false:_cbPlacesFilter-COLLEGE_UNIVERSITY,false:_cbPlacesFilter-SCHOOL,false:_cbPlacesFilter-CONVENTIONS_EVENT_CENTER,false:_cbPlacesFilter-GOVERNMENT,false:_cbPlacesFilter-LIBRARY,false:_cbPlacesFilter-CITY_HALL,false:_cbPlacesFilter-ORGANIZATION_OR_ASSOCIATION,false:_cbPlacesFilter-PRISON_CORRECTIONAL_FACILITY,false:_cbPlacesFilter-COURTHOUSE,false:_cbPlacesFilter-CEMETERY,false:_cbPlacesFilter-FIRE_DEPARTMENT,false:_cbPlacesFilter-POLICE_STATION,false:_cbPlacesFilter-MILITARY,false:_cbPlacesFilter-HOSPITAL_MEDICAL_CARE,false:_cbPlacesFilter-OFFICES,false:_cbPlacesFilter-POST_OFFICE,false:_cbPlacesFilter-RELIGIOUS_CENTER,false:_cbPlacesFilter-KINDERGARDEN,false:_cbPlacesFilter-FACTORY_INDUSTRIAL,false:_cbPlacesFilter-EMBASSY_CONSULATE,false:_cbPlacesFilter-INFORMATION_POINT,false:_cbPlacesFilter-SHOPPING_AND_SERVICES,false:_cbPlacesFilter-ARTS_AND_CRAFTS,false:_cbPlacesFilter-BANK_FINANCIAL,false:_cbPlacesFilter-SPORTING_GOODS,false:_cbPlacesFilter-BOOKSTORE,false:_cbPlacesFilter-PHOTOGRAPHY,false:_cbPlacesFilter-CAR_DEALERSHIP,false:_cbPlacesFilter-FASHION_AND_CLOTHING,false:_cbPlacesFilter-CONVENIENCE_STORE,false:_cbPlacesFilter-PERSONAL_CARE,false:_cbPlacesFilter-DEPARTMENT_STORE,false:_cbPlacesFilter-PHARMACY,false:_cbPlacesFilter-ELECTRONICS,false:_cbPlacesFilter-FLOWERS,false:_cbPlacesFilter-FURNITURE_HOME_STORE,false:_cbPlacesFilter-GIFTS,false:_cbPlacesFilter-GYM_FITNESS,false:_cbPlacesFilter-SWIMMING_POOL,false:_cbPlacesFilter-HARDWARE_STORE,false:_cbPlacesFilter-MARKET,false:_cbPlacesFilter-SUPERMARKET_GROCERY,false:_cbPlacesFilter-JEWELRY,false:_cbPlacesFilter-LAUNDRY_DRY_CLEAN,false:_cbPlacesFilter-SHOPPING_CENTER,false:_cbPlacesFilter-MUSIC_STORE,false:_cbPlacesFilter-PET_STORE_VETERINARIAN_SERVICES,false:_cbPlacesFilter-TOY_STORE,false:_cbPlacesFilter-TRAVEL_AGENCY,false:_cbPlacesFilter-ATM,false:_cbPlacesFilter-CURRENCY_EXCHANGE,false:_cbPlacesFilter-CAR_RENTAL,false:_cbPlacesFilter-FOOD_AND_DRINK,false:_cbPlacesFilter-RESTAURANT,false:_cbPlacesFilter-BAKERY,false:_cbPlacesFilter-DESSERT,false:_cbPlacesFilter-CAFE,false:_cbPlacesFilter-FAST_FOOD,false:_cbPlacesFilter-FOOD_COURT,false:_cbPlacesFilter-BAR,false:_cbPlacesFilter-ICE_CREAM,false:_cbPlacesFilter-CULTURE_AND_ENTERTAINEMENT,false:_cbPlacesFilter-ART_GALLERY,false:_cbPlacesFilter-CASINO,false:_cbPlacesFilter-CLUB,false:_cbPlacesFilter-TOURIST_ATTRACTION_HISTORIC_SITE,false:_cbPlacesFilter-MOVIE_THEATER,false:_cbPlacesFilter-MUSEUM,false:_cbPlacesFilter-MUSIC_VENUE,false:_cbPlacesFilter-PERFORMING_ARTS_VENUE,false:_cbPlacesFilter-GAME_CLUB,false:_cbPlacesFilter-STADIUM_ARENA,false:_cbPlacesFilter-THEME_PARK,false:_cbPlacesFilter-ZOO_AQUARIUM,false:_cbPlacesFilter-RACING_TRACK,false:_cbPlacesFilter-THEATER,false:_cbPlacesFilter-OTHER,false:_cbPlacesFilter-CONSTRUCTION_SITE,false:_cbPlacesFilter-LODGING,false:_cbPlacesFilter-HOTEL,false:_cbPlacesFilter-HOSTEL,false:_cbPlacesFilter-CAMPING_TRAILER_PARK,false:_cbPlacesFilter-COTTAGE_CABIN,false:_cbPlacesFilter-BED_AND_BREAKFAST,false:_cbPlacesFilter-OUTDOORS,false:_cbPlacesFilter-PARK,false:_cbPlacesFilter-PLAYGROUND,false:_cbPlacesFilter-BEACH,false:_cbPlacesFilter-SPORTS_COURT,false:_cbPlacesFilter-GOLF_COURSE,false:_cbPlacesFilter-PLAZA,false:_cbPlacesFilter-PROMENADE,false:_cbPlacesFilter-POOL,false:_cbPlacesFilter-SCENIC_LOOKOUT_VIEWPOINT,false:_cbPlacesFilter-SKI_AREA,false:_cbPlacesFilter-NATURAL_FEATURES,false:_cbPlacesFilter-ISLAND,false:_cbPlacesFilter-SEA_LAKE_POOL,false:_cbPlacesFilter-RIVER_STREAM,false:_cbPlacesFilter-FOREST_GROVE,false:_cbPlacesFilter-FARM,false:_cbPlacesFilter-CANAL,false:_cbPlacesFilter-SWAMP_MARSH,false:_cbPlacesFilter-DAM,false:_cbFilterPrivatePlaces,false:_cbInvertPlacesFilter,false[END]';
defaultSettings += '[UROverviewUROptions][len=1601]:_cbURFilterOutsideArea,false:_cbNoFilterForURInURL,false:_cbFilterWazeAuto,false:_cbFilterIncorrectTurn,false:_cbFilterIncorrectAddress,false:_cbFilterIncorrectRoute,false:_cbFilterMissingRoundabout,false:_cbFilterGeneralError,false:_cbFilterTurnNotAllowed,false:_cbFilterIncorrectJunction,false:_cbFilterMissingBridgeOverpass,false:_cbFilterWrongDrivingDirection,false:_cbFilterMissingExit,false:_cbFilterMissingRoad,false:_cbFilterBlockedRoad,false:_cbFilterMissingLandmark,false:_cbFilterUndefined,false:_cbFilterRoadworks,false:_cbFilterConstruction,false:_cbFilterClosure,false:_cbFilterEvent,false:_cbFilterNote,false:_cbInvertURFilter,false:_cbFilterOpenUR,false:_cbFilterClosedUR,false:_cbFilterSolved,false:_cbFilterUnidentified,false:_cbEnableMinAgeFilter,false:_inputFilterMinDays,:_cbEnableMaxAgeFilter,false:_inputFilterMaxDays,:_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,:_cbHideMyFollowed,false:_cbHideMyUnfollowed,false:_cbEnableCommentAgeFilter2,false:_inputFilterCommentDays2,:_cbEnableCommentAgeFilter,false:_inputFilterCommentDays,:_cbURUserIDFilter,false:_cbURResolverIDFilter,false:_cbNoFilterForTaggedURs,false[END]';
defaultSettings += '[UROverviewCameraOptions][len=798]:_cbShowWorldCams,true:_cbShowUSACams,true:_cbShowNonWorldCams,true:_cbShowOnlyMyCams,false:_cbShowApprovedCams,true:_cbShowNonApprovedCams,true:_cbShowOlderCreatedNonApproved,false:_inputCameraMinCreatedDays,:_cbShowOlderUpdatedNonApproved,false:_inputCameraMinUpdatedDays,:_cbShowSpeedCams,true:_cbShowIfSpeedSet,true:_cbShowIfNoSpeedSet,true:_cbShowRedLightCams,true:_cbShowDummyCams,true:_cbHideCreatedByMe,false:_cbHideCreatedByRank0,false:_cbHideCreatedByRank1,false:_cbHideCreatedByRank2,false:_cbHideCreatedByRank3,false:_cbHideCreatedByRank4,false:_cbHideCreatedByRank5,false:_cbHideUpdatedByMe,false:_cbHideUpdatedByRank0,false:_cbHideUpdatedByRank1,false:_cbHideUpdatedByRank2,false:_cbHideUpdatedByRank3,false:_cbHideUpdatedByRank4,false:_cbHideUpdatedByRank5,false:_cbHideCWLCams,false[END]';
defaultSettings += '[UROverviewMasterEnable][len=4]true[END]';
defaultSettings += '[UROverviewPlaceWatchList][len=0][END]';
defaultSettings += '[UROverviewCurrentVersion][len=0][END]';
defaultSettings += '[UROverviewCWLGroups][len=16]0,No group,false[END]';
document.getElementById('_txtSettings').value = defaultSettings;
uroTextToSettings();
document.getElementById('_txtSettings').value = '';
}
}
function uroSettingsToText()
{
var txtSettings = '';
uroSaveSettings();
for(var lsEntry in localStorage)
{
if(lsEntry.indexOf('UROverview') === 0)
{
txtSettings += '['+lsEntry+'][len=' + localStorage[lsEntry].length + ']' + localStorage[lsEntry] + '[END]';
}
}
document.getElementById('_txtSettings').value = txtSettings;
document.getElementById('_txtSettings').focus();
document.getElementById('_txtSettings').select();
}
function uroTextToSettings()
{
var txtSettings = '';
txtSettings = uroGetElmValue('_txtSettings');
if(txtSettings.indexOf('[END]') == -1) return;
var subText = txtSettings.split('[END]');
for(var i=0;i<subText.length;i++)
{
var bPos = subText[i].indexOf(']');
if(bPos != -1)
{
var settingID = subText[i].substr(1,bPos-1);
subText[i] = subText[i].substr(bPos+1);
bPos = subText[i].indexOf(']');
if(bPos != -1)
{
var settingLength = subText[i].substr(5,bPos-5);
subText[i] = subText[i].substr(bPos+1);
if(subText[i].length == settingLength)
{
localStorage[settingID] = subText[i];
}
}
}
}
uroLoadSettings();
}
function uroClearSettingsText()
{
document.getElementById('_txtSettings').value = '';
}
function uroDateToDays(dateToConvert)
{
var dateNow = new Date();
var elapsedSinceEpoch = dateNow.getTime();
var elapsedSinceEvent = elapsedSinceEpoch - dateToConvert;
dateNow.setHours(0);
dateNow.setMinutes(0);
dateNow.setSeconds(0);
dateNow.setMilliseconds(0);
var elapsedSinceMidnight = elapsedSinceEpoch - dateNow.getTime();
if(elapsedSinceEvent < elapsedSinceMidnight)
{
// event occurred today...
return 0;
}
else
{
// event occurred at some point prior to midnight this morning, so return a minimum value of 1...
return 1 + Math.floor((elapsedSinceEvent - elapsedSinceMidnight) / 86400000);
}
}
function uroGetURAge(urObj,ageType,getRaw)
{
if(ageType === 0)
{
if((urObj.attributes.driveDate === null)||(urObj.attributes.driveDate === 0)) return -1;
if(getRaw) return urObj.attributes.driveDate;
else return uroDateToDays(urObj.attributes.driveDate);
}
else if(ageType === 1)
{
if((urObj.attributes.resolvedOn === null)||(urObj.attributes.resolvedOn === 0)) return -1;
if(getRaw) return urObj.attributes.resolvedOn;
else return uroDateToDays(urObj.attributes.resolvedOn);
}
else
{
return -1;
}
}
function uroGetPURAge(purObj)
{
if(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded !== null)
{
return uroDateToDays(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded);
}
else
{
return -1;
}
}
function uroGetCameraAge(camObj, mode)
{
if(mode === 0)
{
if(camObj.attributes.updatedOn === null) return -1;
return uroDateToDays(camObj.attributes.updatedOn);
}
if(mode === 1)
{
if(camObj.attributes.createdOn === null) return -1;
return uroDateToDays(camObj.attributes.createdOn);
}
}
function uroGetCommentAge(commentObj)
{
if(commentObj.createdOn === null) return -1;
return uroDateToDays(commentObj.createdOn);
}
function uroParseDaysAgo(days)
{
if(days === 0) return 'today';
else if(days === 1) return '1 day ago';
else return days+' days ago';
}
function uroGetCameraSpeedString(camSpeed)
{
if(camSpeed !== null)
{
var conversionFactor = 1; // default to metric
var multipleFactor = 10; // default to limits being set in multiples of 10
var country;
if(W.model.countries.top === undefined)
{
country = W.model.countries.additionalInfo[0].name;
}
else
{
country = W.model.countries.top.name;
}
if(country !== null)
{
// country-specific deviations from the above...
if
(
(country == "United Kingdom") ||
(country == "Jersey") ||
(country == "Guernsey") ||
(country == "United States")
)
{
// countries using MPH
conversionFactor = 1.609;
}
if
(
(country == "United States") ||
(country == "Guernsey")
)
{
// countries with speed limits set in multiples of 5
multipleFactor = 5;
}
}
var speed = Math.round(camSpeed / conversionFactor);
var retval = speed;
if(conversionFactor == 1) retval += "KM/H";
else retval += "MPH";
if(speed % multipleFactor !== 0) retval += " (not valid?)";
return retval;
}
else return "not set";
}
// --------------------------------------------------------------------------------------------------------------------
// AREA FRIENDLYNAME STUFF
// --------------------------------------------------------------------------------------------------------------------
function uroAFNObj(fName, area, server)
{
this.fName = fName;
this.area = area;
this.server = server;
}
function uroUpdateAreaName(name, server, area)
{
var foundExisting = false;
for(var i=0; i<uroFriendlyAreaNames.length; i++)
{
if((uroFriendlyAreaNames[i].server == server) && (uroFriendlyAreaNames[i].area == area))
{
if(name === "")
{
uroFriendlyAreaNames.splice(i,1);
foundExisting = true;
}
else
{
uroFriendlyAreaNames[i].fName = name;
foundExisting = true;
}
}
}
if((foundExisting === false) && (name !== ""))
{
uroFriendlyAreaNames.push(new uroAFNObj(name,area,server));
}
uroReplaceAreaNames(true);
}
function uroAreaNameHover()
{
if((uroAreaNameHoverObj === null) || (uroAreaNameHoverObj != this))
{
uroAreaNameHoverTime = 0;
}
uroAreaNameHoverObj = this;
}
function uroAreaNameUnHover()
{
if(uroANEditHovered === true)
{
return false;
}
if(uroAreaNameOverlayShown)
{
uroAreaNameHoverObj.removeChild(uroANEditBox);
}
uroAreaNameHoverObj = null;
uroAreaNameHoverTime = -1;
uroAreaNameOverlayShown = false;
}
function uroANEditHover()
{
uroANEditHovered = true;
uroAddEventListener('uroANEditBox','mouseout',uroANEditUnHover,false);
uroAddEventListener('uroANEditBox','click',uroANEditClick,false);
}
function uroANEditUnHover()
{
var newName = document.getElementById('_textAreaName').value;
// sanitise name to avoid conflicts with config storage delimiters...
newName = newName.replace(',','');
newName = newName.replace(':','');
var server = W.location.code;
var area = uroGetAreaArea(uroAreaNameHoverObj.parentNode.children[1]);
uroAreaNameHoverObj.removeChild(uroANEditBox);
uroAreaNameOverlayShown = false;
uroANEditHovered = false;
uroUpdateAreaName(newName, server, area);
}
function uroANEditClick(e)
{
// this traps the click to prevent it falling through to the underlying area name element and
// potentially causing the map view to be relocated to that area...
e.stopPropagation();
}
function uroGetAreaArea(listObj)
{
var area = listObj.getElementsByTagName('span')[0].innerHTML;
area = parseFloat(area.split(' ')[0]);
return area;
}
function uroAreaNameOverlaySetup()
{
uroAreaNameOverlayShown = true;
uroANEditBox = document.createElement('div');
uroANEditBox.id = "uroANEditBox";
uroANEditBox.style.position = "absolute";
uroANEditBox.style.top = '7px';
uroANEditBox.style.left = '2px';
uroANEditBox.style.width = "99%";
uroAreaNameHoverObj.appendChild(uroANEditBox);
uroANEditBox.onmouseover = uroANEditHover();
var existingName = uroAreaNameHoverObj.innerHTML;
var italicTagPos = existingName.indexOf(' <i>');
if(italicTagPos == -1)
{
existingName = "";
}
else
{
existingName = existingName.substr(0,italicTagPos);
}
uroANEditBox.innerHTML = '<input type="text" style="font-size:14px; line-height:16px; height:22px; width:100%" id="_textAreaName" value="'+existingName+'">';
}
function uroReplaceAreaNames(replaceAfterNameChange)
{
if(document.getElementById('sidepanel-areas') === undefined)
{
return;
}
if(replaceAfterNameChange === false)
{
if(document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id == "friendlyNamed")
{
return;
}
}
var panelRootObj = document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0];
var areaCount = panelRootObj.children.length;
if(areaCount === 0)
{
return;
}
var localisedManagedArea = I18n.lookup("user.areas.managed_area");
for(var loop=0; loop < areaCount; loop++)
{
var childObjPElems = panelRootObj.children[loop].getElementsByTagName('p');
var title = childObjPElems[0].innerHTML;
if(title.indexOf(localisedManagedArea) > -1)
{
var area = uroGetAreaArea(childObjPElems[1]);
childObjPElems[0].innerHTML = localisedManagedArea;
for(var fnIdx=0; fnIdx < uroFriendlyAreaNames.length; fnIdx++)
{
var fnObj = uroFriendlyAreaNames[fnIdx];
if((fnObj.area == area) && (fnObj.server == W.location.code))
{
childObjPElems[0].innerHTML = fnObj.fName +' <i>('+localisedManagedArea+')</i>';
break;
}
}
var titleObj = panelRootObj.getElementsByClassName('title')[loop];
titleObj.addEventListener("mouseover", uroAreaNameHover, false);
titleObj.addEventListener("mouseout", uroAreaNameUnHover, false);
titleObj.style.cursor = "text";
}
}
document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id = "friendlyNamed";
}
// --------------------------------------------------------------------------------------------------------------------
// WATCHLIST STUFF
// --------------------------------------------------------------------------------------------------------------------
// Generic Functions
function uroTypeCast(varin)
{
if(varin == "null") return null;
if(typeof varin == "string") return parseInt(varin);
return varin;
}
function uroTruncate(val)
{
if(val === null) return val;
if(val < 0) return Math.ceil(val);
return Math.floor(val);
}
function uroOWLGroupObj(groupID, groupName, groupCollapsed)
{
groupID = uroTypeCast(groupID);
this.groupID = groupID;
this.groupName = groupName;
this.groupCount = 0;
this.groupCollapsed = groupCollapsed;
}
// Camera Functions
function uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon)
{
if(type !== null) type = uroTypeCast(type);
if(azymuth !== null) azymuth = uroTruncate(uroTypeCast(azymuth)%360);
if(speed !== null) speed = uroTruncate(uroTypeCast(speed));
if(typeof validated == "string") validated = (validated == "true");
if(lat !== null) lat = uroTruncate(uroTypeCast(lat));
if(lon !== null) lon = uroTruncate(uroTypeCast(lon));
this.type = type;
this.azymuth = azymuth;
this.speed = speed;
this.validated = validated;
this.lat = lat;
this.lon = lon;
}
function uroCamWatchObj(persistent, fid, lon, lat, type, azymuth, speed, validated, groupID, server)
{
fid = uroTypeCast(fid);
groupID = uroTypeCast(groupID);
if(typeof persistent == "string") persistent = (persistent == "true");
this.fid = fid;
this.persistent = persistent;
this.loaded = false;
this.server = server;
this.groupID = groupID;
this.watch = new uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon);
this.current = new uroCamWatchObjCheckProps(null, null, null, null, null, null);
}
function uroCamDataChanged(idx)
{
var camObj = uroCamWatchObjects[idx];
if(camObj.loaded === false) return false;
if(camObj.current.type != camObj.watch.type) return true;
if(camObj.current.azymuth != camObj.watch.azymuth) return true;
if(camObj.current.speed != camObj.watch.speed) return true;
if(camObj.current.validated != camObj.watch.validated) return true;
if(camObj.current.lat != camObj.watch.lat) return true;
if(camObj.current.lon != camObj.watch.lon) return true;
return false;
}
function uroFindCWLGroupByIdx(groupIdx)
{
var groupName = '';
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
if(uroCWLGroups[loop].groupID == groupIdx)
{
groupName = uroCWLGroups[loop].groupName;
break;
}
}
return groupName;
}
function uroIsCamOnWatchList(fid)
{
for(var loop=0;loop<uroCamWatchObjects.length;loop++)
{
if(uroCamWatchObjects[loop].fid == fid) return loop;
}
return -1;
}
function uroAddCurrentCamWatchData(idx, lat, lon, type, azymuth, speed, validated, server)
{
var camObj = uroCamWatchObjects[idx];
camObj.loaded = true;
camObj.server = server;
camObj.current = new uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon);
return(uroCamDataChanged(idx));
}
function uroAddCamToWatchList()
{
if(uroIsCamOnWatchList(uroShownFID) == -1)
{
var camObj = W.model.cameras.objects[uroShownFID];
uroCamWatchObjects.push(new uroCamWatchObj(true, uroShownFID, camObj.geometry.x, camObj.geometry.y, camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, 0, W.location.code));
uroAddCurrentCamWatchData(uroCamWatchObjects.length-1, camObj.geometry.y, camObj.geometry.x, camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, W.location.code);
uroAddLog('added camera '+uroShownFID+' to watchlist');
uroOWLUpdateHTML();
}
}
function uroRemoveCamFromWatchList()
{
var camidx = uroIsCamOnWatchList(uroShownFID);
if(camidx != -1)
{
uroCamWatchObjects.splice(camidx,1);
uroAddLog('removed camera '+uroShownFID+' from watchlist');
uroOWLUpdateHTML();
}
}
function uroUpdateCamWatchList()
{
var camIdx = uroIsCamOnWatchList(uroShownFID);
if(camIdx != -1)
{
var camObj = W.model.cameras.objects[uroShownFID];
uroCamWatchObjects[camIdx].watch = new uroCamWatchObjCheckProps(camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, camObj.geometry.y, camObj.geometry.x);
}
}
function uroClearCamWatchList()
{
if(confirm('Removing all cameras from the OWL cannot be undone\nAre you sure you want to do this?') === true)
{
uroCamWatchObjects = [];
uroOWLUpdateHTML();
}
}
function uroRetrieveCameras(lat, lon)
{
var camPos = new OpenLayers.LonLat();
var camChanged = false;
camPos.lon = lon;
camPos.lat = lat;
camPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
var camURL = 'https://' + document.location.host;
camURL += Waze.Config.api_base;
camURL += '/Features?language=en&cameras=true&bbox=';
var latl = camPos.lat - 0.25;
var latu = camPos.lat + 0.25;
var lonl = camPos.lon - 0.25;
var lonr = camPos.lon + 0.25;
camURL += lonl+','+latl+','+lonr+','+latu;
uroAddLog('retrieving camera data around '+camPos.lon+','+camPos.lat);
var camReq = new XMLHttpRequest();
camReq.open('GET',camURL,false);
try
{
camReq.send();
uroAddLog('response '+camReq.status+' received');
if (camReq.status === 200)
{
var camData = JSON.parse(camReq.responseText);
for(var camIdx = 0; camIdx < camData.cameras.objects.length; camIdx++)
{
var camObj = camData.cameras.objects[camIdx];
var listIdx = uroIsCamOnWatchList(camObj.id);
if(listIdx != -1)
{
camPos.lon = camObj.geometry.coordinates[0];
camPos.lat = camObj.geometry.coordinates[1];
camPos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
camPos.lon = uroTruncate(camPos.lon);
camPos.lat = uroTruncate(camPos.lat);
camChanged |= uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, camObj.validated, W.location.code);
}
else if(camObj.validated === false)
{
}
}
}
else
{
uroAddLog('request failed (status != 200)');
}
}
catch(err)
{
uroAddLog('camera load request failed (exception '+err+' caught)');
}
return camChanged;
}
function uroGetCurrentCamWatchListObjects()
{
var camChanged = false;
var camsChanged = [];
var camsDeleted = [];
var camidx;
var camObj;
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
camObj = uroCamWatchObjects[camidx];
if((camObj.loaded === false) && ((camObj.server == W.location.code) || (camObj.server == '??')))
{
if(typeof W.model.cameras.objects[camObj.fid] == 'object')
{
if(W.model.cameras.objects[camObj.fid].state != "Delete")
{
var wazeObj = W.model.cameras.objects[camObj.fid];
camChanged |= uroAddCurrentCamWatchData(camidx, wazeObj.geometry.y, wazeObj.geometry.x, wazeObj.attributes.type, wazeObj.attributes.azymuth, wazeObj.attributes.speed, wazeObj.attributes.validated);
}
else
{
camChanged |= uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon);
}
}
else
{
camChanged |= uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon);
}
}
}
if(camChanged)
{
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
if(uroCamDataChanged(camidx))
{
camsChanged.push(uroCamWatchObjects[camidx]);
}
}
}
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
camObj = uroCamWatchObjects[camidx];
if((camObj.loaded === false) && (camObj.server == W.location.code))
{
camsDeleted.push(camObj);
}
}
if((camsChanged.length > 0) || (camsDeleted.length > 0))
{
var alertStr = 'Camera WatchList Alert!!!\r\n';
for(camidx=0;camidx<camsChanged.length;camidx++)
{
alertStr += 'Camera ID '+camsChanged[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsChanged[camidx].groupID)+'" has been changed\r\n';
}
for(camidx=0;camidx<camsDeleted.length;camidx++)
{
alertStr += 'Camera ID '+camsDeleted[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsDeleted[camidx].groupID)+'" has been deleted\r\n';
}
alert(alertStr);
}
}
function uroClearDeletedCameras()
{
for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
{
if(uroCamWatchObjects[camidx].loaded === false)
{
uroShownFID = uroCamWatchObjects[camidx].fid;
uroRemoveCamFromWatchList();
}
}
}
function uroClearUnknownServerCameras()
{
var confirmMsg = 'Cameras with an unknown server cannot be automatically verified by URO+.\n';
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.\n';
confirmMsg += 'If the cameras then continue to show up as an unknown server, it is safe to delete them...\n\n';
confirmMsg += 'Do you still wish to proceed with deleting all unknown server cameras?';
if(confirm(confirmMsg) === true)
{
for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
{
if(uroCamWatchObjects[camidx].server == '??')
{
uroShownFID = uroCamWatchObjects[camidx].fid;
uroRemoveCamFromWatchList();
}
}
}
}
function uroRescanCamWatchList()
{
for(var camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
uroCamWatchObjects[camidx].loaded = false;
}
uroGetCurrentCamWatchListObjects();
uroOWLUpdateHTML();
}
function uroGotoCam()
{
var camidx = this.id.substr(13);
var camPos = new OpenLayers.LonLat();
camPos.lon = uroCamWatchObjects[camidx].watch.lon;
camPos.lat = uroCamWatchObjects[camidx].watch.lat;
W.map.setCenter(camPos,4);
W.map.camerasLayer.setVisibility(true);
return false;
}
// Segment Functions
function uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn)
{
if(left !== null) left = uroTruncate(uroTypeCast(left));
if(right !== null) right = uroTruncate(uroTypeCast(right));
if(bottom !== null) bottom = uroTruncate(uroTypeCast(bottom));
if(top !== null) top = uroTruncate(uroTypeCast(top));
if(fromNode !== null) fromNode = uroTypeCast(fromNode);
if(toNode !== null) toNode = uroTypeCast(toNode);
if(fwdDir !== null) fwdDir = uroTypeCast(fwdDir);
if(revDir !== null) revDir = uroTypeCast(revDir);
if(length !== null) length = uroTypeCast(length);
if(level !== null) level = uroTypeCast(level);
if(rank !== null) rank = uroTypeCast(rank);
if(roadType !== null) roadType = uroTypeCast(roadType);
if(updatedOn !== null) updatedOn = uroTypeCast(updatedOn);
this.left = left;
this.right = right;
this.bottom = bottom;
this.top = top;
this.fromNode = fromNode;
this.toNode = toNode;
this.fwdDir = fwdDir;
this.revDir = revDir;
this.length = length;
this.level = level;
this.rank = rank;
this.roadType = roadType;
this.updatedOn = updatedOn;
}
function uroSegWatchObj(persistent, fid, left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn, groupID, server)
{
fid = uroTypeCast(fid);
groupID = uroTypeCast(groupID);
if(typeof persistent == "string") persistent = (persistent == "true");
this.fid = fid;
this.persistent = persistent;
this.loaded = false;
this.server = server;
this.groupID = groupID;
this.watch = new uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn);
this.current = new uroSegWatchObjCheckProps(null, null, null, null, null, null, null, null, null, null, null, null, null);
}
function uroSegDataChanged(idx)
{
var segObj = uroSegWatchObjects[idx];
if(segObj.loaded === false) return false;
if(segObj.current.left != segObj.watch.left) return true;
if(segObj.current.right != segObj.watch.right) return true;
if(segObj.current.bottom != segObj.watch.bottom) return true;
if(segObj.current.top != segObj.watch.top) return true;
if(segObj.current.fromNode != segObj.watch.fromNode) return true;
if(segObj.current.toNode != segObj.watch.toNode) return true;
if(segObj.current.fwdDir != segObj.watch.fwdDir) return true;
if(segObj.current.revDir != segObj.watch.revDir) return true;
if(segObj.current.length != segObj.watch.length) return true;
if(segObj.current.level != segObj.watch.level) return true;
if(segObj.current.rank != segObj.watch.rank) return true;
if(segObj.current.roadType != segObj.watch.roadType) return true;
if(segObj.current.updatedOn != segObj.watch.updatedOn) return true;
return false;
}
function uroIsSegOnWatchList(fid)
{
for(var loop=0;loop<uroSegWatchObjects.length;loop++)
{
if(uroSegWatchObjects[loop].fid == fid) return loop;
}
return -1;
}
function uroAddCurrentSegWatchData(idx, left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn, server)
{
var segObj = uroSegWatchObjects[idx];
segObj.loaded = true;
segObj.server = server;
segObj.current = new uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn);
return(uroSegDataChanged(idx));
}
function uroClearSegWatchList()
{
if(confirm('Removing all segments from the OWL cannot be undone\nAre you sure you want to do this?') === true)
{
uroSegWatchObjects = [];
uroOWLUpdateHTML();
}
}
function uroAddUpdateSegWatchList()
{
var selectedCount = W.selectionManager.selectedItems.length;
if(selectedCount === 0)
{
return;
}
for(var loop=0;loop < selectedCount; loop++)
{
var segObj = W.selectionManager.selectedItems[loop].model.attributes;
var fid = segObj.id;
var idx = uroIsSegOnWatchList(fid);
if(idx != -1)
{
uroSegWatchObjects[idx].watch = new uroSegWatchObjCheckProps(segObj.geometry.bounds.left, segObj.geometry.bounds.right, segObj.geometry.bounds.bottom, segObj.geometry.bounds.top, segObj.fromNodeID, segObj.toNodeID, segObj.fwdDirection, segObj.revDirection, segObj.length, segObj.level, segObj.rank, segObj.roadType, segObj.updatedOn);
uroAddLog('updated watchlist details for segment '+fid);
}
else
{
uroSegWatchObjects.push(new uroSegWatchObj(true, fid, segObj.geometry.bounds.left, segObj.geometry.bounds.right, segObj.geometry.bounds.bottom, segObj.geometry.bounds.top, segObj.fromNodeID, segObj.toNodeID, segObj.fwdDirection, segObj.revDirection, segObj.length, segObj.level, segObj.rank, segObj.roadType, segObj.updatedOn, 0, W.location.code));
uroAddCurrentSegWatchData(uroSegWatchObjects.length-1, segObj.geometry.bounds.left, segObj.geometry.bounds.right, segObj.geometry.bounds.bottom, segObj.geometry.bounds.top, segObj.fromNodeID, segObj.toNodeID, segObj.fwdDirection, segObj.revDirection, segObj.length, segObj.level, segObj.rank, segObj.roadType, segObj.updatedOn, W.location.code);
uroAddLog('added segment '+fid+' to watchlist');
}
}
//uroOWLUpdateHTML();
}
function uroRemoveSegFromWatchList()
{
var selectedCount = W.selectionManager.selectedItems.length;
if(selectedCount === 0)
{
return;
}
for(var loop=0;loop < selectedCount; loop++)
{
var fid = W.selectionManager.selectedItems[loop].model.attributes.id;
var idx = uroIsSegOnWatchList(fid);
if(idx != -1)
{
uroSegWatchObjects.splice(idx,1);
uroAddLog('removed segment '+fid+' from watchlist');
}
}
//uroOWLUpdateHTML();
}
function uroRetrieveSegments(lat, lon)
{
var pos = new OpenLayers.LonLat();
var changed = false;
pos.lon = lon;
pos.lat = lat;
pos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
var URL = 'https://' + document.location.host;
URL += Waze.Config.api_base;
URL += '/Features?roadTypes=1%2C2%2C3%2C4%2C5%2C6%2C7%2C8%2C9%2C10%2C11%2C12%2C13%2C14%2C15%2C16%2C17%2C18%2C19%2C20%2C21';
URL += '&bbox=';
var latl = pos.lat - 0.25;
var latu = pos.lat + 0.25;
var lonl = pos.lon - 0.25;
var lonr = pos.lon + 0.25;
URL += lonl+','+latl+','+lonr+','+latu;
URL += '&language=en';
uroAddLog('retrieving segment data around '+pos.lon+','+pos.lat);
var req = new XMLHttpRequest();
req.open('GET',URL,false);
try
{
req.send();
uroAddLog('response '+req.status+' received');
if (req.status === 200)
{
var data = JSON.parse(req.responseText);
for(var idx = 0; idx < data.segments.objects.length; idx++)
{
var obj = data.segments.objects[idx];
var listIdx = uroIsSegOnWatchList(obj.id);
if(listIdx != -1)
{
//pos.lon = obj.geometry.coordinates[0];
//pos.lat = obj.geometry.coordinates[1];
//pos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
//camPos.lon = uroTruncate(camPos.lon);
//camPos.lat = uroTruncate(camPos.lat);
//camChanged |= uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, camObj.validated, W.location.code);
}
else if(obj.validated === false)
{
}
}
}
else
{
uroAddLog('request failed (status != 200)');
}
}
catch(err)
{
uroAddLog('segment load request failed (exception '+err+' caught)');
}
return changed;
}
function uroGetCurrentSegWatchListObjects()
{
var segChanged = false;
var segsChanged = [];
var segsDeleted = [];
var idx;
var segObj;
for(idx=0;idx<uroSegWatchObjects.length;idx++)
{
segObj = uroSegWatchObjects[idx];
if((segObj.loaded === false) && ((segObj.server == W.location.code) || (segObj.server == '??')))
{
var segLat = (segObj.watch.top + segObj.watch.bottom) / 2;
var segLon = (segObj.watch.right + segObj.watch.left) / 2;
if(typeof W.model.segments.objects[segObj.fid] == 'object')
{
if(W.model.segments.objects[segObj.fid].state != "Delete")
{
var wazeObj = W.model.segments.objects[segObj.fid];
segChanged |= uroAddCurrentSegWatchData(idx, wazeObj.geometry.bounds.left, wazeObj.geometry.bounds.right, wazeObj.geometry.bounds.bottom, wazeObj.geometry.bounds.top, wazeObj.fromNodeID, wazeObj.toNodeID, wazeObj.fwdDirection, wazeObj.revDirection, wazeObj.length, wazeObj.level, wazeObj.rank, wazeObj.roadType, wazeObj.updatedOn, W.location.code);
}
else
{
segChanged |= uroRetrieveSegments(segLat, segLon);
}
}
else
{
segChanged |= uroRetrieveSegments(segLat, segLon);
}
}
}
if(segChanged)
{
for(idx=0;idx<uroSegWatchObjects.length;idx++)
{
if(uroSegDataChanged(idx))
{
segsChanged.push(uroSegWatchObjects[idx]);
}
}
}
for(idx=0;idx<uroSegWatchObjects.length;idx++)
{
segObj = uroSegWatchObjects[idx];
if((segObj.loaded === false) && (segObj.server == W.location.code))
{
segsDeleted.push(segObj);
}
}
if((segsChanged.length > 0) || (segsDeleted.length > 0))
{
var alertStr = 'Segment WatchList Alert!!!\r\n';
for(idx=0;idx<segsChanged.length;idx++)
{
alertStr += 'Segment ID '+segsChanged[idx].fid+' in group "'+uroFindCWLGroupByIdx(segsChanged[idx].groupID)+'" has been changed\r\n';
}
for(idx=0;idx<segsDeleted.length;idx++)
{
alertStr += 'Segment ID '+segsDeleted[idx].fid+' in group "'+uroFindCWLGroupByIdx(segsDeleted[idx].groupID)+'" has been deleted\r\n';
}
alert(alertStr);
}
}
// Places Functions
function uroPlaceWatchObjCheckProps(left, right, bottom, top, name, imageCount, residential, updatedOn)
{
if(left !== null) left = uroTruncate(uroTypeCast(left));
if(right !== null) right = uroTruncate(uroTypeCast(right));
if(bottom !== null) bottom = uroTruncate(uroTypeCast(bottom));
if(top !== null) top = uroTruncate(uroTypeCast(top));
if(imageCount !== null) imageCount = uroTypeCast(imageCount);
if(typeof residential == "string") residential = (residential == "true");
if(updatedOn !== null) updatedOn = uroTypeCast(updatedOn);
this.left = left;
this.right = right;
this.bottom = bottom;
this.top = top;
this.name = name;
this.imageCount = imageCount;
this.residential = residential;
this.updatedOn = updatedOn;
}
function uroPlaceWatchObj(persistent, fid, left, right, bottom, top, imageCount, name, residential, updatedOn, groupID, server)
{
groupID = uroTypeCast(groupID);
if(typeof persistent == "string") persistent = (persistent == "true");
this.fid = fid;
this.persistent = persistent;
this.loaded = false;
this.server = server;
this.groupID = groupID;
this.watch = new uroPlaceWatchObjCheckProps(left, right, bottom, top, name, imageCount, residential, updatedOn);
this.current = new uroPlaceWatchObjCheckProps(null, null, null, null, null, null, null, null);
}
function uroPlaceDataChanged(idx)
{
var placeObj = uroPlaceWatchObjects[idx];
if(placeObj.loaded === false) return false;
if(placeObj.current.left != placeObj.watch.left) return true;
if(placeObj.current.right != placeObj.watch.right) return true;
if(placeObj.current.bottom != placeObj.watch.bottom) return true;
if(placeObj.current.top != placeObj.watch.top) return true;
if(placeObj.current.name != placeObj.watch.name) return true;
if(placeObj.current.imageCount != placeObj.watch.imageCount) return true;
if(placeObj.current.residential != placeObj.watch.residential) return true;
if(placeObj.current.updatedOn != placeObj.watch.updatedOn) return true;
return false;
}
function uroIsPlaceOnWatchList(fid)
{
for(var loop=0;loop<uroPlaceWatchObjects.length;loop++)
{
if(uroPlaceWatchObjects[loop].fid == fid) return loop;
}
return -1;
}
function uroClearPlaceWatchList()
{
if(confirm('Removing all places from the OWL cannot be undone\nAre you sure you want to do this?') === true)
{
uroPlaceWatchObjects = [];
uroOWLUpdateHTML();
}
}
function uroHighlightCWLEntry()
{
this.style.backgroundColor = '#FFFFAA';
return false;
}
function uroUnhighlightCWLEntry()
{
var camidx = this.id.substr(8);
var changed = uroCamDataChanged(camidx);
var deleted = (uroCamWatchObjects[camidx].loaded === false);
if(uroCamWatchObjects[camidx].server != W.location.code)
{
if(uroCamWatchObjects[camidx].server == '??') this.style.backgroundColor = '#A0A0A0';
else this.style.backgroundColor = '#AAFFAA';
}
else if(changed) this.style.backgroundColor = '#AAAAFF';
else if(deleted) this.style.backgroundColor = '#FFAAAA';
else this.style.backgroundColor = '#FFFFFF';
return false;
}
function uroCWLIconHighlight()
{
var iconType = this.id.substr(11,1);
this.src = uroIcons[iconType][0];
return false;
}
function uroCWLIconLowlight()
{
var iconType = this.id.substr(11,1);
this.src = uroIcons[iconType][1];
return false;
}
function uroPopulateCWLGroupSelect()
{
var selector = document.getElementById('_uroCWLGroupSelect');
while(selector.options.length > 0)
{
selector.options.remove(0);
}
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
var groupObj = uroCWLGroups[loop];
if(groupObj.groupID != -1)
{
selector.options.add(new Option(groupObj.groupName,groupObj.groupID));
}
}
}
function uroGetNextCWLGroupID()
{
var nextID = 1;
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
if(uroCWLGroups[loop].groupID >= nextID)
{
nextID = uroCWLGroups[loop].groupID + 1;
}
}
return nextID;
}
function uroFindCWLGroupByName(groupName)
{
var groupID = -1;
for(var loop=0;loop<uroCWLGroups.length;loop++)
{
if((uroCWLGroups[loop].groupName == groupName) && (uroCWLGroups[loop].groupID != -1))
{
groupID = uroCWLGroups[loop].groupID;
break;
}
}
return groupID;
}
function uroAddCWLGroup()
{
var groupID = uroGetNextCWLGroupID();
var groupName = uroGetElmValue('_uroCWLGroupEntry');
if(uroFindCWLGroupByName(groupName) == -1)
{
uroCWLGroups.push(new uroOWLGroupObj(groupID,groupName,false));
uroPopulateCWLGroupSelect();
}
}
function uroRemoveCWLGroup()
{
var loop;
var selector = document.getElementById('_uroCWLGroupSelect');
var groupID = parseInt(selector.selectedOptions[0].value);
if(groupID === 0) return false; // prevent deletion of the default group
for(loop=0;loop<uroCamWatchObjects.length;loop++)
{
var cwObj = uroCamWatchObjects[loop];
if(cwObj.groupID == groupID)
{
cwObj.groupID = 0;
}
}
for(loop=0;loop<uroCWLGroups.length;loop++)
{
var groupObj = uroCWLGroups[loop];
if(groupObj.groupID == groupID)
{
groupObj.groupID = -1;
}
}
uroOWLUpdateHTML();
}
function uroAssignCameraToGroup()
{
var camidx = this.id.substr(13);
var selector = document.getElementById('_uroCWLGroupSelect');
uroCamWatchObjects[camidx].groupID = parseInt(selector.selectedOptions[0].value);
uroOWLUpdateHTML();
return false;
}
function uroAddBtnEvl(btnID, evlType, evlFunction)
{
var btnObj = document.getElementById(btnID);
if(btnObj !== null)
{
btnObj.addEventListener(evlType, evlFunction, true);
}
}
function uroCWLGroupCollapseExpand()
{
var groupidx = this.id.substr(18);
if(uroCWLGroups[groupidx].groupCollapsed === true) uroCWLGroups[groupidx].groupCollapsed = false;
else uroCWLGroups[groupidx].groupCollapsed = true;
uroOWLUpdateHTML();
return false;
}
function uroOWLUpdateHTML()
{
var camTypes = new Array("","","Speed", "Dummy", "Red Light");
var selectedGroup = 0;
var iHTML = '';
var camidx;
var groupidx;
if(document.getElementById('_uroCWLGroupSelect') !== null)
{
selectedGroup = 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)
{
for(groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
{
var groupObj = uroCWLGroups[groupidx];
iHTML += '<div id="_uroCWLGroup-'+groupidx+'">';
if(groupObj.groupCollapsed === true)
{
iHTML += '<img src="'+uroIcons[0][1]+'" id="_uroCWLGroupState-'+groupidx+'">';
}
else
{
iHTML += '<img src="'+uroIcons[0][0]+'" id="_uroCWLGroupState-'+groupidx+'">';
}
iHTML += '<b>'+groupObj.groupName+'</b><br>';
groupObj.groupCount = 0;
if(uroCamWatchObjects.length > 0)
{
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
var camObj = uroCamWatchObjects[camidx];
if(camObj.groupID == groupObj.groupID)
{
groupObj.groupCount++;
var changed = uroCamDataChanged(camidx);
var deleted = (camObj.loaded === false);
iHTML += '<div id="_uroCWL-'+camidx+'" style="padding:3px;border-width:2px;border-style:solid;border-color:#FFFFFF;background-color:';
if(camObj.server != W.location.code)
{
if(camObj.server == '??') iHTML += '#A0A0A0;';
else iHTML += '#AAFFAA;';
}
else if(changed) iHTML += '#AAAAFF;';
else if(deleted) iHTML += '#FFAAAA;';
else iHTML += '#FFFFFF;';
if(groupObj.groupCollapsed === true) iHTML += 'display:none;">';
else iHTML += 'display:block;">';
iHTML += 'ID: '+camObj.fid;
iHTML += ' ('+camObj.server+')';
iHTML += ' Type: '+camTypes[camObj.watch.type];
if(camObj.server != W.location.code)
{
if(camObj.server == '??')
{
iHTML += '<br><i>Unknown server</i>';
}
else
{
iHTML += '<br><i>Not on this server</i>';
}
}
else if(deleted)
{
iHTML += '<br>DELETED';
}
else if(changed)
{
if(camObj.current.type != camObj.watch.type)
{
iHTML += '<br> Type changed';
iHTML += ' ('+camObj.watch.type+' to '+camObj.current.type+')';
}
if(camObj.current.azymuth != camObj.watch.azymuth)
{
iHTML += '<br> Azimuth changed';
iHTML += ' ('+camObj.watch.azymuth+' to '+camObj.current.azymuth+')';
}
if(camObj.current.speed != camObj.watch.speed)
{
iHTML += '<br> Speed changed';
iHTML += ' ('+camObj.watch.speed+' to '+camObj.current.speed+')';
}
if(camObj.current.validated != camObj.watch.validated)
{
iHTML += '<br> Approval state changed';
iHTML += ' ('+camObj.watch.validated+' to '+camObj.current.validated+')';
}
if(camObj.current.lat != camObj.watch.lat)
{
iHTML += '<br> Latitude changed';
iHTML += ' ('+camObj.watch.lat+' to '+camObj.current.lat+')';
}
if(camObj.current.lon != camObj.watch.lon)
{
iHTML += '<br> Longitude changed';
iHTML += ' ('+camObj.watch.lon+' to '+camObj.current.lon+')';
}
}
if(camObj.server == W.location.code)
{
if(deleted === false)
{
iHTML += ' <img id="_uroCWLIcon1-'+camidx+'" src="'+uroIcons[1][1]+'">';
}
iHTML += ' <img id="_uroCWLIcon2-'+camidx+'" src="'+uroIcons[2][1]+'">';
}
iHTML += '</div>';
}
}
}
iHTML += '</div>';
}
}
iHTML += '</div><div id="_uroCWLControls">';
iHTML += '<hr>Group control:<br>';
iHTML += '<select id="_uroCWLGroupSelect" style="width:40%;height:22px;"></select> <input type="button" id="_btnCWLGroupDel" value="Delete group"><br>';
iHTML += '<input type="text" id="_uroCWLGroupEntry" style="width:40%;height:22px;"> <input type="button" id="_btnCWLGroupAdd" value="Add group">';
iHTML += '<br><input type="button" id="_btnRescanCamWatchList" value="Refresh Camera Data"><br><br>';
iHTML += '<b>Remove cameras from OWL:</b><br>';
iHTML += '<input type="button" id="_btnRemoveDeletedCameras" value="Deleted"> ';
iHTML += '<input type="button" id="_btnRemoveUnknownServerCameras" value="Unknown Server"> ';
iHTML += '<input type="button" id="_btnClearCamWatchList" value="ALL Cameras">';
iHTML += '</div>';
uroOWL.innerHTML = iHTML;
if(uroCamWatchObjects.length > 0)
{
for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
{
document.getElementById("_uroCWL-"+camidx).onmouseover = uroHighlightCWLEntry;
document.getElementById("_uroCWL-"+camidx).onmouseleave = uroUnhighlightCWLEntry;
if(uroCamWatchObjects[camidx].server == W.location.code)
{
var icon1 = document.getElementById("_uroCWLIcon1-"+camidx);
var icon2 = document.getElementById("_uroCWLIcon2-"+camidx);
if(icon1 !== null)
{
icon1.onmouseover = uroCWLIconHighlight;
icon1.onmouseleave = uroCWLIconLowlight;
icon1.onclick = uroAssignCameraToGroup;
}
if(icon2 !== null)
{
icon2.onmouseover = uroCWLIconHighlight;
icon2.onmouseleave = uroCWLIconLowlight;
icon2.onclick = uroGotoCam;
}
}
}
}
uroAddBtnEvl('_btnClearCamWatchList', 'click', uroClearCamWatchList);
uroAddBtnEvl('_btnRemoveDeletedCameras', 'click', uroClearDeletedCameras);
uroAddBtnEvl('_btnRemoveUnknownServerCameras', 'click', uroClearUnknownServerCameras);
uroAddBtnEvl('_btnRescanCamWatchList', 'click', uroRescanCamWatchList);
uroAddBtnEvl('_btnCWLGroupDel', 'click', uroRemoveCWLGroup);
uroAddBtnEvl('_btnCWLGroupAdd', 'click', uroAddCWLGroup);
if(document.getElementById('_uroCWLGroupSelect') !== null)
{
uroAddLog('populating CWL group list');
uroPopulateCWLGroupSelect();
var selector = document.getElementById('_uroCWLGroupSelect');
if(selectedGroup >= selector.length)
{
selectedGroup = 0;
}
selector.selectedIndex = selectedGroup;
}
if(uroCWLGroups.length > 0)
{
for(groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
{
if(uroCWLGroups[groupidx].groupCount === 0)
{
uroSetStyleDisplay('_uroCWLGroup-'+groupidx,'none');
}
else
{
uroSetOnClick('_uroCWLGroupState-'+groupidx,uroCWLGroupCollapseExpand);
}
}
}
}
// --------------------------------------------------------------------------------------------------------------------
// END OF WATCHLIST STUFF
// --------------------------------------------------------------------------------------------------------------------
function uroIsOnIgnoreList(fid)
{
if(sessionStorage.UROverview_FID_IgnoreList.indexOf('fid:'+fid) == -1) return false;
else return true;
}
function uroEnableIgnoreListControls()
{
var btnState = "visible";
if(sessionStorage.UROverview_FID_IgnoreList === '')
{
btnState = "hidden";
}
document.getElementById('_btnUndoLastHide').style.visibility = btnState;
document.getElementById('_btnClearSessionHides').style.visibility = btnState;
uroFilterItems();
}
function uroAddToIgnoreList()
{
if(!uroIsOnIgnoreList(uroShownFID))
{
sessionStorage.UROverview_FID_IgnoreList += 'fid:'+uroShownFID;
uroAddLog('added fid '+uroShownFID+' to ignore list');
uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
uroDiv.style.visibility = 'hidden';
uroEnableIgnoreListControls();
W.map.events.register("mousemove", null, uroFilterItemsOnMove);
}
return false;
}
/*
function uroRemoveFromIgnoreList(fid)
{
var ignorelist = sessionStorage.UROverview_FID_IgnoreList;
var fidpos = ignorelist.indexOf('fid:'+fid);
if(fidpos != -1)
{
var preFID = ignorelist.slice(0,fidpos);
ignorelist = ignorelist.slice(fidpos+1);
fidpos = ignorelist.indexOf('fid:');
if(fidpos == -1) ignorelist = '';
else ignorelist = ignorelist.slice(fidpos);
sessionStorage.UROverview_FID_IgnoreList = preFID + ignorelist;
uroAddLog('removed fid '+fid+' from ignore list');
uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
uroEnableIgnoreListControls();
}
}
*/
function uroRemoveLastAddedIgnore()
{
var ignorelist = sessionStorage.UROverview_FID_IgnoreList;
var fidpos = ignorelist.lastIndexOf('fid:');
if(fidpos != -1)
{
ignorelist = ignorelist.slice(0,fidpos);
sessionStorage.UROverview_FID_IgnoreList = ignorelist;
uroAddLog('removed last fid from ignore list');
uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
uroEnableIgnoreListControls();
}
}
function uroRemoveAllIgnores()
{
sessionStorage.UROverview_FID_IgnoreList = '';
uroEnableIgnoreListControls();
}
function uroKeywordPresent(desc, keyword)
{
var re;
if(uroGetCBChecked('_cbCaseInsensitive') === true) re = RegExp(keyword,'i');
else re = RegExp(keyword);
if(desc.search(re) != -1) return true;
else return false;
}
function uroClickify(desc)
{
var linkStartPos = desc.indexOf('http://');
if(linkStartPos == -1) linkStartPos = desc.indexOf('https://');
if(linkStartPos != -1)
{
var descPreLink = desc.slice(0,linkStartPos);
var descURL = desc.slice(linkStartPos);
var linkEndPos = descURL.indexOf(' ');
var descPostLink = '';
if(linkEndPos != -1)
{
descPostLink = descURL.slice(linkEndPos);
descURL = descURL.slice(0,linkEndPos);
}
var linkTarget = '';
if(descURL.indexOf('cryosphere') != -1) linkTarget = '_cryosphere';
else if(descURL.indexOf('waze.com') != -1) linkTarget = '_wazeUR';
desc = descPreLink + '<a target="'+linkTarget+'" href="'+descURL+'">here</a>' + descPostLink;
}
return desc;
}
function uroRefreshUpdateRequestSessions()
{
for (var urID in W.model.mapUpdateRequests.objects)
{
if(W.model.mapUpdateRequests.objects.hasOwnProperty(urID))
{
var reqPos = uroRequestedURSessionIDs.indexOf(urID);
var pendPos = uroPendingURSessionIDs.indexOf(urID);
if(W.model.updateRequestSessions.objects[urID] === undefined)
{
if((reqPos == -1) && (pendPos == -1))
{
uroPendingURSessionIDs.push(urID);
}
}
else
{
if(reqPos != -1)
{
uroRequestedURSessionIDs.splice(reqPos,1);
}
if(pendPos != -1)
{
uroPendingURSessionIDs.splice(pendPos,1);
}
}
}
}
var idList = [];
while((idList.length < 50) && (uroPendingURSessionIDs.length))
{
var id = uroPendingURSessionIDs.pop();
idList.push(id);
uroRequestedURSessionIDs.push(id);
}
if(idList.length > 0)
{
uroAddLog('grabbing '+idList.length+' updateRequestSessions, IDs: '+idList);
W.model.updateRequestSessions.get(idList);
}
if((uroPendingURSessionIDs.length) || (uroRequestedURSessionIDs.length))
{
setTimeout(uroRefreshUpdateRequestSessions,10);
}
}
function uroURHasMyComments(fid)
{
var nComments = W.model.updateRequestSessions.objects[fid].comments.length;
if(nComments === 0) return false;
for(var cidx=0; cidx<nComments; cidx++)
{
if(W.model.updateRequestSessions.objects[fid].comments[cidx].userID == uroUserID) return true;
}
return false;
}
function uroACMObj(urID, customType, hasMyComments, nComments)
{
this.urID = urID;
this.customType = customType;
this.hasMyComments = hasMyComments;
this.nComments = nComments;
}
function uroAddCustomMarkers(urID, customType, hasMyComments, nComments)
{
var useCustomMarker = false;
if(uroGetCBChecked('_cbMasterEnable') === true)
{
if(customType === 0) useCustomMarker = (uroGetCBChecked('_cbCustomRoadworksMarkers'));
else if(customType === 1) useCustomMarker = (uroGetCBChecked('_cbCustomConstructionMarkers'));
else if(customType === 2) useCustomMarker = (uroGetCBChecked('_cbCustomClosuresMarkers'));
else if(customType === 3) useCustomMarker = (uroGetCBChecked('_cbCustomEventsMarkers'));
else if(customType === 4) useCustomMarker = (uroGetCBChecked('_cbCustomNotesMarkers'));
else if(customType === 100) useCustomMarker = (uroGetCBChecked('_cbCustomElginMarkers'));
else if(customType === 101) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficCastMarkers'));
else if(customType === 102) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficMasterMarkers'));
}
if(!useCustomMarker) customType = -1;
uroCustomMarkerList.push(new uroACMObj(urID, customType, hasMyComments, nComments));
}
function uroRenderCustomMarkers(markerType)
{
var urID;
var elmID;
var newSpan;
var divElem;
var objIdx;
var customType;
var customMarker;
if(markerType == 'ur')
{
var useDefaultConvoMarker = false;
var addCommentCount = false;
if(uroGetCBChecked('_cbMasterEnable') === true)
{
if((uroGetCBChecked('_cbNativeConvoMarkers')) && (uroBetaEditor === false)) useDefaultConvoMarker = true;
if((uroGetCBChecked('_cbNativeBetaConvoMarkers')) && (uroBetaEditor === true)) useDefaultConvoMarker = true;
if(uroGetCBChecked('_cbCommentCount')) addCommentCount = true;
}
else
{
useDefaultConvoMarker = true;
}
var uRCM_masterEnable = uroGetCBChecked('_cbMasterEnable');
divElem = document.getElementById(W.map.updateRequestLayer.id);
if(divElem.childNodes.length > 0)
{
for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
{
customType = -1;
var cmlObj = uroCustomMarkerList[objIdx];
if(uRCM_masterEnable === true)
{
customType = cmlObj.customType;
}
if(customType < 100)
{
urID = cmlObj.urID;
var nComments = cmlObj.nComments;
var iconObj = W.map.updateRequestLayer.markers[urID].icon;
newSpan = '';
if(nComments !== 0)
{
var classList = iconObj.imageDiv.classList;
elmID = "commentCount_"+urID;
if(addCommentCount)
{
// add a new comment count bubble if the UR doesn't already have one
if(document.getElementById(elmID) === null)
{
newSpan += '<span id="'+elmID+'" style="position:absolute;top:-9px;left:-11px;pointer-events:none;z-index:1">';
// define the comment-count holding span within the span used to hold the empty bubble image, and before the image is
// added to the HTML, to avoid z-indexing issues when adjacent comment count bubbles are overlapped...
newSpan += '<span id="'+elmID+"_inner"+'" style="position:absolute;top:4px;left:11px;font-size:11px;;pointer-events:none"></span>';
newSpan += '<img src="'+uroMarkers[0]+'">';
newSpan += '</span>';
}
}
else
{
// remove comment count bubble from this UR marker if one has previously been
// added and the user has now disabled the option...
if(document.getElementById(elmID) !== null)
{
document.getElementById(elmID).remove();
}
if(document.getElementById(elmID+"_inner") !== null)
{
document.getElementById(elmID+"_inner").remove();
}
}
elmID = "convoMarker_"+urID;
if(useDefaultConvoMarker === false)
{
if(document.getElementById(elmID) === null)
{
var hasMyComments = cmlObj.hasMyComments;
// z-index needs to be set to 1 here so that when a new comment is added to a UR and WME re-renders the native
// conversation marker, the custom marker remains on top...
newSpan += '<span id="'+elmID+'" style="position:absolute;top:-9px;left:18px;pointer-events:none;z-index:1">';
if(hasMyComments) newSpan += '<img src="'+uroMarkers[2]+'">';
else newSpan += '<img src="'+uroMarkers[1]+'">';
newSpan += '</span>';
classList.remove("has-comments");
}
}
else
{
// remove custom conversation marker from this UR if one has previously been
// added and the user has now disabled this option
if(document.getElementById(elmID) !== null)
{
document.getElementById(elmID).remove();
}
if(nComments > 0)
{
// only replace the native marker class if the UR has comments - if we're just clearing the custom
// marker following a master enable switchoff, we don't then want to add native markers to URs which
// didn't have them in the first place...
classList.add("has-comments");
}
}
}
elmID = "customMarker_"+urID;
customMarker = '';
if(customType != -1)
{
if(document.getElementById(elmID) === null)
{
newSpan += '<span id="'+elmID+'" style="position:absolute;pointer-events:none;"></span>';
}
customType = uroGetCustomMarkerIdx(customType);
if(W.model.mapUpdateRequests.objects[urID] !== undefined)
{
if(W.model.mapUpdateRequests.objects[urID].attributes.open === false) customType += 1;
}
customMarker = '<img src="'+uroIcons[customType][0]+'">';
}
else
{
if(document.getElementById(elmID) !== null)
{
document.getElementById(elmID).remove();
}
}
if(newSpan !== '')
{
iconObj.$div.prepend(newSpan);
if(customMarker !== '')
{
if(document.getElementById(elmID) !== null)
{
document.getElementById(elmID).innerHTML = customMarker;
}
}
if(addCommentCount)
{
var styleLeft;
if(nComments < 10) styleLeft = '11px';
else if(nComments < 100) styleLeft = '8px';
else styleLeft = '5px';
elmID = "commentCount_"+urID;
if(document.getElementById(elmID+"_inner") !== null)
{
document.getElementById(elmID+"_inner").innerHTML = nComments;
document.getElementById(elmID+"_inner").style.left = styleLeft;
}
}
}
}
}
}
}
else if(markerType == 'mp')
{
divElem = document.getElementById(W.map.problemLayer.id);
if(divElem.childNodes.length > 0)
{
for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
{
customType = uroCustomMarkerList[objIdx].customType;
if((customType >= 100) || (customType == -1))
{
urID = uroCustomMarkerList[objIdx].urID;
elmID = "customMarker_"+urID;
if(customType != -1)
{
if(document.getElementById(elmID) === null)
{
newSpan = '<span id="'+elmID+'" style="position:absolute;pointer-events:none;"></span>';
if(W.map.problemLayer.markers[urID] !== undefined)
{
W.map.problemLayer.markers[urID].icon.$div.prepend(newSpan);
}
}
if(document.getElementById(elmID) !== null)
{
customType = uroGetCustomMarkerIdx(customType);
if(W.model.problems.objects[urID] !== undefined)
{
if(W.model.problems.objects[urID].attributes.open === false) customType += 1;
}
customMarker = '<img src="'+uroIcons[customType][0]+'">';
document.getElementById(elmID).innerHTML = customMarker;
}
}
else
{
if(document.getElementById(elmID) !== null)
{
document.getElementById(elmID).remove();
}
}
}
}
}
}
}
function uroChangeCustomMarkers(urID,isHighlighted,customType,markerType)
{
if(document.getElementById('customMarker_'+urID) !== null)
{
if(markerType == "ur")
{
if(W.model.updateRequestSessions.objects[urID].open === false) customType += 1;
}
else if(markerType == "mp")
{
if(W.model.problems.objects[urID].attributes.open === false) customType += 1;
}
if(isHighlighted === true)
{
document.getElementById('customMarker_'+urID).innerHTML = '<img src="'+uroIcons[customType][1]+'">';
}
else
{
document.getElementById('customMarker_'+urID).innerHTML = '<img src="'+uroIcons[customType][0]+'">';
}
}
}
function uroFilterPlaces()
{
if(uroFilterPreamble() === false) return;
if(uroPlaceSelected === true) return;
if(uroGetCBChecked('_cbDisablePlacesFiltering') === true) return;
var filterCats = [];
for(var i=0; i<W.Config.venues.categories.length; i++)
{
var parentCategory = W.Config.venues.categories[i];
var subCategory;
if(uroGetCBChecked('_cbPlacesFilter-'+parentCategory) === true)
{
filterCats.push(parentCategory);
for(var i1=0; i1<W.Config.venues.subcategories[parentCategory].length; i1++)
{
subCategory = W.Config.venues.subcategories[parentCategory][i1];
filterCats.push(subCategory);
}
}
else
{
for(var i2=0; i2<W.Config.venues.subcategories[parentCategory].length; i2++)
{
subCategory = W.Config.venues.subcategories[parentCategory][i2];
if(uroGetCBChecked('_cbPlacesFilter-'+subCategory) === true)
{
filterCats.push(subCategory);
}
}
}
}
var placeStyle;
var uFP_filterEditedLessThan = uroGetCBChecked('_cbPlaceFilterEditedLessThan');
var uFP_filterEditedMoreThan = uroGetCBChecked('_cbPlaceFilterEditedMoreThan');
var uFP_filterL0 = uroGetCBChecked('_cbHidePlacesL0');
var uFP_filterL1 = uroGetCBChecked('_cbHidePlacesL1');
var uFP_filterL2 = uroGetCBChecked('_cbHidePlacesL2');
var uFP_filterL3 = uroGetCBChecked('_cbHidePlacesL3');
var uFP_filterL4 = uroGetCBChecked('_cbHidePlacesL4');
var uFP_filterL5 = uroGetCBChecked('_cbHidePlacesL5');
var uFP_filterOnLockLevel = (uFP_filterL0 || uFP_filterL1 || uFP_filterL2 || uFP_filterL3 || uFP_filterL4 || uFP_filterL5);
var uFP_filterNoPhotos = uroGetCBChecked('_cbHideNoPhotoPlaces');
var uFP_filterWithPhotos = uroGetCBChecked('_cbHidePhotoPlaces');
var uFP_filterPrivate = uroGetCBChecked('_cbFilterPrivatePlaces');
var uFP_invertFilters = uroGetCBChecked('_cbInvertPlacesFilter');
var uFP_masterEnable = uroGetCBChecked('_cbMasterEnable');
var uFP_thresholdMinDays = document.getElementById('_inputFilterPlaceEditMinDays').value;
var uFP_thresholdMaxDays = document.getElementById('_inputFilterPlaceEditMaxDays').value;
for(var v=0; v<W.map.landmarkLayer.features.length; v++)
{
placeStyle = 'visible';
if(uFP_masterEnable === true)
{
var lmObj = W.map.landmarkLayer.features[v];
// when an area place is selected, the drag points for editing the place outline now get added as objects into W.map.landmarkLayer.features,
// however none of these objects have the .model property - we must therefore check each entry in features[] to see if it has .model before
// attempting to filter it...
if(lmObj.model !== undefined)
{
if(lmObj.model.attributes.id < 0)
{
// don't apply filtering to newly-created places - this allows the user to leave their filtering settings unchanged whilst
// adding a new place which, once saved, would then be hidden...
break;
}
if((uFP_filterEditedLessThan) || (uFP_filterEditedMoreThan))
{
var editDate = lmObj.model.attributes.updatedOn;
if(editDate === undefined)
{
// where a place has never been edited since its creation, use the creation date instead...
editDate = lmObj.model.attributes.createdOn;
}
if(editDate !== undefined)
{
var editDaysAgo = uroDateToDays(editDate);
if(uFP_filterEditedLessThan)
{
if(editDaysAgo < uFP_thresholdMinDays)
{
placeStyle = 'hidden';
}
}
if(uFP_filterEditedMoreThan)
{
if(editDaysAgo > uFP_thresholdMaxDays)
{
placeStyle = 'hidden';
}
}
}
}
if(placeStyle == 'visible')
{
if(uFP_filterOnLockLevel)
{
var lockLevel = lmObj.model.attributes.lockRank;
if ((uFP_filterL0) && (lockLevel === 0)) placeStyle = 'hidden';
if ((uFP_filterL1) && (lockLevel === 1)) placeStyle = 'hidden';
if ((uFP_filterL2) && (lockLevel === 2)) placeStyle = 'hidden';
if ((uFP_filterL3) && (lockLevel === 3)) placeStyle = 'hidden';
if ((uFP_filterL4) && (lockLevel === 4)) placeStyle = 'hidden';
if ((uFP_filterL5) && (lockLevel === 5)) placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
if(uFP_filterNoPhotos || uFP_filterWithPhotos)
{
var nPhotos = 0;
for(var loop=0; loop<lmObj.model.attributes.images.length; loop++)
{
if(lmObj.model.attributes.images[loop].attributes.approved) nPhotos++;
}
if((uFP_filterNoPhotos) && (nPhotos === 0)) placeStyle = 'hidden';
if((uFP_filterWithPhotos) && (nPhotos !== 0)) placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
if((uFP_filterPrivate === true) && (lmObj.model.attributes.residential === true))
{
placeStyle = 'hidden';
}
else
{
for(var cat=0; cat<filterCats.length; cat++)
{
if(lmObj.model.attributes.categories.contains(filterCats[cat]))
{
placeStyle = 'hidden';
break;
}
}
}
}
}
if(uFP_invertFilters === true)
{
if(placeStyle == 'hidden') placeStyle = 'visible';
else placeStyle = 'hidden';
}
}
var geoID = W.map.landmarkLayer.features[v].geometry.id;
if(document.getElementById(geoID) !== null)
{
document.getElementById(geoID).style.visibility = placeStyle;
}
}
var uFP_filterUneditable = uroGetCBChecked('_cbFilterUneditablePlaceUpdates');
var uFP_filterLockRanked = uroGetCBChecked('_cbFilterLockRankedPlaceUpdates');
var uFP_filterFlagged = uroGetCBChecked("_cbFilterFlaggedPUR");
var uFP_filterNewPlace = uroGetCBChecked("_cbFilterNewPlacePUR");
var uFP_filterUpdatedDetails = uroGetCBChecked("_cbFilterUpdatedDetailsPUR");
var uFP_filterNewPhoto = uroGetCBChecked("_cbFilterNewPhotoPUR");
var uFP_filterMinPURAge = uroGetCBChecked('_cbEnablePURMinAgeFilter');
var uFP_filterMaxPURAge = uroGetCBChecked('_cbEnablePURMaxAgeFilter');
var uFP_invertPURFilters = uroGetCBChecked('_cbInvertPURFilters');
var uFP_filterHighSeverity = uroGetCBChecked('_cbPURFilterHighSeverity');
var uFP_filterMedSeverity = uroGetCBChecked('_cbPURFilterMediumSeverity');
var uFP_filterLowSeverity = uroGetCBChecked('_cbPURFilterLowSeverity');
var uFP_leavePURGeos = uroGetCBChecked('_cbLeavePURGeos');
var uFP_thresholdMinPURDays = uroGetElmValue('_inputPURFilterMinDays');
var uFP_thresholdMaxPURDays = uroGetElmValue('_inputPURFilterMaxDays');
var uFP_isLoggedIn = W.loginManager.isLoggedIn();
var uFP_userRank = W.loginManager.user.rank;
var purAge = null;
for(var pu in W.map.placeUpdatesLayer.markers)
{
if(W.map.placeUpdatesLayer.markers.hasOwnProperty(pu))
{
var puObj = W.map.placeUpdatesLayer.markers[pu];
if(W.map.placeUpdatesLayer.getVisibility() === true)
{
placeStyle = 'visible';
if(uFP_filterUneditable === true)
{
if(puObj.model.attributes.permissions === 0)
{
placeStyle = 'hidden';
}
if((placeStyle == 'visible') && (uFP_isLoggedIn))
{
if(uFP_userRank < puObj.model.attributes.lockRank)
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (puObj.model.attributes.adLocked))
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterLockRanked === true))
{
if(puObj.model.attributes.lockRank !== 0)
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterFlagged === true))
{
if(puObj.icon.imageDiv.className.indexOf('flag') != -1)
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterNewPlace === true))
{
if(puObj.icon.imageDiv.className.indexOf('add_venue') != -1)
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterUpdatedDetails === true))
{
if((puObj.icon.imageDiv.className.indexOf('update_venue') != -1) || (puObj.icon.imageDiv.className.indexOf('multiple') != -1))
{
placeStyle = 'hidden';
}
}
if((placeStyle == 'visible') && (uFP_filterNewPhoto === true))
{
if(puObj.icon.imageDiv.className.indexOf('add_image') != -1)
{
placeStyle = 'hidden';
}
}
if(uFP_invertPURFilters === true)
{
if(placeStyle == 'hidden') placeStyle = 'visible';
else placeStyle = 'hidden';
}
if(uFP_filterMinPURAge || uFP_filterMaxPURAge)
{
purAge = uroGetPURAge(puObj.model);
if(uFP_filterMinPURAge === true)
{
if(purAge < uFP_thresholdMinPURDays) placeStyle = 'hidden';
}
if(uFP_filterMaxPURAge === true)
{
if(purAge > uFP_thresholdMaxPURDays) placeStyle = 'hidden';
}
}
if(placeStyle == 'visible')
{
var purSeverity = puObj._getSeverity();
if((uFP_filterHighSeverity) && (purSeverity == "high")) placeStyle = 'hidden';
if((placeStyle == 'visible') && (uFP_filterMedSeverity) && (purSeverity == "medium")) placeStyle = 'hidden';
if((placeStyle == 'visible') && (uFP_filterLowSeverity) && (purSeverity == "low")) placeStyle = 'hidden';
}
puObj.icon.imageDiv.style.visibility = placeStyle;
if(uFP_leavePURGeos === false)
{
if(puObj.model !== undefined)
{
if(puObj.model.geometry !== undefined)
{
var puGeo = document.getElementById(puObj.model.geometry.id);
if(puGeo !== null)
{
puGeo.style.visibility = placeStyle;
}
}
}
}
}
}
}
}
function uroFilterCameras()
{
if(uroFilterPreamble() === false) return;
var camLayer = document.getElementById(uroRootContainer+'_svgRoot');
if(camLayer === null)
{
if(uroNullCamLayer === false)
{
uroAddLog('caught null camLayer');
uroNullCamLayer = true;
}
return;
}
uroNullCamLayer = false;
if(uroMouseIsDown === false) W.map.camerasLayer.redraw();
if(uroGetCBChecked('_cbMasterEnable') === true)
{
for (var uroCamObj in W.model.cameras.objects)
{
if(W.model.cameras.objects.hasOwnProperty(uroCamObj))
{
var uroCamUpdater = '';
var uroCamUpdaterRank = -1;
var uroCamCreator = '';
var uroCamCreatorRank = -1;
var uroCam = W.model.cameras.objects[uroCamObj];
var uroCamStyle = 'visible';
if(uroCam.attributes.createdBy !== null)
{
if(W.model.users.objects[uroCam.attributes.createdBy] !== undefined)
{
uroCamCreator = W.model.users.objects[uroCam.attributes.createdBy].userName;
uroCamCreatorRank = W.model.users.objects[uroCam.attributes.createdBy].rank;
}
}
if(uroCam.attributes.updatedBy !== null)
{
if(W.model.users.objects[uroCam.attributes.updatedBy] !== undefined)
{
uroCamUpdater = W.model.users.objects[uroCam.attributes.updatedBy].userName;
uroCamUpdaterRank = W.model.users.objects[uroCam.attributes.updatedBy].rank;
}
}
var uroCamApproved = uroCam.attributes.validated;
var uroCamType = uroCam.attributes.type;
if(uroGetCBChecked('_cbShowOnlyMyCams') === true)
{
if((uroUserID != uroCam.attributes.createdBy)&&(uroUserID != uroCam.attributes.updatedBy)) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbShowWorldCams') === false) || (uroGetCBChecked('_cbShowUSACams') === false) || (uroGetCBChecked('_cbShowNonWorldCams') === false))
{
var posWorld = uroCamCreator.indexOf('world_');
var posUSA = uroCamCreator.indexOf('usa_');
if((uroGetCBChecked('_cbShowWorldCams') === false) && (posWorld === 0)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowUSACams') === false) && (posUSA === 0)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowNonWorldCams') === false) && (posWorld !== 0) && (posUSA !== 0)) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbShowApprovedCams') === false) || (uroGetCBChecked('_cbShowNonApprovedCams') === false))
{
if((uroGetCBChecked('_cbShowApprovedCams') === false) && (uroCamApproved === true)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowNonApprovedCams') === false) && (uroCamApproved === false)) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbShowNonApprovedCams') === true) && (uroCamApproved === false))
{
if(((uroGetCBChecked('_cbShowOlderCreatedNonApproved') === true)) && (uroGetCameraAge(uroCam,1) <= uroGetElmValue('_inputCameraMinCreatedDays'))) uroCamStyle = 'hidden';
if(((uroGetCBChecked('_cbShowOlderUpdatedNonApproved') === true)) && (uroGetCameraAge(uroCam,0) <= uroGetElmValue('_inputCameraMinUpdatedDays'))) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbShowSpeedCams') === false) || (uroGetCBChecked('_cbShowRedLightCams') === false) || (uroGetCBChecked('_cbShowDummyCams') === false))
{
if((uroGetCBChecked('_cbShowSpeedCams') === false) && (uroCamType == 2)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowRedLightCams') === false) && (uroCamType == 4)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowDummyCams') === false) && (uroCamType == 3)) uroCamStyle = 'hidden';
}
if(uroGetCBChecked('_cbShowSpeedCams') === true)
{
if((uroGetCBChecked('_cbShowIfNoSpeedSet') === false) && (uroCam.attributes.speed === null)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbShowIfSpeedSet') === false) && (uroCam.attributes.speed !== null)) uroCamStyle = 'hidden';
}
if(uroGetCBChecked('_cbHideCreatedByMe') === true)
{
if(uroUserID == uroCam.attributes.createdBy) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbHideCreatedByRank0') === true) && (uroCamCreatorRank === 0)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank1') === true) && (uroCamCreatorRank == 1)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank2') === true) && (uroCamCreatorRank == 2)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank3') === true) && (uroCamCreatorRank == 3)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank4') === true) && (uroCamCreatorRank == 4)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCreatedByRank5') === true) && (uroCamCreatorRank == 5)) uroCamStyle = 'hidden';
if(uroGetCBChecked('_cbHideUpdatedByMe') === true)
{
if(uroUserID == uroCam.attributes.updatedBy) uroCamStyle = 'hidden';
}
if((uroGetCBChecked('_cbHideUpdatedByRank0') === true) && (uroCamUpdaterRank === 0)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank1') === true) && (uroCamUpdaterRank == 1)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank2') === true) && (uroCamUpdaterRank == 2)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank3') === true) && (uroCamUpdaterRank == 3)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank4') === true) && (uroCamUpdaterRank == 4)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideUpdatedByRank5') === true) && (uroCamUpdaterRank == 5)) uroCamStyle = 'hidden';
if((uroGetCBChecked('_cbHideCWLCams') === true) && (uroIsCamOnWatchList(uroCam.attributes.id) != -1)) uroCamStyle = 'hidden';
var uroCamGeometryID = uroCam.geometry.id;
if(camLayer.getElementById(uroCamGeometryID) !== null)
{
if(uroCamStyle == "hidden")
{
camLayer.getElementById(uroCamGeometryID).remove();
}
}
}
}
}
}
function uroFilterURs_onObjectsChanged()
{
if(uroBackfilling === false)
{
if(uroURDialogIsOpen === false)
{
uroURBackfill();
}
uroFilterURs();
}
}
function uroFilterURs_onObjectsAdded()
{
if(uroBackfilling === false)
{
uroURBackfill();
uroFilterURs();
}
}
function uroFilterURs_onObjectsRemoved()
{
if(uroBackfilling === false)
{
uroURBackfill();
uroFilterURs();
}
}
function uroURBackfill_GetData(lon, lat, blockSize)
{
lon = parseFloat(lon);
lat = parseFloat(lat);
blockSize = parseFloat(blockSize);
uroAddLog('Backfill square '+lon+','+lat);
var xmlReq = new XMLHttpRequest();
var tURL = 'https://' + document.location.host;
tURL += Waze.Config.api_base;
tURL += '/Features?language=en&mapUpdateRequestFilter=0';
if(uroPracticeMode === true) tURL += '&sandbox=true';
tURL += '&bbox='+(lon)+','+(lat)+','+(lon + blockSize)+','+(lat + blockSize);
xmlReq.open('GET',tURL,false);
try
{
xmlReq.send();
uroAddLog('response '+xmlReq.status+' received');
if (xmlReq.status === 200)
{
var tResp = JSON.parse(xmlReq.responseText);
var urCount = tResp.mapUpdateRequests.objects.length;
uroAddLog(urCount+' URs loaded for backfill processing');
if(urCount == 500)
{
uroAddLog('WARNING - backfill data may have been pre-filtered by server');
}
var backfilled = 0;
for(var i=0; i<urCount; i++)
{
var urID = tResp.mapUpdateRequests.objects[i].id;
if(W.model.mapUpdateRequests.objects[urID] === undefined)
{
var newUR = require('Waze/Feature/Vector/UpdateRequest');
var tUR = new newUR(tResp.mapUpdateRequests.objects[i]);
var tPoint = new OpenLayers.Geometry.Point();
tPoint.x = tResp.mapUpdateRequests.objects[i].geometry.coordinates[0];
tPoint.y = tResp.mapUpdateRequests.objects[i].geometry.coordinates[1];
tPoint.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
tUR.geometry = tPoint;
var tReqBounds = new OpenLayers.Geometry.Polygon();
var tBounds = new OpenLayers.Bounds();
tBounds.left = tPoint.x;
tBounds.right = tPoint.x;
tBounds.top = tPoint.y;
tBounds.bottom = tPoint.y;
tReqBounds.bounds = tBounds;
tUR.requestBounds = tReqBounds;
W.model.mapUpdateRequests.put(tUR);
backfilled++;
}
}
uroAddLog(backfilled+' URs backfilled');
}
else
{
uroAddLog('request failed (status != 200)');
}
}
catch(err)
{
uroAddLog('UR backfill request failed (exception '+err+' caught)');
}
}
function uroURBackfill()
{
if((uroGetCBChecked('_cbURBackfill') === false) || (uroGetCBChecked('_cbMasterEnable') === false))
{
return;
}
var nativeURCount = Object.keys(W.model.mapUpdateRequests.objects).length;
if(nativeURCount < 500)
{
return;
}
uroAddLog('exactly 500 URs loaded, possible server-side filtering requiring backfill...');
var subSize = 0.1;
var vpWidth = W.map.getExtent().getWidth();
var vpHeight = W.map.getExtent().getHeight();
var vpCentre = W.map.getCenter();
var vpLL = new OpenLayers.LonLat();
var vpUR = new OpenLayers.LonLat();
vpLL.lon = vpCentre.lon - (vpWidth / 2);
vpLL.lat = vpCentre.lat - (vpHeight / 2);
vpUR.lon = vpCentre.lon + (vpWidth / 2);
vpUR.lat = vpCentre.lat + (vpHeight / 2);
vpLL = vpLL.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
vpUR = vpUR.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
vpLL.lon -= (subSize / 2);
vpLL.lat -= (subSize / 2);
vpUR.lon += (subSize / 2);
vpUR.lat += (subSize / 2);
vpLL.lon = +vpLL.lon.toFixed(1);
vpLL.lat = +vpLL.lat.toFixed(1);
vpUR.lon = +vpUR.lon.toFixed(1);
vpUR.lat = +vpUR.lat.toFixed(1);
uroBackfilling = true;
for(var bfLat = vpLL.lat; bfLat <= vpUR.lat; bfLat += subSize)
{
for(var bfLon = vpLL.lon; bfLon <= vpUR.lon; bfLon += subSize)
{
uroURBackfill_GetData(bfLon, bfLat, subSize);
}
}
uroBackfilling = false;
uroFilterURs();
}
function uroStopThatDamnedRecentering()
{
return W.map.getExtent();
}
function uroFilterURs()
{
// compatibility fix for URComments - based on code supplied by RickZabel
var hasActiveURFilters = false;
if(uroGetCBChecked('_cbMasterEnable') === true)
{
var urTabInputs = document.getElementById('uroCtrlURs').getElementsByTagName('input');
for(var loop = 0; loop < urTabInputs.length; loop++)
{
if(urTabInputs[loop].type == 'checkbox')
{
var ignoreCB = false;
ignoreCB = ignoreCB || (urTabInputs[loop].id == '_cbCaseInsensitive');
ignoreCB = ignoreCB || (urTabInputs[loop].id == '_cbNoFilterForTaggedURs');
if((urTabInputs[loop].checked) && (ignoreCB === false))
{
hasActiveURFilters = true;
break;
}
}
}
}
sessionStorage.UROverview_hasActiveURFilters = hasActiveURFilters;
if(uroFilterPreamble() === false) return;
uroRefreshUpdateRequestSessions();
var selectorResolver = document.getElementById('_selectURResolverID');
var selectorCommentUser = document.getElementById('_selectURUserID');
if(uroGetCBChecked('_cbURResolverIDFilter') === false)
{
while(selectorResolver.options.length > 0)
{
selectorResolver.options.remove(0);
}
}
if(uroGetCBChecked('_cbURUserIDFilter') === false)
{
while(selectorCommentUser.options.length > 0)
{
selectorCommentUser.options.remove(0);
}
}
if(Object.keys(W.model.updateRequestSessions.objects).length === 0)
{
return;
}
var commenterUser = null;
if(uroGetCBChecked('_cbURUserIDFilter') === true)
{
if(selectorCommentUser.options.length === 0)
{
uroUpdateUserList();
}
if(selectorCommentUser.selectedOptions[0] !== undefined)
{
commenterUser = parseInt(selectorCommentUser.selectedOptions[0].value);
}
}
var resolverUser = null;
if(uroGetCBChecked('_cbURResolverIDFilter') === true)
{
if(selectorResolver.options.length === 0)
{
uroUpdateResolverList();
}
if(selectorResolver.selectedOptions[0] !== undefined)
{
resolverUser = parseInt(selectorResolver.selectedOptions[0].value);
}
}
uroCustomMarkerList = [];
var uFURs_masterEnable = uroGetCBChecked('_cbMasterEnable');
var filterOutsideEditableArea = uroGetCBChecked('_cbURFilterOutsideArea');
var filterSolved = uroGetCBChecked('_cbFilterSolved');
var filterUnidentified = uroGetCBChecked('_cbFilterUnidentified');
var filterClosed = uroGetCBChecked('_cbFilterClosedUR');
var filterOpen = uroGetCBChecked('_cbFilterOpenUR');
var filterDescMustBePresent = uroGetCBChecked('_cbURDescriptionMustBePresent');
var filterDescMustBeAbsent = uroGetCBChecked('_cbURDescriptionMustBeAbsent');
var filterKeywordMustBePresent = uroGetCBChecked('_cbEnableKeywordMustBePresent');
var filterKeywordMustBeAbsent = uroGetCBChecked('_cbEnableKeywordMustBeAbsent');
var filterMinURAge = uroGetCBChecked('_cbEnableMinAgeFilter');
var filterMaxURAge = uroGetCBChecked('_cbEnableMaxAgeFilter');
var filterMinComments = uroGetCBChecked('_cbEnableMinCommentsFilter');
var filterMaxComments = uroGetCBChecked('_cbEnableMaxCommentsFilter');
var filterReporterLastCommenter = uroGetCBChecked('_cbHideIfReporterLastCommenter');
var filterReporterNotLastCommenter = uroGetCBChecked('_cbHideIfReporterNotLastCommenter');
var filterHideAnyComments = uroGetCBChecked('_cbHideAnyComments');
var filterHideNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
var filterHideMyComments = uroGetCBChecked('_cbHideMyComments');
var filterIfLastCommenter = uroGetCBChecked('_cbHideIfLastCommenter');
var filterIfNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
var filterCommentMinAge = uroGetCBChecked('_cbEnableCommentAgeFilter2');
var filterCommentMaxAge = uroGetCBChecked('_cbEnableCommentAgeFilter');
var filterUserID = uroGetCBChecked('_cbURUserIDFilter');
var filterMyFollowed = uroGetCBChecked('_cbHideMyFollowed');
var filterMyUnfollowed = uroGetCBChecked('_cbHideMyUnfollowed');
var filterWazeAuto = uroGetCBChecked('_cbFilterWazeAuto');
var filterRoadworks = uroGetCBChecked('_cbFilterRoadworks');
var filterConstruction = uroGetCBChecked('_cbFilterConstruction');
var filterClosure = uroGetCBChecked('_cbFilterClosure');
var filterEvent = uroGetCBChecked('_cbFilterEvent');
var filterNote = uroGetCBChecked('_cbFilterNote');
var filterIncorrectTurn = uroGetCBChecked('_cbFilterIncorrectTurn');
var filterIncorrectAddress = uroGetCBChecked('_cbFilterIncorrectAddress');
var filterIncorrectRoute = uroGetCBChecked('_cbFilterIncorrectRoute');
var filterMissingRoundabout = uroGetCBChecked('_cbFilterMissingRoundabout');
var filterGeneralError = uroGetCBChecked('_cbFilterGeneralError');
var filterTurnNotAllowed = uroGetCBChecked('_cbFilterTurnNotAllowed');
var filterIncorrectJunction = uroGetCBChecked('_cbFilterIncorrectJunction');
var filterMissingBridgeOverpass = uroGetCBChecked('_cbFilterMissingBridgeOverpass');
var filterWrongDrivingDirection = uroGetCBChecked('_cbFilterWrongDrivingDirection');
var filterMissingExit = uroGetCBChecked('_cbFilterMissingExit');
var filterMissingRoad = uroGetCBChecked('_cbFilterMissingRoad');
var filterMissingLandmark = uroGetCBChecked('_cbFilterMissingLandmark');
var filterBlockedRoad = uroGetCBChecked('_cbFilterBlockedRoad');
var filterUndefined = uroGetCBChecked('_cbFilterUndefined');
var invertURFilters = uroGetCBChecked('_cbInvertURFilter');
var noFilterTaggedURs = uroGetCBChecked('_cbNoFilterForTaggedURs');
var noFilterURInURL = uroGetCBChecked('_cbNoFilterForURInURL');
var keywordPresent = uroGetElmValue('_textKeywordPresent');
var keywordAbsent = uroGetElmValue('_textKeywordAbsent');
var thresholdMinAge = uroGetElmValue('_inputFilterMinDays');
var thresholdMaxAge = uroGetElmValue('_inputFilterMaxDays');
var thresholdMinComments = uroGetElmValue('_inputFilterMinComments');
var thresholdMaxComments = uroGetElmValue('_inputFilterMaxComments');
var thresholdMaxCommentAge = uroGetElmValue('_inputFilterCommentDays');
var thresholdMinCommentAge = uroGetElmValue('_inputFilterCommentDays2');
var urcFilteringIsActive = false;
var urcCB = document.getElementById('URCommentsFilterEnabled');
if(urcCB !== null)
{
if(urcCB.checked)
{
urcFilteringIsActive = true;
}
}
urcCB = document.getElementById('URCommentUROOnlyMyUR');
if(urcCB !== null)
{
if(urcCB.checked)
{
urcFilteringIsActive = true;
}
}
urcCB = document.getElementById('URCommentUROHideTagged');
if(urcCB !== null)
{
if(urcCB.checked)
{
urcFilteringIsActive = true;
}
}
for (var urobj in W.model.mapUpdateRequests.objects)
{
if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
{
var ureq = W.model.mapUpdateRequests.objects[urobj];
if(uroGetCBChecked('_cbInhibitURCentering') === true)
{
// seems the easiest way to prevent WME recentering the mapview on the UR when it's clicked, is to replace the
// getBounds() function in each UR object with one that returns the current mapview extents...
ureq.getRequestBounds().getBounds = uroStopThatDamnedRecentering;
}
var urStyle = 'visible';
var ureqID = null;
if(ureq.fid === null) ureqID = ureq.attributes.id;
else ureqID = ureq.fid;
var inhibitFiltering = ((ureqID == uroURIDInURL) && (noFilterURInURL));
var hasMyComments = false;
var nComments = 0;
var customType = uroGetCustomType(ureqID, "ur");
if(W.model.updateRequestSessions.objects[ureqID] !== undefined)
{
nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
if((uFURs_masterEnable === false) && (nComments === 0))
{
// when master enable is turned off, we want to make sure that all URs, including ones that were previously hidden, are correctly
// displayed in their native form - i.e. no comment count or custom conversation bubbles. The easiest way to achieve this is to
// force the uroRenderCustomMarkers code to test for the presence of these bubbles on each UR, which we do by setting a non-zero
// comment count for each UR... For URs which genuinely do have no comments we use -1 to indicate that we're not really setting
// a comment count, but that we still need to do something that wouldn't be achieved by using 0.
nComments = -1;
}
}
if((uFURs_masterEnable === true) && (inhibitFiltering === false))
{
var wazeauto_ur = false;
var ukroadworks_ur = false;
var construction_ur = false;
var closure_ur = false;
var event_ur = false;
var note_ur = false;
var filterByNotIncludedKeyword = false;
var filterByIncludedKeyword = true;
var desc = '';
if(ureq.attributes.description !== null) desc = ureq.attributes.description.replace(/<\/?[^>]+(>|$)/g, "");
if(customType === 0) ukroadworks_ur = true;
else if(customType === 1) construction_ur = true;
else if(customType === 2) closure_ur = true;
else if(customType === 3) event_ur = true;
else if(customType === 4) note_ur = true;
// check UR against editable area...
if(filterOutsideEditableArea === true)
{
if(ureq.canEdit() === false) urStyle = 'hidden';
}
// check UR against current session ignore list...
if(uroIsOnIgnoreList(ureqID)) urStyle = 'hidden';
// check against closed/not identified filtering if enabled...
if(filterSolved === true)
{
if(ureq.attributes.resolution === 0) urStyle = 'hidden';
}
if(filterUnidentified === true)
{
if(ureq.attributes.resolution == 1) urStyle = 'hidden';
}
if((ureq.attributes.resolvedOn !== null) && (filterClosed === true))
{
urStyle = 'hidden';
}
if((ureq.attributes.resolvedOn === null) && (filterOpen === true))
{
urStyle = 'hidden';
}
if(urStyle == 'visible')
{
// check UR against keyword filtering if enabled...
if(filterDescMustBePresent === true)
{
if(desc === '') urStyle = 'hidden';
}
if(filterDescMustBeAbsent === true)
{
if(desc !== '') urStyle = 'hidden';
}
if(filterKeywordMustBePresent === true)
{
var keywordIsPresentInDesc = uroKeywordPresent(desc,keywordPresent);
filterByIncludedKeyword &= (!keywordIsPresentInDesc);
}
if(filterKeywordMustBeAbsent === true)
{
var keywordIsAbsentInDesc = uroKeywordPresent(desc,keywordAbsent);
filterByNotIncludedKeyword |= keywordIsAbsentInDesc;
}
}
if(urStyle == 'visible')
{
// do age-based filtering if enabled
if(filterMinURAge === true)
{
if(uroGetURAge(ureq,0,false) < thresholdMinAge) urStyle = 'hidden';
}
if(filterMaxURAge === true)
{
if(uroGetURAge(ureq,0,false) > thresholdMaxAge) urStyle = 'hidden';
}
}
if(urStyle == 'visible')
{
if(resolverUser !== null)
{
if(ureq.attributes.resolvedBy != resolverUser) urStyle = 'hidden';
}
}
if(urStyle == 'visible')
{
// do comments/following filtering
if(W.model.updateRequestSessions.objects[ureqID] !== undefined)
{
nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
var commentDaysOld = -1;
if(filterMinComments === true)
{
if(nComments < thresholdMinComments) urStyle = 'hidden';
}
if(filterMaxComments === true)
{
if(nComments > thresholdMaxComments) urStyle = 'hidden';
}
if(nComments > 0)
{
var reporterIsLastCommenter = false;
if(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1].userID == -1) reporterIsLastCommenter = true;
if(filterReporterLastCommenter === true)
{
if(reporterIsLastCommenter === true) urStyle = 'hidden';
}
else if(filterReporterNotLastCommenter === true)
{
if(reporterIsLastCommenter === false) urStyle = 'hidden';
}
hasMyComments = uroURHasMyComments(ureqID);
if(hasMyComments === false)
{
if(filterHideAnyComments === true) urStyle = 'hidden';
if(filterHideNotLastCommenter === true) urStyle = 'hidden';
}
else
{
if(filterHideMyComments === true) urStyle = 'hidden';
var userIsLastCommenter = false;
if(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1].userID == uroUserID) userIsLastCommenter = true;
if(filterIfLastCommenter === true)
{
if(userIsLastCommenter === true) urStyle = 'hidden';
}
else if(filterIfNotLastCommenter === true)
{
if(userIsLastCommenter === false) urStyle = 'hidden';
}
}
commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1]);
if((filterCommentMinAge === true) && (commentDaysOld != -1))
{
if(thresholdMinCommentAge > commentDaysOld) urStyle = 'hidden';
}
if((filterCommentMaxAge === true) && (commentDaysOld != -1))
{
if(thresholdMaxCommentAge < commentDaysOld) urStyle = 'hidden';
}
var cidx;
if((commenterUser !== null) && (urStyle != 'hidden'))
{
urStyle = 'hidden';
for(cidx=0; cidx<nComments; cidx++)
{
if(W.model.updateRequestSessions.objects[ureqID].comments[cidx].userID == commenterUser)
{
urStyle = 'visible';
break;
}
}
}
var commentText = '';
for(cidx=0; cidx<nComments; cidx++)
{
commentText += W.model.updateRequestSessions.objects[ureqID].comments[cidx].text;
}
if(filterKeywordMustBePresent === true)
{
var keywordIsPresentInComments = uroKeywordPresent(commentText,keywordPresent);
filterByIncludedKeyword &= (!keywordIsPresentInComments);
}
if(filterKeywordMustBeAbsent === true)
{
var keywordIsAbsentInComments = uroKeywordPresent(commentText,keywordAbsent);
filterByNotIncludedKeyword |= keywordIsAbsentInComments;
}
}
else
{
if(filterUserID === true)
{
urStyle = 'hidden';
}
}
filterByNotIncludedKeyword &= filterKeywordMustBeAbsent;
filterByIncludedKeyword &= filterKeywordMustBePresent;
if(filterByNotIncludedKeyword || filterByIncludedKeyword)
{
urStyle = 'hidden';
}
if(W.model.updateRequestSessions.objects[ureqID].isFollowing === true)
{
if(filterMyFollowed === true) urStyle = 'hidden';
}
else
{
if(filterMyUnfollowed === true) urStyle = 'hidden';
}
}
}
if(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(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(filterUndefined === true) urStyle = 'hidden';
if(invertURFilters === true)
{
if(urStyle == 'hidden') urStyle = 'visible';
else urStyle = 'hidden';
}
}
// 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';
}
}
}
// only touch marker visibility if we've got active filter settings, or if URComments is not
// doing any filtering of its own
if((hasActiveURFilters === true) || (urcFilteringIsActive === false) || (uFURs_masterEnable === false))
{
W.map.updateRequestLayer.markers[urobj].icon.imageDiv.style.visibility = urStyle;
}
if(urStyle != 'hidden')
{
uroAddCustomMarkers(ureqID,customType, hasMyComments, nComments);
}
}
}
uroRenderCustomMarkers('ur');
}
function uroFilterProblems()
{
if(uroFilterPreamble() === false) return;
var selector;
if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === false) && (uroGetCBChecked('_cbMPClosedUserIDFilter') === false))
{
selector = document.getElementById('_selectMPUserID');
while(selector.options.length > 0)
{
selector.options.remove(0);
}
}
var solverUser = null;
if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === true) || (uroGetCBChecked('_cbMPClosedUserIDFilter') === true))
{
selector = document.getElementById('_selectMPUserID');
if(selector.options.length === 0)
{
uroUpdateMPSolverList();
}
if(selector.selectedOptions[0] !== undefined)
{
solverUser = parseInt(selector.selectedOptions[0].value);
}
}
var urobj;
var problem;
var problemStyle;
var problem_marker_img;
for (urobj in W.model.problems.objects)
{
if(W.model.problems.objects.hasOwnProperty(urobj))
{
problem = W.model.problems.objects[urobj];
problemStyle = 'visible';
var ureqID = null;
var customType = null;
if(uroGetCBChecked('_cbMasterEnable') === true)
{
ureqID = problem.attributes.id;
customType = uroGetCustomType(ureqID, "mp");
// check problem against current session ignore list...
if(uroIsOnIgnoreList(ureqID)) problemStyle = 'hidden';
if(uroGetCBChecked('_cbMPFilterOutsideArea') === true)
{
if(problem.canEdit() === false)
{
problemStyle = 'hidden';
}
}
// check against closed/not identified filtering if enabled...
problem_marker_img = '';
if(problem.geometry.id !== null)
{
if(document.getElementById(problem.geometry.id) !== null)
{
problem_marker_img = document.getElementById(problem.geometry.id).href.baseVal;
if(uroGetCBChecked('_cbMPFilterSolved') === true)
{
if(problem_marker_img.indexOf('_solved') != -1) problemStyle = 'hidden';
}
if(uroGetCBChecked('_cbMPFilterUnidentified') === true)
{
if(problem_marker_img.indexOf('_rejected') != -1) problemStyle = 'hidden';
}
}
}
if(uroGetCBChecked('_cbMPFilterClosed') === true)
{
if(problem.attributes.open === false)
{
problemStyle = 'hidden';
}
}
if(problemStyle == 'visible')
{
if(solverUser !== null)
{
if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === true) && (problem.attributes.resolvedBy == solverUser)) problemStyle = 'hidden';
if((uroGetCBChecked('_cbMPClosedUserIDFilter') === true) && (problem.attributes.resolvedBy != solverUser)) problemStyle = 'hidden';
}
}
if(problemStyle == 'visible')
{
var problemType = null;
if(uroDOMHasTurnProblems)
{
problemType = problem.attributes.problemType;
}
else
{
problemType = problem.attributes.subType;
}
if(problemType == 101)
{
if(uroGetCBChecked('_cbMPFilterDrivingDirectionMismatch') === true) problemStyle = 'hidden';
}
else if(problemType == 102)
{
if(uroGetCBChecked('_cbMPFilterMissingJunction') === true) problemStyle = 'hidden';
}
else if(problemType == 103)
{
if(uroGetCBChecked('_cbMPFilterMissingRoad') === true) problemStyle = 'hidden';
}
else if(problemType == 104)
{
if(uroGetCBChecked('_cbMPFilterCrossroadsJunctionMissing') === true) problemStyle = 'hidden';
}
else if(problemType == 105)
{
if(uroGetCBChecked('_cbMPFilterRoadTypeMismatch') === true) problemStyle = 'hidden';
}
else if(problemType == 106)
{
if(uroGetCBChecked('_cbMPFilterRestrictedTurn') === true) problemStyle = 'hidden';
}
else if(problemType == 200)
{
if(uroGetCBChecked('_cbMPFilterTurnProblem') === true) problemStyle = 'hidden';
}
else if(problemType == 300)
{
if(uroGetCBChecked('_cbMPFilterRoadClosureProblem') === true) problemStyle = 'hidden';
}
else if(uroGetCBChecked('_cbMPFilterUnknownProblem') === true) problemStyle = 'hidden';
if(uroGetCBChecked('_cbMPFilterReopenedProblem') === true)
{
if((problem.attributes.open === true) && (problem.attributes.resolvedOn !== null))
{
problemStyle = 'hidden';
}
}
if(uroGetCBChecked('_cbInvertMPFilter') === true)
{
if(problemStyle == 'hidden') problemStyle = 'visible';
else problemStyle = 'hidden';
}
if(problem.attributes.weight <= 3)
{
if(uroGetCBChecked('_cbMPFilterLowSeverity') === true) problemStyle = 'hidden';
}
else if(problem.attributes.weight <= 7)
{
if(uroGetCBChecked('_cbMPFilterMediumSeverity') === true) problemStyle = 'hidden';
}
else if(uroGetCBChecked('_cbMPFilterHighSeverity') === true) problemStyle = 'hidden';
}
}
W.map.problemLayer.markers[urobj].icon.imageDiv.style.visibility = problemStyle;
if((problemStyle != 'hidden') && (ureqID !== null) && (customType !== null))
{
uroAddCustomMarkers(ureqID,customType, false, 0);
}
}
}
if(uroDOMHasTurnProblems)
{
for (urobj in W.model.turnProblems.objects)
{
if(W.model.turnProblems.objects.hasOwnProperty(urobj))
{
problem = W.model.turnProblems.objects[urobj];
problemStyle = 'visible';
if(uroGetCBChecked('_cbMasterEnable') === true)
{
// check problem against current session ignore list...
if(uroIsOnIgnoreList(problem.attributes.id)) problemStyle = 'hidden';
// check against closed/not identified filtering if enabled...
problem_marker_img = '';
if(problem.geometry.id !== null)
{
if(document.getElementById(problem.geometry.id) !== null)
{
problem_marker_img = document.getElementById(problem.geometry.id).href.baseVal;
if(uroGetCBChecked('_cbMPFilterSolved') === true)
{
if(problem_marker_img.indexOf('_solved') != -1) problemStyle = 'hidden';
}
if(uroGetCBChecked('_cbMPFilterUnidentified') === true)
{
if(problem_marker_img.indexOf('_rejected') != -1) problemStyle = 'hidden';
}
}
}
if(uroGetCBChecked('_cbMPFilterClosed') === true)
{
if(problem.attributes.open === false)
{
problemStyle = 'hidden';
}
}
if(problemStyle == 'visible')
{
if(uroGetCBChecked('_cbMPFilterTurnProblem') === true) problemStyle = 'hidden';
if(uroGetCBChecked('_cbMPFilterReopenedProblem') === true)
{
if((problem.attributes.open === true) && (problem.attributes.resolvedOn !== null))
{
problemStyle = 'hidden';
}
}
if(uroGetCBChecked('_cbInvertMPFilter') === true)
{
if(problemStyle == 'hidden') problemStyle = 'visible';
else problemStyle = 'hidden';
}
}
}
W.map.problemLayer.markers[urobj].icon.imageDiv.style.visibility = problemStyle;
}
}
}
uroRenderCustomMarkers('mp');
}
function uroToHex(decValue,digits)
{
var modifier = 1;
for(var i=0; i<digits; i++)
{
modifier *= 16;
}
decValue = parseInt(decValue);
decValue += modifier;
var retval = decValue.toString(16);
retval = retval.substr(-digits);
retval = retval.toUpperCase();
return retval;
}
function uroFilterPreamble()
{
var mapviewport = document.getElementsByClassName("olMapViewport")[0];
if(mapviewport === null)
{
if(uroNullMapViewport === false)
{
uroAddLog('caught null mapviewport');
uroNullMapViewport = true;
}
return false;
}
uroNullMapViewport = false;
if((uroGetCBChecked('_cbWhiteBackground') === true) && (uroGetCBChecked('_cbMasterEnable') === true))
{
var customColour = '#' + uroToHex(uroGetElmValue('_inputCustomBackgroundRed'),2);
customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundGreen'),2);
customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundBlue'),2);
mapviewport.style.backgroundColor = customColour;
}
else
{
mapviewport.style.backgroundColor = "#C2C2C2";
}
if((uroGetCBChecked('_cbHideAMLayer')) && (uroGetCBChecked('_cbMasterEnable') === true))
{
W.map.managedAreasLayer.setOpacity(0);
}
else
{
W.map.managedAreasLayer.setOpacity(1);
}
return true;
}
function uroFilterItems_URTabClick()
{
uroFilterURs();
}
function uroFilterItems_MPTabClick()
{
uroFilterProblems();
}
function uroFilterItems_PlacesTabClick()
{
uroFilterPlaces();
}
function uroFilterItems_CamerasTabClick()
{
uroFilterCameras();
}
function uroFilterItems_MiscTabClick()
{
uroFilterItems();
}
function uroFilterItems_MasterEnableClick()
{
if(uroGetCBChecked('_cbMasterEnable') === false)
{
uroHidePopup();
}
uroFilterItems();
}
function uroFilterItems()
{
uroFilterProblems();
uroFilterPlaces();
uroFilterCameras();
uroFilterURs();
}
function uroFilterItemsOnMove()
{
W.map.events.unregister('mousemove',null,uroFilterItemsOnMove);
uroFilterItems();
}
function uroDeleteObject()
{
uroAddLog('delete camera ID '+uroShownFID);
if(W.model.cameras.objects[uroShownFID] === null)
{
uroAddLog('camera object not found...');
return false;
}
uroRemoveCamFromWatchList();
var actionObj = require('Waze/Action/DeleteObject');
var deleteAction = new actionObj(W.model.cameras.objects[uroShownFID], null);
W.model.actionManager.add(deleteAction);
uroExitPopup();
uroHidePopup();
return false;
}
function uroGetUserNameAndRank(userID)
{
var userName;
var userLevel;
if(W.model.users.objects[userID] !== undefined)
{
userName = W.model.users.objects[userID].userName;
if(userName === undefined)
{
userName = userID;
}
userLevel = W.model.users.objects[userID].rank + 1;
}
else
{
userName = userID;
userLevel = '?';
}
return userName + ' (' + userLevel + ')';
}
function uroCheckCommentsForKeyword(idSrc, keyword)
{
var ursObj = W.model.updateRequestSessions.objects[idSrc];
if(typeof(ursObj) == 'undefined') return false;
if(ursObj.comments.length === 0) return false;
for(var idx=0; idx<ursObj.comments.length; idx++)
{
if(ursObj.comments[idx].text.indexOf(keyword) != -1)
{
return true;
}
}
return false;
}
function uroGetCustomMarkerIdx(customType)
{
if(customType === 0) return 3;
if(customType === 1) return 3;
if(customType === 2) return 5;
if(customType === 3) return 7;
if(customType === 4) return 9;
if(customType === 100) return 11;
if(customType === 101) return 13;
if(customType === 102) return 15;
return -1;
}
function uroGetCustomType(idSrc, markerType)
{
var desc = '';
if(markerType == "ur")
{
var ureq = W.model.mapUpdateRequests.objects[idSrc];
if(ureq.attributes.description !== null)
{
desc = ureq.attributes.description;
}
}
else if(markerType == "mp")
{
var mp = W.model.problems.objects[idSrc];
if(mp.attributes.description !== null)
{
desc = mp.attributes.description;
}
}
if(desc !== '')
{
if(desc.indexOf('[ROADWORKS]') != -1) return 0;
if(desc.indexOf('[CONSTRUCTION]') != -1) return 1;
if(desc.indexOf('[CLOSURE]') != -1) return 2;
if(desc.indexOf('[EVENT]') != -1) return 3;
if(desc.indexOf('[NOTE]') != -1) return 4;
if(desc.indexOf('[Elgin]') != -1) return 100;
if(desc.indexOf('[TrafficCast]') != -1) return 101;
if(desc.indexOf('[TM]') != -1) return 102;
}
if(markerType == "ur")
{
if(uroCheckCommentsForKeyword(idSrc,'[ROADWORKS]')) return 0;
if(uroCheckCommentsForKeyword(idSrc,'[CONSTRUCTION]')) return 1;
if(uroCheckCommentsForKeyword(idSrc,'[CLOSURE]')) return 2;
if(uroCheckCommentsForKeyword(idSrc,'[EVENT]')) return 3;
if(uroCheckCommentsForKeyword(idSrc,'[NOTE]')) return 4;
}
return -1;
}
function uroFormatRestriction(restObj)
{
var retval = '<tr>';
retval += '<td style="text-align:center;">';
if((restObj.days & 1) == 1) retval += 'S';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 2) == 2) retval += 'M';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 4) == 4) retval += 'T';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 8) == 8) retval += 'W';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 16) == 16) retval += 'T';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 32) == 32) retval += 'F';
else retval += '-';
retval += '</td><td style="text-align:center;">';
if((restObj.days & 64) == 64) retval += 'S';
else retval += '-';
retval += '</td><td>';
if(restObj.fromDate === null) retval += 'All dates';
else retval += restObj.fromDate+' to '+restObj.toDate;
retval += '</td><td>';
if(restObj.allDay === true) retval += 'All day';
else retval += restObj.fromTime+' to '+restObj.toTime;
retval += '</td><td>';
if(restObj.allVehicleTypes == restObj.vehicleTypes) retval += 'All vehicles';
else retval += 'Some vehicles';
retval += '</td><td>';
if(restObj.description !== null)
{
var desc = restObj.description.replace(/<\/?[^>]+(>|$)/g, "");
desc = uroClickify(desc);
retval += desc;
}
retval += '</td></tr>';
return retval;
}
function uroHidePopup()
{
if(uroPopupShown)
{
uroDiv.style.visibility = 'hidden';
uroPopupShown = false;
uroPopupTimer = -2;
uroShownFID = -1;
}
}
function uroRecentreSessionOnUR()
{
W.map.updateRequestLayer.markers[uroShownFID].icon.imageDiv.click();
W.map.moveTo(W.map.updateRequestLayer.markers[uroShownFID].lonlat, 5);
uroHidePopup();
return false;
}
function uroRecentreSessionOnMP()
{
W.map.problemLayer.markers[uroShownFID].icon.imageDiv.click();
W.map.moveTo(W.map.problemLayer.markers[uroShownFID].lonlat, 5);
uroHidePopup();
return false;
}
function uroRecentreSessionOnPUR()
{
W.map.placeUpdatesLayer.markers[uroShownFID].icon.imageDiv.click();
W.map.moveTo(W.map.placeUpdatesLayer.markers[uroShownFID].lonlat, 5);
uroHidePopup();
return false;
}
function uroRecentreSessionOnVenueNavPoint()
{
W.map.moveTo(uroGetVenueNavPoint(uroShownFID), 5);
uroHidePopup();
return false;
}
function uroGetDateTimeString(ts)
{
var tDateObj = new Date(ts);
var dateLocale;
var timeLocale;
if(uroGetCBChecked('_cbDateFmtDDMMYY')) dateLocale = 'en-gb';
if(uroGetCBChecked('_cbDateFmtMMDDYY')) dateLocale = 'en-us';
if(uroGetCBChecked('_cbDateFmtYYMMDD')) dateLocale = 'ja';
if(uroGetCBChecked('_cbTimeFmt24H')) timeLocale = 'en-gb';
if(uroGetCBChecked('_cbTimeFmt12H')) timeLocale = 'en-us';
return tDateObj.toLocaleDateString(dateLocale) + ' ' + tDateObj.toLocaleTimeString(timeLocale);
}
function uroParsePxString(pxString)
{
return parseInt(pxString.split("px")[0]);
}
function uroStackListObj(fid,x,y)
{
this.fid = fid;
this.x = uroTypeCast(x);
this.y = uroTypeCast(y);
}
function uroRestackMarkers()
{
if(uroStackList.length === 0) return;
var markerCollection = null;
if(uroStackType == 1) markerCollection = W.map.updateRequestLayer.markers;
else if(uroStackType == 2) markerCollection = W.map.problemLayer.markers;
else if(uroStackType == 3) markerCollection = W.map.placeUpdatesLayer.markers;
if(markerCollection !== null)
{
for(var idx=0; idx<uroStackList.length; idx++)
{
var orig_x = uroStackList[idx].x + 'px';
var orig_y = uroStackList[idx].y + 'px';
var fid = uroStackList[idx].fid;
if(markerCollection[fid] !== undefined)
{
markerCollection[fid].icon.imageDiv.style.left = orig_x;
markerCollection[fid].icon.imageDiv.style.top = orig_y;
}
}
uroStackList = [];
uroUnstackedMasterID = null;
uroStackType = null;
}
}
function uroIsIDAlreadyUnstacked(idSrc)
{
if(uroStackList.length === 0) return false;
for(var idx=0; idx<uroStackList.length; idx++)
{
if(uroStackList[idx].fid == idSrc) return true;
}
return false;
}
function uroCheckStacking(stackType, masterID, unstackedX, unstackedY)
{
if(W.map.getZoom() < uroGetElmValue('_inputUnstackZoomLevel')) return;
if(uroIsIDAlreadyUnstacked(masterID) === true) return;
if(uroPopupDwellTimer > 0) return;
uroAddLog('checking for marker stack, type '+stackType+'...');
var stackList = [];
var threshSquared = uroGetElmValue('_inputUnstackSensitivity');
threshSquared *= threshSquared;
var markerCollection = null;
var marker;
var tempX = 1000000000;
if(stackType == 1) markerCollection = W.map.updateRequestLayer.markers;
else if(stackType == 2) markerCollection = W.map.problemLayer.markers;
else if(stackType == 3) markerCollection = W.map.placeUpdatesLayer.markers;
if(markerCollection !== null)
{
for(marker in markerCollection)
{
if(markerCollection.hasOwnProperty(marker))
{
var testMarkerObj = markerCollection[marker];
var includeInStack = (testMarkerObj.icon.imageDiv.style.visibility != 'hidden');
var suppressClosed = (testMarkerObj.icon.imageDiv.classList.contains("recently-closed") & (W.map.updateRequestLayer.showHidden === false));
if((includeInStack) && (!suppressClosed))
{
if(testMarkerObj.id != masterID)
{
var xdiff = unstackedX - uroParsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.left);
var ydiff = unstackedY - uroParsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.top);
var distSquared = ((xdiff * xdiff) + (ydiff * ydiff));
if(distSquared < threshSquared)
{
testMarkerObj.model.attributes.geometry.x = tempX;
tempX++;
stackList.push(testMarkerObj.id);
}
}
}
}
}
}
if(stackList.length > 0)
{
uroAddLog('markers are stacked!');
if(uroUnstackedMasterID != masterID)
{
uroAddLog('unstacked ID mismatch, relocating markers...');
uroRestackMarkers();
uroStackType = stackType;
uroUnstackedMasterID = masterID;
uroStackList = [];
// push the highlighted marker onto the stacklist so uroIsIDAlreadyUnstacked() will return true
uroStackList.push(new uroStackListObj(masterID,unstackedX,unstackedY));
for(var shoveIdx=0; shoveIdx < stackList.length; shoveIdx++)
{
var fid = stackList[shoveIdx];
var x = uroParsePxString(markerCollection[fid].icon.imageDiv.style.left);
var y = uroParsePxString(markerCollection[fid].icon.imageDiv.style.top);
// store the unstacked marker positions so they can be reinstated later
uroStackList.push(new uroStackListObj(fid,x,y));
unstackedX += 10;
unstackedY -= 30;
markerCollection[fid].icon.imageDiv.style.left = unstackedX + 'px';
markerCollection[fid].icon.imageDiv.style.top = unstackedY + 'px';
}
// hide other markers to prevent confusion with the unstacked markers
for(marker in markerCollection)
{
if(markerCollection.hasOwnProperty(marker))
{
var toHideID = markerCollection[marker].id;
if(uroIsIDAlreadyUnstacked(toHideID) === false)
{
markerCollection[toHideID].icon.imageDiv.style.visibility = 'hidden';
}
}
}
}
}
else
{
uroRestackMarkers();
}
}
function uroGetVenueNavPoint(uroFID)
{
for(var vObj in W.model.venues.objects)
{
if(W.model.venues.objects.hasOwnProperty(vObj))
{
if(uroFID == vObj)
{
return W.model.venues.objects[vObj].getNavigationPoint().point.toLonLat();
}
}
}
// just in case... return a safe value if the requested venue object wasn't found
return W.map.getCenter();
}
function uroOpenNewTab()
{
// flush the current settings into localStorage before the new tab opens, so that when its instance of
// URO+ fires up it'll have the same settings as this one
uroSaveSettings();
return true;
}
function uroEditTBR()
{
if(uroTBRObj === null)
{
return;
}
uroTBRObj.click();
return false;
}
function uroNewLookHighlightedItemsCheck(e)
{
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;
var rc = document.getElementById(uroRootContainer);
if(rc === null)
{
if(uroNullRootContainer === false)
{
uroAddLog('caught null rootContainer');
uroNullRootContainer = true;
}
return;
}
uroNullRootContainer = false;
if(W.map.updateRequestLayer === null)
{
if(uroNullURLayer === false)
{
uroAddLog('caught null UR layer');
uroNullURLayer = true;
}
return;
}
uroNullURLayer = false;
if(W.map.problemLayer === null)
{
if(uroNullProblemLayer === false)
{
uroAddLog('caught null problem layer');
uroNullProblemLayer = true;
}
return;
}
uroNullProblemLayer = false;
if(uroGetCBChecked('_cbMasterEnable') === false)
{
return;
}
var mouseX;
var mouseY;
if(e == 'dwellTimeout')
{
mouseX = uroPrevMouseX;
mouseY = uroPrevMouseY;
}
else
{
mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top;
var maxJitter = uroGetElmValue('_inputMaxJitter');
if((Math.abs(uroPrevMouseX - mouseX) > maxJitter) || (Math.abs(uroPrevMouseY - mouseY) > maxJitter))
{
uroPopupDwellTimer = uroGetElmValue('_inputPopupDwellTimeout');
}
uroPrevMouseX = mouseX;
uroPrevMouseY = mouseY;
}
var result = '';
var rw;
var rh;
var popupXOffset = uroParsePxString(window.getComputedStyle(document.getElementById('sidebar')).getPropertyValue("width"));
var popupYOffset = $(document.getElementById("WazeMap")).offset().top - 80;
var uroPopupX = mouseX + popupXOffset + 10;
var uroPopupY = mouseY + popupYOffset + 10;
var objHasIgnoreLink = false;
var objHasDeleteLink = false;
var objHasAddWatchLink = false;
var objHasRemoveWatchLink = false;
var objHasUpdateWatchLink = false;
var objHasRecentreSessionLink = false;
var objHasOpenInNewTabLink = false;
var isVenue = false;
var newPopupType = null;
var markerObj;
var markerPos;
var markerImg;
var ureq;
var idx;
var hovered;
// popup for segment restrictions
if(uroGetCBChecked('_cbInhibitSegPopup') === false)
{
for(var slIdx=0; slIdx < W.map.segmentLayer.features.length; slIdx++)
{
if(W.map.segmentLayer.features[slIdx].renderIntent == 'highlight')
{
var doPopUp = false;
var segObj;
var restObj;
if(W.map.segmentLayer.features[slIdx].fid === null) segObj = W.map.segmentLayer.features[slIdx].model;
else segObj = W.map.segmentLayer.features[slIdx];
result += '<table cellpadding=4 border=1">';
if(segObj.attributes.fwdRestrictions.length > 0)
{
doPopUp = true;
result += '<tr><td colspan=11><b>A-B restrictions:</b></td></tr>';
for(idx = 0; idx < segObj.attributes.fwdRestrictions.length; idx++)
{
restObj = segObj.attributes.fwdRestrictions[idx];
result += uroFormatRestriction(restObj);
}
}
if (segObj.attributes.revRestrictions.length > 0)
{
doPopUp = true;
result += '<tr><td colspan=11><b>B-A restrictions:</b></td></tr>';
for(idx = 0; idx < segObj.attributes.revRestrictions.length; idx++)
{
restObj = segObj.attributes.revRestrictions[idx];
result += uroFormatRestriction(restObj);
}
}
result += '</table>';
if(W.map.closuresMarkerLayer.getVisibility() === true)
{
result += '<table cellpadding=4 border=1" width="100%">';
if(segObj.attributes.hasClosures === true)
{
var hasFwd = false;
var hasRev = false;
var rcObj;
var roadClosure;
for(roadClosure in W.model.roadClosures.objects)
{
if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
{
rcObj = W.model.roadClosures.objects[roadClosure];
if(rcObj.segID == segObj.attributes.id)
{
if(rcObj.forward === true)
{
if(hasFwd === false)
{
result += '<tr><td colspan=3><b>A-B closures:</b></td></tr>';
hasFwd = true;
}
if(rcObj.active === true)
{
result += '<tr>';
}
else
{
result += '<tr bgcolor="#C0C0C0">';
}
var startDate = rcObj.startDate;
var endDate = "unknown";
if(rcObj.endDate !== null)
{
endDate = rcObj.endDate;
}
var provider = "---";
if(rcObj.provider !== null)
{
provider = rcObj.provider;
}
var reason = "---";
if(rcObj.reason !== null)
{
reason = rcObj.reason;
}
result += '<td>' + startDate + ' to ' + endDate + '</td>';
result += '<td>' + provider + '</td>';
result += '<td>' + reason + '</td>';
result += '</td></tr>';
}
else
{
hasRev = true;
}
}
}
}
if(hasRev === true)
{
result += '<tr><td colspan=3><b>B-A closures:</b></td></tr>';
for(roadClosure in W.model.roadClosures.objects)
{
if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
{
rcObj = W.model.roadClosures.objects[roadClosure];
if(rcObj.segID == segObj.attributes.id)
{
if(rcObj.forward === false)
{
if(rcObj.active === true)
{
result += '<tr>';
}
else
{
result += '<tr bgcolor="#C0C0C0">';
}
result += '<td>' + rcObj.startDate + ' to ' + rcObj.endDate + '</td>';
result += '<td>' + rcObj.provider + '</td>';
result += '<td>' + rcObj.reason + '</td>';
result += '</td></tr>';
}
}
}
}
}
if((hasFwd === true) || (hasRev === true))
{
doPopUp = true;
}
}
result += '</table>';
}
if(doPopUp === true)
{
if(segObj.attributes.id === null) uroFID = segObj.id;
else uroFID = segObj.attributes.id;
newPopupType = 'segment_restriction';
}
break;
}
}
}
// popup for restricted turns
if(newPopupType === null)
{
if(uroGetCBChecked('_cbInhibitTurnsPopup') === false)
{
var turnMarkerCount = W.map.layers[uroTurnsLayerIdx].markers.length;
if(turnMarkerCount > 0)
{
for(idx=0; idx<turnMarkerCount; idx++)
{
markerObj = W.map.layers[uroTurnsLayerIdx].markers[idx];
var arrowElm = markerObj.icon.imageDiv.childNodes[0];
markerImg = window.getComputedStyle(arrowElm).getPropertyValue("background-image");
markerPos = window.getComputedStyle(arrowElm).getPropertyValue("background-position");
markerPos = markerPos.split(' ');
markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
hovered = false;
if(markerImg.indexOf('turns-sa7bd56a5e6.png') != -1)
{
if(markerPos == -222)
{
hovered = true;
}
}
if(hovered === true)
{
uroAddLog('hover over restricted turn marker');
uroTBRObj = arrowElm.childNodes[0];
var trObj = ($(arrowElm).data('model'));
var resObj = null;
if(trObj.fromSeg.attributes.fromRestrictions !== undefined)
{
resObj = trObj.fromSeg.attributes.fromRestrictions[trObj.toSeg.attributes.id];
}
if(resObj === null)
{
if(trObj.fromSeg.attributes.toRestrictions !== undefined)
{
resObj = trObj.fromSeg.attributes.toRestrictions[trObj.toSeg.attributes.id];
}
}
result += '<label id="_editTBR">Click to edit</label><br>';
result += '<table cellpadding=4 border=1">';
for(var resIdx=0; resIdx < resObj.length; resIdx++)
{
result += uroFormatRestriction(resObj[resIdx]);
}
result += '</table>';
uroFID = markerObj.icon.imageDiv._eventCacheID;
newPopupType = 'turn_restriction';
break;
}
}
}
}
}
var targetTab = '';
// popup for landmarks
if((newPopupType === null) && (uroGetCBChecked('_cbInhibitLandmarkPopup') === false))
{
uroPlaceSelected = false;
for(var llFeatureIdx=0; llFeatureIdx < W.map.landmarkLayer.features.length; llFeatureIdx++)
{
var renderIntent = W.map.landmarkLayer.features[llFeatureIdx].renderIntent;
if(renderIntent == 'highlight')
{
var venueObj;
if(W.map.landmarkLayer.features[llFeatureIdx].fid === null) venueObj = W.map.landmarkLayer.features[llFeatureIdx].model;
else venueObj = W.map.landmarkLayer.features[llFeatureIdx];
if(venueObj.attributes.id === null) uroFID = venueObj.id;
else uroFID = venueObj.attributes.id;
var navpointPos=new OpenLayers.LonLat();
navpointPos = uroGetVenueNavPoint(uroFID);
navpointPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
result += '<b>';
if(venueObj.attributes.name === '') result += 'Unnamed landmark';
else result += venueObj.attributes.name;
result += '</b><br>';
result += '<ul>';
for(idx = 0; idx < venueObj.attributes.categories.length; idx++)
{
result += '<li>' + I18n.lookup("venues.categories")[venueObj.attributes.categories[idx]];
}
result += '</ul>';
if(venueObj.attributes.residential === true)
{
result += '<i>Residential</i>';
}
var npLink = document.location.href;
var npLayers = '&layers='+W.map.mapState.getLayerVisibilityBitmask();
npLink = npLink.substr(0,npLink.indexOf('?zoom'));
npLink += '?zoom=5&lat='+navpointPos.lat+'&lon='+navpointPos.lon+npLayers;
targetTab = "_uroTab_" + Math.round(Math.random()*1000000);
result += '<hr>Jump to nav point: <a href="'+npLink+'" id="_openInNewTab" target="'+targetTab+'">in new tab</a> - ';
objHasOpenInNewTabLink = true;
result += '<a href="#" id="_recentreSession">in this tab</a>';
objHasRecentreSessionLink = true;
newPopupType = 'venue';
isVenue = true;
break;
}
else if((renderIntent == 'select') || (renderIntent == 'highlightselected'))
{
uroPlaceSelected = true;
}
}
}
var unstackedX;
var unstackedY;
var ureqID = null;
var isUR = false;
var isProblem = false;
var isTurnProb = false;
var isPlaceUpdate = false;
// look for URs, place updates and problems
if(newPopupType === null)
{
var customIdx;
var idSrc = null;
if(uroGetCBChecked('_cbInhibitURPopup') === false)
{
hovered = false;
for(var markerURL in W.map.updateRequestLayer.markers)
{
if(W.map.updateRequestLayer.markers.hasOwnProperty(markerURL))
{
markerObj = W.map.updateRequestLayer.markers[markerURL];
markerImg = markerObj.icon.$div.css('background-image');
markerPos = markerObj.icon.$div.css('background-position');
markerPos = markerPos.split(' ');
markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
var urIDSrc = markerObj.id;
if(markerImg.indexOf('problems-s8f369ca968.png') != -1)
{
if((markerPos == -403) || (markerPos == -483) || (markerPos == -563) || (markerPos == -643))
{
hovered = true;
uroAddLog('UR image type 1');
}
}
else if(markerImg.indexOf('problems-se224ab677e.png') != -1)
{
if((markerPos == -40) || (markerPos == -160) || (markerPos == -200) || (markerPos == -240))
{
hovered = true;
uroAddLog('UR image type 2');
}
}
else
{
if(markerPos > -200)
{
hovered = true;
uroAddLog('UR image type 3');
}
}
if(hovered === true)
{
isUR = true;
newPopupType = 'ur';
uroAddLog('hover over UR ID '+urIDSrc);
unstackedX = uroParsePxString(W.map.updateRequestLayer.markers[urIDSrc].icon.imageDiv.style.left);
unstackedY = uroParsePxString(W.map.updateRequestLayer.markers[urIDSrc].icon.imageDiv.style.top);
// override popup base position
uroPopupX = unstackedX + popupXOffset + 6;
uroPopupY = unstackedY + popupYOffset + 46;
uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
// check for stacking...
if(uroShownFID != idSrc)
{
uroCheckStacking(1,urIDSrc, unstackedX, unstackedY);
}
if(urIDSrc != uroCustomMarkerFID)
{
if(uroCustomMarkerFID !== null)
{
customIdx = uroGetCustomMarkerIdx(uroGetCustomType(uroCustomMarkerFID, uroCustomMarkerType));
uroChangeCustomMarkers(uroCustomMarkerFID,false,customIdx,uroCustomMarkerType);
uroCustomMarkerFID = null;
}
customIdx = uroGetCustomMarkerIdx(uroGetCustomType(urIDSrc, "ur"));
if(customIdx != -1)
{
uroCustomMarkerFID = urIDSrc;
uroCustomMarkerType = "ur";
uroChangeCustomMarkers(urIDSrc,hovered,customIdx,uroCustomMarkerType);
}
}
idSrc = urIDSrc;
break;
}
}
}
if((hovered === false) && (uroStackType == 1))
{
uroRestackMarkers();
uroFilterURs();
}
if((idSrc === null) && (uroCustomMarkerFID !== null))
{
customIdx = uroGetCustomMarkerIdx(uroGetCustomType(uroCustomMarkerFID, uroCustomMarkerType));
uroChangeCustomMarkers(uroCustomMarkerFID,false, customIdx,uroCustomMarkerType);
uroCustomMarkerFID = null;
uroCustomMarkerType = null;
}
}
if((newPopupType === null) && (uroGetCBChecked('_cbInhibitPUPopup') === false))
{
hovered = false;
for(var markerPUL in W.map.placeUpdatesLayer.markers)
{
if(W.map.placeUpdatesLayer.markers.hasOwnProperty(markerPUL))
{
markerObj = W.map.placeUpdatesLayer.markers[markerPUL];
markerImg = markerObj.icon.$div.css('background-image');
markerPos = markerObj.icon.$div.css('background-position');
markerPos = markerPos.split(' ');
markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
if(markerImg.indexOf('placeUpdates-sb30471988c.png') != -1)
{
// absolute offsets: 0 = new place, -120 = flagged, -240 = new photo, -360 = updated details
// relative offsets: 0 = green, -40 = highlighted, -80 = default
if(((markerPos + 40) % 120) === 0)
{
hovered = true;
uroAddLog('PUR marker type 1');
}
}
else if(markerImg.indexOf('placeUpdates-s2e8d9c5ce4.png') != -1)
{
if
(
(markerObj.icon.$div.css("filter") == "brightness(110%)") ||
(markerObj.icon.$div.css("webkit-filter") == "brightness(1.1)")
)
{
hovered = true;
uroAddLog('PUR marker type 2');
}
}
if(hovered === true)
{
idSrc = markerObj.id;
unstackedX = uroParsePxString(W.map.placeUpdatesLayer.markers[idSrc].icon.imageDiv.style.left);
unstackedY = uroParsePxString(W.map.placeUpdatesLayer.markers[idSrc].icon.imageDiv.style.top);
// override popup base position
uroPopupX = unstackedX + popupXOffset + 6;
uroPopupY = unstackedY + popupYOffset + 46;
uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
if(uroShownFID != idSrc)
{
// check for stacking...
uroCheckStacking(3,idSrc, unstackedX, unstackedY);
}
isPlaceUpdate = true;
newPopupType = 'pur';
uroAddLog('hover over placeUpdate ID '+idSrc);
break;
}
}
}
if((hovered === false) && (uroStackType == 3))
{
uroRestackMarkers();
uroFilterPlaces();
}
}
if((newPopupType === null) && (uroGetCBChecked('_cbInhibitMPPopup') === false))
{
hovered = false;
for(var markerPL in W.map.problemLayer.markers)
{
if(W.map.problemLayer.markers.hasOwnProperty(markerPL))
{
markerObj = W.map.problemLayer.markers[markerPL];
markerImg = markerObj.icon.$div.css('background-image');
markerPos = markerObj.icon.$div.css('background-position');
markerPos = markerPos.split(' ');
markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
if(markerImg.indexOf('problems-s8f369ca968.png') != -1)
{
if((markerPos == -65) || (markerPos == -145) || (markerPos == -225) || (markerPos == -305))
{
hovered = true;
uroAddLog('Problem image type 1');
}
}
else if(markerImg.indexOf('problems-se224ab677e.png') != -1)
{
if((markerPos == -320) || (markerPos == -560) || (markerPos == -520) || (markerPos == -440))
{
hovered = true;
uroAddLog('Problem image type 2');
}
}
else
{
if(markerPos > -200)
{
hovered = true;
uroAddLog('Problem image type 3');
}
}
if(hovered === true)
{
idSrc = null;
if(markerObj.model.fid === null) idSrc = markerObj.id;
else idSrc = markerObj.model.fid;
unstackedX = uroParsePxString(W.map.problemLayer.markers[idSrc].icon.imageDiv.style.left);
unstackedY = uroParsePxString(W.map.problemLayer.markers[idSrc].icon.imageDiv.style.top);
// override popup base position
uroPopupX = unstackedX + popupXOffset + 6;
uroPopupY = unstackedY + popupYOffset + 46;
uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
// check for stacking...
if(uroShownFID != idSrc)
{
uroCheckStacking(2,idSrc, unstackedX, unstackedY);
}
if(idSrc != uroCustomMarkerFID)
{
if(uroCustomMarkerFID !== null)
{
customIdx = uroGetCustomMarkerIdx(uroGetCustomType(uroCustomMarkerFID, uroCustomMarkerType));
uroChangeCustomMarkers(uroCustomMarkerFID,false,customIdx,uroCustomMarkerType);
uroCustomMarkerFID = null;
uroCustomMarkerType = null;
}
customIdx = uroGetCustomMarkerIdx(uroGetCustomType(idSrc, "mp"));
if(customIdx != -1)
{
uroCustomMarkerFID = idSrc;
uroCustomMarkerType = "mp";
uroChangeCustomMarkers(idSrc,hovered,customIdx,uroCustomMarkerType);
}
}
isProblem = true;
newPopupType = 'map_problem';
uroAddLog('hover over problem ID '+idSrc);
break;
}
}
}
if((hovered === false) && (uroStackType == 2))
{
uroRestackMarkers();
uroFilterProblems();
}
if((idSrc === null) && (uroCustomMarkerFID !== null))
{
var newCustomIdx = uroGetCustomMarkerIdx(uroGetCustomType(uroCustomMarkerFID, uroCustomMarkerType));
uroChangeCustomMarkers(uroCustomMarkerFID, false, newCustomIdx, uroCustomMarkerType);
uroCustomMarkerFID = null;
uroCustomMarkerType = null;
}
}
if (idSrc !== null)
{
ureq = null;
if(isUR) ureq = W.model.mapUpdateRequests.objects[idSrc];
else if(isProblem)
{
ureq = W.model.problems.objects[idSrc];
if(ureq === undefined)
{
if(uroDOMHasTurnProblems)
{
ureq = W.model.turnProblems.objects[idSrc];
if(ureq !== undefined) isTurnProb = true;
}
}
}
else if(isPlaceUpdate) ureq = W.map.placeUpdatesLayer.markers[idSrc].model;
if(ureq.fid !== null) ureqID = ureq.fid;
else if(ureq.id !== null) ureqID = ureq.id;
else if(ureq.attributes.id !== null) ureqID = ureq.attributes.id;
uroFID = ureqID;
}
else
{
uroFID = -1;
}
//if((uroFID != uroShownFID) && (uroFID != -1))
if(uroFID != -1)
{
var uroDaysResolved;
if(isUR)
{
uroAddLog('building popup for UR '+idSrc);
result = '<b>Update Request ('+idSrc+'): ' + I18n.lookup("update_requests.types")[ureq.attributes.type] + '</b><br>';
if(ureq.attributes.description !== null)
{
var desc = ureq.attributes.description.replace(/<\/?[^>]+(>|$)/g, "");
if(desc != "null")
{
desc = uroClickify(desc);
result += desc + '<br>';
}
}
var uroDaysOld = uroGetURAge(ureq,0,false);
var uroSubmittedTS = uroGetURAge(ureq,0,true);
if(uroSubmittedTS != -1)
{
uroSubmittedTS = uroGetDateTimeString(uroSubmittedTS);
}
if(uroDaysOld != -1)
{
result += '<i>Submitted ' + uroParseDaysAgo(uroDaysOld) + ' ';
if(uroSubmittedTS != -1) result += '(' + uroSubmittedTS + ') ';
if(ureq.attributes.guestUserName !== undefined)
{
result += 'via Livemap';
if(ureq.attributes.guestUserName !== '')
{
result += ' by '+ureq.attributes.guestUserName.replace(/<\/?[^>]+(>|$)/g, "");
}
}
result += '</i>';
}
if(ureq.attributes.resolvedOn !== null)
{
uroDaysResolved = uroGetURAge(ureq,1,false);
var uroResolvedTS = uroGetURAge(ureq,1,true);
if(uroResolvedTS != -1)
{
uroResolvedTS = uroGetDateTimeString(uroResolvedTS);
}
if(uroDaysResolved != -1)
{
result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' ';
if(uroResolvedTS != -1) result += '(' + uroResolvedTS + ')</i>';
result += '<br><i>Marked as ';
if(ureq.attributes.resolution === 0) result += 'solved';
else if(ureq.attributes.resolution == 1) result += 'not identified';
else result += 'unknown';
if(ureq.attributes.resolvedBy !== null)
{
result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy);
}
result += '</i>';
}
}
if(W.model.updateRequestSessions.objects[ureqID] !== undefined)
{
var hasMyComments = uroURHasMyComments(ureqID);
var nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
result += '<br>' + nComments + ' comment';
if(nComments != 1) result += 's';
if((hasMyComments === false) && (nComments > 0)) result += ' (none by me)';
if(nComments > 0)
{
var commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1]);
if(commentDaysOld != -1)
{
result += ', last update '+uroParseDaysAgo(commentDaysOld);
}
}
}
}
else if(isProblem)
{
uroAddLog('building popup for problem '+idSrc);
if(isTurnProb) result = '<b>Turn Problem ('+idSrc+'): ' + I18n.lookup("problems.types").turn.title;
else
{
result = '<b>Map Problem ('+idSrc+'): ';
var problemType = null;
if(uroDOMHasTurnProblems)
{
problemType = ureq.attributes.problemType;
}
else
{
problemType = ureq.attributes.subType;
}
if(problemType == 300)
{
result += I18n.lookup("problems.panel.closure.title");
}
else
{
if(I18n.lookup("problems.types")[problemType] === undefined) result += 'Unknown problem type ('+problemType+')';
else result += I18n.lookup("problems.types")[problemType].title;
}
}
result += '</b><br>';
if(ureq.attributes.description !== null)
{
result += 'Description: ' + ureq.attributes.description + '<br>';
}
if(ureq.attributes.extraInfo !== null)
{
result += 'ExtraInfo: ' + ureq.attributes.extraInfo + '<br>';
}
if(ureq.attributes.provider !== null)
{
result += 'Provider: ' + ureq.attributes.provider + '<br>';
}
if(ureq.attributes.resolvedOn !== null)
{
uroDaysResolved = uroGetURAge(ureq,1,false);
if(uroDaysResolved != -1)
{
result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' ';
if(ureq.attributes.resolvedBy !== null)
{
result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy);
}
if((ureq.attributes.open === true) && (ureq.attributes.resolvedOn !== null))
{
result += '<br>Reopened by Waze';
}
result += '</i>';
}
}
}
else if(isPlaceUpdate)
{
uroAddLog('building popup for placeUpdate '+idSrc);
result = '<b>';
if(ureq.attributes.name === '') result += 'Unnamed landmark';
else result += ureq.attributes.name;
result += '</b><br>';
result += '<ul>';
for(idx = 0; idx < ureq.attributes.categories.length; idx++)
{
result += '<li>' + I18n.lookup("venues.categories")[ureq.attributes.categories[idx]];
}
result += '</ul>';
if(ureq.attributes.residential === true)
{
result += '<i>Residential</i>';
}
var daysOld = uroGetPURAge(ureq);
if(daysOld != -1)
{
result += '<br><i>Submitted '+uroParseDaysAgo(daysOld)+'</i>';
}
}
// add "open new WME tab" link
var urPos=new OpenLayers.LonLat();
if(isPlaceUpdate)
{
urPos=ureq.geometry.bounds.centerLonLat.clone();
}
else
{
urPos.lon=ureq.geometry.x;
urPos.lat=ureq.geometry.y;
}
urPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
var urLink = document.location.href;
var urLayers = '&layers='+W.map.mapState.getLayerVisibilityBitmask();
urLink = urLink.substr(0,urLink.indexOf('?zoom'));
urLink += '?zoom=5&lat='+urPos.lat+'&lon='+urPos.lon+urLayers;
if(isUR) urLink += '&mapUpdateRequest='+idSrc;
else if(isTurnProb) urLink += '&showturn='+idSrc+'&endshow';
else if(isProblem) urLink += '&mapProblem='+idSrc;
else if(isPlaceUpdate) urLink += '&showpur='+idSrc+'&endshow';
targetTab = "_uroTab_" + Math.round(Math.random()*1000000);
result += '<hr><ul><li><a href="'+urLink+'" id="_openInNewTab" target="'+targetTab+'">Open in new tab</a> - ';
objHasOpenInNewTabLink = true;
result += '<a href="#" id="_recentreSession">centre in current tab</a>';
objHasRecentreSessionLink = true;
// add "open new livemap tab" link
var lmLink = null;
if(document.getElementsByClassName("waze-header-menu").length === 0)
{
uroAddLog('Livemap link in livemap element');
lmLink = document.getElementById('livemap').href;
}
else
{
uroAddLog('Livemap link in header menu, locating...');
var menuItems = document.getElementsByClassName("waze-header-menu")[0];
for(var miloop = 0; miloop<menuItems.childElementCount; miloop++)
{
if(menuItems.children[miloop].innerHTML.indexOf('livemap') != -1)
{
uroAddLog('found link in menu entry '+miloop);
lmLink = menuItems.children[miloop].getElementsByTagName('a')[0].href;
uroAddLog(lmLink);
}
}
}
if(lmLink !== null)
{
var zpos = lmLink.indexOf('?');
if(zpos > -1) lmLink = lmLink.substr(0,zpos);
lmLink += '?zoom=17&lat='+urPos.lat+'&lon='+urPos.lon+'&layers=BTTTT';
result += '<li><a href="'+lmLink+'" target="_lmTab">Open in new livemap tab</a>';
}
if(!isPlaceUpdate)
{
// add "ignore for this session" link
result += '<li><a href="#" id="_addtoignore">Hide for this session</a></ul>';
objHasIgnoreLink = true;
}
}
}
// look for cameras
if((newPopupType === null) && (uroGetCBChecked('_cbInhibitCamPopup') === false))
{
for(var clFeatureIdx = 0; clFeatureIdx < W.map.camerasLayer.features.length; clFeatureIdx++)
{
if(W.map.camerasLayer.features[clFeatureIdx].renderIntent == 'highlight')
{
if(W.map.camerasLayer.features[clFeatureIdx].fid === null) ureq = W.map.camerasLayer.features[clFeatureIdx].model;
else ureq = W.map.camerasLayer.features[clFeatureIdx];
if(ureq.fid === null) ureqID = ureq.attributes.id;
else ureqID = ureq.fid;
// test isSelected() so that we only do overview data on cameras that are being hovered over
if(ureq.isSelected() === false)
{
newPopupType = 'camera';
uroFID = ureqID;
uroAddLog('generating popup for camera '+uroFID);
if(I18n.lookup("edit.camera.fields.type") === undefined)
{
result += '<b>Camera: ' + ureq.TYPES[ureq.attributes.type] + '</b><br>';
}
else
{
result += '<b>Camera: ' + I18n.lookup("edit.camera.fields.type")[ureq.attributes.type] + '</b><br>';
}
result += 'ID: '+uroFID+'<br>';
result += 'Created by ';
var userID;
if(W.model.users.get(ureq.attributes.createdBy) !== undefined)
{
userID = ureq.attributes.createdBy;
result += uroGetUserNameAndRank(userID);
}
else result += 'unknown';
result += ', ';
var camAge = uroGetCameraAge(ureq,1);
if(camAge != -1)
{
result += uroParseDaysAgo(camAge);
}
else result += 'unknown days ago';
result += '<br>Updated by ';
if(W.model.users.get(ureq.attributes.updatedBy) !== undefined)
{
userID = ureq.attributes.updatedBy;
var userName = W.model.users.objects[userID].userName;
var userLevel = W.model.users.objects[userID].rank + 1;
result += userName + ' (' + userLevel + ')';
}
else result += 'unknown';
result += ', ';
camAge = uroGetCameraAge(ureq,0);
if(camAge != -1)
{
result += uroParseDaysAgo(camAge);
}
else result += 'unknown days ago';
result += '<br>Speed data: ';
result += uroGetCameraSpeedString(ureq.attributes.speed);
result += '<hr><ul>';
if(uroIsCamOnWatchList(uroFID) != -1)
{
result += '<li><a href="#" id="_updatewatchlist">Update watchlist entry</a>';
result += '<li><a href="#" id="_removefromwatchlist">Remove from watchlist</a>';
objHasUpdateWatchLink = true;
objHasRemoveWatchLink = true;
}
else
{
result += '<li><a href="#" id="_addtowatchlist">Add to watchlist</a>';
objHasAddWatchLink = true;
}
if(ureq.attributes.permissions !== 0)
{
result += '<li><a href="#" id="_deleteobject">Delete Camera</a>';
objHasDeleteLink = true;
}
result += '</ul>';
}
break;
}
}
}
if((newPopupType !== null) && (uroPopupDwellTimer === 0))
{
if((uroFID != uroShownFID) || (newPopupType != uroShownPopupType))
{
if(uroFID != uroShownFID) uroAddLog('FID mismatch, show popup: '+uroFID+'/'+uroShownFID);
else uroAddLog('Popup type mismatch: '+newPopupType+'/'+uroShownPopupType);
uroShownFID = uroFID;
uroShownPopupType = newPopupType;
uroPopupShown = false;
}
if(uroPopupShown === false)
{
uroAddLog('display popup at '+uroPopupX+','+uroPopupY);
uroPopupShown = true;
uroDiv.style.height = "auto";
uroDiv.style.width = "auto";
uroDiv.innerHTML = result;
if((uroFID != -1) && (objHasIgnoreLink === true))
{
uroAddEventListener('_addtoignore','click', uroAddToIgnoreList, true);
}
if(objHasDeleteLink === true)
{
uroAddEventListener('_deleteobject','click', uroDeleteObject, true);
}
if(objHasRemoveWatchLink === true)
{
uroAddEventListener('_removefromwatchlist','click', uroRemoveCamFromWatchList, true);
}
if(objHasAddWatchLink === true)
{
uroAddEventListener('_addtowatchlist','click', uroAddCamToWatchList, true);
}
if(objHasUpdateWatchLink === true)
{
uroAddEventListener('_updatewatchlist','click', uroUpdateCamWatchList, true);
}
if(objHasOpenInNewTabLink === true)
{
uroAddEventListener('_openInNewTab','mouseup', uroOpenNewTab, true);
}
if(objHasRecentreSessionLink === true)
{
if(isUR) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnUR, true);
else if((isProblem)||(isTurnProb)) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnMP, true);
else if(isPlaceUpdate) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnPUR, true);
else if(isVenue) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnVenueNavPoint, true);
}
if(newPopupType == 'turn_restriction')
{
uroAddEventListener('_editTBR','click', uroEditTBR, true);
}
// restrict the popup width to be no wider than half the window width to avoid it completely
// overlapping the marker it's associated with - by keeping it to half the window width we
// guarantee that it'll fit either to the left or the right of the marker no matter how far
// across the screen the marker is located...
rw = parseInt(uroDiv.clientWidth);
if(rw > (window.innerWidth / 2))
{
rw = (window.innerWidth / 2);
uroDiv.style.width = rw+'px';
}
// get the div height after any adjustment of the width above, to account for whatever content
// reflow may have occurred as a result of reducing the width...
rh = parseInt(uroDiv.clientHeight);
if((uroPopupX + rw) > window.innerWidth)
{
// where the popup would be off the right hand side of the screen, move it completely over to the
// other side of the mouse pointer
uroPopupX -= (rw + 20);
if(uroPopupX < 0) uroPopupX = 0;
}
if((uroPopupY + rh) > window.innerHeight)
{
// where the popup would be off the bottom of the screen, shift it up just far enough to be
// fully visible
uroPopupY -= (((uroPopupY + rh) - window.innerHeight) + 30);
if(uroPopupY < 0) uroPopupY = 0;
}
uroDiv.style.top = uroPopupY+'px';
uroDiv.style.left = uroPopupX+'px';
uroDiv.style.visibility = 'visible';
}
uroPopupTimer = -1;
}
else if((newPopupType === null) && (uroPopupDwellTimer !== 0) && (uroPopupShown === true))
{
uroHidePopup();
}
else
{
if((uroPopupTimer == -1) && (uroFID != uroShownFID))
{
uroPopupTimer = uroGetElmValue('_inputPopupEntryTimeout');
}
}
}
function uroRestyleWMETabs()
{
// The nav-tabs class is now also used for the General/Closures tabs on the segment edit panel, so we have
// to restrict the scope of this code to just those nav-tab classed elements within the user-tabs element.
var navTabs = document.getElementById('user-tabs').getElementsByClassName("nav-tabs")[0].children;
for(var loop = 0; loop<navTabs.length; loop++)
{
navTabs[loop].children[0].style.padding = "4px";
}
var panelDisplay = '';
if(uroGetCBChecked('_cbHideEditorInfo'))
{
panelDisplay = "none";
}
document.getElementById("user-details").style.display = panelDisplay;
setTimeout(uroRestyleWMETabs,1000);
}
function uroExclusiveCB()
{
var cbChecked = uroGetCBChecked(this.id);
if(cbChecked === true)
{
var pairedList = this.attributes.pairedWith.value.split(',');
for(var i=0; i<pairedList.length; i++)
{
uroSetCBChecked(pairedList[i], false);
}
}
}
function uroGetAMs(e)
{
var amList = '';
if(W.map.managedAreasLayer.getVisibility() === true)
{
var mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
var mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top - document.getElementById('toolbar').clientHeight;
var mousePixel = new OL.Pixel(mouseX, mouseY);
var mousePoint = W.map.getLonLatFromPixel(mousePixel).toPoint();
for(var amObj in W.model.managedAreas.objects)
{
if(W.model.managedAreas.objects[amObj].geometry.containsPoint(mousePoint))
{
if(amList !== '') amList += ', ';
amList += uroGetUserNameAndRank(W.model.managedAreas.objects[amObj].userID);
}
}
if(amList === '')
{
amList = 'none';
}
amList = "<b>Area Managers:</b> "+amList;
}
document.getElementById("uroAMList").innerHTML = amList;
}
function uroMouseDown()
{
uroMouseIsDown = true;
}
function uroMouseUp()
{
uroMouseIsDown = false;
}
function uroUREvent_onObjectsChanged()
{
}
function uroUREvent_onObjectsAdded()
{
if(uroGetCBChecked('_cbURResolverIDFilter') === true)
{
uroUpdateResolverList();
}
uroFilterURs();
}
function uroUREvent_onObjectsRemoved()
{
}
function uroGetSelectedURCommentCount()
{
if(W.model.updateRequestSessions.objects[uroSelectedURID] !== undefined)
{
var cachedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
uroAddLog(uroSelectedURID+':'+cachedCommentCount+' '+uroExpectedCommentCount);
// if there aren't the same number of cached comments as there are comments in the UR dialog list, initiate
// a refresh of the comment data...
if(cachedCommentCount != uroExpectedCommentCount)
{
if(uroPendingCommentDataRefresh === true)
{
if(cachedCommentCount > 0)
{
uroCachedLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].comments[cachedCommentCount-1].id;
}
else
{
uroCachedLastCommentID = null;
}
uroAddLog('updateRequestSessions refresh required for UR '+uroSelectedURID);
if(uroCachedLastCommentID !== null)
{
uroAddLog('last comment ID for this UR is '+uroCachedLastCommentID);
}
else
{
uroAddLog('first comment for this UR, no previous comment to ID');
}
var idList = [];
idList.push(uroSelectedURID);
// need to delete the existing cache object first, as .get() is only capable of creating new objects,
// it doesn't seem able to update an existing object with new data
W.model.updateRequestSessions.remove(W.model.updateRequestSessions.objects[uroSelectedURID]);
W.model.updateRequestSessions.get(idList);
// the call to .get() initiates a XMLHttpRequest for the data, so we now need to switch modes - the
// refresh process has started so we're no longer pending, but we are now waiting for the XMLHttpRequest
// to return something...
uroPendingCommentDataRefresh = false;
uroWaitingCommentDataRefresh = true;
}
else
{
if(cachedCommentCount > 0)
{
var currentLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].comments[cachedCommentCount-1].id;
if(currentLastCommentID == uroCachedLastCommentID)
{
// most recent comment loaded for this UR is the same one that was present at the start of this
// refresh process, so kick back into pending mode so we can retry the .get()...
uroAddLog('latest comment ID still the same, reverting to pending mode...');
uroPendingCommentDataRefresh = true;
}
else
{
// something may have gone awry here - the most recent comment loaded for this UR doesn't have the
// same ID as the one present at the start of the refresh process, yet the comment counts still don't
// match up, which suggests either a comment got lost along the way or someone else has commented on
// the same UR at almost the same time. To get out of the loop this would create, assume that a
// mismatch in the IDs means the .get() has completed successfully no matter what the new comment
// count is, and take this new count to be the count we were expecting all along...
uroAddLog('latest comment ID different, but expected count not correct...');
uroExpectedCommentCount = cachedCommentCount;
}
}
else
{
uroAddLog('first comment on this UR not received yet, reverting to pending mode...');
uroPendingCommentDataRefresh = true;
}
}
}
else
{
// if the WME session is loaded with a UR already selected, such that WME has opened the UR dialog as part
// of the session startup process, adding new comments to the UR cause the cached data to be updated immediately.
// This prevents URO+ from switching into waiting mode in the above block of code, so we have to instead do
// it here by comparing the cached count against the expected count following the Send click event.
if(cachedCommentCount >= uroExpectedCommentCount)
{
uroPendingCommentDataRefresh = false;
uroWaitingCommentDataRefresh = true;
uroExpectedCommentCount = null;
}
// once the cached data has been updated, refilter the URs so that the new comment count is taken into account
// immediately for filtering and display purposes
if(uroWaitingCommentDataRefresh === true)
{
uroWaitingCommentDataRefresh = false;
uroFilterURs();
uroAddLog('refresh complete');
}
}
}
}
function uroAddedComment()
{
// when the user clicks the Send button to submit a new UR comment, this event handler fires before the new comment is
// posted to the server and thus also before the comment list gets updated in the UR dialog. So we take the current
// comment count and, if the new comment edit box isn't empty, increment it by 1 to get the expected count. Then we
// set the pending flag true to initiate a session refresh on the next 100ms tick
uroExpectedCommentCount = W.map.panelRegion.currentView.conversationView.conversation.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();
document.getElementsByClassName('close-panel')[0].click();
}
function uroTenthSecondTick()
{
if(uroSetupListeners)
{
if((W.loginManager.isLoggedIn()) || (uroPracticeMode === true))
{
uroSetupListeners = false;
if(uroPracticeMode === true)
{
document.getElementsByClassName("sandbox")[0].style.display = "none";
}
// 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.events.register("objectschanged", null, uroFilterURs_onObjectsChanged);
W.model.mapUpdateRequests.events.register("objectsadded", null, uroFilterURs_onObjectsAdded);
W.model.mapUpdateRequests.events.register("objectsremoved", null, uroFilterURs_onObjectsRemoved);
W.model.updateRequestSessions.events.register("objectschanged", null, uroUREvent_onObjectsChanged);
W.model.updateRequestSessions.events.register("objectsadded", null, uroUREvent_onObjectsAdded);
W.model.updateRequestSessions.events.register("objectsremoved", null, uroUREvent_onObjectsRemoved);
W.model.cameras.events.register("objectschanged", null, uroFilterCameras);
W.model.cameras.events.register("objectsadded", null, uroFilterCameras);
W.model.cameras.events.register("objectsremoved", null, uroFilterCameras);
W.model.problems.events.register("objectschanged", null, uroFilterProblems);
W.model.problems.events.register("objectsadded", null, uroFilterProblems);
W.model.problems.events.register("objectsremoved", null, uroFilterProblems);
W.model.venues.events.register("objectschanged", null, uroFilterPlaces);
W.model.venues.events.register("objectsadded", null, uroFilterPlaces);
W.model.venues.events.register("objectsremoved", null, uroFilterPlaces);
var userTabs = document.getElementById(uroUserTabId);
var tabContent = null;
if(uroPracticeMode === false)
{
var navTabs = userTabs.getElementsByClassName('nav-tabs')[0]; ////uroTimbonesGetElementsByClassName('nav-tabs', userTabs)[0];
tabContent = document.getElementById('user-info').getElementsByClassName('tab-content')[0]; ////uroTimbonesGetElementsByClassName('tab-content', userTabs)[0];
var newtabUR = document.createElement('li');
newtabUR.innerHTML = '<a href="#sidepanel-uroverview" data-toggle="tab">URO+</a>';
navTabs.appendChild(newtabUR);
}
uroControls.id = "sidepanel-uroverview";
uroControls.className = "tab-pane";
if(uroPracticeMode === false)
{
tabContent.appendChild(uroControls);
}
else
{
userTabs.appendChild(uroControls);
}
uroAddEventListener('_btnUndoLastHide',"click", uroRemoveLastAddedIgnore, true);
uroAddEventListener('_btnClearSessionHides',"click", uroRemoveAllIgnores, true);
uroEnableIgnoreListControls();
uroAddEventListener('_btnClearCamWatchList',"click", uroClearCamWatchList, true);
uroAddEventListener('_btnSettingsToText',"click", uroSettingsToText, true);
uroAddEventListener('_btnTextToSettings',"click", uroTextToSettings, true);
uroAddEventListener('_btnResetSettings',"click", uroDefaultSettings, true);
uroAddEventListener('_btnClearSettingsText',"click", uroClearSettingsText, true);
uroAddEventListener('_cbMasterEnable',"click", uroFilterItems_MasterEnableClick, true);
uroSetOnClick("_linkSelectUserRequests",uroShowURTab);
uroSetOnClick("_linkSelectMapProblems",uroShowMPTab);
uroSetOnClick("_linkSelectPlaces",uroShowPlacesTab);
uroSetOnClick("_linkSelectCameras",uroShowCameraTab);
uroSetOnClick("_linkSelectMisc",uroShowMiscTab);
uroSetOnClick("_linkSelectOWL",uroShowOWLTab);
for(var idx=0;idx<W.Config.venues.categories.length;idx++)
{
uroSetOnClick('_uroPlacesGroupState-'+idx,uroPlacesGroupCollapseExpand);
}
// add exclusiveCB click handlers to all checkboxes with a pairedWith attribute
var cbList = document.getElementsByTagName('input');
for (var optIdx=0;optIdx<cbList.length;optIdx++)
{
if((cbList[optIdx].id.indexOf('_cb') === 0) && (cbList[optIdx].attributes.pairedWith !== undefined))
{
uroSetOnClick(cbList[optIdx].id,uroExclusiveCB);
}
}
var dbgMode = "none";
if(uroShowDebugOutput)
{
dbgMode = "inline";
}
document.getElementById('_uroDebugMode').style.display = dbgMode;
uroAddEventListener('_uroVersion',"click", uroToggleDebug, true);
uroAddLog('finalise onload');
uroLoadSettings();
uroNewLookCheckDetailsRequest();
if(uroGetCBChecked('_cbEnableDTE'))
{
if(dteControlsIdx != -1)
{
dteSetNewTabLength();
}
else
{
uroAddLog('ERROR - archive panel not found!');
uroSetStyleDisplay(uroUserTabId,'');
}
}
// filter markers as and when the map is moved
W.map.events.register("moveend", null, uroFilterItems);
W.map.events.register("mousemove", null, uroGetAMs);
W.map.events.register("mousemove", null, uroNewLookHighlightedItemsCheck);
W.map.events.registerPriority("mousedown", null, uroMouseDown);
// trap mousedown on Streetview marker drag
document.getElementsByClassName('street-view-control')[0].onmousedown = uroMouseDown;
W.map.events.register("mouseup", null, uroMouseUp);
uroSetStyles(uroCtrlURs);
uroSetStyles(uroCtrlMPs);
uroSetStyles(uroCtrlPlaces);
uroSetStyles(uroCtrlCameras);
uroSetStyles(uroCtrlMisc);
uroSetStyles(uroOWL);
uroShowURTab();
uroRestyleWMETabs();
uroUserID = W.loginManager.getLoggedInUser().id;
uroFilterItems();
}
}
else
{
var mousePos = document.getElementsByClassName('mouse-position')[0].innerHTML;
if(document.getElementsByClassName('panel')[0] === undefined)
{
uroHidePopupOnPanelOpen = true;
}
if(uroPopupShown === true)
{
var hidePopup = false;
if((mousePos == '00.00000, 00.00000') && (uroMouseInPopup === false))
{
hidePopup = true;
}
if(document.getElementsByClassName('panel')[0] !== undefined)
{
if(uroHidePopupOnPanelOpen === true)
{
hidePopup = true;
uroHidePopupOnPanelOpen = false;
}
}
if(hidePopup === true)
{
uroHidePopup();
}
}
if((uroAreaNameHoverObj !== null) && (uroAreaNameHoverTime != -1) && (uroAreaNameOverlayShown === false))
{
if(++uroAreaNameHoverTime > 5)
{
uroAreaNameOverlaySetup();
}
}
uroReplaceAreaNames(false);
if(uroPopupTimer > 0)
{
if(uroMouseInPopup === false)
{
uroPopupTimer--;
}
}
if(uroPopupTimer === 0)
{
uroHidePopup();
}
if(uroPopupDwellTimer > 0)
{
uroPopupDwellTimer--;
if(uroPopupDwellTimer === 0)
{
uroNewLookHighlightedItemsCheck('dwellTimeout');
}
}
if(document.getElementsByClassName("archive-panel")[0] === undefined)
{
if(dteClearHighlightsOnPanelClose)
{
dteClearListHighlight();
dteClearHighlightsOnPanelClose = false;
}
}
else
{
if(dteArmClearHighlightsOnPanelClose)
{
dteArmClearHighlightsOnPanelClose = false;
dteClearHighlightsOnPanelClose = true;
}
}
// test for the opening or closing of the UR editing dialog
var URDialogIsOpen = (document.getElementsByClassName('new-comment-form').length == 1);
if(URDialogIsOpen)
{
var thisSelectedURID = W.map.panelRegion.currentView.conversationView.conversation.getID();
if(thisSelectedURID != uroSelectedURID)
{
// if the user selects a new UR whilst the editing dialog is still open, treat it in the
// same way as if the user had selected that UR with the dialog closed
uroURDialogIsOpen = false;
}
if(uroURDialogIsOpen === false)
{
// user is editing a new UR
uroSelectedURID = thisSelectedURID;
// add our own click event handler to the Send button, so we can do stuff whenever a new comment is added
document.getElementsByClassName('new-comment-form')[0].getElementsByClassName('btn')[0].addEventListener("click", uroAddedComment, false);
// make the "Next update request" button behave like the good old days...
if(uroGetCBChecked('_cbInhibitNURButton') === true)
{
var nurButton = document.getElementsByClassName('btn btn-block next')[0];
if(nurButton !== undefined)
{
// first change its label...
nurButton.innerHTML = "Done";
// ...then add a new click handler that'll inhibit the native ones
nurButton.addEventListener("click", uroInhibitNextUpdateRequestButton, false);
uroAddLog('inhibit Next update request button');
}
}
uroAddLog('user is editing UR '+uroSelectedURID);
uroExpectedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
}
}
else if(uroURDialogIsOpen === true)
{
// dialog was open and has now been closed
uroSelectedURID = null;
}
uroURDialogIsOpen = URDialogIsOpen;
if(((uroPendingCommentDataRefresh === true) || (uroWaitingCommentDataRefresh === true)) && (uroSelectedURID !== null))
{
uroAddLog('check completion of comment data refresh for UR '+uroSelectedURID+' ('+uroPendingCommentDataRefresh+','+uroWaitingCommentDataRefresh+')');
uroGetSelectedURCommentCount();
}
var selectedTotal = W.selectionManager.selectedItems.length;
if((selectedTotal > 0) && (document.getElementById('_uroDivOWLBtns') === null))
{
var selectedClass = W.selectionManager.selectedItems[0].model.CLASS_NAME;
var displayAddToOWLBtn = false;
var displayUpdateOWLBtn = false;
var displayRemoveFromOWLBtn = false;
var selectedSegments = false;
var selectedLandmarks = false;
var fid;
var loop;
// WME only seems to allow multi-object selections for segments, so testing the class of the first object in the
// selection list tells us the class of any other objects in the list too...
if(selectedClass == "Waze.Feature.Vector.Segment")
{
selectedSegments = true;
for(loop=0; loop<selectedTotal; loop++)
{
fid = W.selectionManager.selectedItems[loop].model.attributes.id;
var segIdx = uroIsSegOnWatchList(fid);
if(segIdx == -1)
{
displayAddToOWLBtn = true;
}
else
{
if(uroSegDataChanged(segIdx))
{
displayUpdateOWLBtn = true;
}
displayRemoveFromOWLBtn = true;
}
}
}
else if(selectedClass == "Waze.Feature.Vector.Landmark")
{
selectedLandmarks = true;
for(loop=0; loop<selectedTotal; loop++)
{
fid = W.selectionManager.selectedItems[loop].model.attributes.id;
var placeIdx = uroIsPlaceOnWatchList(fid);
if(placeIdx == -1)
{
displayAddToOWLBtn = true;
}
else
{
if(uroPlaceDataChanged(placeIdx))
{
displayUpdateOWLBtn = true;
}
displayRemoveFromOWLBtn = true;
}
}
}
var btnHTML = '<div id="_uroDivOWLBtns">';
if((displayAddToOWLBtn === true) && (displayUpdateOWLBtn === false))
{
btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Add to OWL</button>';
}
else if((displayUpdateOWLBtn === true) && (displayAddToOWLBtn === false))
{
btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Update OWL</button>';
}
else if((displayAddToOWLBtn === true) && (displayUpdateOWLBtn === true))
{
btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Add to & Update OWL</button>';
}
if(displayRemoveFromOWLBtn === true)
{
btnHTML += '<button class="btn btn-default" id="_btnRemoveOWL">Remove from OWL</button>';
}
btnHTML += '</div>';
/*
// once we get around to enabling these again, remember that altering the inner HTML of the
// segment-edit-general panel when the selected segment is part of a roundabout then disables
// the onclick handler for the select roundabout button...
//
// also remember that the current WME beta has yet another different side panel arrangement
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);
}
}
*/
}
// fix the livemap link in WME beta...
if(uroBetaEditor === true)
{
var lmLink = document.getElementsByClassName('waze-header-menu')[0].children[0].children[0].getAttribute('href');
if(lmLink.indexOf('https:') === -1)
{
uroAddLog('fixing livemap link...');
lmLink = 'https://www.waze.com' + lmLink;
document.getElementsByClassName('waze-header-menu')[0].children[0].children[0].setAttribute('href',lmLink);
}
}
}
}
function uroToggleURCtrls()
{
uroCtrlsHidden = !uroCtrlsHidden;
if (!uroCtrlsHidden)
{
document.getElementById('_hideUCCtrl').innerHTML = "hide";
if(uroCurrentTab == 1) uroShowURTab();
else if(uroCurrentTab == 2) uroShowMPTab();
else if(uroCurrentTab == 3) uroShowPlacesTab();
else if(uroCurrentTab == 4) uroShowCameraTab();
else if(uroCurrentTab == 5) uroShowOWLTab();
else if(uroCurrentTab == 6) uroShowMiscTab();
}
else
{
document.getElementById('_hideUCCtrl').innerHTML = "show";
uroSetStyleDisplay('uroCtrlURs','none');
uroSetStyleDisplay('uroCtrlMPs','none');
uroSetStyleDisplay('uroCtrlCameras','none');
uroSetStyleDisplay('uroCtrlMisc','none');
uroSetStyleDisplay('uroOWL','none');
}
return false;
}
function uroActiveTab(_id)
{
var e = document.getElementById(_id);
e.style.backgroundColor = "aliceblue";
e.style.borderTop = "1px solid";
e.style.borderLeft = "1px solid";
e.style.borderRight = "1px solid";
e.style.borderBottom = "0px solid";
}
function uroInactiveTab(_id)
{
var e = document.getElementById(_id);
e.style.backgroundColor = "white";
e.style.borderTop = "0px solid";
e.style.borderLeft = "0px solid";
e.style.borderRight = "0px solid";
e.style.borderBottom = "1px solid";
}
function uroInactiveAllTabs()
{
uroInactiveTab("_tabSelectCameras");
uroInactiveTab("_tabSelectMapProblems");
uroInactiveTab("_tabSelectMisc");
uroInactiveTab("_tabSelectUserRequests");
uroInactiveTab("_tabSelectCWL");
uroInactiveTab("_tabSelectPlaces");
if(!uroCtrlsHidden)
{
uroSetStyleDisplay('uroCtrlURs','none');
uroSetStyleDisplay('uroCtrlMPs','none');
uroSetStyleDisplay('uroCtrlCameras','none');
uroSetStyleDisplay('uroCtrlMisc','none');
uroSetStyleDisplay('uroOWL','none');
uroSetStyleDisplay('uroCtrlPlaces','none');
}
}
function uroShowURTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectUserRequests");
uroCurrentTab = 1;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlURs','block');
return false;
}
function uroShowMPTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectMapProblems");
uroCurrentTab = 2;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMPs','block');
return false;
}
function uroShowPlacesTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectPlaces");
uroCurrentTab = 3;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlPlaces','block');
for(var idx=0;idx<uroPlacesGroupsCollapsed.length;idx++)
{
uroPlacesGroupCEHandler(idx);
}
return false;
}
function uroShowCameraTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectCameras");
uroCurrentTab = 4;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlCameras','block');
return false;
}
function uroShowOWLTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectCWL");
uroCurrentTab = 5;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroOWL','block');
uroOWLUpdateHTML();
return false;
}
function uroShowMiscTab()
{
uroInactiveAllTabs();
uroActiveTab("_tabSelectMisc");
uroCurrentTab = 6;
if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMisc','block');
return false;
}
function uroTimbonesGetElementsByClassName(classname, node)
{
if(!node) node = document.getElementsByTagName("body")[0];
var a = [];
var re = new RegExp('\\b' + classname + '\\b');
var els = node.getElementsByTagName("*");
for (var i=0,j=els.length; i<j; i++)
{
if (re.test(els[i].className)) a.push(els[i]);
}
return a;
}
function uroNewLookCheckDetailsRequest()
{
var thisurl = document.location.href;
var doRetry = true;
var urID;
var endmarkerpos = thisurl.indexOf('&endshow');
var showmarkerpos = thisurl.indexOf('&showturn=');
if((endmarkerpos != -1) && (showmarkerpos != -1))
{
showmarkerpos += 10;
uroAddLog('showturn tab opened');
urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
uroAddLog(' turn problem ID = '+urID);
try
{
W.map.problemLayer.markers[urID].icon.imageDiv.click();
doRetry = false;
}
catch(err)
{
uroAddLog('problems not fully loaded, retrying...');
}
if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
}
else
{
showmarkerpos = thisurl.indexOf('&showpur=');
if((endmarkerpos != -1) && (showmarkerpos != -1))
{
showmarkerpos += 9;
uroAddLog('showPUR tab opened');
urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
uroAddLog(' PUR ID = '+urID);
try
{
W.map.placeUpdatesLayer.markers[urID].icon.imageDiv.click();
doRetry = false;
}
catch(err)
{
uroAddLog('PURs not fully loaded, retrying...');
}
if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
}
}
}
function uroUpdateMPSolverList()
{
if(Object.keys(W.model.problems.objects).length === 0)
{
return;
}
var resolverList = [];
var selector = document.getElementById('_selectMPUserID');
var selectedUser = null;
if(selector.selectedOptions[0] !== undefined)
{
selectedUser = parseInt(selector.selectedOptions[0].value);
}
while(selector.options.length > 0)
{
selector.options.remove(0);
}
var selectedIdx = 0;
var idx = 0;
for (var mpobj in W.model.problems.objects)
{
if(W.model.problem.objects.hasOwnProperty(mpobj))
{
var prob = W.model.problems.objects[mpobj];
if(prob.attributes.resolvedBy !== null)
{
var userID = prob.attributes.resolvedBy;
var userName = W.model.users.objects[userID].userName;
if(resolverList.indexOf(userName) == -1)
{
resolverList.push(userName);
selector.options.add(new Option(userName, userID));
if(userID == selectedUser)
{
selectedIdx = idx;
}
idx++;
}
}
}
}
if(selectedIdx !== null)
{
selector.selectedIndex = selectedIdx;
}
}
function uroUpdateResolverList()
{
if(Object.keys(W.model.mapUpdateRequests.objects).length === 0)
{
return;
}
var resolverList = [];
var selector = document.getElementById('_selectURResolverID');
var selectedUser = null;
if(selector.selectedOptions[0] !== undefined)
{
selectedUser = parseInt(selector.selectedOptions[0].value);
}
while(selector.options.length > 0)
{
selector.options.remove(0);
}
var selectedIdx = 0;
var idx = 0;
for (var urobj in W.model.mapUpdateRequests.objects)
{
if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
{
var ureq = W.model.mapUpdateRequests.objects[urobj];
if(ureq.attributes.resolvedBy !== null)
{
var userID = ureq.attributes.resolvedBy;
var userName = W.model.users.objects[userID].userName;
if(resolverList.indexOf(userName) == -1)
{
resolverList.push(userName);
selector.options.add(new Option(userName, userID));
if(userID == selectedUser)
{
selectedIdx = idx;
}
idx++;
}
}
}
}
if(selectedIdx !== null)
{
selector.selectedIndex = selectedIdx;
}
}
function uroUpdateUserList()
{
if(Object.keys(W.model.updateRequestSessions.objects).length === 0) return;
var selector = document.getElementById('_selectURUserID');
var selectedUser = null;
if(selector.selectedOptions[0] !== undefined)
{
selectedUser = parseInt(selector.selectedOptions[0].value);
}
while(selector.options.length > 0)
{
selector.options.remove(0);
}
var selectedIdx = null;
var listedIDs = [];
for(var ursIdx in W.model.updateRequestSessions.objects)
{
if(W.model.updateRequestSessions.objects.hasOwnProperty(ursIdx))
{
var ursObj = W.model.updateRequestSessions.objects[ursIdx];
if(ursObj.comments.length > 0)
{
for(var cidx=0; cidx < ursObj.comments.length; cidx++)
{
var userID = ursObj.comments[cidx].userID;
if((listedIDs.indexOf(userID) == -1) && (userID != -1))
{
listedIDs.push(userID);
}
}
}
}
}
if(listedIDs.length > 0)
{
var users = W.model.users.getByIds(listedIDs);
for(var idx=0; idx<listedIDs.length; idx++)
{
selector.options.add(new Option(users[idx].userName, listedIDs[idx]));
if(listedIDs[idx] == selectedUser)
{
selectedIdx = idx;
}
}
}
if(selectedIdx !== null)
{
selector.selectedIndex = selectedIdx;
}
}
function uroSetStyles(obj)
{
obj.style.fontSize = '12px';
obj.style.lineHeight = '100%';
obj.style.overflow = 'auto';
obj.style.height = (window.innerHeight * 0.55) + 'px';
}
function uroPlacesGroupCEHandler(groupidx)
{
if(uroPlacesGroupsCollapsed[groupidx] === false)
{
document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "block";
document.getElementById('_uroPlacesGroupState-'+groupidx).src = uroIcons[0][0];
}
else
{
document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "none";
document.getElementById('_uroPlacesGroupState-'+groupidx).src = uroIcons[0][1];
}
}
function uroPlacesGroupCollapseExpand()
{
var groupidx = this.id.substr(21);
if(uroPlacesGroupsCollapsed[groupidx] === true) uroPlacesGroupsCollapsed[groupidx] = false;
else uroPlacesGroupsCollapsed[groupidx] = true;
uroPlacesGroupCEHandler(groupidx);
return false;
}
function uroPopulatePlacesTab()
{
var tHTML = '';
tHTML += '<b>Filter PURs by category/status:</b><br>';
tHTML += '<input type="checkbox" id="_cbFilterUneditablePlaceUpdates">Ones I can\'t edit</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterLockRankedPlaceUpdates">Ones with non-zero lockRanks</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterNewPlacePUR">Ones for new places</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterUpdatedDetailsPUR">Ones for updated place details</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterNewPhotoPUR">Ones for new photos</input><br>';
tHTML += '<input type="checkbox" id="_cbFilterFlaggedPUR">Ones flagged for attention</input><br>';
tHTML += '<br><input type="checkbox" id="_cbLeavePURGeos">Don\'t hide place polygons/points</input><br>';
tHTML += '<br><input type="checkbox" id="_cbInvertPURFilters">Invert PUR filters</input><br>';
tHTML += '<br><b>Filter PURs by severity:</b><br>';
tHTML += '<input type="checkbox" id="_cbPURFilterLowSeverity">Low</input> ';
tHTML += '<input type="checkbox" id="_cbPURFilterMediumSeverity">Medium</input> ';
tHTML += '<input type="checkbox" id="_cbPURFilterHighSeverity">High</input>';
tHTML += '<br><b>Filter PURs by age of submission:</b><br>';
tHTML += '<input type="checkbox" id="_cbEnablePURMinAgeFilter">Hide PURs less than </input>';
tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPURFilterMinDays"> days old<br>';
tHTML += '<input type="checkbox" id="_cbEnablePURMaxAgeFilter">Hide PURs more than </input>';
tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPURFilterMaxDays"> days old<br>';
tHTML += '<hr>';
tHTML += '<br><b>Filter Places by state:</b><br>';
tHTML += 'Hide if last edited<br>';
tHTML += '<input type="checkbox" id="_cbPlaceFilterEditedLessThan"> less than </input>';
tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterPlaceEditMinDays"> days ago<br>';
tHTML += '<input type="checkbox" id="_cbPlaceFilterEditedMoreThan"> more than </input>';
tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterPlaceEditMaxDays"> days ago<br>';
tHTML += '<br>Hide if locked at level:<br>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL0">1</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL1">2</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL2">3</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL3">4</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL4">5</input>';
tHTML += ' <input type="checkbox" id="_cbHidePlacesL5">6</input>';
tHTML += '<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 += '<br><br><b>Filter Places by category:</b><br>';
var nCategories = W.Config.venues.categories.length;
var i;
if(uroPlacesGroupsCollapsed.length != nCategories)
{
for(i=0; i<nCategories; i++)
{
uroPlacesGroupsCollapsed.push(false);
}
}
for(i=0; i<nCategories; i++)
{
var parentCategory = W.Config.venues.categories[i];
var localisedName = I18n.lookup("venues.categories")[parentCategory];
if(uroPlacesGroupsCollapsed[i] === true)
{
tHTML += '<img src="'+uroIcons[0][1]+'" id="_uroPlacesGroupState-'+i+'">';
}
else
{
tHTML += '<img src="'+uroIcons[0][0]+'" id="_uroPlacesGroupState-'+i+'">';
}
tHTML += ' <input type="checkbox" id="_cbPlacesFilter-'+parentCategory+'"><b>'+localisedName+'</b></input><br>';
tHTML += '<div id="_uroPlacesGroup-'+i+'" style="padding:3px;border-width:2px;border-style:solid;border-color:#FFFFFF">';
for(var ii=0; ii<W.Config.venues.subcategories[parentCategory].length; ii++)
{
var subCategory = W.Config.venues.subcategories[parentCategory][ii];
localisedName = I18n.lookup("venues.categories")[subCategory];
tHTML += ' <input type="checkbox" id="_cbPlacesFilter-'+subCategory+'">'+localisedName+'</input><br>';
}
tHTML += '</div>';
}
tHTML += '<input type="checkbox" id="_cbFilterPrivatePlaces"><b>Residential Places</b></input><br>';
tHTML += '<br><input type="checkbox" id="_cbInvertPlacesFilter">Invert Place filters?</input>';
uroCtrlPlaces.innerHTML = tHTML;
}
function uroWazeBits()
{
// "fake" uroWazeBits() function which only performs layer scan, to stop the uroWazeBits() call in WMETB from
// messing around with other stuff in the actual uroWazeBits() function (now renamed uroRealWazeBits...) that
// really only ought to be called once.
var i;
for(i=0;i<W.map.layers.length;i++)
{
if(W.map.layers[i].name == 'Spotlight') uroMaskLayer = i;
if(W.map.layers[i].name.indexOf('Waze.Control.SelectHighlightFeature') != -1) uroRootContainer = W.map.layers[i].div.id;
if(W.map.layers[i].name == 'Node Connections') uroTurnsLayerIdx = i;
}
uroPlacesRoot = W.map.landmarkLayer.id + '_vroot';
for(i=0;i<W.map.controls.length;i++)
{
if(W.map.controls[i].CLASS_NAME == 'Waze.View.ArchivePanel') dteControlsIdx = i;
else if(W.map.controls[i].CLASS_NAME == 'Waze.Control.Archive') dteControlsIdx = i;
if(W.map.controls[i].id !== null)
{
if(W.map.controls[i].id.indexOf('UpdateRequests') != -1) uroURControlsIdx = i;
if(W.map.controls[i].id.indexOf('MapProblems') != -1) uroProblemControlsIdx = i;
}
}
uroAddLog('uroMaskLayer at idx '+uroMaskLayer);
uroAddLog('Turns layer at idx '+uroTurnsLayerIdx);
uroAddLog('uroRootContainer = '+uroRootContainer);
uroAddLog('Places root layer = '+uroPlacesRoot);
}
function uroRealWazeBits()
{
if(document.getElementsByClassName("sandbox").length > 0)
{
uroPracticeMode = true;
uroAddLog('WME practice mode detected, script is disabled...');
return;
}
else
{
uroPracticeMode = false;
}
uroAddLog('adding WazeBits...'+uroToHex(uroWazeBitsPresent,4));
if((uroWazeBitsPresent & 0x0001) === 0)
{
if(typeof W != "undefined")
{
if(typeof W.map != "undefined")
{
uroAddLog(' W.map OK');
uroWazeBitsPresent |= 0x0001;
}
}
}
if((uroWazeBitsPresent & 0x0002) === 0)
{
if(typeof W != "undefined")
{
if(typeof W.model != "undefined")
{
uroAddLog(' W.model OK');
uroWazeBitsPresent |= 0x0002;
}
}
}
if((uroWazeBitsPresent & 0x0004) === 0)
{
if(typeof W != "undefined")
{
if(typeof W.loginManager != "undefined")
{
uroAddLog(' loginManager OK');
uroWazeBitsPresent |= 0x0004;
}
}
}
if((uroWazeBitsPresent & 0x0008) === 0)
{
if(typeof W != "undefined")
{
if(typeof W.selectionManager != "undefined")
{
uroAddLog(' selectionManager OK');
uroWazeBitsPresent |= 0x0008;
}
}
}
if((uroWazeBitsPresent & 0x0010) === 0)
{
if(typeof OpenLayers != "undefined")
{
uroAddLog(' OpenLayers OK');
uroWazeBitsPresent |= 0x0010;
}
}
if((uroWazeBitsPresent & 0x0020) === 0)
{
if(typeof Waze != "undefined")
{
uroAddLog(' Waze OK');
uroWazeBitsPresent |= 0x0020;
}
}
if((uroWazeBitsPresent & 0x0040) === 0)
{
if(document.getElementById('user-tabs') !== null)
{
uroUserTabId = 'user-tabs';
uroAddLog(' user-tabs OK');
uroWazeBitsPresent |= 0x0040;
}
}
if((uroWazeBitsPresent & 0x0080) === 0)
{
if(uroPracticeMode === true)
{
uroAddLog(' sidepanel-drives ignored in practice mode');
uroWazeBitsPresent |= 0x0080;
}
else
{
if(document.getElementById('sidepanel-drives') !== null)
{
uroAddLog(' sidepanel-drives OK');
uroWazeBitsPresent |= 0x0080;
}
}
}
if((uroWazeBitsPresent & 0x0100) === 0)
{
if(typeof I18n != "undefined")
{
uroAddLog(' I18n OK');
uroWazeBitsPresent |= 0x0100;
}
}
if(uroWazeBitsPresent !== 0x01FF) setTimeout(uroRealWazeBits,250);
else if((W.loginManager.isLoggedIn() === false) && (uroPracticeMode === false))
{
uroAddLog('Waiting for user log-in...');
setTimeout(uroRealWazeBits,1000);
}
else
{
uroAddLog('All WazeBits present and correct...');
uroDOMHasTurnProblems = (W.model.turnProblems !== undefined);
uroPopulatePlacesTab();
uroControls.appendChild(uroCtrlURs);
uroControls.appendChild(uroCtrlMPs);
uroControls.appendChild(uroCtrlPlaces);
uroControls.appendChild(uroCtrlCameras);
uroControls.appendChild(uroOWL);
uroControls.appendChild(uroCtrlMisc);
uroControls.appendChild(uroCtrlHides);
uroControls.appendChild(uroAMList);
uroCtrlURs.onclick = uroFilterItems_URTabClick;
uroCtrlMPs.onclick = uroFilterItems_MPTabClick;
uroCtrlPlaces.onclick = uroFilterItems_PlacesTabClick;
uroCtrlCameras.onclick = uroFilterItems_CamerasTabClick;
uroCtrlMisc.onclick = uroFilterItems_MiscTabClick;
uroWazeBits();
uroDiv.addEventListener("mouseover", uroEnterPopup, false);
uroDiv.addEventListener("mouseout", uroExitPopup, false);
if(sessionStorage.UROverview_FID_IgnoreList === undefined) sessionStorage.UROverview_FID_IgnoreList = '';
if(sessionStorage.UROverview_FID_WatchList === undefined) sessionStorage.UROverview_FID_WatchList = '';
if(uroConfirmIntercepted === false) uroAddInterceptor();
if(uroBetaEditor === true)
{
uroAddLog('fixing header links...');
var nLinks = document.getElementsByClassName('waze-header-menu')[0].children.length;
for(var link=0; link<nLinks; link++)
{
var relLink = document.getElementsByClassName('waze-header-menu')[0].children[link].children[0].getAttribute('href');
relLink = 'https://www.waze.com' + relLink;
document.getElementsByClassName('waze-header-menu')[0].children[link].children[0].setAttribute('href',relLink);
}
}
setInterval(uroTenthSecondTick,100);
uroShowDebugOutput = uroPersistentDebugOutput;
}
}
function uroAddInterceptor()
{
uroAddLog('Adding interceptor function...');
// add interceptor function for confirm(), so that we can auto-select the "OK" option when solving URs
// which have pending question...
var _confirm = confirm;
confirm = function(msg)
{
if((I18n.lookup("update_requests.panel.confirm") == msg) && (uroGetCBChecked('_cbDisablePendingQuestions') === true))
{
uroAddLog('Intercepted pending comments confirmation...');
return true;
}
else if(typeof(msg) == 'undefined')
{
uroAddLog('Intercepted blank confirmation...');
return true;
}
else
{
return _confirm(msg);
}
};
uroConfirmIntercepted = true;
}
function uroEnterPopup()
{
uroMouseInPopup = true;
}
function uroExitPopup()
{
uroMouseInPopup = false;
}
function uroToggleDebug()
{
uroShowDebugOutput = !uroShowDebugOutput;
var dbgMode = "none";
if(uroShowDebugOutput)
{
dbgMode = "inline";
}
document.getElementById('_uroDebugMode').style.display = dbgMode;
}
function uroInitialise()
{
if(document.URL.indexOf('editor-beta') != -1) uroBetaEditor = true;
var urlBits = document.URL.split("&mapUpdateRequest=");
if(urlBits.length == 2)
{
uroURIDInURL = parseInt(urlBits[1].split('&')[0]);
uroAddLog('found UR ID '+uroURIDInURL+' in URL');
}
// create a new div to display the UR details floaty-box
uroDiv = document.createElement('div');
uroDiv.id = "uroDiv";
uroDiv.style.position = 'absolute';
uroDiv.style.visibility = 'hidden';
uroDiv.style.top = '0';
uroDiv.style.left = '0';
uroDiv.style.zIndex = 100;
uroDiv.style.backgroundColor = 'aliceblue';
uroDiv.style.borderWidth = '3px';
uroDiv.style.borderStyle = 'solid';
uroDiv.style.borderRadius = '10px';
uroDiv.style.boxShadow = '5px 5px 10px Silver';
uroDiv.style.padding = '4px';
document.body.appendChild(uroDiv);
uroControls = document.createElement('section');
uroControls.style.fontSize = '12px';
uroControls.id = 'uroControls';
var updateURL;
if(navigator.userAgent.indexOf('Chrome') == -1)
{
updateURL = 'https://gf.qytechs.cn/scripts/1952-uroverview-plus-uro';
}
else
{
updateURL = 'https://chrome.google.com/webstore/detail/uroverview/amdamgkgchnbaopmphhjapmjcdghdphi';
}
var tabbyHTML = '<b><a href="'+updateURL+'" target="_blank">UROverview Plus</a></b> <label id="_uroVersion">'+uroVersion+'</label>';
tabbyHTML += '<label id="_uroDebugMode">(dbg)</label>';
tabbyHTML += ' <input type="checkbox" id="_cbMasterEnable" checked>Enabled</input>';
tabbyHTML += '<p><table border=0 width="100%"><tr>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectUserRequests"><a href="#" id="_linkSelectUserRequests" style="text-decoration:none;font-size:12px">URs</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectMapProblems"><a href="#" id="_linkSelectMapProblems" style="text-decoration:none;font-size:12px">MPs</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectPlaces"><a href="#" id="_linkSelectPlaces" style="text-decoration:none;font-size:12px">Places</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectCameras"><a href="#" id="_linkSelectCameras" style="text-decoration:none;font-size:12px">Cams</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectCWL"><a href="#" id="_linkSelectOWL" style="text-decoration:none;font-size:12px">OWL</a></td>';
tabbyHTML += '<td valign="center" align="center" id="_tabSelectMisc"><a href="#" id="_linkSelectMisc" style="text-decoration:none;font-size:12px">Misc</a></td>';
tabbyHTML += '</tr></table>';
uroControls.innerHTML = tabbyHTML;
uroCtrlURs = document.createElement('p');
uroCtrlMPs = document.createElement('p');
uroCtrlCameras = document.createElement('p');
uroOWL = document.createElement('p');
uroCtrlMisc = document.createElement('p');
uroAMList = document.createElement('div');
uroCtrlHides = document.createElement('div');
uroCtrlPlaces = document.createElement('p');
// UR controls tab
uroCtrlURs.id = "uroCtrlURs";
uroCtrlURs.innerHTML = '<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURFilterOutsideArea">Hide URs outside my editable area</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbNoFilterForURInURL">Don\'t filter UR in URL</input><br><br>';
uroCtrlURs.innerHTML += '<b>Filter by type:</b><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterWazeAuto">Waze Automatic</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectTurn">Incorrect turn</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectAddress">Incorrect address</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectRoute">Incorrect route</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingRoundabout">Missing roundabout</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterGeneralError">General error</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterTurnNotAllowed">Turn not allowed</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectJunction">Incorrect junction</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingBridgeOverpass">Missing bridge overpass</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterWrongDrivingDirection">Wrong driving direction</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingExit">Missing exit</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingRoad">Missing road</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterBlockedRoad">Blocked road</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingLandmark">Missing Landmark</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterUndefined">Undefined</input><br>';
uroCtrlURs.innerHTML += ' <i>Specially tagged types</i><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterRoadworks">[ROADWORKS]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterConstruction">[CONSTRUCTION]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterClosure">[CLOSURE]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterEvent">[EVENT]</input><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbFilterNote">[NOTE]</input><br><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbInvertURFilter">Invert operation of type filters?</input><br>';
uroCtrlURs.innerHTML += '<hr>';
uroCtrlURs.innerHTML += '<br><b>Hide by state:</b><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterOpenUR">Open</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterClosedUR">Closed</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterSolved">Solved</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterUnidentified">Not identified</input><br><br>';
uroCtrlURs.innerHTML += '<br><b>Filter by age of submission:</b><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMinAgeFilter">Hide URs less than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinDays"> days old<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMaxAgeFilter">Hide URs more than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxDays"> days old<br>';
uroCtrlURs.innerHTML += '<br><b>Filter by description/comments/following:</b><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURDescriptionMustBePresent" pairedWith="_cbURDescriptionMustBeAbsent">Hide</input> or ';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURDescriptionMustBeAbsent" pairedWith="_cbURDescriptionMustBePresent">show</input> URs with no description<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableKeywordMustBePresent">Hide URs not including </input>';
uroCtrlURs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPresent"><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableKeywordMustBeAbsent">Hide URs including </input>';
uroCtrlURs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordAbsent"><br>';
uroCtrlURs.innerHTML += ' <input type="checkbox" id="_cbCaseInsensitive"><i>Case-insensitive matches?</i></input><br><br>';
uroCtrlURs.innerHTML += 'With comments from me?<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyComments" pairedWith="_cbHideAnyComments">Yes </input>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideAnyComments" pairedWith="_cbHideMyComments">No</input><br>';
uroCtrlURs.innerHTML += 'If last comment made by me?<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfLastCommenter" pairedWith="_cbHideIfNotLastCommenter">Yes </input>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfNotLastCommenter" pairedWith="_cbHideIfLastCommenter">No </input><br>';
uroCtrlURs.innerHTML += 'If last comment made by UR reporter?<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfReporterLastCommenter" pairedWith="_cbHideIfReporterNotLastCommenter">Yes </input>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfReporterNotLastCommenter" pairedWith="_cbHideIfReporterLastCommenter">No</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMinCommentsFilter">With less than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinComments"> comments<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMaxCommentsFilter">With more than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="0" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxComments"> comments<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyFollowed" pairedWith="_cbHideMyUnfollowed">Ones I am or </input>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyUnfollowed" pairedWith="_cbHideMyFollowed">am not following</input><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter2">Last comment less than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays2"> days ago<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter">Last comment more than </input>';
uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays"> days ago<br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURUserIDFilter">Without comments from user</input>';
uroCtrlURs.innerHTML += '<select id="_selectURUserID" style="width:80%; height:22px;"></select><br>';
uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURResolverIDFilter">Not resolved by user</input>';
uroCtrlURs.innerHTML += '<select id="_selectURResolverID" style="width:80%; height:22px;"></select>';
uroCtrlURs.innerHTML += '<br><br><input type="checkbox" id="_cbNoFilterForTaggedURs"><b>Don\'t apply state/age filters to tagged URs</b></input><br>';
// Map problems controls tab
uroCtrlMPs.id = "uroCtrlMPs";
uroCtrlMPs.innerHTML = '<br>';
uroCtrlMPs.innerHTML += '<b>Filter MPs by type:</b><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMissingJunction">Missing junction</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMissingRoad">Missing road</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterCrossroadsJunctionMissing">Missing crossroads</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterDrivingDirectionMismatch">Driving direction mismatch</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRoadTypeMismatch">Road type mismatch</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRestrictedTurn">Restricted turn might be allowed</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRoadClosureProblem">Road closure</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterUnknownProblem">Unknown problem type</input><br><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterTurnProblem">Turn Problems</input><br><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterReopenedProblem">Reopened Problems</input><br><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbInvertMPFilter">Invert operation of type filters?</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterOutsideArea">Hide MPs outside my editable area</input><br>';
uroCtrlMPs.innerHTML += '<br><b>Hide closed/solved/unidentified Problems:</b><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterClosed">Closed</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterSolved">Solved</input><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterUnidentified">Not identified</input><br><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPClosedUserIDFilter" pairedWith="_cbMPNotClosedUserIDFilter">Closed</input> or ';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPNotClosedUserIDFilter" pairedWith="_cbMPClosedUserIDFilter">Not Closed</input> by user';
uroCtrlMPs.innerHTML += '<select id="_selectMPUserID" style="width:80%; height:22px;"></select><br>';
uroCtrlMPs.innerHTML += '<br><b>Hide problems (not turn) by severity:</b><br>';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterLowSeverity">Low</input> ';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMediumSeverity">Medium</input> ';
uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterHighSeverity">High</input><br>';
// Places filtering tab
uroCtrlPlaces.id = "uroCtrlPlaces";
uroCtrlPlaces.innerHTML = 'Places filter list being populated, please wait...';
// Camera controls tab
uroCtrlCameras.id = "uroCtrlCameras";
uroCtrlCameras.innerHTML = '<br><b>Show Cameras by creator:</b><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowWorldCams" checked>world_* users</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowUSACams" checked>usa_* users</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowNonWorldCams" checked>other users</input><br>';
uroCtrlCameras.innerHTML += '<br><input type="checkbox" id="_cbShowOnlyMyCams">Show ONLY cameras created/edited by me</input><br>';
uroCtrlCameras.innerHTML += '<br><b>Show Cameras by approval status:</b><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowApprovedCams" checked>approved</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowNonApprovedCams" checked>non-approved</input><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbShowOlderCreatedNonApproved"> if created more than </input>';
uroCtrlCameras.innerHTML += '<input type="number" min="1" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCameraMinCreatedDays"> days ago<br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbShowOlderUpdatedNonApproved"> if updated more than </input>';
uroCtrlCameras.innerHTML += '<input type="number" min="1" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCameraMinUpdatedDays"> days ago<br>';
uroCtrlCameras.innerHTML += '<br><b>Show Cameras by type:</b><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowSpeedCams" checked>Speed</input><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbShowIfSpeedSet" checked> with speed data</input><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbShowIfNoSpeedSet" checked> with no speed data</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowRedLightCams" checked>Red Light</input><br>';
uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowDummyCams" checked>Dummy</input><br>';
uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by creator:</b><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByMe">me</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank0">L1</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank1">L2</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank2">L3</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank3">L4</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank4">L5</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideCreatedByRank5">L6</input>';
uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by updater:</b><br>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByMe">me</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank0">L1</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank1">L2</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank2">L3</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank3">L4</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank4">L5</input>';
uroCtrlCameras.innerHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank5">L6</input>';
uroCtrlCameras.innerHTML += '<br><br><b><input type="checkbox" id="_cbHideCWLCams">Hide cameras on watchlist</input></b><br>';
// Object watchlist tab
uroOWL.id = "uroOWL";
uroOWLUpdateHTML();
// Misc controls tab
uroCtrlMisc.id = "uroCtrlMisc";
uroCtrlMisc.innerHTML = '<br><b>Use default conversation markers:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbNativeConvoMarkers" checked>in public WME</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbNativeBetaConvoMarkers" checked>in beta WME</input><br>';
uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbCommentCount">Show comment count on UR markers</input></b><br>';
uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbURBackfill">Backfill UR data</input></b><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Marker Unstacking:</b><br>';
uroCtrlMisc.innerHTML += 'Distance threshold: <input type="number" min="1" max="30" value="15" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackSensitivity"><br>';
uroCtrlMisc.innerHTML += 'Disable below zoom: <input type="number" min="0" max="10" value="3" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackZoomLevel"><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Use custom marker for URs tagged as:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomRoadworksMarkers">[ROADWORKS]</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomConstructionMarkers">[CONSTRUCTION]</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomClosuresMarkers">[CLOSURE]</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomEventsMarkers">[EVENT]</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomNotesMarkers">[NOTE]</input><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Use custom marker for MPs tagged as:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomElginMarkers">[Elgin]</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTrafficMasterMarkers">[TM]</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTrafficCastMarkers">[TrafficCast]</input><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Popup mouse behaviour:</b><br>';
uroCtrlMisc.innerHTML += 'Mouse idle <input type="number" min="1" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupDwellTimeout"> *100ms<br>';
uroCtrlMisc.innerHTML += 'Mouse over <input type="number" min="1" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupEntryTimeout"> *100ms<br>';
uroCtrlMisc.innerHTML += 'Distance <input type="number" min="0" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMaxJitter"> pixels<br>';
uroCtrlMisc.innerHTML += '<br><br><b>Disable popup for:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitURPopup">URs</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitMPPopup">MPs</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitCamPopup">Cameras</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitSegPopup">Segments</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitTurnsPopup">Restricted Turns</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitLandmarkPopup">Places</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitPUPopup">Place Updates</input><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Date/Time formatting for popups:</b><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtDDMMYY" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtYYMMDD" checked>day/month/year</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtMMDDYY" pairedWith="_cbDateFmtDDMMYY,_cbDateFmtYYMMDD">month/day/year</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtYYMMDD" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtDDMMYY">year/month/day</input><br><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbTimeFmt24H" pairedWith="_cbTimeFmt12H" checked>24 hour</input><br>';
uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbTimeFmt12H" pairedWith="_cbTimeFmt24H">12 hour</input><br><br>';
uroCtrlMisc.innerHTML += '<i>Unticked uses browser default setting</i>';
uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbWhiteBackground">Use custom background colour</input></b><br>';
uroCtrlMisc.innerHTML += 'R:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundRed">';
uroCtrlMisc.innerHTML += 'G:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundGreen">';
uroCtrlMisc.innerHTML += 'B:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundBlue"><br>';
uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbHideAMLayer">Hide Area Manager polygons</input></b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisablePlacesFiltering">Disable Places filtering</input></b><br>';
////uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisablePendingQuestions">Disable UR Pending Questions confirmation</input></b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbInhibitNURButton">Replace "Next update request" button</input></b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbInhibitURCentering">Disable map move on UR selection</input></b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbHideEditorInfo">Hide sidebar editor info</input></b><br>';
uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbEnableDTE">Drive Tab Enhancement (DTE)</input></b><br>';
uroCtrlMisc.innerHTML += '<br><br><b>Settings backup/restore/reset:</b><br>';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnSettingsToText" value="Backup"> ';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnTextToSettings" value="Restore"> | ';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnResetSettings" value="Reset"><br><br>';
uroCtrlMisc.innerHTML += '<textarea id="_txtSettings" value="" /><br>';
uroCtrlMisc.innerHTML += '<input type="button" id="_btnClearSettingsText" value="Clear"><br>';
// footer for tabs container
uroCtrlHides.id = 'uroCtrlHides';
uroCtrlHides.innerHTML = '<input type="button" id="_btnUndoLastHide" value="Undo last hide"> ';
uroCtrlHides.innerHTML += '<input type="button" id="_btnClearSessionHides" value="Undo all hides"><p>';
// footer for AM list
uroAMList.id = 'uroAMList';
window.addEventListener("beforeunload", uroSaveSettings, false);
uroRealWazeBits();
}
function dteAddHeader()
{
var rlcObj = document.getElementsByClassName("result-list-container");
if(typeof rlcObj == "undefined") return;
if(typeof rlcObj[0].children[0] == "undefined") return;
if(typeof rlcObj[0].children[0].innerHTML == "undefined") return;
var thtml = rlcObj[0].children[0].innerHTML;
if(thtml.indexOf('Full drive history') == -1)
{
thtml += '<br><br><i><small>Full drive history goes back to '+dteOldestFullDrive.toDateString()+'</small></i>';
rlcObj[0].children[0].innerHTML = thtml;
}
}
function dteSetNewTabLength()
{
uroAddLog('altering ResultsPerPage parameter...');
var t = document.getElementById('sidepanel-drives');
t.style.overflow = 'auto';
t.style.height = (window.innerHeight * 0.6) + 'px';
var baseloc = 'https://'+window.location.hostname+Waze.Config.api_base+'/Archive/MyList?minDistance=1000';
var a = new XMLHttpRequest();
var fullDrives = 0;
var nPages = 1;
var foundMissingDrive = false;
while(!foundMissingDrive)
{
////uroSetStyleDisplay(uroUserTabId,'none');
var loc = baseloc+'&offset='+fullDrives+'&count=5';
uroAddLog('requesting '+loc);
var retries = 0;
while(retries < 3)
{
a.open('GET',loc,false);
a.send();
if(a.responseText !== "")
{
break;
}
retries++;
}
if(a.responseText !== "")
{
var b = JSON.parse(a.responseText);
var loadedDrives = b.archives.objects.length;
uroAddLog('received '+loadedDrives+' drives');
if(loadedDrives != 5) foundMissingDrive = true;
for(var loop=0; loop < loadedDrives; loop++)
{
if(b.archives.objects[loop].hasFullSession === false)
{
foundMissingDrive = true;
}
else
{
fullDrives++;
dteOldestFullDrive = new Date(b.archives.objects[loop].startTime);
}
}
}
else
{
foundMissingDrive = true;
}
}
uroAddLog(fullDrives+' full drives in history');
uroAddLog('oldest drives are on '+dteOldestFullDrive.toDateString());
if(fullDrives < 5)
{
fullDrives = 5;
uroAddLog('insufficient full drives, using standard drives tab');
}
else if(fullDrives > 50)
{
nPages = Math.ceil(fullDrives / 50);
uroAddLog('too many full drives for a single tab page, splitting over '+nPages+' pages...');
fullDrives = Math.ceil(fullDrives/nPages);
}
////uroSetStyleDisplay(uroUserTabId,'');
if((dteOldestFullDrive - dteEpoch) > 0)
{
var totalDrives = 0;
if(W.model.archives.additionalInfo !== null)
{
totalDrives = W.model.archives.additionalInfo.totalSessions;
}
if(totalDrives !== null)
{
uroAddLog('updating drives tab...');
W.map.controls[dteControlsIdx].sidePanelView.ResultsPerPage = fullDrives;
uroAddLog(totalDrives+' drives in history');
W.map.controls[dteControlsIdx].sidePanelView.setSessions(totalDrives);
W.map.controls[dteControlsIdx].loadSessions(0);
}
setInterval(dteAddHeader,250);
setInterval(dteCheckDriveListChanges,250);
}
}
function dteListClick()
{
dteClearListHighlight();
this.style.backgroundColor = "lightgreen";
dteArmClearHighlightsOnPanelClose = true;
}
function dteClearListHighlight()
{
var drivesShown = document.getElementById('sidepanel-drives').getElementsByClassName('result session').length;
if(drivesShown > 0)
{
for(var loop = 0;loop < drivesShown; loop++)
{
var listEntry = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[loop];
listEntry.style.backgroundColor = "";
}
}
}
function dteCheckDriveListChanges()
{
var drivesShown = document.getElementById('sidepanel-drives').getElementsByClassName('result session').length;
if(drivesShown > 0)
{
var topID = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[0].getAttribute('data-id');
if(topID != dteTopID)
{
dteTopID = topID;
for(var loop = 0;loop < drivesShown; loop++)
{
var listEntry = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[loop];
var driveID = listEntry.getAttribute('data-id');
var driveObj = W.model.archives.objects[driveID];
var driveSecs = Math.floor((driveObj.endTime - driveObj.startTime) / 1000);
var driveHours = Math.floor(driveSecs / 3600);
driveSecs -= (driveHours * 3600);
var driveMins = Math.floor(driveSecs / 60);
driveSecs -= (driveMins * 60);
var trueTime = (driveHours+':'+("0"+driveMins).slice(-2)+'.'+("0"+driveSecs).slice(-2));
listEntry.getElementsByTagName('span')[1].innerHTML = trueTime;
listEntry.addEventListener("click", dteListClick, false);
}
}
}
}
function uroBootstrap()
{
console.debug('uroBootstrap()');
var bGreasemonkeyServiceDefined = false;
/*
try {
bGreasemonkeyServiceDefined = (typeof Components.interfaces.gmIGreasemonkeyService === "object");
}
catch (err) { }
*/
/*
if (typeof unsafeWindow === "undefined" || ! bGreasemonkeyServiceDefined) {
unsafeWindow = ( function () {
var dummyElem = document.createElement('p');
dummyElem.setAttribute('onclick', 'return window;');
return dummyElem.onclick();
}) ();
}
*/
/* begin running the code! */
uroInitialise();
}
uroBootstrap();