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