您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Adds a whole bunch of features to WME, which someday I may get around to documenting properly...
当前为
// ==UserScript== // @name UROverview Plus (URO+) // @namespace http://greasemonkey.chizzum.com // @description Adds a whole bunch of features to WME, which someday I may get around to documenting properly... // @include https://*.waze.com/*editor* // @include https://editor-beta.waze.com/* // @include https://beta.waze.com/* // @exclude https://www.waze.com/user/*editor/* // @exclude https://www.waze.com/*/user/*editor/* // @grant none // @version 3.224 // ==/UserScript== /* ======================================================================================================================= Bug fixes - MUST BE CLEARED BEFORE RELEASE ======================================================================================================================= ======================================================================================================================= Things to be checked ======================================================================================================================= */ /* JSHint Directives */ /* globals $: */ /* globals W: true */ /* globals I18n: */ /* globals OpenLayers: true */ /* globals require: */ /* globals _: */ /* globals trustedTypes: */ /* jshint bitwise: false */ /* jshint eqnull: true */ /* jshint esversion: 11 */ const uroVersion = "3.224"; const uroReleaseDate = "20231211"; // list of changes affecting all users const uroChanges = [ "RTC details in popup now indicates how far in advance a future closure will occur - e.g. 'Today', '5 days' etc.", "Expired closures are now handled appropriately wherever URO interacts with closures...", "Further compatibility updates for latest WME release", "Optimised layout of segment details popup", "Optimised duplicate UR check" ]; // list of changes affecting only WME Beta users (at least until the next production release including these parts of the beta code...) const uroBetaChanges = [ ]; // true enables debug output during script startup let uroShowDebugOutput = true; // true keeps debug output enabled after script startup const uroPersistentDebugOutput = false; // true enables performance monitoring debug output const uroPerformanceMonitoringOutput = false; // var uroRecentDebug = []; let uroURDupes = []; let uroCtrlsHidden = false; let uroFID = -1; let uroShownFID = -1; let uroShownPopupType = null; let uroInhibitSave = true; let uroPopupTimer = -2; let uroPopupDwellTimer = -1; let uroPopupAutoHideTimer = 0; let uroPopupShown = false; let uroPopupSuppressed = false; let uroSetupListeners = true; let uroMouseInPopup = false; let uroConfirmIntercepted = false; let uroRootContainer = null; let uroPlacesRoot = null; let uroMCLayer = null; let uroVenueLayer = null; let uroMO_URLayer = null; const URO_TMARKER = { UR: 0, MP: 1, PUR: 2, PPUR: 3, RPUR: 4, RTC: 5 }; let uroMarkerLayers = []; const URO_TRTC = { UNKNOWN: 0, WME: 1, WAZEFEED: 2, WAZEOTHER: 3 }; const URO_SRTC = { UNKNOWN: 0, EXPIRED: 1, ACTIVE: 2, FUTURE: 3 }; let uroCustomMarkerList = []; let uroPendingURSessionIDs = []; let uroRequestedURSessionIDs = []; let uroPlacesGroupsCollapsed = []; let uroKnownProblemTypeIDs = []; let uroKnownProblemTypeNames = []; let uroSelectedItems = []; let uroPURsToHide = []; let uroNullOpenLayers = false; let uroNullURLayer = false; let uroNullProblemLayer = false; let uroNullMapViewport = false; let uroURDialogIsOpen = false; let uroHoveredURID = null; let uroSelectedURID = null; let uroURReclickAttempts = 0; let uroPendingCommentDataRefresh = false; let uroWaitingCommentDataRefresh = false; let uroExpectedCommentCount = null; let uroCachedLastCommentID = null; let uroMCSelected = false; let uroMouseIsDown = false; let uroBackfilling = false; let uroPopulatingRequestSessions = false; let uroHidePopupOnPanelOpen = false; let uroPointerWithinMap = false; let uroUserID = -1; let uroDOMHasTurnProblems = false; let uroBetaEditor = false; let uroMTEMode = false; let uroFinalisingListenerSetup = false; let uroInitialised = false; let uroDiv = null; let uroAlerts = null; let uroControls = null; let uroCtrlHides = null; let uroAMList = []; let uroManagedAreas = []; let uroIgnoreAreasUserID = null; let uroCWLGroups = []; let uroCamWatchObjects = []; let uroSegWatchObjects = []; let uroPlaceWatchObjects = []; let uroFriendlyAreaNames = []; let uroAreaNameHoverTime = -1; let uroAreaNameHoverObj = null; let uroAreaNameOverlayShown = false; let uroANEditHovered = false; let uroANEditBox = null; let uroPrevMouseX = -1; let uroPrevMouseY = -1; let uroMousedOverMapComment = null; let uroMousedOverOtherObjectWithinMapComment = false; let uroLastZoom = -1; let uroTBRObj = null; let uroRTCObjs = null; let uroBackfillQueue = []; let uroUnstackedMasterID = null; let uroStackList = []; let uroStackType = null; let uroMousedOverMarkerID = null; let uroMousedOverMarkerType = null; let uroClickedOnMarkerID = null; let uroClickedOnMarkerType = null; let uroClickedOnMarkerForCenterInterceptor = null; let uroClickedOnMarkerMapCenter = null; let uroNodeLayerScanAttempts = 0; let uroAlertBoxStack = []; let uroAlertBoxTickAction = null; let uroAlertBoxCrossAction = null; let uroAlertBoxInUse = false; let uroMainTickHandlerID = null; let uroMainTickStage = 0; let uroSettingsApplied = false; let uroInhibitURFiltering = false; const uroCustomURTags = ['[ROADWORKS]','[CONSTRUCTION]','[CLOSURE]','[EVENT]','[NOTE]','[WSLM]','[BOG]','[DIFFICULT]']; const uroAltMarkers = [ // each altMarker has 4 variants: 0 = normal open, 1 = selected open, 2 = normal closed, 3 = selected closed // 0: closure UR [ "", "", "", "" ], // 1: roadworks UR [ "", "", "", "" ], // 2: custom keyword UR [ "", "", "", "" ], // 3: note UR [ "", "", "", "" ], // 4: event UR [ "", "", "", "" ], // 5: WMSL/SLUR UR [ "", "", "", "" ], // 6: Elgin MP [ "", "", "", "" ], // 7: TrafficCast MP [ "", "", "", "" ], // 8: TrafficMaster MP [ "", "", "", "" ], // 9: CalTrans [ '', '', '', '' ], // 10: TfL [ '', '', '', '' ], // 11: BOG [ '', '', '', '' ], // 12: Difficult turn [ '', '', '', '' ] ]; const uroMarkers = [ // 0 = comment count circle [""], // 1 = green comment marker [""], // 2 = yellow (own) comment marker [""] ]; const uroHighlightedCameraImages = [ // speed [""], // dummy [""], // rlc [""] ]; const URO_MAX_TABS_PER_ROW = 6; const URO_TABS_ID = { URS: 0, MPS: 1, MCS: 2, RTCS: 3, RAS: 4, PLACES: 5, CAMS: 6, OWL: 7, MISC: 8 }; const URO_TABS_FIELD = { TABHEADER: 0, TABBODY: 1, LINKID: 2, TABTITLE: 3, SHOWFN: 4, CLICKFN: 5, STORAGE: 6, POPULATEFN: 7 }; var uroCtrlTabs = [ ['_tabURs', null, '_linkURs', 'URs', uroShowURsTab, uroFilterItems_URsTabClick, 'UROverviewUROptions', uroPopulateURTab], ['_tabMPs', null, '_linkMPs', 'MPs', uroShowMPsTab, uroFilterItems_MPsTabClick, 'UROverviewMPOptions', uroPopulateMPTab], ['_tabMCs', null, '_linkMCs', 'MCs', uroShowMCsTab, uroFilterItems_MCsTabClick, 'UROverviewMCOptions', uroPopulateMCTab], ['_tabRTCs', null, '_linkRTCs', 'RTCs', uroShowRTCsTab, uroFilterItems_RTCsTabClick, 'UROverviewRTCOptions', uroPopulateRTCTab], ['_tabRAs', null, '_linkRAs', 'RAs', uroShowRAsTab, uroFilterItems_RAsTabClick, 'UROverviewRAOptions', uroPopulateRATab], ['_tabPlaces', null, '_linkPlaces', 'Places', uroShowPlacesTab, uroFilterItems_PlacesTabClick, 'UROverviewPlacesOptions', uroPopulatePlacesTab], ['_tabCams', null, '_linkCams', 'Cams', uroShowCamsTab, uroFilterItems_CamsTabClick, 'UROverviewCameraOptions', uroPopulateCamsTab], ['_tabOWL', null, '_linkOWL', 'OWL', uroShowOWLTab, null, null, null], ['_tabMisc', null, '_linkMisc', 'Misc', uroShowMiscTab, uroFilterItems_MiscTabClick, 'UROverviewMiscOptions', uroPopulateMiscTab] ]; function uroModifyHTML(htmlIn) { if(typeof trustedTypes === "undefined") { return htmlIn; } else { const escapeHTMLPolicy = trustedTypes.createPolicy("forceInner", {createHTML: (to_escape) => to_escape}); return escapeHTMLPolicy.createHTML(htmlIn); } } function uroPopulateMPTab() { 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>'; tHTML += '<br><b>Show MPs based on start/end dates:</b><br>'; tHTML += '<input type="checkbox" id="_cbMPFilterStartDate">Start</input> '; tHTML += '<input type="number" min="1" max="31" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterStartDay"> / '; tHTML += '<input type="number" min="1" max="12" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterStartMonth"> / '; tHTML += '<input type="number" min="2010" max="2100" size="4" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterStartYear"><br>'; tHTML += '<input type="checkbox" id="_cbMPFilterEndDate">End</input> '; tHTML += '<input type="number" min="1" max="31" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterEndDay"> / '; tHTML += '<input type="number" min="1" max="12" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterEndMonth"> / '; tHTML += '<input type="number" min="2010" max="2100" size="4" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMPFilterEndYear"><br>'; tHTML += '<input type="checkbox" id="_cbMPFilterEndDatePassed">End date in the past</input>'; return tHTML; } function uroPopulatePlacesTab() { 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="_cbPURFilterInsideManagedAreas">Ones within AM areas</input>'; tHTML += ' (<input type="checkbox" id="_cbPURExcludeUserArea">except my area)</input><br>'; tHTML += '<i>Requires Area Manager layer to be enabled</i><br>'; tHTML += '<input type="checkbox" id="_cbFilterLockRankedPlaceUpdates">Ones with non-zero lockRanks</input><br>'; tHTML += '<input type="checkbox" id="_cbFilterNewPlacePUR">Ones for new places</input><br>'; tHTML += '<input type="checkbox" id="_cbFilterUpdatedDetailsPUR">Ones for updated place details</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFPhone">Phone number</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFName">Name</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFEntryExitPoints">Entry//exit points</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFOpeningHours">Opening hours</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFAliases">Aliases</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFServices">Services</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFGeometry">Geometry</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFHouseNumber">House number</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFCategories">Categories</input><br>'; tHTML += ' <input type="checkbox" id="_cbPURFilterCFDescription">Description</input><br>'; tHTML += '<input type="checkbox" id="_cbFilterNewPhotoPUR">Ones for new photos</input><br>'; tHTML += '<input type="checkbox" id="_cbFilterFlaggedPUR">Ones flagged for attention</input><br>'; tHTML += '<br><input type="checkbox" id="_cbInvertPURFilters">Invert PUR filters</input><br>'; tHTML += '<br><b>Filter PURs by severity:</b><br>'; tHTML += '<input type="checkbox" id="_cbPURFilterLowSeverity">Low</input> '; 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="_cbHideDescribedPlaces" pairedWith="_cbHideNonDescribedPlaces">Hide or </input>'; tHTML += '<input type="checkbox" id="_cbHideNonDescribedPlaces" pairedWith="_cbHideDescribedPlaces">show ones with descriptive text</input><br>'; tHTML += '<input type="checkbox" id="_cbHideKeywordPlaces" pairedWith="_cbHideNoKeywordPlaces">Hide or </input>'; tHTML += '<input type="checkbox" id="_cbHideNoKeywordPlaces" pairedWith="_cbHideKeywordPlaces">show ones with a name including</input><br>'; tHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPlace"><br>'; tHTML += '<br><b>Show Places touched by a specific editor:</b><br>'; tHTML += '<input type="checkbox" id="_cbShowOnlyPlacesCreatedBy">Created by</input> / '; tHTML += '<input type="checkbox" id="_cbShowOnlyPlacesEditedBy">edited by</input><br>'; tHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textPlacesEditor"><br>'; tHTML += '<select id="_selectPlacesUserID" style="width:80%; height:22px;"></select><br>'; tHTML += '<br><b>Hide Places touched by a specific editor:</b><br>'; tHTML += '<input type="checkbox" id="_cbHideOnlyPlacesCreatedBy">Created by</input> / '; tHTML += '<input type="checkbox" id="_cbHideOnlyPlacesEditedBy">edited by</input><br>'; tHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textHidePlacesEditor"><br>'; tHTML += '<select id="_selectHidePlacesUserID" style="width:80%; height:22px;"></select><br>'; tHTML += '<br><br><b>Filter Places by category:</b><br>'; tHTML += '<input type="checkbox" id="_cbLeavePURGeos" pairedWith="_cbHidePURsForFilteredPlaces">Keep place visible if linked PUR is hidden, or</input><br>'; tHTML += '<input type="checkbox" id="_cbHidePURsForFilteredPlaces" pairedWith="_cbLeavePURGeos">Hide PURs linked to hidden places</input><br><br>'; 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>'; return tHTML; } function uroPopulateURTab() { var iHTML = ''; iHTML = '<br>'; iHTML += '<input type="checkbox" id="_cbURFilterOutsideArea">Hide URs outside my editable area</input><br>'; iHTML += '<input type="checkbox" id="_cbURFilterInsideManagedAreas">Hide URs within AM areas</input>'; iHTML += ' (<input type="checkbox" id="_cbURExcludeUserArea">except my area)</input><br>'; iHTML += ' <i>Requires Area Manager layer to be enabled</i><br>'; iHTML += '<input type="checkbox" id="_cbNoFilterForURInURL">Don\'t filter selected UR</input><br><br>'; iHTML += '<input type="checkbox" id="_cbURFilterDupes">Show only duplicate URs</input><br><br>'; iHTML += '<b>Filter by type:</b><br>'; iHTML += '<input type="checkbox" id="_cbFilterWazeAuto">Waze Automatic</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterIncorrectTurn">Incorrect turn</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterIncorrectAddress">Incorrect address</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterIncorrectRoute">Incorrect route</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterMissingRoundabout">Missing roundabout</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterGeneralError">General error</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterTurnNotAllowed">Turn not allowed</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterIncorrectJunction">Incorrect junction</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterMissingBridgeOverpass">Missing bridge overpass</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterWrongDrivingDirection">Wrong driving direction</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterMissingExit">Missing exit</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterMissingRoad">Missing road</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterBlockedRoad">Blocked road</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterMissingLandmark">Missing Landmark</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterSpeedLimits">Missing or Invalid Speed limit</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterUndefined">Undefined</input><br>'; iHTML += ' <i>Specially tagged types</i><br>'; iHTML += ' <input type="checkbox" id="_cbFilterRoadworks">[ROADWORKS]</input><br>'; iHTML += ' <input type="checkbox" id="_cbFilterConstruction">[CONSTRUCTION]</input><br>'; iHTML += ' <input type="checkbox" id="_cbFilterClosure">[CLOSURE]</input><br>'; iHTML += ' <input type="checkbox" id="_cbFilterEvent">[EVENT]</input><br>'; iHTML += ' <input type="checkbox" id="_cbFilterNote">[NOTE]</input><br>'; iHTML += ' <input type="checkbox" id="_cbFilterBOG">[BOG]</input><br>'; iHTML += ' <input type="checkbox" id="_cbFilterDifficult">[DIFFICULT]</input><br>'; iHTML += ' <input type="checkbox" id="_cbFilterWSLM">[WSLM]</input><br><br>'; iHTML += '<input type="checkbox" id="_cbInvertURFilter">Invert operation of type filters?</input><br>'; iHTML += '<hr>'; iHTML += '<br><b>Hide by state:</b><br>'; iHTML += '<input type="checkbox" id="_cbFilterOpenUR">Open</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterClosedUR">Closed</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterSolved">Solved</input><br>'; iHTML += '<input type="checkbox" id="_cbFilterUnidentified">Not identified</input><br><br>'; iHTML += '<br><b>Filter by age of submission:</b><br>'; iHTML += '<input type="checkbox" id="_cbEnableMinAgeFilter">Hide URs less than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinDays"> days old<br>'; iHTML += '<input type="checkbox" id="_cbEnableMaxAgeFilter">Hide URs more than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxDays"> days old<br>'; iHTML += '<br><b>Filter by description/comments/following:</b><br>'; iHTML += '<input type="checkbox" id="_cbHideMyFollowed" pairedWith="_cbHideMyUnfollowed">Ones I am or </input>'; iHTML += '<input type="checkbox" id="_cbHideMyUnfollowed" pairedWith="_cbHideMyFollowed">am not following</input><br><br>'; iHTML += '<input type="checkbox" id="_cbURDescriptionMustBePresent" pairedWith="_cbURDescriptionMustBeAbsent">Hide</input> or '; iHTML += '<input type="checkbox" id="_cbURDescriptionMustBeAbsent" pairedWith="_cbURDescriptionMustBePresent">show</input> URs with no description<br>'; iHTML += '<input type="checkbox" id="_cbEnableKeywordMustBePresent">Hide URs not including </input>'; iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPresent"><br>'; iHTML += '<input type="checkbox" id="_cbEnableKeywordMustBeAbsent">Hide URs including </input>'; iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordAbsent"><br>'; iHTML += ' <input type="checkbox" id="_cbCaseInsensitive"><i>Case-insensitive matches?</i></input><br><br>'; iHTML += 'With comments from me?<br>'; iHTML += '<input type="checkbox" id="_cbHideMyComments" pairedWith="_cbHideAnyComments">Yes </input>'; iHTML += '<input type="checkbox" id="_cbHideAnyComments" pairedWith="_cbHideMyComments">No</input><br>'; iHTML += 'If last comment made by me?<br>'; iHTML += '<input type="checkbox" id="_cbHideIfLastCommenter" pairedWith="_cbHideIfNotLastCommenter">Yes </input>'; iHTML += '<input type="checkbox" id="_cbHideIfNotLastCommenter" pairedWith="_cbHideIfLastCommenter">No </input><br>'; iHTML += 'If last comment made by UR reporter?<br>'; iHTML += '<input type="checkbox" id="_cbHideIfReporterLastCommenter" pairedWith="_cbHideIfReporterNotLastCommenter">Yes </input>'; iHTML += '<input type="checkbox" id="_cbHideIfReporterNotLastCommenter" pairedWith="_cbHideIfReporterLastCommenter">No</input><br>'; iHTML += '<input type="checkbox" id="_cbEnableMinCommentsFilter">With less than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinComments"> comments<br>'; iHTML += '<input type="checkbox" id="_cbEnableMaxCommentsFilter">With more than </input>'; iHTML += '<input type="number" min="0" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxComments"> comments<br><br>'; iHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter2">Last comment less than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays2"> days ago<br>'; iHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter">Last comment more than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays"> days ago<br>'; iHTML += ' <input type="checkbox" id="_cbIgnoreOtherEditorComments"><i>Ignore other editor comments?</i></input><br><br>'; iHTML += '<input type="checkbox" id="_cbURUserIDFilter">Without comments from user</input>'; iHTML += '<select id="_selectURUserID" style="width:80%; height:22px;"></select><br>'; iHTML += '<input type="checkbox" id="_cbURResolverIDFilter">Not resolved by user</input>'; iHTML += '<select id="_selectURResolverID" style="width:80%; height:22px;"></select>'; iHTML += '<br><br><input type="checkbox" id="_cbInvertURStateFilter">Invert operation of state/age filters?</input><br>'; iHTML += '<input type="checkbox" id="_cbNoFilterForTaggedURs">Don\'t apply state/age filters to tagged URs</input><br>'; return iHTML; } function uroPopulateMCTab() { var iHTML = ''; iHTML = '<br>'; iHTML += ' <i>Specially tagged types</i><br>'; iHTML += ' <input type="checkbox" id="_cbMCFilterRoadworks">[ROADWORKS]</input><br>'; iHTML += ' <input type="checkbox" id="_cbMCFilterConstruction">[CONSTRUCTION]</input><br>'; iHTML += ' <input type="checkbox" id="_cbMCFilterClosure">[CLOSURE]</input><br>'; iHTML += ' <input type="checkbox" id="_cbMCFilterEvent">[EVENT]</input><br>'; iHTML += ' <input type="checkbox" id="_cbMCFilterNote">[NOTE]</input><br>'; iHTML += ' <input type="checkbox" id="_cbMCFilterBOG">[BOG]</input><br>'; iHTML += ' <input type="checkbox" id="_cbMCFilterDifficult">[DIFFICULT]</input><br>'; iHTML += ' <input type="checkbox" id="_cbMCFilterWSLM">[WSLM]</input><br><br>'; iHTML += '<input type="checkbox" id="_cbInvertMCFilter">Invert operation of type filters?</input><br>'; iHTML += '<hr>'; iHTML += '<br><b>Filter by description/comments/following:</b><br>'; iHTML += '<input type="checkbox" id="_cbMCHideMyFollowed" pairedWith="_cbMCHideMyUnfollowed">Ones I am or </input>'; iHTML += '<input type="checkbox" id="_cbMCHideMyUnfollowed" pairedWith="_cbMCHideMyFollowed">am not following</input><br><br>'; iHTML += '<input type="checkbox" id="_cbMCDescriptionMustBePresent" pairedWith="_cbMCDescriptionMustBeAbsent">Hide</input> or '; iHTML += '<input type="checkbox" id="_cbMCDescriptionMustBeAbsent" pairedWith="_cbMCDescriptionMustBePresent">show</input> MCs with no description<br>'; iHTML += '<input type="checkbox" id="_cbMCCommentsMustBePresent" pairedWith="_cbMCCommentsMustBeAbsent">Hide</input> or '; iHTML += '<input type="checkbox" id="_cbMCCommentsMustBeAbsent" pairedWith="_cbMCCommentsMustBePresent">show</input> MCs with no comments<br>'; iHTML += '<input type="checkbox" id="_cbMCExpiryMustBePresent" pairedWith="_cbMCExpiryMustBeAbsent">Hide</input> or '; iHTML += '<input type="checkbox" id="_cbMCExpiryMustBeAbsent" pairedWith="_cbMCExpiryMustBePresent">show</input> MCs with no expiry date<br>'; iHTML += '<input type="checkbox" id="_cbMCEnableKeywordMustBePresent">Hide MCs not including </input>'; iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textMCKeywordPresent"><br>'; iHTML += '<input type="checkbox" id="_cbMCEnableKeywordMustBeAbsent">Hide MCs including </input>'; iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textMCKeywordAbsent"><br>'; iHTML += ' <input type="checkbox" id="_cbMCCaseInsensitive"><i>Case-insensitive matches?</i></input><br>'; iHTML += '<input type="checkbox" id="_cbMCCreatorIDFilter">Show MCs created by user</input>'; iHTML += '<select id="_selectMCCreatorID" style="width:80%; height:22px;"></select><br>'; iHTML += '<br><input type="checkbox" id="_cbHideWRCMCs"><b>Hide Waze_roadclosures MCs</b></input><br>'; iHTML += '<br><b>Hide MCs with lock level:</b><br>'; iHTML += ' <input type="checkbox" id="_cbHideMCRank0">L1</input>'; iHTML += ' <input type="checkbox" id="_cbHideMCRank1">L2</input>'; iHTML += ' <input type="checkbox" id="_cbHideMCRank2">L3</input>'; iHTML += ' <input type="checkbox" id="_cbHideMCRank3">L4</input>'; iHTML += ' <input type="checkbox" id="_cbHideMCRank4">L5</input>'; iHTML += ' <input type="checkbox" id="_cbHideMCRank5">L6</input>'; iHTML += '<hr>'; iHTML += '<input type="checkbox" id="_cbMCEnhancePointMCVisibility">Enhance visibility of point MCs</input>'; return iHTML; } function uroPopulateCamsTab() { var iHTML = ''; iHTML = '<br><b>Show Cameras created by:</b><br>'; iHTML += '<input type="checkbox" id="_cbShowWorldCams" checked>world_* users</input><br>'; iHTML += '<input type="checkbox" id="_cbShowUSACams" checked>usa_* users</input><br>'; iHTML += '<input type="checkbox" id="_cbShowNonWorldCams" checked>other users</input><br>'; iHTML += '<br><b>Show Cameras touched by a specific editor:</b><br>'; iHTML += '<input type="checkbox" id="_cbShowOnlyCamsCreatedBy">Created by</input> / '; iHTML += '<input type="checkbox" id="_cbShowOnlyCamsEditedBy">edited by</input><br>'; iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textCameraEditor"><br>'; iHTML += '<select id="_selectCameraUserID" style="width:80%; height:22px;"></select><br>'; iHTML += '<br><input type="checkbox" id="_cbShowOnlyMyCams">Show ONLY cameras created/edited by me</input><br>'; iHTML += '<br><b>Show Cameras by type:</b><br>'; iHTML += '<input type="checkbox" id="_cbShowSpeedCams" checked>Speed</input><br>'; iHTML += ' <input type="checkbox" id="_cbShowIfSpeedSet" checked> with speed data</input><br>'; iHTML += ' <input type="checkbox" id="_cbShowIfNoSpeedSet" checked> with no speed data</input><br>'; iHTML += ' <input type="checkbox" id="_cbShowIfInvalidSpeedSet" checked> with invalid speed data (zoom 16+)</input><br>'; iHTML += '<input type="checkbox" id="_cbShowRedLightCams" checked>Red Light</input><br>'; iHTML += ' <input type="checkbox" id="_cbShowRLCIfZeroSpeedSet" checked> with speed limit = 0</input><br>'; iHTML += ' <input type="checkbox" id="_cbShowRLCIfNonZeroSpeedSet" checked> with speed limit > 0</input><br>'; iHTML += ' <input type="checkbox" id="_cbShowRLCIfNoSpeedSet" checked> with no speed data</input><br>'; iHTML += '<input type="checkbox" id="_cbShowDummyCams" checked>Dummy</input><br>'; iHTML += '<br><b>Hide Cameras by creator:</b><br>'; iHTML += ' <input type="checkbox" id="_cbHideCreatedByMe">me</input>'; iHTML += ' <input type="checkbox" id="_cbHideCreatedByRank0">L1</input>'; iHTML += ' <input type="checkbox" id="_cbHideCreatedByRank1">L2</input>'; iHTML += ' <input type="checkbox" id="_cbHideCreatedByRank2">L3</input>'; iHTML += ' <input type="checkbox" id="_cbHideCreatedByRank3">L4</input>'; iHTML += ' <input type="checkbox" id="_cbHideCreatedByRank4">L5</input>'; iHTML += ' <input type="checkbox" id="_cbHideCreatedByRank5">L6</input>'; iHTML += '<br><b>Hide Cameras by updater:</b><br>'; iHTML += ' <input type="checkbox" id="_cbHideUpdatedByMe">me</input>'; iHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank0">L1</input>'; iHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank1">L2</input>'; iHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank2">L3</input>'; iHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank3">L4</input>'; iHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank4">L5</input>'; iHTML += ' <input type="checkbox" id="_cbHideUpdatedByRank5">L6</input>'; iHTML += '<br><br><b><input type="checkbox" id="_cbHideCWLCams">Hide cameras on watchlist</input></b><br>'; iHTML += '<br><br><b><input type="checkbox" id="_cbInvertCamFilters">Invert operation of camera filters?</input></b><br>'; iHTML += '<br><br><b><input type="checkbox" id="_cbHighlightInsteadOfHideCams">Highlight instead of hide</input></b><br>'; return iHTML; } function uroPopulateRTCTab() { var iHTML = ''; iHTML = '<br><b>Hide Road Closures:</b><br>'; // Hidden checkbox to avoid errors when applying settings from previous versions of the script where this was an active control... iHTML += '<input type="checkbox" id="_cbHideUserRTCs" style="display: none;" />'; iHTML += '<table style="text-align:center;">'; iHTML += '<tr><td/><td><div class="map-marker road-closure status-finished" style="margin-left:0px;margin-top:0px;" /></td><td><div class="map-marker road-closure status-active" style="margin-left:0px;margin-top:0px;" /></td><td><div class="map-marker road-closure status-not-started" style="margin-left:0px;margin-top:0px;" /></td></tr>'; iHTML += '<tr><td>From WME</td><td><input type="checkbox" id="_cbHideExpiredEditorRTCs" /></td><td><input type="checkbox" id="_cbHideEditorRTCs" /></td><td><input type="checkbox" id="_cbHideFutureEditorRTCs" /></td></tr>'; iHTML += '<tr><td>From WazeFeed</td><td><input type="checkbox" id="_cbHideExpiredWazeFeedRTCs" /></td><td><input type="checkbox" id="_cbHideWazeFeedRTCs" /></td><td><input type="checkbox" id="_cbHideFutureWazeFeedRTCs" /></td></tr>'; iHTML += '<tr><td>From Staff</td><td><input type="checkbox" id="_cbHideExpiredWazeRTCs" /></td><td><input type="checkbox" id="_cbHideWazeRTCs" /></td><td><input type="checkbox" id="_cbHideFutureWazeRTCs" /></td></tr>'; iHTML += '</table><br>'; iHTML += '<input type="checkbox" id="_cbShowMTERTCs" pairedWith="_cbHideMTERTCs">Show</input> or '; iHTML += '<input type="checkbox" id="_cbHideMTERTCs" pairedWith="_cbShowMTERTCs">hide RTCs associated with MTE: </input>'; iHTML += '<select id="_selectRTCMTE" style="width:80%; height:22px;"></select><br>'; iHTML += '<br>'; iHTML += 'Hide if:<br>'; iHTML += '<input type="checkbox" id="_cbEnableRTCDurationFilterLessThan">Duration less than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterRTCDurationLessThan"> days<br>'; iHTML += '<input type="checkbox" id="_cbEnableRTCDurationFilterMoreThan">Duration more than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterRTCDurationMoreThan"> days<br>'; iHTML += '<br><b>Filter by date/time:</b><br>'; iHTML += '<input type="checkbox" id="_cbRTCFilterShowForTS" pairedWith="_cbRTCFilterHideForTS">Show</input> or '; iHTML += '<input type="checkbox" id="_cbRTCFilterHideForTS" pairedWith="_cbRTCFilterShowForTS">hide</input> RTCs active at<br>'; iHTML += 'Date: <input type="number" min="1" max="31" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterDay"> / '; iHTML += '<input type="number" min="1" max="12" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterMonth"> / '; iHTML += '<input type="number" min="2010" max="2100" size="4" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterYear"><br>'; iHTML += 'Time: <input type="number" min="0" max="23" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterHour">:'; iHTML += '<input type="number" min="0" max="59" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputRTCFilterMin">'; return iHTML; } function uroPopulateRATab() { var iHTML = ''; iHTML = '<br><b>Filter Restricted Areas:</b><br>'; iHTML += '<input type="checkbox" id="_cbShowSpecificRA">Show a specific area: </input>'; iHTML += '<select id="_selectRA" style="width:80%; height:22px;"></select><br><br>'; iHTML += '<input type="checkbox" id="_cbRAEditorIDFilter">Show areas edited by user: </input>'; iHTML += '<select id="_selectRAEditorID" style="width:80%; height:22px;"></select><br><br>'; iHTML += 'Hide if:<br>'; iHTML += '<input type="checkbox" id="_cbEnableRAAgeFilterLessThan">Last modified less than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterRAAgeLessThan"> days ago<br>'; iHTML += '<input type="checkbox" id="_cbEnableRAAgeFilterMoreThan">Last modified more than </input>'; iHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterRAAgeMoreThan"> days ago<br>'; return iHTML; } function uroPopulateMiscTab() { var iHTML = ''; iHTML += '<br><b><input type="checkbox" id="_cbHideSegmentsWhenRoadsHidden" />Hide segment layer when road layer is hidden</b><br>'; iHTML += '<br><b><input type="checkbox" id="_cbKillInertialPanning" />Stop inertial panning when mouse moves out of map area</b><br>'; iHTML += '<br><br><b>Use default conversation markers:</b><br>'; iHTML += '<input type="checkbox" id="_cbNativeConvoMarkers" checked />in public WME<br>'; iHTML += '<input type="checkbox" id="_cbNativeBetaConvoMarkers" checked />in beta WME<br>'; iHTML += '<br><br><b><input type="checkbox" id="_cbCommentCount" />Show comment count on UR markers</b><br>'; // Hidden checkbox to avoid errors when applying settings from previous versions of the script where this was an active control... iHTML += '<input type="checkbox" style="display: none;" id="_cbEnableDeleteFeedEntries" />'; iHTML += '<br><br><b><input type="checkbox" id="_cbAutoApplyClonedClosure" />Auto-apply cloned closures</b><br>'; iHTML += '<b><input type="checkbox" id="_cbAutoScrollClosureList" />Auto-scroll to end of closures</b><br>'; iHTML += '<br><br><b><input type="checkbox" id="_cbURBackfill" />Backfill UR data</b><br>'; iHTML += '<br><br><b>Disable filtering above zoom level </b>'; iHTML += '<input type="number" min="12" max="22" value="22" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinZoomLevel" /><br>'; iHTML += '<br><br><b>Marker Unstacking:</b><br>'; iHTML += 'Distance threshold: <input type="number" min="1" max="30" value="15" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackSensitivity" /><br>'; iHTML += 'Disable below zoom: <input type="number" min="12" max="22" value="15" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackZoomLevel" /><br>'; iHTML += '<br><br><b>Use custom marker for URs tagged as:</b><br>'; iHTML += '<input type="checkbox" id="_cbCustomRoadworksMarkers" />[ROADWORKS]<br>'; iHTML += '<input type="checkbox" id="_cbCustomConstructionMarkers" />[CONSTRUCTION]<br>'; iHTML += '<input type="checkbox" id="_cbCustomClosuresMarkers" />[CLOSURE]<br>'; iHTML += '<input type="checkbox" id="_cbCustomEventsMarkers" />[EVENT]<br>'; iHTML += '<input type="checkbox" id="_cbCustomNotesMarkers" />[NOTE]<br>'; iHTML += '<input type="checkbox" id="_cbCustomBOGMarkers" />[BOG]<br>'; iHTML += '<input type="checkbox" id="_cbCustomDifficultMarkers" />[DIFFICULT]<br>'; iHTML += '<input type="checkbox" id="_cbCustomWSLMMarkers" />[WSLM]<br>'; iHTML += '<input type="checkbox" id="_cbCustomNativeSLMarkers" />Native speed limit reports<br>'; iHTML += '<input type="checkbox" id="_cbCustomKeywordMarkers" />'; iHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textCustomKeyword" /><br>'; iHTML += '<br><br><b>Use custom marker for MPs tagged as:</b><br>'; iHTML += '<input type="checkbox" id="_cbCustomElginMarkers" />[Elgin]<br>'; iHTML += '<input type="checkbox" id="_cbCustomTrafficMasterMarkers" />[TM]<br>'; iHTML += '<input type="checkbox" id="_cbCustomTrafficCastMarkers" />[TrafficCast]<br>'; iHTML += '<input type="checkbox" id="_cbCustomCaltransMarkers" />[Caltrans]<br>'; iHTML += '<input type="checkbox" id="_cbCustomTFLMarkers" />[TfL Open Data]<br>'; iHTML += '<br><br><b>Popup mouse behaviour:</b><br>'; iHTML += 'Mouse idle <input type="number" min="1" max="10" value="2" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupDwellTimeout" /> *100ms<br>'; iHTML += 'Mouse over <input type="number" min="1" max="10" value="2" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupEntryTimeout" /> *100ms<br>'; iHTML += 'Distance <input type="number" min="0" max="10" value="2" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMaxJitter" /> pixels<br>'; iHTML += 'Auto-hide after <input type="number" min="0" max="10" value="0" size="2" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupAutoHideTimeout" /> seconds<br>'; iHTML += '<br><br><b>Disable popup for:</b><br>'; iHTML += '<input type="checkbox" id="_cbInhibitURPopup" />URs<br>'; iHTML += '<input type="checkbox" id="_cbInhibitMPPopup" />MPs<br>'; iHTML += '<input type="checkbox" id="_cbInhibitCamPopup" />Cameras<br>'; iHTML += '<input type="checkbox" id="_cbInhibitSegPopup" />Segments<br>'; iHTML += ' <input type="checkbox" id="_cbInhibitSegGenericPopup" />Speed limit info<br>'; iHTML += '<input type="checkbox" id="_cbInhibitTurnsPopup" />Restricted Turns<br>'; iHTML += '<input type="checkbox" id="_cbInhibitLandmarkPopup" />Places<br>'; iHTML += '<input type="checkbox" id="_cbInhibitPUPopup" />Place Updates<br>'; iHTML += '<input type="checkbox" id="_cbInhibitMapCommentPopup" />Map Comments<br>'; iHTML += '<input type="checkbox" id="_cbInhibitNodesPopup" />Junction Nodes<br>'; iHTML += '<br><br><b>Date/Time formatting for popups:</b><br>'; iHTML += '<input type="checkbox" id="_cbDateFmtDDMMYY" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtYYMMDD" checked />day/month/year<br>'; iHTML += '<input type="checkbox" id="_cbDateFmtMMDDYY" pairedWith="_cbDateFmtDDMMYY,_cbDateFmtYYMMDD" />month/day/year<br>'; iHTML += '<input type="checkbox" id="_cbDateFmtYYMMDD" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtDDMMYY" />year/month/day<br><br>'; iHTML += '<input type="checkbox" id="_cbTimeFmt24H" pairedWith="_cbTimeFmt12H" checked />24 hour<br>'; iHTML += '<input type="checkbox" id="_cbTimeFmt12H" pairedWith="_cbTimeFmt24H" />12 hour<br><br>'; iHTML += '<i>Unticked uses browser default setting</i>'; iHTML += '<br><br><b><input type="checkbox" id="_cbWhiteBackground" />Use custom background colour</b><br>'; iHTML += 'R:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundRed" />'; iHTML += 'G:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundGreen" />'; iHTML += 'B:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundBlue" /><br>'; iHTML += '<br><br><b>Replace "Next ..." button with "Done" for:</b><br>'; iHTML += '<input type="checkbox" id="_cbInhibitNURButton" />URs<br>'; iHTML += '<input type="checkbox" id="_cbInhibitNMPButton" />MPs<br>'; iHTML += '<input type="checkbox" id="_cbInhibitNPURButton" />PURs<br>'; iHTML += '<br><br><b>Disable on-click recentering for:</b><br>'; iHTML += '<input type="checkbox" id="_cbInhibitURCentering" />URs<br>'; iHTML += '<input type="checkbox" id="_cbInhibitMPCentering" />MPs<br>'; iHTML += '<input type="checkbox" id="_cbInhibitPURCentering" />PURs<br>'; iHTML += '<input type="checkbox" id="_cbInhibitPPURCentering" />PPURs<br>'; iHTML += '<input type="checkbox" id="_cbInhibitRPURCentering" />RPURs<br>'; iHTML += '<br><br><b><input type="checkbox" id="_cbHideAMLayer" />Hide Area Manager polygons</b><br>'; iHTML += '<b><input type="checkbox" id="_cbMoveAMList" />Show AMs in topbar when AM layer is active</b><br>'; iHTML += '<br><b><input type="checkbox" id="_cbDisablePlacesFiltering" />Disable Places filtering</b><br>'; // Hidden checkbox to avoid errors when applying settings from previous versions of the script where this was an active control... iHTML += '<b><input type="checkbox" style="display: none;" id="_cbDisableTabStyling" />Use default tab styling</b><br>'; if(document.getElementById("user-details") !== null) { iHTML += '<b><input type="checkbox" id="_cbHideEditorInfo" />Hide sidebar editor info</b><br>'; } iHTML += '<br><br><b>Settings backup/restore/reset:</b><br>'; iHTML += '<input type="button" id="_btnSettingsToText" value="Backup" /> '; iHTML += '<input type="button" id="_btnTextToSettings" value="Restore" /> | '; iHTML += '<input type="button" id="_btnResetSettings" value="Reset" /><br><br>'; iHTML += '<textarea id="_txtSettings" value=""></textarea><br>'; iHTML += '<input type="button" id="_btnClearSettingsText" value="Clear" /><br>'; /* iHTML += '<br><br><b>Debug:</b><br>'; iHTML += '<input type="button" id="_btnDebugToScreen" value="Show debug data" />'; */ return iHTML; } /* function uroAddDebug(debugtext) { 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 = uroModifyHTML('<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() { // Auto-selects the "None" event in the closures tab to avoid the user having to manually select it for almost // every single closure they'll ever add - the only time you'd need to select anything other than None is when // adding a closure related to a MTE... var retval = false; var tObj = document.getElementById('closure_eventId'); // Make sure the closure event list is available, and that we haven't already messed with it. if((tObj !== null) && (tObj.tag != "touchedByURO")) { // The event dropdown is now some byzantine piece of DOM manipulation to generate something which looks like a // regular select list, but which can't be manipulated like one... The first gotcha is that the selected item // exists only within a shadow DOM section within the dropdown rather than simply being part of the list from // which we'd be able to read off its selected index. So to check whether or not the user has selected an // event already, we need to drill down into this shadow DOM to get its text contents, and compare those against // the I18n translation for the choose event text. What a palaver... var shadowElm = tObj.shadowRoot.querySelectorAll('.selected-value')[0]; if(shadowElm !== undefined) { var eventText = tObj.shadowRoot.querySelectorAll('.selected-value')[0].innerText; // Sometimes we get here before WME has finished rendering, so if the event text hasn't been set yet then we // need to return false and let the caller deal with it... if(eventText !== '') { if(eventText == I18n.lookup('closures.choose_event')) { // Having now established that, yes, the closure hasn't yet been associated with any event, it's surprisingly // easy to change it to "None" - we just generate a click event on the first child element in the main DOM (not // the shadow DOM this time), which replicates what the user would do to select None manually. tObj.children[0].click(); } // Tag the event list to prevent further processing attempts whilst this closure remains open. tObj.tag = "touchedByURO"; retval = true; } } } return retval; } function uroPerformanceMonitoring(source, ts) { if(uroPerformanceMonitoringOutput === true) { console.log(source+': '+(performance.now() - ts)); } } function uroAddLog(logtext) { if(uroShowDebugOutput) console.log('URO+: '+Date()+' '+logtext); } function uroGetCBChecked(cbID) { try { return(document.getElementById(cbID).checked); } catch(err) { uroAddLog('uroGetCBChecked() - '+cbID+' not found!'); return null; } } function uroSetCBChecked(cbID, state) { try { document.getElementById(cbID).checked = state; } catch(err) { uroAddLog('uroSetCBChecked() - '+cbID+' not found!'); } } function uroGetElmValue(elmID) { try { return(document.getElementById(elmID).value); } catch(err) { uroAddLog('uroGetElmValue() - '+elmID+' not found!'); return null; } } function uroSetStyleDisplay(elm,style) { try { if(typeof elm == 'object') { elm.style.display = style; } else { document.getElementById(elm).style.display = style; } } catch(err) { uroAddLog('uroSetStyleDisplay() - '+elm+' not found!'); } } function uroSetOnClick(elm,fn) { try { if(typeof elm == 'object') { elm.onclick = fn; } else { document.getElementById(elm).onclick = fn; } } catch(err) { uroAddLog('uroSetOnClick() - '+elm+' not found!'); } } function uroAddEventListener(elm,eventType,eventFn,eventBool) { try { document.getElementById(elm).addEventListener(eventType, eventFn, eventBool); } catch(err) { uroAddLog('uroAddEventListener() - '+elm+' not found!'); } } function uroAlertBoxObj(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction) { this.headericon = headericon; this.title = title; this.content = content; this.hasCross = hasCross; this.tickText = tickText; this.crossText = crossText; this.tickAction = tickAction; this.crossAction = crossAction; } function uroCloseAlertBox() { document.getElementById('uroAlerts').childNodes[0].innerHTML = uroModifyHTML(''); document.getElementById('uroAlerts').childNodes[1].innerHTML = uroModifyHTML(''); document.getElementById('uroAlertTickBtnCaption').innerHTML = uroModifyHTML(''); document.getElementById('uroAlertCrossBtnCaption').innerHTML = uroModifyHTML(''); uroAlertBoxTickAction = null; uroAlertBoxCrossAction = null; document.getElementById('uroAlerts').style.visibility = "hidden"; document.getElementById('uroAlertCrossBtn').style.visibility = "hidden"; uroAlertBoxInUse = false; if(uroAlertBoxStack.length > 0) { uroBuildAlertBoxFromStack(); } } function uroCloseAlertBoxWithTick() { if(typeof uroAlertBoxTickAction === 'function') { uroAlertBoxTickAction(); } uroCloseAlertBox(); } function uroCloseAlertBoxWithCross() { if(typeof uroAlertBoxCrossAction === 'function') { uroAlertBoxCrossAction(); } uroCloseAlertBox(); } function uroShowAlertBox(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction) { uroAlertBoxStack.push(new uroAlertBoxObj(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction)); if(uroAlertBoxInUse === false) { uroBuildAlertBoxFromStack(); } } function uroBuildAlertBoxFromStack() { uroAlertBoxInUse = true; uroAlertBoxTickAction = null; uroAlertBoxCrossAction = null; 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 = uroModifyHTML(titleContent); document.getElementById('uroAlerts').childNodes[1].innerHTML = uroModifyHTML(uroAlertBoxStack[0].content); document.getElementById('uroAlertTickBtnCaption').innerHTML = uroModifyHTML(uroAlertBoxStack[0].tickText); if(uroAlertBoxStack[0].hasCross) { document.getElementById('uroAlertCrossBtnCaption').innerHTML = uroModifyHTML(uroAlertBoxStack[0].crossText); document.getElementById('uroAlertCrossBtn').style.visibility = "visible"; if(typeof uroAlertBoxStack[0].crossAction === "function") { uroAlertBoxCrossAction = uroAlertBoxStack[0].crossAction; } } else { document.getElementById('uroAlertCrossBtn').style.visibility = "hidden"; } if(typeof uroAlertBoxStack[0].tickAction === "function") { uroAlertBoxTickAction = uroAlertBoxStack[0].tickAction; } document.getElementById('uroAlerts').style.visibility = ""; uroAlertBoxStack.shift(); } function uroFirstTimerWelcomePack() { uroAddLog('welcome new users to Club URO...'); // for now, just show the update notes... uroShowUpdateNotes(); } function uroShowUpdateNotes() { uroAddLog('let existing users know what\'s new in this release'); var releaseNotes = ''; releaseNotes += '<p>Thanks for upgrading to URO+ '+uroVersion+' ('+uroReleaseDate+'). What\'s changed?</p>'; var loop; if(uroChanges.length > 0) { releaseNotes += '<ul>'; for(loop=0; loop < uroChanges.length; loop++) { releaseNotes += '<li>'+uroChanges[loop]; } releaseNotes += '</ul>'; } else if(uroBetaChanges.length > 0) { if(!uroBetaEditor) { releaseNotes += '<ul><li>Nothing of interest, unless you\'re a WME beta tester, in which case log into the beta and find out...</ul>'; } } if((uroBetaEditor) && (uroBetaChanges.length > 0)) { releaseNotes += '<p>For WME Beta:<p>'; releaseNotes += '<ul>'; for(loop=0; loop < uroBetaChanges.length; loop++) { releaseNotes += '<li>'+uroBetaChanges[loop]; } releaseNotes += '</ul>'; } uroShowAlertBox('fa-info-circle', 'URO+ Release Notes', releaseNotes, false, "OK", "", null, null); } function uroAdvertiseCustomIcons() { uroAddLog('advertise the benefits of custom UR icons...'); var confirmMsg = ''; confirmMsg += '<p>Hi there. One of the features of URO+ that a lot of users find useful is the ability to use a custom marker for URs and MPs which have been tagged with a specific keyword in their description text.</p>'; confirmMsg += '<p>Markers are defined for <b>[ROADWORKS]</b>, <b>[CONSTRUCTION]</b>, <b>[CLOSURE]</b>, <b>[EVENT]</b>, <b>[NOTE]</b>, <b>[WSLM]</b>, <b>[BOG]</b> and <b>[DIFFICULT]</b> tags in URs, and <b>[TfL Open Data]</b>, <b>[Elgin]</b>, <b>[TM]</b>, <b>[TrafficCast]</b> and <b>[Caltrans]</b> in MPs.</p>'; confirmMsg += '<img src="'+uroAltMarkers[1][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[0][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[4][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[3][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[5][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[11][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[12][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[10][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[6][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[8][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[7][0]+'">'; confirmMsg += '<img src="'+uroAltMarkers[9][0]+'">'; confirmMsg += '<p style="clear:left;">Would you like me to automatically enable these custom markers?</p>'; confirmMsg += '<p>If you change your mind later on, they can be enabled/disabled via the Misc tab within the URO+ settings</p>'; uroShowAlertBox('fa-info-circle', 'URO+ Message to Users', confirmMsg, true, 'Yes please', 'No thanks', uroSetMarkerCBs, null); } function uroSetMarkerCBs() { uroSetCBChecked('_cbCustomRoadworksMarkers', true); uroSetCBChecked('_cbCustomConstructionMarkers', true); uroSetCBChecked('_cbCustomClosuresMarkers', true); uroSetCBChecked('_cbCustomEventsMarkers', true); uroSetCBChecked('_cbCustomNotesMarkers', true); uroSetCBChecked('_cbCustomBOGMarkers', true); uroSetCBChecked('_cbCustomDifficultMarkers', true); uroSetCBChecked('_cbCustomWSLMMarkers', true); uroSetCBChecked('_cbCustomNativeSLMarkers', true); uroSetCBChecked('_cbCustomElginMarkers', true); uroSetCBChecked('_cbCustomTrafficMasterMarkers', true); uroSetCBChecked('_cbCustomTrafficCastMarkers', true); uroSetCBChecked('_cbCustomCaltransMarkers', true); uroSetCBChecked('_cbCustomTFLMarkers', true); } function uroGatherSettings(container) { let options = ''; if(typeof(container) == 'string') { container = document.getElementById(container); } let urOptions = container.getElementsByTagName('input'); for (let optIdx=0;optIdx<urOptions.length;optIdx++) { // Don't save settings for any of the legacy input elements we've now hidden... if(urOptions[optIdx].style.display != "none") { let id = urOptions[optIdx].id; if((id.indexOf('_cb') === 0)||(id.indexOf('_text') === 0)||(id.indexOf('_input') === 0)) { options += ':' + id; if(urOptions[optIdx].type == 'checkbox') options += ',' + urOptions[optIdx].checked.toString(); else if((urOptions[optIdx].type == 'text')||(urOptions[optIdx].type == 'number')) options += ',' + urOptions[optIdx].value.toString(); } } } return options; } function uroGatherCamWatchList() { 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.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) || (uroMTEMode === true) || (uroSettingsApplied === false)) { uroAddLog('save inhibited'); return; } if (localStorage) { try { for(var i = 0; i < uroCtrlTabs.length; ++i) { localStorage[uroCtrlTabs[i][URO_TABS_FIELD.STORAGE]] = uroGatherSettings(uroCtrlTabs[i][URO_TABS_FIELD.TABBODY]); } 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'); } catch(err) { uroAddLog('exception thrown during save - probably script reload whilst in MTE mode...'); } } else { uroAddLog('no localStorage, save blocked'); } } function uroApplySettings(settings) { uroSettingsApplied = true; if(settings != undefined) { if(document.querySelector('#_cbMasterEnable') === null) { uroSettingsApplied = false; } else { 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 == 9) { // CWL entries with 9 fields include the validated property which is now redundant, so we need to strip this property before adding // the camera to the object collection. Whilst WME no longer displays unapproved cameras, it's preferable at this stage to leave // any watched unapproved cameras in the object collection, just in case any of them were approved (and will therefore still be // present in WME) inbetween the last time the user ran URO and now. For those unapproved cameras which were still unapproved when // removed from WME, URO will then list them as deleted and the user can then perform a single manual tidy-up of their watchlist to // remove them there as well. uroCamWatchObjects.push(new uroCamWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[7],fields[8])); } else if(fields.length == 8) { uroCamWatchObjects.push(new uroCamWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7])); } } } } /* function 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 uroTranslateLegacyZoomLevels() { var tZoom = parseInt(document.getElementById("_inputFilterMinZoomLevel").value); if(tZoom < 12) { tZoom += 12; document.getElementById("_inputFilterMinZoomLevel").value = tZoom; } tZoom = parseInt(document.getElementById("_inputUnstackZoomLevel").value); if(tZoom < 12) { tZoom += 12; document.getElementById("_inputUnstackZoomLevel").value = tZoom; } } function uroLoadSettings() { var isNewInstall = true; var isUpgradeInstall = true; var notifyAboutCustomIcons = true; uroAddLog('loadSettings()'); for(var i = 0; i < uroCtrlTabs.length; ++i) { if (uroCtrlTabs[i][URO_TABS_FIELD.STORAGE] != null) { uroAddLog('recover '+uroCtrlTabs[i][URO_TABS_FIELD.TABTITLE]+' tab settings'); uroApplySettings(localStorage[uroCtrlTabs[i][URO_TABS_FIELD.STORAGE]]); isNewInstall = false; } } if(localStorage.UROverviewCurrentVersion != null) { notifyAboutCustomIcons = false; } else { if(uroGetCBChecked('_cbCustomRoadworksMarkers') === true) notifyAboutCustomIcons = false; if(uroGetCBChecked('_cbCustomConstructionMarkers')=== true) notifyAboutCustomIcons = false; if(uroGetCBChecked('_cbCustomClosuresMarkers') === true) notifyAboutCustomIcons = false; if(uroGetCBChecked('_cbCustomEventsMarkers') === true) notifyAboutCustomIcons = false; if(uroGetCBChecked('_cbCustomNotesMarkers') === true) notifyAboutCustomIcons = false; if(uroGetCBChecked('_cbCustomWSLMMarkers') === true) notifyAboutCustomIcons = false; if(uroGetCBChecked('_cbCustomBOGMarkers') === true) notifyAboutCustomIcons = false; if(uroGetCBChecked('_cbCustomDifficultMarkers') === true) notifyAboutCustomIcons = false; if(uroGetCBChecked('_cbCustomNativeSLMarkers') === true) notifyAboutCustomIcons = false; } if (localStorage.UROverviewMPOptions != null) { uroTranslateLegacyMPTabSettings(); } if (localStorage.UROverviewMiscOptions != null) { uroTranslateLegacyZoomLevels(); } if(localStorage.UROverviewCWLGroups != null) { uroAddLog('recover CWL groups'); uroApplyCWLGroups(); isNewInstall = false; } else { uroAddLog('set default CWL group'); uroCWLGroups.push(new uroOWLGroupObj(0,'No group',false)); } if(localStorage.UROverviewCamWatchList != null) { uroAddLog('recover camera watchlist'); uroApplyCamWatchList(); uroGetCurrentCamWatchListObjects(); isNewInstall = false; } /* if(localStorage.UROverviewSegWatchList != null) { uroAddLog('recover segment watchlist'); uroApplySegWatchList(); uroGetCurrentSegWatchListObjects(); isNewInstall = false; } if(localStorage.UROverviewPlaceWatchList != null) { uroAddLog('recover places watchlist'); uroApplyPlaceWatchList(); //uroGetCurrentPlaceWatchListObjects(); isNewInstall = false; } if(localStorage.UROverviewPlacesGroups != null) { uroAddLog('recover places groups'); uroApplyPlacesGroups(); isNewInstall = false; } */ if(localStorage.UROverviewCurrentVersion != null) { uroAddLog('comparing install versions'); if(localStorage.UROverviewCurrentVersion == uroVersion) { isUpgradeInstall = false; } } if(localStorage.UROverviewFriendlyAreaNames != null) { uroAddLog('recover friendly area names'); uroApplyFriendlyAreaNames(); isNewInstall = false; } if(localStorage.UROverviewMasterEnable != null) { uroAddLog('recover master enable state'); document.getElementById('_cbMasterEnable').checked = (localStorage.UROverviewMasterEnable == "true"); uroAddLog('enable checkbox state set...'); } if(isNewInstall) { uroFirstTimerWelcomePack(); } else if(isUpgradeInstall) { uroShowUpdateNotes(); } if(notifyAboutCustomIcons) { uroAdvertiseCustomIcons(); } uroInhibitSave = false; } function uroDefaultSettings() { uroShowAlertBox("fa-warning", "URO+ Warning", "Resetting URO+ settings <b>cannot</b> be undone.<br>Are you <i>sure</i> you want to do this?", true, "Reset settings", "Keep settings", uroDefaultSettingsAction, null); } function uroDefaultSettingsAction() { var defaultSettings = ''; defaultSettings += '[UROverviewCWLGroups][len=16]0,No group,false[END]'; defaultSettings += '[UROverviewCamWatchList][len=0][END]'; defaultSettings += '[UROverviewCameraOptions][len=852]:_cbShowWorldCams,true:_cbShowUSACams,true:_cbShowNonWorldCams,true:_cbShowOnlyCamsCreatedBy,false:_cbShowOnlyCamsEditedBy,false:_textCameraEditor,:_cbShowOnlyMyCams,false:_cbShowSpeedCams,true:_cbShowIfSpeedSet,true:_cbShowIfNoSpeedSet,true:_cbShowIfInvalidSpeedSet,true:_cbShowRedLightCams,true:_cbShowRLCIfZeroSpeedSet,true:_cbShowRLCIfNonZeroSpeedSet,true:_cbShowRLCIfNoSpeedSet,true:_cbShowDummyCams,true:_cbHideCreatedByMe,false:_cbHideCreatedByRank0,false:_cbHideCreatedByRank1,false:_cbHideCreatedByRank2,false:_cbHideCreatedByRank3,false:_cbHideCreatedByRank4,false:_cbHideCreatedByRank5,false:_cbHideUpdatedByMe,false:_cbHideUpdatedByRank0,false:_cbHideUpdatedByRank1,false:_cbHideUpdatedByRank2,false:_cbHideUpdatedByRank3,false:_cbHideUpdatedByRank4,false:_cbHideUpdatedByRank5,false:_cbHideCWLCams,false:_cbHighlightInsteadOfHideCams,false[END]'; defaultSettings += '[UROverviewCurrentVersion][len=5]3.146[END]'; defaultSettings += '[UROverviewFriendlyAreaNames][len=0][END]'; defaultSettings += '[UROverviewMCOptions][len=529]:_cbMCFilterRoadworks,false:_cbMCFilterConstruction,false:_cbMCFilterClosure,false:_cbMCFilterEvent,false:_cbMCFilterNote,false:_cbMCFilterBOG,false:_cbMCFilterDifficult,false:_cbMCFilterWSLM,false:_cbInvertMCFilter,false:_cbMCHideMyFollowed,false:_cbMCHideMyUnfollowed,false:_cbMCDescriptionMustBePresent,false:_cbMCDescriptionMustBeAbsent,false:_cbMCEnableKeywordMustBePresent,false:_textMCKeywordPresent,:_cbMCEnableKeywordMustBeAbsent,false:_textMCKeywordAbsent,:_cbMCCaseInsensitive,false:_cbMCEnhancePointMCVisibility,false[END]'; defaultSettings += '[UROverviewMPOptions][len=1219]:_cbMPFilterOutsideArea,false:_cbMPFilter_T1,false:_cbMPFilter_T2,false:_cbMPFilter_T3,false:_cbMPFilter_T5,false:_cbMPFilter_T6,false:_cbMPFilter_T7,false:_cbMPFilter_T8,false:_cbMPFilter_T10,false:_cbMPFilter_T11,false:_cbMPFilter_T12,false:_cbMPFilter_T13,false:_cbMPFilter_T14,false:_cbMPFilter_T15,false:_cbMPFilter_T16,false:_cbMPFilter_T17,false:_cbMPFilter_T19,false:_cbMPFilter_T20,false:_cbMPFilter_T21,false:_cbMPFilter_T22,false:_cbMPFilter_T23,false:_cbMPFilter_T50,false:_cbMPFilter_T51,false:_cbMPFilter_T52,false:_cbMPFilter_T53,false:_cbMPFilter_T70,false:_cbMPFilter_T71,false:_cbMPFilter_T101,false:_cbMPFilter_T102,false:_cbMPFilter_T103,false:_cbMPFilter_T104,false:_cbMPFilter_T105,false:_cbMPFilter_T106,false:_cbMPFilter_T200,false:_cbMPFilter_T300,false:_cbMPFilterUnknownProblem,false:_cbFilterElgin,false:_cbFilterTrafficCast,false:_cbFilterTrafficMaster,false:_cbFilterCaltrans,false:_cbFilterTFL,false:_cbMPFilterReopenedProblem,false:_cbInvertMPFilter,false:_cbMPFilterClosed,false:_cbMPFilterSolved,false:_cbMPFilterUnidentified,false:_cbMPClosedUserIDFilter,false:_cbMPNotClosedUserIDFilter,false:_cbMPFilterLowSeverity,false:_cbMPFilterMediumSeverity,false:_cbMPFilterHighSeverity,false[END]'; defaultSettings += '[UROverviewMasterEnable][len=4]true[END]'; defaultSettings += '[UROverviewMiscOptions][len=1698]:_cbHideUserRTCs,false:_cbHideEditorRTCs,false:_cbHideFutureEditorRTCs,false:_cbHideWazeRTCs,false:_cbHideFutureWazeRTCs,false:_cbHideSegmentsWhenRoadsHidden,false:_cbKillInertialPanning,false:_cbNativeConvoMarkers,true:_cbNativeBetaConvoMarkers,true:_cbCommentCount,false:_cbEnableDeleteFeedEntries,false:_cbURBackfill,false:_inputFilterMinZoomLevel,10:_inputUnstackSensitivity,15:_inputUnstackZoomLevel,3:_cbCustomRoadworksMarkers,false:_cbCustomConstructionMarkers,false:_cbCustomClosuresMarkers,false:_cbCustomEventsMarkers,false:_cbCustomNotesMarkers,false:_cbCustomBOGMarkers,false:_cbCustomDifficultMarkers,false:_cbCustomWSLMMarkers,false:_cbCustomNativeSLMarkers,false:_cbCustomKeywordMarkers,false:_textCustomKeyword,:_cbCustomElginMarkers,false:_cbCustomTrafficMasterMarkers,false:_cbCustomTrafficCastMarkers,false:_cbCustomCaltransMarkers,false:_cbCustomTFLMarkers,false:_inputPopupDwellTimeout,2:_inputPopupEntryTimeout,2:_inputMaxJitter,2:_inputPopupAutoHideTimeout,0:_cbInhibitURPopup,false:_cbInhibitMPPopup,false:_cbInhibitCamPopup,false:_cbInhibitSegPopup,false:_cbInhibitSegGenericPopup,false:_cbInhibitTurnsPopup,false:_cbInhibitLandmarkPopup,false:_cbInhibitPUPopup,false:_cbInhibitMapCommentPopup,false:_cbInhibitNodesPopup,false:_cbDateFmtDDMMYY,true:_cbDateFmtMMDDYY,false:_cbDateFmtYYMMDD,false:_cbTimeFmt24H,true:_cbTimeFmt12H,false:_cbWhiteBackground,false:_inputCustomBackgroundRed,255:_inputCustomBackgroundGreen,255:_inputCustomBackgroundBlue,255:_cbInhibitNURButton,false:_cbInhibitNMPButton,false:_cbInhibitNPURButton,false:_cbHideAMLayer,false:_cbMoveAMList,false:_cbDisablePlacesFiltering,false:_cbDisableTabStyling,false:_cbHideEditorInfo,false:_cbEnableDTE,false[END]'; defaultSettings += '[UROverviewPlaceWatchList][len=0][END]'; defaultSettings += '[UROverviewPlacesGroups][len=65]false:false:false:false:false:false:false:false:false:false:false[END]'; defaultSettings += '[UROverviewPlacesOptions][len=6062]:_cbFilterUneditablePlaceUpdates,false:_cbFilterLockRankedPlaceUpdates,false:_cbFilterNewPlacePUR,false:_cbFilterUpdatedDetailsPUR,false:_cbPURFilterCFPhone,false:_cbPURFilterCFName,false:_cbPURFilterCFEntryExitPoints,false:_cbPURFilterCFOpeningHours,false:_cbPURFilterCFAliases,false:_cbPURFilterCFServices,false:_cbPURFilterCFGeometry,false:_cbPURFilterCFHouseNumber,false:_cbPURFilterCFCategories,false:_cbPURFilterCFDescription,false:_cbFilterNewPhotoPUR,false:_cbFilterFlaggedPUR,false:_cbLeavePURGeos,false:_cbInvertPURFilters,false:_cbPURFilterLowSeverity,false:_cbPURFilterMediumSeverity,false:_cbPURFilterHighSeverity,false:_cbEnablePURMinAgeFilter,false:_inputPURFilterMinDays,:_cbEnablePURMaxAgeFilter,false:_inputPURFilterMaxDays,:_cbPlaceFilterEditedLessThan,false:_inputFilterPlaceEditMinDays,:_cbPlaceFilterEditedMoreThan,false:_inputFilterPlaceEditMaxDays,:_cbHidePlacesL0,false:_cbHidePlacesL1,false:_cbHidePlacesL2,false:_cbHidePlacesL3,false:_cbHidePlacesL4,false:_cbHidePlacesL5,false:_cbHidePlacesStaff,false:_cbHidePlacesAdLocked,false:_cbHideAreaPlaces,false:_cbHidePointPlaces,false:_cbHidePhotoPlaces,false:_cbHideNoPhotoPlaces,false:_cbHideLinkedPlaces,false:_cbHideNoLinkedPlaces,false:_cbHideDescribedPlaces,false:_cbHideNonDescribedPlaces,false:_cbHideKeywordPlaces,false:_cbHideNoKeywordPlaces,false:_textKeywordPlace,:_cbShowOnlyPlacesCreatedBy,false:_cbShowOnlyPlacesEditedBy,false:_textPlacesEditor,:_cbHideOnlyPlacesCreatedBy,false:_cbHideOnlyPlacesEditedBy,false:_textHidePlacesEditor,:_cbPlacesFilter-CAR_SERVICES,false:_cbPlacesFilter-GAS_STATION,false:_cbPlacesFilter-GARAGE_AUTOMOTIVE_SHOP,false:_cbPlacesFilter-CAR_WASH,false:_cbPlacesFilter-CHARGING_STATION,false:_cbPlacesFilter-TRANSPORTATION,false:_cbPlacesFilter-AIRPORT,false:_cbPlacesFilter-BUS_STATION,false:_cbPlacesFilter-FERRY_PIER,false:_cbPlacesFilter-SEAPORT_MARINA_HARBOR,false:_cbPlacesFilter-SUBWAY_STATION,false:_cbPlacesFilter-TRAIN_STATION,false:_cbPlacesFilter-BRIDGE,false:_cbPlacesFilter-TUNNEL,false:_cbPlacesFilter-TAXI_STATION,false:_cbPlacesFilter-JUNCTION_INTERCHANGE,false:_cbPlacesFilter-REST_AREAS,false:_cbPlacesFilter-PROFESSIONAL_AND_PUBLIC,false:_cbPlacesFilter-COLLEGE_UNIVERSITY,false:_cbPlacesFilter-SCHOOL,false:_cbPlacesFilter-CONVENTIONS_EVENT_CENTER,false:_cbPlacesFilter-GOVERNMENT,false:_cbPlacesFilter-LIBRARY,false:_cbPlacesFilter-CITY_HALL,false:_cbPlacesFilter-ORGANIZATION_OR_ASSOCIATION,false:_cbPlacesFilter-PRISON_CORRECTIONAL_FACILITY,false:_cbPlacesFilter-COURTHOUSE,false:_cbPlacesFilter-CEMETERY,false:_cbPlacesFilter-FIRE_DEPARTMENT,false:_cbPlacesFilter-POLICE_STATION,false:_cbPlacesFilter-MILITARY,false:_cbPlacesFilter-HOSPITAL_URGENT_CARE,false:_cbPlacesFilter-DOCTOR_CLINIC,false:_cbPlacesFilter-OFFICES,false:_cbPlacesFilter-POST_OFFICE,false:_cbPlacesFilter-RELIGIOUS_CENTER,false:_cbPlacesFilter-KINDERGARDEN,false:_cbPlacesFilter-FACTORY_INDUSTRIAL,false:_cbPlacesFilter-EMBASSY_CONSULATE,false:_cbPlacesFilter-INFORMATION_POINT,false:_cbPlacesFilter-EMERGENCY_SHELTER,false:_cbPlacesFilter-TRASH_AND_RECYCLING_FACILITIES,false:_cbPlacesFilter-SHOPPING_AND_SERVICES,false:_cbPlacesFilter-ARTS_AND_CRAFTS,false:_cbPlacesFilter-BANK_FINANCIAL,false:_cbPlacesFilter-SPORTING_GOODS,false:_cbPlacesFilter-BOOKSTORE,false:_cbPlacesFilter-PHOTOGRAPHY,false:_cbPlacesFilter-CAR_DEALERSHIP,false:_cbPlacesFilter-FASHION_AND_CLOTHING,false:_cbPlacesFilter-CONVENIENCE_STORE,false:_cbPlacesFilter-PERSONAL_CARE,false:_cbPlacesFilter-DEPARTMENT_STORE,false:_cbPlacesFilter-PHARMACY,false:_cbPlacesFilter-ELECTRONICS,false:_cbPlacesFilter-FLOWERS,false:_cbPlacesFilter-FURNITURE_HOME_STORE,false:_cbPlacesFilter-GIFTS,false:_cbPlacesFilter-GYM_FITNESS,false:_cbPlacesFilter-SWIMMING_POOL,false:_cbPlacesFilter-HARDWARE_STORE,false:_cbPlacesFilter-MARKET,false:_cbPlacesFilter-SUPERMARKET_GROCERY,false:_cbPlacesFilter-JEWELRY,false:_cbPlacesFilter-LAUNDRY_DRY_CLEAN,false:_cbPlacesFilter-SHOPPING_CENTER,false:_cbPlacesFilter-MUSIC_STORE,false:_cbPlacesFilter-PET_STORE_VETERINARIAN_SERVICES,false:_cbPlacesFilter-TOY_STORE,false:_cbPlacesFilter-TRAVEL_AGENCY,false:_cbPlacesFilter-ATM,false:_cbPlacesFilter-CURRENCY_EXCHANGE,false:_cbPlacesFilter-CAR_RENTAL,false:_cbPlacesFilter-TELECOM,false:_cbPlacesFilter-FOOD_AND_DRINK,false:_cbPlacesFilter-RESTAURANT,false:_cbPlacesFilter-BAKERY,false:_cbPlacesFilter-DESSERT,false:_cbPlacesFilter-CAFE,false:_cbPlacesFilter-FAST_FOOD,false:_cbPlacesFilter-FOOD_COURT,false:_cbPlacesFilter-BAR,false:_cbPlacesFilter-ICE_CREAM,false:_cbPlacesFilter-CULTURE_AND_ENTERTAINEMENT,false:_cbPlacesFilter-ART_GALLERY,false:_cbPlacesFilter-CASINO,false:_cbPlacesFilter-CLUB,false:_cbPlacesFilter-TOURIST_ATTRACTION_HISTORIC_SITE,false:_cbPlacesFilter-MOVIE_THEATER,false:_cbPlacesFilter-MUSEUM,false:_cbPlacesFilter-MUSIC_VENUE,false:_cbPlacesFilter-PERFORMING_ARTS_VENUE,false:_cbPlacesFilter-GAME_CLUB,false:_cbPlacesFilter-STADIUM_ARENA,false:_cbPlacesFilter-THEME_PARK,false:_cbPlacesFilter-ZOO_AQUARIUM,false:_cbPlacesFilter-RACING_TRACK,false:_cbPlacesFilter-THEATER,false:_cbPlacesFilter-OTHER,false:_cbPlacesFilter-CONSTRUCTION_SITE,false:_cbPlacesFilter-LODGING,false:_cbPlacesFilter-HOTEL,false:_cbPlacesFilter-HOSTEL,false:_cbPlacesFilter-CAMPING_TRAILER_PARK,false:_cbPlacesFilter-COTTAGE_CABIN,false:_cbPlacesFilter-BED_AND_BREAKFAST,false:_cbPlacesFilter-OUTDOORS,false:_cbPlacesFilter-PARK,false:_cbPlacesFilter-PLAYGROUND,false:_cbPlacesFilter-BEACH,false:_cbPlacesFilter-SPORTS_COURT,false:_cbPlacesFilter-GOLF_COURSE,false:_cbPlacesFilter-PLAZA,false:_cbPlacesFilter-PROMENADE,false:_cbPlacesFilter-POOL,false:_cbPlacesFilter-SCENIC_LOOKOUT_VIEWPOINT,false:_cbPlacesFilter-SKI_AREA,false:_cbPlacesFilter-NATURAL_FEATURES,false:_cbPlacesFilter-ISLAND,false:_cbPlacesFilter-SEA_LAKE_POOL,false:_cbPlacesFilter-RIVER_STREAM,false:_cbPlacesFilter-FOREST_GROVE,false:_cbPlacesFilter-FARM,false:_cbPlacesFilter-CANAL,false:_cbPlacesFilter-SWAMP_MARSH,false:_cbPlacesFilter-DAM,false:_cbPlacesFilter-PARKING_LOT,false:_cbFilterPrivatePlaces,false:_cbInvertPlacesFilter,false[END]'; defaultSettings += '[UROverviewSegWatchList][len=0][END]'; defaultSettings += '[UROverviewUROptions][len=1779]:_cbURFilterDupes,false:_cbURFilterOutsideArea,false:_cbNoFilterForURInURL,false:_cbFilterWazeAuto,false:_cbFilterIncorrectTurn,false:_cbFilterIncorrectAddress,false:_cbFilterIncorrectRoute,false:_cbFilterMissingRoundabout,false:_cbFilterGeneralError,false:_cbFilterTurnNotAllowed,false:_cbFilterIncorrectJunction,false:_cbFilterMissingBridgeOverpass,false:_cbFilterWrongDrivingDirection,false:_cbFilterMissingExit,false:_cbFilterMissingRoad,false:_cbFilterBlockedRoad,false:_cbFilterMissingLandmark,false:_cbFilterSpeedLimits,false:_cbFilterUndefined,false:_cbFilterRoadworks,false:_cbFilterConstruction,false:_cbFilterClosure,false:_cbFilterEvent,false:_cbFilterNote,false:_cbFilterBOG,false:_cbFilterDifficult,false:_cbFilterWSLM,false:_cbInvertURFilter,false:_cbFilterOpenUR,false:_cbFilterClosedUR,false:_cbFilterSolved,false:_cbFilterUnidentified,false:_cbEnableMinAgeFilter,false:_inputFilterMinDays,:_cbEnableMaxAgeFilter,false:_inputFilterMaxDays,:_cbHideMyFollowed,false:_cbHideMyUnfollowed,false:_cbURDescriptionMustBePresent,false:_cbURDescriptionMustBeAbsent,false:_cbEnableKeywordMustBePresent,false:_textKeywordPresent,:_cbEnableKeywordMustBeAbsent,false:_textKeywordAbsent,:_cbCaseInsensitive,false:_cbHideMyComments,false:_cbHideAnyComments,false:_cbHideIfLastCommenter,false:_cbHideIfNotLastCommenter,false:_cbHideIfReporterLastCommenter,false:_cbHideIfReporterNotLastCommenter,false:_cbEnableMinCommentsFilter,false:_inputFilterMinComments,:_cbEnableMaxCommentsFilter,false:_inputFilterMaxComments,:_cbEnableCommentAgeFilter2,false:_inputFilterCommentDays2,:_cbEnableCommentAgeFilter,false:_inputFilterCommentDays,:_cbIgnoreOtherEditorComments,false:_cbURUserIDFilter,false:_cbURResolverIDFilter,false:_cbInvertURStateFilter,false:_cbNoFilterForTaggedURs,false[END]'; document.getElementById('_txtSettings').value = defaultSettings; uroTextToSettings(); document.getElementById('_txtSettings').value = ''; } function uroSettingsToText() { 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(); dateNow.setHours(24); var pendingUntilMidnight = elapsedSinceEpoch - dateNow.getTime(); if((elapsedSinceEvent < elapsedSinceMidnight) && (elapsedSinceEvent > pendingUntilMidnight)) { // event occurred today... return 0; } else if(elapsedSinceEvent < 0) { // event occurrs at some point in the future after midnight today, so return a minimum value of -1... return -1 - Math.floor((pendingUntilMidnight - elapsedSinceEvent) / 86400000); } else { // event occurred at some point prior to midnight this morning, so return a minimum value of 1... return 1 + Math.floor((elapsedSinceEvent - elapsedSinceMidnight) / 86400000); } } function uroGetURAge(urObj,ageType,getRaw) { if(ageType === 0) { if((urObj.attributes.driveDate === null)||(urObj.attributes.driveDate === 0)) return -1; if(getRaw) return urObj.attributes.driveDate; else return uroDateToDays(urObj.attributes.driveDate); } else if(ageType === 1) { if((urObj.attributes.resolvedOn === null)||(urObj.attributes.resolvedOn === 0)) return -1; if(getRaw) return urObj.attributes.resolvedOn; else return uroDateToDays(urObj.attributes.resolvedOn); } else { return -1; } } function uroGetMCAge(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 if(ageType === 2) { if((mcObj.attributes.endDate === null)||(mcObj.attributes.endDate === 0)) return -1; var tDate = new Date(mcObj.attributes.endDate); if(getRaw) return tDate; else return uroDateToDays(tDate); } else { return -1; } } function uroGetPURAge(purObj) { if(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded !== null) { return uroDateToDays(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded); } else { return -1; } } function uroGetCameraAge(camObj, mode) { if(mode === 0) { if(camObj.attributes.updatedOn === null) return -1; return uroDateToDays(camObj.attributes.updatedOn); } if(mode === 1) { if(camObj.attributes.createdOn === null) return -1; return uroDateToDays(camObj.attributes.createdOn); } } function uroGetCommentAge(commentObj) { if(commentObj.createdOn === null) return -1; return uroDateToDays(commentObj.createdOn); } function uroParseDaysAgo(days) { if(days === 0) return 'today'; else if(days === 1) return '1 day ago'; else return days+' days ago'; } function uroParseDaysToGo(days) { days = 0 - days; if(days === 0) return 'today'; else if(days === 1) return 'in 1 day'; else return 'in '+days+' days'; } function uroGetLocalisedSpeedString(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 = null; if((W.model.getTopCountry()) && (W.model.getTopCountry().attributes.name !== undefined)) { country = W.model.getTopCountry().attributes.name; } if(country !== null) { // country-specific deviations from the above... if ( (country == "United Kingdom") || (country == "Jersey") || (country == "Guernsey") || (country == "United States") ) { // countries using MPH conversionFactor = 1.609; } if ( (country == "United States") || (country == "Guernsey") ) { // countries with speed limits set in multiples of 5 multipleFactor = 5; } } 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.app.getAppRegionCode(); var area = uroGetAreaArea(uroAreaNameHoverObj.parentNode.children[1].innerText); uroAreaNameHoverObj.removeChild(uroANEditBox); uroAreaNameOverlayShown = false; uroANEditHovered = false; uroUpdateAreaName(newName, server, area); } function uroANEditClick(e) { // this traps the click to prevent it falling through to the underlying area name element and // potentially causing the map view to be relocated to that area... e.stopPropagation(); } function uroGetAreaArea(area) { area = parseFloat(area.split(' ')[0]); return area; } function uroAreaNameOverlaySetup() { uroAreaNameOverlayShown = true; uroANEditBox = document.createElement('div'); uroANEditBox.id = "uroANEditBox"; uroANEditBox.style.position = "absolute"; uroANEditBox.style.top = '7px'; uroANEditBox.style.left = '2px'; uroANEditBox.style.width = "99%"; uroAreaNameHoverObj.appendChild(uroANEditBox); uroANEditBox.onmouseover = uroANEditHover(); var existingName = uroAreaNameHoverObj.innerHTML; var italicTagPos = existingName.indexOf(' <i>'); if(italicTagPos == -1) { existingName = ""; } else { existingName = existingName.substr(0,italicTagPos); } uroANEditBox.innerHTML = uroModifyHTML('<input type="text" style="font-size:14px; line-height:16px; height:22px; width:100%" id="_textAreaName" value="'+existingName+'">'); } function uroReplaceAreaNames(replaceAfterNameChange) { if(document.getElementById('sidepanel-areas') === undefined) { return; } if(document.getElementById('sidepanel-areas').getElementsByClassName('result-list').length === 0) { return; } if(replaceAfterNameChange === false) { if(document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id == "friendlyNamed") { return; } } let panelRootObj = document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0]; if(panelRootObj === undefined) { // we get here if the user doesn't have any areas defined... return; } let areaNameObjs = panelRootObj.getElementsByClassName('list-item-card-info'); if(areaNameObjs.length === 0) { return; } var localisedManagedArea = I18n.lookup("user.areas.managed_area"); for(let loop=0; loop < areaNameObjs.length; loop++) { if(areaNameObjs[loop].children.length === 2) { let title = areaNameObjs[loop].children[0].innerText; if(title.indexOf(localisedManagedArea) > -1) { let area = uroGetAreaArea(areaNameObjs[loop].children[1].innerText); areaNameObjs[loop].children[0].innerHTML = uroModifyHTML(localisedManagedArea); for(let fnIdx=0; fnIdx < uroFriendlyAreaNames.length; fnIdx++) { let fnObj = uroFriendlyAreaNames[fnIdx]; if((fnObj.area == area) && (fnObj.server == W.app.getAppRegionCode())) { areaNameObjs[loop].children[0].innerHTML = uroModifyHTML(fnObj.fName +' <i>('+localisedManagedArea+')</i>'); break; } } var titleObj = areaNameObjs[loop].getElementsByClassName('list-item-card-title')[0]; titleObj.addEventListener("mouseover", uroAreaNameHover, false); titleObj.addEventListener("mouseout", uroAreaNameUnHover, false); titleObj.style.cursor = "text"; } } } document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id = "friendlyNamed"; } // -------------------------------------------------------------------------------------------------------------------- // WATCHLIST STUFF // -------------------------------------------------------------------------------------------------------------------- // Generic Functions function uroTypeCast(varin) { if(varin == "null") return null; if(typeof varin == "string") return parseInt(varin); return varin; } function uroTruncate(val) { if(val === null) return val; if(val < 0) return Math.ceil(val); return Math.floor(val); } function uroOWLGroupObj(groupID, groupName, groupCollapsed) { groupID = uroTypeCast(groupID); this.groupID = groupID; this.groupName = groupName; this.groupCount = 0; this.groupCollapsed = groupCollapsed; } // Camera Functions function uroCamWatchObjCheckProps(type, azymuth, speed, lat, lon) { if(type !== null) type = uroTypeCast(type); if(azymuth !== null) azymuth = uroTruncate(uroTypeCast(azymuth)%360); if(speed !== null) speed = uroTruncate(uroTypeCast(speed)); if(lat !== null) lat = uroTruncate(uroTypeCast(lat)); if(lon !== null) lon = uroTruncate(uroTypeCast(lon)); this.type = type; this.azymuth = azymuth; this.speed = speed; this.lat = lat; this.lon = lon; } function uroCamWatchObj(persistent, fid, lon, lat, type, azymuth, speed, groupID, server) { fid = uroTypeCast(fid); groupID = uroTypeCast(groupID); if(typeof persistent == "string") persistent = (persistent == "true"); if(server === "undefined") server = "??"; this.fid = fid; this.persistent = persistent; this.loaded = false; this.server = server; this.groupID = groupID; this.watch = new uroCamWatchObjCheckProps(type, azymuth, speed, lat, lon); this.current = new uroCamWatchObjCheckProps(null, null, null, null, null); } function uroCamDataChanged(idx) { 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.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, server) { var camObj = uroCamWatchObjects[idx]; camObj.loaded = true; camObj.server = server; camObj.current = new uroCamWatchObjCheckProps(type, azymuth, speed, lat, lon); return(uroCamDataChanged(idx)); } function uroAddCamToWatchList() { if(uroIsCamOnWatchList(uroShownFID) == -1) { 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, 0, W.app.getAppRegionCode())); uroAddCurrentCamWatchData(uroCamWatchObjects.length-1, camObj.geometry.y, camObj.geometry.x, camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, W.app.getAppRegionCode()); uroAddLog('added camera '+uroShownFID+' to watchlist'); uroOWLUpdateHTML(); } } function uroRemoveCamFromWatchList() { 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.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 += W.Config.api_base; camURL += '/Features?language=en&cameras=true&bbox='; var latl = camPos.lat - 0.25; var latu = camPos.lat + 0.25; var lonl = camPos.lon - 0.25; var lonr = camPos.lon + 0.25; camURL += lonl+','+latl+','+lonr+','+latu; uroAddLog('retrieving camera data around '+camPos.lon+','+camPos.lat); var camReq = new XMLHttpRequest(); camReq.open('GET',camURL,false); try { camReq.send(); uroAddLog('response '+camReq.status+' received for camera data request'); if (camReq.status === 200) { var camData = JSON.parse(camReq.responseText); for(var camIdx = 0; camIdx < camData.cameras.objects.length; camIdx++) { var camObj = camData.cameras.objects[camIdx]; var listIdx = uroIsCamOnWatchList(camObj.id); if(listIdx != -1) { camPos.lon = camObj.geometry.coordinates[0]; camPos.lat = camObj.geometry.coordinates[1]; camPos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913")); camPos.lon = uroTruncate(camPos.lon); camPos.lat = uroTruncate(camPos.lat); camChanged = (uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, W.app.getAppRegionCode()) || camChanged); } } } else { uroAddLog('camera data request failed (status != 200)'); } } catch(err) { uroAddLog('camera data request failed (exception '+err+' caught)'); } return camChanged; } function uroGetCurrentCamWatchListObjects() { 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.app.getAppRegionCode()) || (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, W.app.getAppRegionCode()) || camChanged); } else { camChanged = (uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon) || camChanged); } } else { camChanged = (uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon) || camChanged); } } } if(camChanged) { for(camidx=0;camidx<uroCamWatchObjects.length;camidx++) { if(uroCamDataChanged(camidx)) { camsChanged.push(uroCamWatchObjects[camidx]); } } } for(camidx=0;camidx<uroCamWatchObjects.length;camidx++) { camObj = uroCamWatchObjects[camidx]; if((camObj.loaded === false) && (camObj.server == W.app.getAppRegionCode())) { camsDeleted.push(camObj); } } if((camsChanged.length > 0) || (camsDeleted.length > 0)) { var alertStr = ''; for(camidx=0;camidx<camsChanged.length;camidx++) { alertStr += 'Camera ID '+camsChanged[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsChanged[camidx].groupID)+'" has been changed<br>'; } alertStr += '<br>'; for(camidx=0;camidx<camsDeleted.length;camidx++) { alertStr += 'Camera ID '+camsDeleted[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsDeleted[camidx].groupID)+'" has been deleted<br>'; } uroShowAlertBox("fa-info-circle", "URO+ Camera Watchlist Alert", alertStr, false, "OK", null, null, null); } } function uroClearDeletedCameras() { for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--) { if(uroCamWatchObjects[camidx].loaded === false) { uroShownFID = uroCamWatchObjects[camidx].fid; uroRemoveCamFromWatchList(); } } } function uroAcceptCameraChanges() { for(var camidx=0; camidx < uroCamWatchObjects.length; camidx++) { if(uroCamDataChanged(camidx)) { uroCamWatchObjects[camidx].watch.type = uroCamWatchObjects[camidx].current.type; uroCamWatchObjects[camidx].watch.azymuth = uroCamWatchObjects[camidx].current.azymuth; uroCamWatchObjects[camidx].watch.speed = uroCamWatchObjects[camidx].current.speed; uroCamWatchObjects[camidx].watch.lat = uroCamWatchObjects[camidx].current.lat; uroCamWatchObjects[camidx].watch.lon = uroCamWatchObjects[camidx].current.lon; } } uroOWLUpdateHTML(); } function uroClearUnknownServerCameras() { var confirmMsg = '<p>Cameras with an unknown server <i>cannot</i> be automatically verified by URO+</p>'; confirmMsg += 'It is recommended that you manually load WME from each server (World, USA/Canada and Israel) to give URO+ a chance of locating these cameras.<br>'; confirmMsg += 'If the cameras then continue to show up as an unknown server, it is safe to delete them...<br><br>'; confirmMsg += 'Do you still wish to proceed with deleting all unknown server cameras?'; uroShowAlertBox("fa-warning", "URO+ Warning", confirmMsg, true, "Delete unknown cameras", "Keep unknown cameras", uroClearUnknownServerCamerasAction, null); } function uroClearUnknownServerCamerasAction() { for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--) { if(uroCamWatchObjects[camidx].server == '??') { uroShownFID = uroCamWatchObjects[camidx].fid; uroRemoveCamFromWatchList(); } } } function uroRescanCamWatchList() { for(var camidx=0;camidx<uroCamWatchObjects.length;camidx++) { uroCamWatchObjects[camidx].loaded = false; } uroGetCurrentCamWatchListObjects(); uroOWLUpdateHTML(); } function uroGotoCam() { var camidx = this.id.substr(13); var camPos = new OpenLayers.LonLat(); camPos.lon = uroCamWatchObjects[camidx].watch.lon; camPos.lat = uroCamWatchObjects[camidx].watch.lat; W.map.setCenter(camPos,16); 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.app.getAppRegionCode())); 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.app.getAppRegionCode()); 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 += W.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.app.getAppRegionCode()); } 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.app.getAppRegionCode()) || (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.app.getAppRegionCode()); } 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.app.getAppRegionCode())) { 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.app.getAppRegionCode()) { if(uroCamWatchObjects[camidx].server == '??') this.style.backgroundColor = '#A0A0A0'; else this.style.backgroundColor = '#AAFFAA'; } else if(changed) this.style.backgroundColor = '#AAAAFF'; else if(deleted) this.style.backgroundColor = '#FFAAAA'; else this.style.backgroundColor = '#FFFFFF'; return false; } function uroCWLIconHighlight() { this.style.color="#0000ff"; return false; } function uroCWLIconLowlight() { this.style.color="#ccccff"; return false; } function uroPopulateCWLGroupSelect() { 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.app.getAppRegionCode()) { if(camObj.server == '??') iHTML += '#A0A0A0;'; else iHTML += '#AAFFAA;'; } else if(changed) iHTML += '#AAAAFF;'; else if(deleted) iHTML += '#FFAAAA;'; else iHTML += '#FFFFFF;'; if(groupObj.groupCollapsed === true) iHTML += 'display:none;">'; else iHTML += 'display:block;">'; iHTML += 'ID: '+camObj.fid; iHTML += ' ('+camObj.server+')'; iHTML += ' Type: '+camTypes[camObj.watch.type]; if(camObj.server != W.app.getAppRegionCode()) { if(camObj.server == '??') { iHTML += '<br><i>Unknown server</i>'; } else { iHTML += '<br><i>Not on this server</i>'; } } else if(deleted) { iHTML += '<br>DELETED'; } else if(changed) { if(camObj.current.type != camObj.watch.type) { iHTML += '<br> 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.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.app.getAppRegionCode()) { 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 += '<input type="button" id="_btnUpdateCamValues" value="Accept all changes"><br><br>'; iHTML += '<b>Remove cameras from OWL:</b><br>'; iHTML += '<input type="button" id="_btnRemoveDeletedCameras" value="Deleted"> '; iHTML += '<input type="button" id="_btnRemoveUnknownServerCameras" value="Unknown Server"> '; iHTML += '<input type="button" id="_btnClearCamWatchList" value="ALL Cameras">'; iHTML += '</div>'; uroCtrlTabs[URO_TABS_ID.OWL][URO_TABS_FIELD.TABBODY].innerHTML = uroModifyHTML(iHTML); uroFinaliseOWLHTMLUpdate(); } function uroFinaliseOWLHTMLUpdate() { if(uroCamWatchObjects.length > 0) { if(document.getElementById("_uroCWL-0") == null) { window.setTimeout(uroFinaliseOWLHTMLUpdate,100); return; } for(var camidx=0;camidx<uroCamWatchObjects.length;camidx++) { document.getElementById("_uroCWL-"+camidx).onmouseover = uroHighlightCWLEntry; document.getElementById("_uroCWL-"+camidx).onmouseleave = uroUnhighlightCWLEntry; if(uroCamWatchObjects[camidx].server == W.app.getAppRegionCode()) { 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) { window.setTimeout(uroFinaliseOWLHTMLUpdate,100); return; } uroAddBtnEvl('_btnClearCamWatchList', 'click', uroClearCamWatchList); uroAddBtnEvl('_btnRemoveDeletedCameras', 'click', uroClearDeletedCameras); uroAddBtnEvl('_btnRemoveUnknownServerCameras', 'click', uroClearUnknownServerCameras); uroAddBtnEvl('_btnRescanCamWatchList', 'click', uroRescanCamWatchList); uroAddBtnEvl('_btnUpdateCamValues', 'click', uroAcceptCameraChanges); uroAddBtnEvl('_btnCWLGroupDel', 'click', uroRemoveCWLGroup); uroAddBtnEvl('_btnCWLGroupAdd', 'click', uroAddCWLGroup); if(document.getElementById('_uroCWLGroupSelect') !== null) { uroAddLog('populating CWL group list'); uroPopulateCWLGroupSelect(); 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"; } try { document.getElementById('_btnUndoLastHide').style.visibility = btnState; document.getElementById('_btnClearSessionHides').style.visibility = btnState; uroFilterItems(); } catch(err) { uroAddLog('exception thrown in uroEnableIgnoreListControls()'); } } function uroAddToIgnoreList() { if(!uroIsOnIgnoreList(uroShownFID)) { sessionStorage.UROverview_FID_IgnoreList += 'fid:'+uroShownFID; uroAddLog('added fid '+uroShownFID+' to ignore list'); uroAddLog(sessionStorage.UROverview_FID_IgnoreList); uroDiv.style.visibility = 'hidden'; uroEnableIgnoreListControls(); W.map.events.register("mousemove", null, uroFilterItemsOnMove); } return false; } function uroRemoveLastAddedIgnore() { var ignorelist = sessionStorage.UROverview_FID_IgnoreList; var fidpos = ignorelist.lastIndexOf('fid:'); if(fidpos != -1) { ignorelist = ignorelist.slice(0,fidpos); sessionStorage.UROverview_FID_IgnoreList = ignorelist; uroAddLog('removed last fid from ignore list'); uroAddLog(sessionStorage.UROverview_FID_IgnoreList); uroEnableIgnoreListControls(); } } function uroRemoveAllIgnores() { sessionStorage.UROverview_FID_IgnoreList = ''; uroEnableIgnoreListControls(); } function uroKeywordPresent(desc, keyword, caseInsensitive) { var re; if(caseInsensitive) re = RegExp(keyword,'i'); else re = RegExp(keyword); if(desc.search(re) != -1) return true; else return false; } function uroClickify(desc, suffix) { // The terminators array consists of pairs of characters which may be found at either // end of a URL. The first entry in each pair indicates which character needs to be // found immediately prior to the URL ('' indicates any character) in order for the // second entry to be considered as a potential end of URL marker let terminators = [ ['', ' '], ['', ','], ['(', ')'], ['[', ']'], ['', '\r'], ['', '\n'] ]; if(desc === null) return ''; if(desc === undefined) return ''; if(desc === '') return ''; if(desc.indexOf("https: one.network") == 0) { desc = desc.replaceAll(' ','/'); } desc = desc.replace(/<\/?[^>]+(>|$)/g, ""); if(desc !== "null") { // At the moment we can only clickify links that start with http or https... if(desc.indexOf('http') != -1) { let links = desc.split("http"); desc = ''; let i, j, linkEndPos, descPostLink; for(i=0; i<links.length; i++) { if(links[i].indexOf('://') != -1) { let prefix = links[i - 1][links[i - 1].length - 1]; links[i] = "http" + links[i]; linkEndPos = links[i].length + 1; // work out where the end of the URL is, based on what the character immediately // preceding the "http" is for(j=0; j<terminators.length; j++) { if(links[i].indexOf(terminators[j][1]) !== -1) { if((prefix === terminators[j][0]) || (terminators[j][0] === '')) { linkEndPos = Math.min(linkEndPos, links[i].indexOf(terminators[j][1])); } } } descPostLink = ''; if(linkEndPos < links[i].length) { descPostLink = links[i].slice(linkEndPos); links[i] = links[i].slice(0,linkEndPos); } desc += '<a target="_wazeUR" href="'+links[i]+'">'+links[i]+'</a>' + descPostLink; } else { desc += links[i]; } } } desc = desc.replace(/\n/g,"<br>"); return desc + suffix; } else { return ''; } } var uroPendingURSessionsTotal; var uroFinalizeTimeoutHandle = null; function uroFinalizeURSessionsGet() { if(uroPendingURSessionsTotal != uroPendingURSessionIDs.length) { uroPendingURSessionsTotal = uroPendingURSessionIDs.length; if(uroFinalizeTimeoutHandle !== null) { window.clearTimeout(uroFinalizeTimeoutHandle); uroFinalizeTimeoutHandle = null; } uroFinalizeTimeoutHandle = window.setTimeout(uroFinalizeURSessionsGet, 500); return; } 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.getAsync(idList); } if((uroPendingURSessionIDs.length) || (uroRequestedURSessionIDs.length)) { window.setTimeout(uroGetUpdateRequestSessions,1000); } else { uroPopulatingRequestSessions = false; } } function uroGetUpdateRequestSessions() { uroPendingURSessionsTotal = uroPendingURSessionIDs.length; if(uroFinalizeTimeoutHandle !== null) { window.clearTimeout(uroFinalizeTimeoutHandle); uroFinalizeTimeoutHandle = null; } uroFinalizeTimeoutHandle = window.setTimeout(uroFinalizeURSessionsGet,500); } function uroRefreshUpdateRequestSessions() { var urcount = 0; uroPendingURSessionIDs = []; uroRequestedURSessionIDs = []; uroPopulatingRequestSessions = true; 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].attributes.comments.length; if(nComments === 0) { return false; } for(var cidx=0; cidx<nComments; cidx++) { if(W.model.updateRequestSessions.objects[fid].attributes.comments[cidx].userID == uroUserID) { return true; } } return false; } function uroACMObj(urID, markerType, customType, hasMyComments, nComments) { this.urID = urID; this.markerType = markerType; this.customType = customType; this.hasMyComments = hasMyComments; this.nComments = nComments; } function uroAddCustomMarkers(urID, markerType, customType, hasMyComments, nComments) { var useCustomMarker = false; if(uroGetCBChecked('_cbMasterEnable') === true) { if(customType === 0) useCustomMarker = (uroGetCBChecked('_cbCustomRoadworksMarkers')); else if(customType === 1) useCustomMarker = (uroGetCBChecked('_cbCustomConstructionMarkers')); else if(customType === 2) useCustomMarker = (uroGetCBChecked('_cbCustomClosuresMarkers')); else if(customType === 3) useCustomMarker = (uroGetCBChecked('_cbCustomEventsMarkers')); else if(customType === 4) useCustomMarker = (uroGetCBChecked('_cbCustomNotesMarkers')); else if(customType === 5) useCustomMarker = (uroGetCBChecked('_cbCustomWSLMMarkers')); else if(customType === 6) useCustomMarker = (uroGetCBChecked('_cbCustomBOGMarkers')); else if(customType === 7) useCustomMarker = (uroGetCBChecked('_cbCustomDifficultMarkers')); else if(customType === 98) useCustomMarker = (uroGetCBChecked('_cbCustomNativeSLMarkers')); else if(customType === 99) useCustomMarker = (uroGetCBChecked('_cbCustomKeywordMarkers')); else if(customType === 100) useCustomMarker = (uroGetCBChecked('_cbCustomElginMarkers')); else if(customType === 101) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficCastMarkers')); else if(customType === 102) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficMasterMarkers')); else if(customType === 103) useCustomMarker = (uroGetCBChecked('_cbCustomCaltransMarkers')); else if(customType === 104) useCustomMarker = (uroGetCBChecked('_cbCustomTFLMarkers')); } if(!useCustomMarker) customType = -1; uroCustomMarkerList.push(new uroACMObj(urID, markerType, customType, hasMyComments, nComments)); } function uroChangeCustomMarkers(urID,isHighlighted,markerType) { if(document.getElementById('customMarker_'+urID) !== null) { var customType = null; var customVariant = 0; var marker = uroGetMarker(markerType, urID); if(marker !== null) { customType = marker.uroCustomType; if(markerType == URO_TMARKER.UR) { if(W.model.mapUpdateRequests.objects[urID].attributes.open === false) customVariant = 2; } else if(markerType == URO_TMARKER.MP) { if(W.model.problems.objects[urID].attributes.open === false) customVariant = 2; } } if(isHighlighted === true) { customVariant += 1; } if((customType !== null) && (customType !== undefined)) { document.getElementById('customMarker_'+urID).innerHTML = uroModifyHTML('<img src="'+uroAltMarkers[customType][customVariant]+'">'); } } } function uroRenderCustomMarkers(markerType) { var urID; var elmID; var newSpan; var divElem; var objIdx; var customType; var customVariant; var cmlObj; var customMarker; var touchedByURO = false; if(markerType == URO_TMARKER.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.getLayerByName("update_requests").id); if(divElem.childNodes.length > 0) { for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++) { customType = -1; cmlObj = uroCustomMarkerList[objIdx]; if(cmlObj.markerType == URO_TMARKER.UR) { if(uRCM_masterEnable === true) { customType = cmlObj.customType; } if(customType < 100) { urID = cmlObj.urID; var nComments = cmlObj.nComments; let marker = uroGetMarker(URO_TMARKER.UR, urID); if(marker !== null) { let iconObj = marker.element; let classList = iconObj.classList; newSpan = ''; if(nComments !== 0) { elmID = "commentCount_"+urID; if((addCommentCount) && (nComments > 0)) { // add a new comment count bubble newSpan += '<span id="'+elmID+'" style="position:absolute;top:-12px;left:-16px;pointer-events:none;z-index:1">'; // define the comment-count holding span within the span used to hold the empty bubble image, and before the image is // added to the HTML, to avoid z-indexing issues when adjacent comment count bubbles are overlapped... newSpan += '<span id="'+elmID+"_inner"+'" style="position:absolute;top:3px;left:0px;font-size:11px;pointer-events:none"></span>'; newSpan += '<img src="'+uroMarkers[0]+'">'; newSpan += '</span>'; } else { // remove comment count bubble from this UR marker if one has previously been // added and the user has now disabled the option... if(document.getElementById(elmID) !== null) { document.getElementById(elmID).remove(); } if(document.getElementById(elmID+"_inner") !== null) { document.getElementById(elmID+"_inner").remove(); } } if(nComments == -1) { // if we've set nComments to -1 to force a marker refresh when filtering is disabled, // we now need to locally reset it to 0 to prevent the remainder of this code assuming // the UR has a non-zero comment count... nComments = 0; } } if(nComments !== 0) { elmID = "convoMarker_"+urID; if(useDefaultConvoMarker === false) { 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 = marker.touchedByURO; elmID = "customMarker_"+urID; customMarker = ''; if(customType != -1) { if(document.getElementById(elmID) === null) { newSpan += '<span id="'+elmID+'" style="position:absolute;pointer-events:none;top:-3px;left:-2px;"></span>'; } customType = uroGetCustomMarkerIdx(customType); marker.uroCustomType = customType; customVariant = 0; if(W.model.mapUpdateRequests.objects[urID] !== undefined) { if(W.model.mapUpdateRequests.objects[urID].attributes.open === false) customVariant = 2; } customMarker = '<img src="'+uroAltMarkers[customType][customVariant]+'">'; } else { if(document.getElementById(elmID) !== null) { document.getElementById(elmID).remove(); } } if(newSpan !== '') { iconObj.innerHTML = newSpan; } if((customMarker !== '') && (document.getElementById(elmID) !== null)) { document.getElementById(elmID).innerHTML = uroModifyHTML(customMarker); } if(addCommentCount) { elmID = "commentCount_"+urID+"_inner"; if(document.getElementById(elmID) !== null) { var styleLeft; if(nComments < 10) styleLeft = '11px'; else if(nComments < 100) styleLeft = '8px'; else styleLeft = '5px'; document.getElementById(elmID).innerHTML = uroModifyHTML(nComments); document.getElementById(elmID).style.left = styleLeft; } } } } } } } } else if(markerType == URO_TMARKER.MP) { divElem = document.getElementById(W.map.getLayerByName("mapProblems").id); if(divElem.childNodes.length > 0) { for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++) { cmlObj = uroCustomMarkerList[objIdx]; if(cmlObj.markerType == URO_TMARKER.MP) { customType = cmlObj.customType; if((customType >= 100) || (customType == -1)) { urID = cmlObj.urID; let markerObj = uroGetMarker(URO_TMARKER.MP, urID); // change main marker if required if(markerObj !== null) { touchedByURO = markerObj.touchedByURO; } elmID = "customMarker_"+urID; customMarker = ''; if(customType != -1) { if(document.getElementById(elmID) === null) { newSpan = '<span id="'+elmID+'" style="position:absolute;pointer-events:none;top:-3px;left:-2px;"></span>'; if(markerObj !== null) { markerObj.icon.$div.prepend(newSpan); } } if(document.getElementById(elmID) !== null) { customType = uroGetCustomMarkerIdx(customType); if(markerObj !== null) { markerObj.uroCustomType = customType; } customVariant = 0; if(W.model.problems.objects[urID] !== undefined) { if(W.model.problems.objects[urID].attributes.open === false) customVariant = 2; } customMarker = '<img src="'+uroAltMarkers[customType][customVariant]+'">'; document.getElementById(elmID).innerHTML = uroModifyHTML(customMarker); } } else { if(document.getElementById(elmID) !== null) { document.getElementById(elmID).remove(); } } } } } } } } function uroIsFilteringEnabled(ignoreZoom) { var retval = false; if ( (uroGetCBChecked('_cbMasterEnable') === true) && ( (ignoreZoom === true) || (W.map.getZoom() <= uroGetElmValue('_inputFilterMinZoomLevel')) ) ) { retval = true; } return retval; } function uroUpdateMTEList() { if(Object.keys(W.model.majorTrafficEvents.objects).length === 0) return; var selectedIdx = null; var idx; var mteNames = []; var mteIDs = []; for(idx in W.model.majorTrafficEvents.objects) { if(W.model.majorTrafficEvents.objects.hasOwnProperty(idx)) { var name = W.model.majorTrafficEvents.objects[idx].attributes.names[0]?.value; if(mteNames.indexOf(name) == -1) { mteNames.push(name); mteIDs.push(idx); } } } // check for any previously selected ID in the list, then clear it and repopulate // using the newly gathered ID collection from above, and finally reselect the // previously selected MTE if its still present in the new list... var selector; var selectedID; var selectorEntry; selector = document.getElementById('_selectRTCMTE'); selectedID = null; if(selector.selectedOptions[0] != null) { selectedID = selector.selectedOptions[0].value; } while(selector.options.length > 0) { selector.options.remove(0); } selector.options.add(new Option('<select a MTE>', null)); if(mteNames.length > 0) { selectorEntry = ''; for(idx=0; idx<mteNames.length; idx++) { selectorEntry = mteNames[idx]; selector.options.add(new Option(selectorEntry, mteIDs[idx])); if(mteIDs[idx] == selectedID) { selectedIdx = idx+1; } } } if(selectedIdx !== null) { selector.selectedIndex = selectedIdx; } } function uroFilterRTCs() { let pmTStart = performance.now(); let pmFunction = "uroFilterRTCs"; if(uroFilterPreamble() === false) return; let uFR_filterActiveFromWME = uroGetCBChecked('_cbHideEditorRTCs'); let uFR_filterActiveFromWazeFeed = uroGetCBChecked('_cbHideWazeFeedRTCs'); let uFR_filterActiveFromWazeOther = uroGetCBChecked('_cbHideWazeRTCs'); let uFR_filterFutureFromWME = uroGetCBChecked('_cbHideFutureEditorRTCs'); let uFR_filterFutureFromWazeFeed = uroGetCBChecked('_cbHideFutureWazeFeedRTCs'); let uFR_filterFutureFromWazeOther = uroGetCBChecked('_cbHideFutureWazeRTCs'); let uFR_filterExpiredFromWME = uroGetCBChecked('_cbHideExpiredEditorRTCs'); let uFR_filterExpiredFromWazeFeed = uroGetCBChecked('_cbHideExpiredWazeFeedRTCs'); let uFR_filterExpiredFromWazeOther = uroGetCBChecked('_cbHideExpiredWazeRTCs'); let uFR_filterShowForMTE = uroGetCBChecked('_cbShowMTERTCs'); let uFR_filterHideForMTE = uroGetCBChecked('_cbHideMTERTCs'); let uFR_filterHideDurationLessThan = uroGetCBChecked('_cbEnableRTCDurationFilterLessThan'); let uFR_filterHideDurationMoreThan = uroGetCBChecked('_cbEnableRTCDurationFilterMoreThan'); let uFR_thresholdDurationLessThan = uroGetElmValue('_inputFilterRTCDurationLessThan'); let uFR_thresholdDurationMoreThan = uroGetElmValue('_inputFilterRTCDurationMoreThan'); let uFR_filterShowForTS = uroGetCBChecked('_cbRTCFilterShowForTS'); let uFR_filterHideForTS = uroGetCBChecked('_cbRTCFilterHideForTS'); let tsD = uroGetElmValue('_inputRTCFilterDay'); let tsMo = uroGetElmValue('_inputRTCFilterMonth'); let tsY = uroGetElmValue('_inputRTCFilterYear'); let tsH = uroGetElmValue('_inputRTCFilterHour'); let tsMi = uroGetElmValue('_inputRTCFilterMin'); let filterTS = uroGetTS(tsD, tsMo, tsY, tsH, tsMi); uroUpdateMTEList(); let mteID = null; let selectorMTE = document.getElementById('_selectRTCMTE'); if(selectorMTE.selectedOptions[0] != null) { mteID = selectorMTE.selectedOptions[0].value; } let uFR_masterEnable = uroIsFilteringEnabled(false); let markerIdx = 0; for (let rtcObj in W.model.roadClosures.objects) { let rtcStyle = 'visible'; if(uFR_masterEnable === true) { let rtcModel = W.model.roadClosures.objects[rtcObj]; if(mteID !== null) { if((uFR_filterShowForMTE === true) && (rtcModel.attributes.eventId !== mteID)) { rtcStyle = 'hidden'; } if((uFR_filterHideForMTE === true) && (rtcModel.attributes.eventId === mteID)) { rtcStyle = 'hidden'; } } let rtcType = uroGetRTCOrigin(rtcModel); let rtcState = uroGetRTCState(rtcModel); if(rtcType == URO_TRTC.WAZEFEED) { if ( ((rtcState === URO_SRTC.ACTIVE) && (uFR_filterActiveFromWazeFeed === true)) || ((rtcState === URO_SRTC.FUTURE) && (uFR_filterFutureFromWazeFeed === true)) || ((rtcState === URO_SRTC.EXPIRED) && (uFR_filterExpiredFromWazeFeed === true)) ) { rtcStyle = 'hidden'; } } else if(rtcType == URO_TRTC.WAZEOTHER) { if ( ((rtcState === URO_SRTC.ACTIVE) && (uFR_filterActiveFromWazeOther === true)) || ((rtcState === URO_SRTC.FUTURE) && (uFR_filterFutureFromWazeOther === true)) || ((rtcState === URO_SRTC.EXPIRED) && (uFR_filterExpiredFromWazeOther === true)) ) { rtcStyle = 'hidden'; } } else { if ( ((rtcState === URO_SRTC.ACTIVE) && (uFR_filterActiveFromWME === true)) || ((rtcState === URO_SRTC.FUTURE) && (uFR_filterFutureFromWME === true)) || ((rtcState === URO_SRTC.EXPIRED) && (uFR_filterExpiredFromWME === true)) ) { rtcStyle = 'hidden'; } } let rtcDuration = uroGetRTCDuration(rtcModel); if(uFR_filterHideDurationLessThan === true) { if(rtcDuration < uFR_thresholdDurationLessThan) { rtcStyle = 'hidden'; } } if(uFR_filterHideDurationMoreThan === true) { if(rtcDuration > uFR_thresholdDurationMoreThan) { rtcStyle = 'hidden'; } } if((uFR_filterShowForTS === true) || (uFR_filterHideForTS === true)) { let startTS = new Date(rtcModel.attributes.startDate).getTime(); let endTS = new Date(rtcModel.attributes.endDate).getTime(); if(uFR_filterShowForTS === true) { if((filterTS < startTS) || (filterTS > endTS)) { rtcStyle = 'hidden'; } } if (uFR_filterHideForTS === true) { if((filterTS >= startTS) && (filterTS <= endTS)) { rtcStyle = 'hidden'; } } } } W.map.getLayerByName("closures").markers[markerIdx].element.style.visibility = rtcStyle; ++markerIdx; } uroPerformanceMonitoring(pmFunction, pmTStart); } function uroUpdateRAList() { if(Object.keys(W.model.restrictedDrivingAreas.objects).length === 0) return; var selectedIdx = null; var idx; var raNames = []; for(idx in W.model.restrictedDrivingAreas.objects) { if(W.model.restrictedDrivingAreas.objects.hasOwnProperty(idx)) { var name = W.model.restrictedDrivingAreas.objects[idx].attributes.name; if(raNames.indexOf(name) == -1) { raNames.push(name); } } } // check for any previously selected name in the list, then clear it and repopulate // using the newly gathered collection from above, and finally reselect the // previously selected MTE if its still present in the new list... var selector; var selectedName; var selectorEntry; selector = document.getElementById('_selectRA'); selectedName = null; if(selector.selectedOptions[0] != null) { selectedName = selector.selectedOptions[0].value; } while(selector.options.length > 0) { selector.options.remove(0); } selector.options.add(new Option('<select a RA>', null)); if(raNames.length > 0) { selectorEntry = ''; for(idx=0; idx<raNames.length; idx++) { selectorEntry = raNames[idx]; selector.options.add(new Option(selectorEntry, selectorEntry)); if(selectorEntry == selectedName) { selectedIdx = idx+1; } } } if(selectedIdx !== null) { selector.selectedIndex = selectedIdx; } } function uroUpdateEditorList(modelObj, listElement, useCreated, useUpdated, useResolved, useCommenter) { if(Object.keys(modelObj).length === 0) return; var selector = document.getElementById(listElement); 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 modelObj) { if(modelObj.hasOwnProperty(idx)) { var obj; if(useCommenter == true) { obj = modelObj[idx]; if(obj.attributes.comments.length > 0) { for(var cidx=0; cidx < obj.attributes.comments.length; cidx++) { var userID = obj.attributes.comments[cidx].userID; if((listedIDs.indexOf(userID) == -1) && (userID != -1)) { listedIDs.push(userID); } } } } else { obj = modelObj[idx].attributes; var cbID = null; var ubID = null; var rbID = null; if(useCreated == true) cbID = obj.createdBy; if(useUpdated == true) ubID = obj.updatedBy; if(useResolved == true) ubID = obj.resolvedBy; if((cbID !== null) && (listedIDs.indexOf(cbID) == -1)) { listedIDs.push(cbID); } if((ubID !== null) && (listedIDs.indexOf(ubID) == -1)) { listedIDs.push(ubID); } if((rbID !== null) && (listedIDs.indexOf(rbID) == -1)) { listedIDs.push(rbID); } } } } selector.options.add(new Option('<select a user>', null)); if(listedIDs.length > 0) { var users = W.model.users.getByIds(listedIDs); var selectorEntry = ''; for(idx=0; idx<users.length; idx++) { if(listedIDs.indexOf(users[idx].id) != -1) { listedIDs.splice(listedIDs.indexOf(users[idx]), 1); } if(users[idx].attributes.userName === undefined) { selectorEntry = users[idx].attributes.id; } else { selectorEntry = users[idx].attributes.userName; } selector.options.add(new Option(selectorEntry, users[idx].id)); if(users[idx].attributes.id == selectedUser) { selectedIdx = idx+1; } } } if(selectedIdx !== null) { selector.selectedIndex = selectedIdx; } } function uroGetUserID(filterNameID, tbUserName) { if(filterNameID === null) { for(var idx in W.model.users.objects) { if(W.model.users.objects.hasOwnProperty(idx)) { if(W.model.users.objects[idx].attributes.userName == tbUserName) { filterNameID = W.model.users.objects[idx].attributes.id; break; } } } } return filterNameID; } function uroFilterRAs() { var pmTStart = performance.now(); var pmFunction = "uroFilterRAs"; if(uroFilterPreamble() === false) return; var uFURs_masterEnable = uroIsFilteringEnabled(false); var filterByArea = uroGetCBChecked('_cbShowSpecificRA'); var filterByLastEditor = uroGetCBChecked('_cbRAEditorIDFilter'); var filterByMinAge = uroGetCBChecked('_cbEnableRAAgeFilterLessThan'); var filterByMaxAge = uroGetCBChecked('_cbEnableRAAgeFilterMoreThan'); var thresholdMinAge = uroGetElmValue('_inputFilterRAAgeLessThan'); var thresholdMaxAge = uroGetElmValue('_inputFilterRAAgeMoreThan'); var selectorRA = document.getElementById('_selectRA'); if(filterByArea === false) { while(selectorRA.options.length > 0) { selectorRA.options.remove(0); } } var shownRA = null; if(filterByArea === true) { if(selectorRA.options.length === 0) { uroUpdateRAList(); } if(selectorRA.selectedOptions[0] != null) { shownRA = selectorRA.selectedOptions[0].value; } } var filterNameID = null; if(filterByLastEditor == true) { uroUpdateEditorList(W.model.restrictedDrivingAreas.objects, '_selectRAEditorID', true, true, false, false); var selector = document.getElementById('_selectRAEditorID'); if(selector.selectedIndex > 0) { filterNameID = document.getElementById('_selectRAEditorID').selectedOptions[0].value; } } let nRANames = document.querySelectorAll('.restricted-driving-area-name-marker').length; for (var raIdx = 0; raIdx < W.map.restrictedDrivingAreaLayer.features.length; raIdx++) { var raObj = W.map.restrictedDrivingAreaLayer.features[raIdx].attributes.wazeFeature; if(raObj !== undefined) { var raStyle = 'visible'; if(uFURs_masterEnable === true) { if(shownRA !== null) { if(raObj._wmeObject.attributes.name != shownRA) raStyle = 'hidden'; } if(filterNameID !== null) { if((raObj._wmeObject.attributes.createdBy != filterNameID) && (raObj._wmeObject.attributes.updatedBy != filterNameID)) { raStyle = 'hidden'; } } var raAge = uroDateToDays(raObj._wmeObject.attributes.updatedOn); if(filterByMinAge == true) { if(raAge < thresholdMinAge) raStyle = 'hidden'; } if(filterByMaxAge == true) { if(raAge > thresholdMaxAge) raStyle = 'hidden'; } } var geoID = W.map.restrictedDrivingAreaLayer.features[raIdx].geometry.id; if(document.getElementById(geoID) !== null) { document.getElementById(geoID).style.visibility = raStyle; } // This doesn't always work, as the order in which the markers are listed on their layer isn't guaranteed // to match the order in which the corresponding RA polys are listed on theirs... if(raIdx < nRANames) { document.querySelectorAll('.restricted-driving-area-name-marker')[raIdx].style.visibility = raStyle; } } } uroPerformanceMonitoring(pmFunction, pmTStart); } var uro_uFP_filterUneditable; var uro_uFP_filterInsideManagedAreas; var uro_uFP_excludeMyAreas; var uro_uFP_filterLockRanked; var uro_uFP_filterFlagged; var uro_uFP_filterNewPlace; var uro_uFP_filterUpdatedDetails; var uro_uFP_filterNewPhoto; var uro_uFP_filterMinPURAge; var uro_uFP_filterMaxPURAge; var uro_uFP_invertPURFilters; var uro_uFP_filterHighSeverity; var uro_uFP_filterMedSeverity; var uro_uFP_filterLowSeverity; var uro_uFP_leavePURGeos; var uro_uFP_filterCFPhone; var uro_uFP_filterCFName; var uro_uFP_filterCFEntryExitPoints; var uro_uFP_filterCFOpeningHours; var uro_uFP_filterCFAliases; var uro_uFP_filterCFServices; var uro_uFP_filterCFGeometry; var uro_uFP_filterCFHouseNumber; var uro_uFP_filterCFCategories; var uro_uFP_filterCFDescription; var uro_uFP_filterOnCFs; var uro_uFP_thresholdMinPURDays; var uro_uFP_thresholdMaxPURDays; var uro_uFP_isLoggedIn; var uro_uFP_userRank; function uroFilterPlaceMarker(mObj, vObj, uFP_masterEnable) { if((mObj === undefined) || (vObj === undefined)) { return; } var purAge = null; var placeStyle = 'visible'; if(uFP_masterEnable === true) { if(uro_uFP_filterInsideManagedAreas === true) { var tPt = new OpenLayers.Geometry.Point(); tPt.x = mObj.lonlat.lon; tPt.y = mObj.lonlat.lat; if(uroCheckGeometryWithinManagedAreas(tPt) === true) placeStyle = 'hidden'; } if((placeStyle == 'visible') && (uro_uFP_filterUneditable === true)) { if(vObj.attributes.permissions === 0) { placeStyle = 'hidden'; } if((placeStyle == 'visible') && (uro_uFP_isLoggedIn)) { if(uro_uFP_userRank < vObj.attributes.lockRank) { placeStyle = 'hidden'; } } if((placeStyle == 'visible') && (vObj.attributes.adLocked)) { placeStyle = 'hidden'; } } if((placeStyle == 'visible') && (uro_uFP_filterLockRanked === true)) { if(vObj.attributes.lockRank !== 0) { placeStyle = 'hidden'; } } if((placeStyle == 'visible') && (uro_uFP_filterFlagged === true)) { if(mObj.element.className.indexOf('flag') != -1) { placeStyle = 'hidden'; } } if((placeStyle == 'visible') && (uro_uFP_filterNewPlace === true)) { if(mObj.element.className.indexOf('add_venue') != -1) { placeStyle = 'hidden'; } } if((placeStyle == 'visible') && (uro_uFP_filterUpdatedDetails === true)) { if((mObj.element.className.indexOf('update_venue') != -1) || (mObj.element.className.indexOf('multiple') != -1)) { placeStyle = 'hidden'; } } if((placeStyle == 'visible') && (uro_uFP_filterOnCFs === true)) { var nVUR = vObj.attributes.venueUpdateRequests.length; while(nVUR > 0) { nVUR--; var tCF = vObj.attributes.venueUpdateRequests[nVUR].attributes.changedFields; if(tCF !== undefined) { if(tCF.length > 0) { var tFN = tCF[0].attributes.fieldName; if((tFN == "phone") && (uro_uFP_filterCFPhone === true)) { placeStyle = 'hidden'; } if((tFN == "name") && (uro_uFP_filterCFName === true)) { placeStyle = 'hidden'; } if((tFN == "entryExitPoints") && (uro_uFP_filterCFEntryExitPoints === true)) { placeStyle = 'hidden'; } if((tFN == "openingHours") && (uro_uFP_filterCFOpeningHours === true)) { placeStyle = 'hidden'; } if((tFN == "aliases") && (uro_uFP_filterCFAliases === true)) { placeStyle = 'hidden'; } if((tFN == "services") && (uro_uFP_filterCFServices === true)) { placeStyle = 'hidden'; } if((tFN == "geometry") && (uro_uFP_filterCFGeometry === true)) { placeStyle = 'hidden'; } if((tFN == "houseNumber") && (uro_uFP_filterCFHouseNumber === true)) { placeStyle = 'hidden'; } if((tFN == "categories") && (uro_uFP_filterCFCategories === true)) { placeStyle = 'hidden'; } if((tFN == "description") && (uro_uFP_filterCFDescription === true)) { placeStyle = 'hidden'; } } } } } if((placeStyle == 'visible') && (uro_uFP_filterNewPhoto === true)) { if(mObj.element.className.indexOf('add_image') != -1) { placeStyle = 'hidden'; } } if(uro_uFP_invertPURFilters === true) { if(placeStyle == 'hidden') placeStyle = 'visible'; else placeStyle = 'hidden'; } if(uro_uFP_filterMinPURAge || uro_uFP_filterMaxPURAge) { purAge = uroGetPURAge(vObj); if(uro_uFP_filterMinPURAge === true) { if(purAge < uro_uFP_thresholdMinPURDays) placeStyle = 'hidden'; } if(uro_uFP_filterMaxPURAge === true) { if(purAge > uro_uFP_thresholdMaxPURDays) placeStyle = 'hidden'; } } if(placeStyle == 'visible') { if((uro_uFP_filterHighSeverity) && (mObj.element.className.indexOf("high") != -1)) placeStyle = 'hidden'; if((placeStyle == 'visible') && (uro_uFP_filterMedSeverity) && (mObj.element.className.indexOf("medium") != -1)) placeStyle = 'hidden'; if((placeStyle == 'visible') && (uro_uFP_filterLowSeverity) && (mObj.element.className.indexOf("low") != -1)) placeStyle = 'hidden'; } if(uroPURsToHide.indexOf(vObj.attributes.id) !== -1) { placeStyle = 'hidden'; } } mObj.element.style.visibility = placeStyle; if((uro_uFP_leavePURGeos === false) && (placeStyle === 'hidden')) { if(vObj.model != null) { if(vObj.attributes.geometry != null) { var puGeo = document.getElementById(vObj.attributes.geometry.id); if(puGeo !== null) { puGeo.style.visibility = 'hidden'; } } } } } function uroFilterPlaces() { var pmTStart = performance.now(); var pmFunction = "uroFilterPlaces"; if(uroFilterPreamble() === false) return; if(W.selectionManager._mouseInFeature !== null) { var renderIntent = W.selectionManager._mouseInFeature.renderIntent; var moObj = W?.selectionManager?._mouseInFeature?.attributes?.wazeFeature?._wmeObject; if(moObj === undefined) moObj = null; if(moObj != null) { if(moObj.type === 'venue') { if((renderIntent == 'select') || (renderIntent == 'highlightselected')) { return; } } } } if(uroGetCBChecked('_cbDisablePlacesFiltering') === true) return; uroUpdateVenueEditorLists(); 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; } } filterNameID = uroGetUserID(filterNameID, tbUserName); var filterHideNameID = null; var tbHideUserName = uroGetElmValue('_textHidePlacesEditor'); var selectorHide = document.getElementById('_selectHidePlacesUserID'); if(selectorHide.selectedIndex > 0) { var selHideUserName = document.getElementById('_selectHidePlacesUserID').selectedOptions[0].innerHTML; if(selHideUserName == tbHideUserName) { filterHideNameID = document.getElementById('_selectHidePlacesUserID').selectedOptions[0].value; } } filterHideNameID = uroGetUserID(filterHideNameID, tbHideUserName); var filterCats = []; for(var i=0; i<W.Config.venues.categories.length; i++) { var parentCategory = W.Config.venues.categories[i]; var subCategory; if(uroGetCBChecked('_cbPlacesFilter-'+parentCategory) === true) { filterCats.push(parentCategory); for(var i1=0; i1<W.Config.venues.subcategories[parentCategory].length; i1++) { subCategory = W.Config.venues.subcategories[parentCategory][i1]; filterCats.push(subCategory); } } else { for(var i2=0; i2<W.Config.venues.subcategories[parentCategory].length; i2++) { subCategory = W.Config.venues.subcategories[parentCategory][i2]; if(uroGetCBChecked('_cbPlacesFilter-'+subCategory) === true) { filterCats.push(subCategory); } } } } var placeStyle; var uFP_filterEditedLessThan = uroGetCBChecked('_cbPlaceFilterEditedLessThan'); var uFP_filterEditedMoreThan = uroGetCBChecked('_cbPlaceFilterEditedMoreThan'); var uFP_filterL0 = uroGetCBChecked('_cbHidePlacesL0'); var uFP_filterL1 = uroGetCBChecked('_cbHidePlacesL1'); var uFP_filterL2 = uroGetCBChecked('_cbHidePlacesL2'); var uFP_filterL3 = uroGetCBChecked('_cbHidePlacesL3'); var uFP_filterL4 = uroGetCBChecked('_cbHidePlacesL4'); var uFP_filterL5 = uroGetCBChecked('_cbHidePlacesL5'); var uFP_filterStaff = uroGetCBChecked('_cbHidePlacesStaff'); var uFP_filterAL = uroGetCBChecked('_cbHidePlacesAdLocked'); var uFP_filterOnLockLevel = (uFP_filterL0 || uFP_filterL1 || uFP_filterL2 || uFP_filterL3 || uFP_filterL4 || uFP_filterL5 || uFP_filterStaff); var uFP_filterNoPhotos = uroGetCBChecked('_cbHideNoPhotoPlaces'); var uFP_filterWithPhotos = uroGetCBChecked('_cbHidePhotoPlaces'); var uFP_filterNoLinks = uroGetCBChecked('_cbHideNoLinkedPlaces'); var uFP_filterWithLinks = uroGetCBChecked('_cbHideLinkedPlaces'); var uFP_filterNoDescription = uroGetCBChecked('_cbHideNonDescribedPlaces'); var uFP_filterWithDescription = uroGetCBChecked('_cbHideDescribedPlaces'); var uFP_filterNoKeyword = uroGetCBChecked('_cbHideKeywordPlaces'); var uFP_filterKeyword = uroGetCBChecked('_cbHideNoKeywordPlaces'); var uFP_filterPrivate = uroGetCBChecked('_cbFilterPrivatePlaces'); var uFP_invertFilters = uroGetCBChecked('_cbInvertPlacesFilter'); var uFP_masterEnable = uroIsFilteringEnabled(false); var uFP_filterAreaPlaces = uroGetCBChecked('_cbHideAreaPlaces'); var uFP_filterPointPlaces = uroGetCBChecked('_cbHidePointPlaces'); var uFP_filterCreatedBy = uroGetCBChecked('_cbShowOnlyPlacesCreatedBy'); var uFP_filterEditedBy = uroGetCBChecked('_cbShowOnlyPlacesEditedBy'); var uFP_filterHideCreatedBy = uroGetCBChecked('_cbHideOnlyPlacesCreatedBy'); var uFP_filterHideEditedBy = uroGetCBChecked('_cbHideOnlyPlacesEditedBy'); var uFP_hidePURsForFilteredPlaces = uroGetCBChecked('_cbHidePURsForFilteredPlaces'); var uFP_NameKeyword = document.getElementById('_textKeywordPlace').value.toLowerCase(); var uFP_thresholdMinDays = document.getElementById('_inputFilterPlaceEditMinDays').value; var uFP_thresholdMaxDays = document.getElementById('_inputFilterPlaceEditMaxDays').value; uroPURsToHide = []; for(var v=0; v<uroVenueLayer.features.length; v++) { placeStyle = 'visible'; if(uFP_masterEnable === true) { var lmObj = uroVenueLayer.features[v]; // when an area place is selected, the drag points for editing the place outline now get added as objects into uroVenueLayer.features, // however none of these objects had the .attributes.repositoryObject property - whilst the devs have now replaced this with the almost // identical .wazeFeature._wmeObject property, it's unclear if drag points still need to be excluded from this scan, so the check // remains in place as a "let's just make sure it has it before trying to use it"... if(lmObj?.attributes?.wazeFeature?._wmeObject != null) { lmObj = lmObj.attributes.wazeFeature._wmeObject.attributes; if(lmObj.id < 0) { // don't apply filtering to newly-created places - this allows the user to leave their filtering settings unchanged whilst // adding a new place which, once saved, would then be hidden... break; } if(uFP_filterAreaPlaces) { if(lmObj.geometry.id.indexOf('Polygon') !== -1) { placeStyle = 'hidden'; } } if(uFP_filterPointPlaces) { if(lmObj.geometry.id.indexOf('Point') !== -1) { placeStyle = 'hidden'; } } if(placeStyle == 'visible') { if((uFP_filterEditedLessThan) || (uFP_filterEditedMoreThan)) { var editDate = lmObj.updatedOn; if(editDate === undefined) { // where a place has never been edited since its creation, use the creation date instead... editDate = lmObj.createdOn; } if(editDate != null) { 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.lockRank; if ((uFP_filterL0) && (lockLevel === 0)) placeStyle = 'hidden'; if ((uFP_filterL1) && (lockLevel === 1)) placeStyle = 'hidden'; if ((uFP_filterL2) && (lockLevel === 2)) placeStyle = 'hidden'; if ((uFP_filterL3) && (lockLevel === 3)) placeStyle = 'hidden'; if ((uFP_filterL4) && (lockLevel === 4)) placeStyle = 'hidden'; if ((uFP_filterL5) && (lockLevel === 5)) placeStyle = 'hidden'; if ((uFP_filterStaff) && (lockLevel === 6)) placeStyle = 'hidden'; } } if(placeStyle == 'visible') { if(uFP_filterAL) { if(lmObj.adLocked) placeStyle = 'hidden'; } } if(placeStyle == 'visible') { if(uFP_filterNoPhotos || uFP_filterWithPhotos) { var nPhotos = 0; for(var loop=0; loop<lmObj.images.length; loop++) { if(lmObj.images[loop].attributes.approved) nPhotos++; } if((uFP_filterNoPhotos) && (nPhotos === 0)) placeStyle = 'hidden'; if((uFP_filterWithPhotos) && (nPhotos !== 0)) placeStyle = 'hidden'; } } if(placeStyle == 'visible') { if(uFP_filterNoLinks || uFP_filterWithLinks) { var nLinks = lmObj.externalProviderIDs.length; if((uFP_filterNoLinks) && (nLinks === 0)) placeStyle = 'hidden'; if((uFP_filterWithLinks) && (nLinks !== 0)) placeStyle = 'hidden'; } } if(placeStyle == 'visible') { if(uFP_filterNoDescription || uFP_filterWithDescription) { var lDesc = lmObj.description.length; if((uFP_filterNoDescription) && (lDesc === 0)) placeStyle = 'hidden'; if((uFP_filterWithDescription) && (lDesc !== 0)) placeStyle = 'hidden'; } } if(placeStyle == 'visible') { if((uFP_filterPrivate === true) && (lmObj.residential === true)) { placeStyle = 'hidden'; } else { for(var cat=0; cat<filterCats.length; cat++) { if(_.includes(lmObj.categories, filterCats[cat])) { placeStyle = 'hidden'; break; } } } } if(placeStyle == 'visible') { if(uFP_filterNoKeyword || uFP_filterKeyword) { var venueName = lmObj.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.createdBy) placeStyle = 'hidden'; } if(uFP_filterEditedBy === true) { if(filterNameID != lmObj.updatedBy) placeStyle = 'hidden'; } } } if(placeStyle == 'visible') { if(filterHideNameID != null) { if(uFP_filterHideCreatedBy === true) { if(filterHideNameID == lmObj.createdBy) placeStyle = 'hidden'; } if(uFP_filterHideEditedBy === true) { if(filterHideNameID == lmObj.updatedBy) placeStyle = 'hidden'; } } } if(uFP_invertFilters === true) { if(placeStyle == 'hidden') placeStyle = 'visible'; else placeStyle = 'hidden'; } } if((placeStyle == 'hidden') && (uFP_hidePURsForFilteredPlaces === true)) { uroPURsToHide.push(lmObj.id); } } var geoID = uroVenueLayer.features[v].geometry.id; if(document.getElementById(geoID) !== null) { document.getElementById(geoID).style.visibility = placeStyle; } } uro_uFP_filterUneditable = uroGetCBChecked('_cbFilterUneditablePlaceUpdates'); uro_uFP_filterInsideManagedAreas = uroGetCBChecked('_cbPURFilterInsideManagedAreas'); uro_uFP_excludeMyAreas = uroGetCBChecked('_cbPURExcludeUserArea'); uro_uFP_filterLockRanked = uroGetCBChecked('_cbFilterLockRankedPlaceUpdates'); uro_uFP_filterFlagged = uroGetCBChecked("_cbFilterFlaggedPUR"); uro_uFP_filterNewPlace = uroGetCBChecked("_cbFilterNewPlacePUR"); uro_uFP_filterUpdatedDetails = uroGetCBChecked("_cbFilterUpdatedDetailsPUR"); uro_uFP_filterNewPhoto = uroGetCBChecked("_cbFilterNewPhotoPUR"); uro_uFP_filterMinPURAge = uroGetCBChecked('_cbEnablePURMinAgeFilter'); uro_uFP_filterMaxPURAge = uroGetCBChecked('_cbEnablePURMaxAgeFilter'); uro_uFP_invertPURFilters = uroGetCBChecked('_cbInvertPURFilters'); uro_uFP_filterHighSeverity = uroGetCBChecked('_cbPURFilterHighSeverity'); uro_uFP_filterMedSeverity = uroGetCBChecked('_cbPURFilterMediumSeverity'); uro_uFP_filterLowSeverity = uroGetCBChecked('_cbPURFilterLowSeverity'); uro_uFP_leavePURGeos = uroGetCBChecked('_cbLeavePURGeos'); uro_uFP_filterCFPhone = uroGetCBChecked('_cbPURFilterCFPhone'); uro_uFP_filterCFName = uroGetCBChecked('_cbPURFilterCFName'); uro_uFP_filterCFEntryExitPoints = uroGetCBChecked('_cbPURFilterCFEntryExitPoints'); uro_uFP_filterCFOpeningHours = uroGetCBChecked('_cbPURFilterCFOpeningHours'); uro_uFP_filterCFAliases = uroGetCBChecked('_cbPURFilterCFAliases'); uro_uFP_filterCFServices = uroGetCBChecked('_cbPURFilterCFServices'); uro_uFP_filterCFGeometry = uroGetCBChecked('_cbPURFilterCFGeometry'); uro_uFP_filterCFHouseNumber = uroGetCBChecked('_cbPURFilterCFHouseNumber'); uro_uFP_filterCFCategories = uroGetCBChecked('_cbPURFilterCFCategories'); uro_uFP_filterCFDescription = uroGetCBChecked('_cbPURFilterCFDescription'); uro_uFP_filterOnCFs = (uro_uFP_filterCFPhone || uro_uFP_filterCFName || uro_uFP_filterCFEntryExitPoints || uro_uFP_filterCFOpeningHours); uro_uFP_filterOnCFs = (uro_uFP_filterOnCFs || uro_uFP_filterCFAliases || uro_uFP_filterCFServices || uro_uFP_filterCFGeometry); uro_uFP_filterOnCFs = (uro_uFP_filterOnCFs || uro_uFP_filterCFHouseNumber || uro_uFP_filterCFCategories || uro_uFP_filterCFDescription); uro_uFP_thresholdMinPURDays = uroGetElmValue('_inputPURFilterMinDays'); uro_uFP_thresholdMaxPURDays = uroGetElmValue('_inputPURFilterMaxDays'); uro_uFP_isLoggedIn = W.loginManager.isLoggedIn(); uro_uFP_userRank = W.loginManager.user.attributes.rank; uro_uFP_filterInsideManagedAreas = uro_uFP_filterInsideManagedAreas && (uroGetManagedAreas() !== 0); if(uroGetCBChecked('_cbPURExcludeUserArea') == true) { uroIgnoreAreasUserID = W.loginManager.user.attributes.id; } uroPrepForFilterPlaceMarker(URO_TMARKER.PUR, uFP_masterEnable); uroPrepForFilterPlaceMarker(URO_TMARKER.PPUR, uFP_masterEnable); uroPrepForFilterPlaceMarker(URO_TMARKER.RPUR, uFP_masterEnable); uroPerformanceMonitoring(pmFunction, pmTStart); } function uroPrepForFilterPlaceMarker(markerType, masterEnable) { var markerLayer; if(markerType == URO_TMARKER.PUR) { markerLayer = W.map.getLayerByName("place_updates"); } else if(markerType == URO_TMARKER.PPUR) { markerLayer = W.map.getLayerByName("PARKING_PLACE_UPDATES"); } else if(markerType == URO_TMARKER.RPUR) { markerLayer = W.map.getLayerByName("RESIDENTIAL_PLACE_UPDATES"); } else { markerLayer = null; } if((markerLayer !== null) && (markerLayer.getVisibility() === true)) { var pu; var mObj; var vObj; var idList = uroGetMarkerIDs(markerType); for(pu of idList) { mObj = uroGetMarker(markerType, pu); if(mObj !== null) { vObj = W.model.venues.objects[pu]; uroFilterPlaceMarker(mObj, vObj, masterEnable); } } } } function uroGetClosestSegmentToPoint(p) { let retval = null; if(W.map.getZoom() >= 16) { let minDist = 99999999; for(let s in W.model.segments.objects) { if(W.model.segments.objects.hasOwnProperty(s)) { let dist = W.model.segments.getObjectById(s).attributes.geometry.distanceTo(p); if(dist < minDist) { minDist = dist; retval = s; } } } } return retval; } function uroFilterCameras() { let pmTStart = performance.now(); let pmFunction = "uroFilterCameras"; if(uroFilterPreamble() === false) { return; } if(uroMouseIsDown === false) W.map.camerasLayer.redraw(); if(uroIsFilteringEnabled(false) === true) { uroUpdateEditorList(W.model.cameras.objects, '_selectCameraUserID', true, true, false, false); let tbUserName = uroGetElmValue('_textCameraEditor'); let selector = document.getElementById('_selectCameraUserID'); let filterNameID = null; if(selector.selectedIndex > 0) { let selUserName = document.getElementById('_selectCameraUserID').selectedOptions[0].innerHTML; if(selUserName == tbUserName) { filterNameID = document.getElementById('_selectCameraUserID').selectedOptions[0].value; } } filterNameID = uroGetUserID(filterNameID, tbUserName); let isChecked_cbShowOnlyCamsCreatedBy = uroGetCBChecked('_cbShowOnlyCamsCreatedBy'); let isChecked_cbShowOnlyCamsEditedBy = uroGetCBChecked('_cbShowOnlyCamsEditedBy'); let isChecked_cbShowOnlyMyCams = uroGetCBChecked('_cbShowOnlyMyCams'); let isChecked_cbShowWorldCams = uroGetCBChecked('_cbShowWorldCams'); let isChecked_cbShowUSACams = uroGetCBChecked('_cbShowUSACams'); let isChecked_cbShowNonWorldCams = uroGetCBChecked('_cbShowNonWorldCams'); let isChecked_cbShowSpeedCams = uroGetCBChecked('_cbShowSpeedCams'); let isChecked_cbShowRedLightCams = uroGetCBChecked('_cbShowRedLightCams'); let isChecked_cbShowDummyCams = uroGetCBChecked('_cbShowDummyCams'); let isChecked_cbShowIfNoSpeedSet = uroGetCBChecked('_cbShowIfNoSpeedSet'); let isChecked_cbShowIfSpeedSet = uroGetCBChecked('_cbShowIfSpeedSet'); let isChecked_cbShowIfInvalidSpeedSet = uroGetCBChecked('_cbShowIfInvalidSpeedSet'); let isChecked_cbShowRLCIfNoSpeedSet = uroGetCBChecked('_cbShowRLCIfNoSpeedSet'); let isChecked_cbShowRLCIfNonZeroSpeedSet = uroGetCBChecked('_cbShowRLCIfNonZeroSpeedSet'); let isChecked_cbShowRLCIfZeroSpeedSet = uroGetCBChecked('_cbShowRLCIfZeroSpeedSet'); let isChecked_cbHideCreatedByMe = uroGetCBChecked('_cbHideCreatedByMe'); let isChecked_cbHideCreatedByRank0 = uroGetCBChecked('_cbHideCreatedByRank0'); let isChecked_cbHideCreatedByRank1 = uroGetCBChecked('_cbHideCreatedByRank1'); let isChecked_cbHideCreatedByRank2 = uroGetCBChecked('_cbHideCreatedByRank2'); let isChecked_cbHideCreatedByRank3 = uroGetCBChecked('_cbHideCreatedByRank3'); let isChecked_cbHideCreatedByRank4 = uroGetCBChecked('_cbHideCreatedByRank4'); let isChecked_cbHideCreatedByRank5 = uroGetCBChecked('_cbHideCreatedByRank5'); let isChecked_cbHideUpdatedByMe = uroGetCBChecked('_cbHideUpdatedByMe'); let isChecked_cbHideUpdatedByRank0 = uroGetCBChecked('_cbHideUpdatedByRank0'); let isChecked_cbHideUpdatedByRank1 = uroGetCBChecked('_cbHideUpdatedByRank1'); let isChecked_cbHideUpdatedByRank2 = uroGetCBChecked('_cbHideUpdatedByRank2'); let isChecked_cbHideUpdatedByRank3 = uroGetCBChecked('_cbHideUpdatedByRank3'); let isChecked_cbHideUpdatedByRank4 = uroGetCBChecked('_cbHideUpdatedByRank4'); let isChecked_cbHideUpdatedByRank5 = uroGetCBChecked('_cbHideUpdatedByRank5'); let isChecked_cbHideCWLCams = uroGetCBChecked('_cbHideCWLCams'); let isChecked_cbHighlightInsteadOfHideCams = uroGetCBChecked('_cbHighlightInsteadOfHideCams'); let isChecked_InvertFiltere = uroGetCBChecked('_cbInvertCamFilters'); for (let uroCamObj in W.model.cameras.objects) { if(W.model.cameras.objects.hasOwnProperty(uroCamObj)) { let uroCamUpdater = ''; let uroCamUpdaterRank = -1; let uroCamCreator = ''; let uroCamCreatorRank = -1; let uroCam = W.model.cameras.objects[uroCamObj]; let uroCamStyle = 'visible'; let uroCamSpeedString = null; if(uroCam.attributes.createdBy !== null) { if(W.model.users.objects[uroCam.attributes.createdBy] != null) { uroCamCreator = W.model.users.objects[uroCam.attributes.createdBy].attributes.userName; uroCamCreatorRank = W.model.users.objects[uroCam.attributes.createdBy].attributes.rank; } } if(uroCam.attributes.updatedBy !== null) { if(W.model.users.objects[uroCam.attributes.updatedBy] != null) { uroCamUpdater = W.model.users.objects[uroCam.attributes.updatedBy].attributes.userName; uroCamUpdaterRank = W.model.users.objects[uroCam.attributes.updatedBy].attributes.rank; } } let uroCamType = uroCam.attributes.type; if(filterNameID != null) { if(isChecked_cbShowOnlyCamsCreatedBy === true) { if(filterNameID != uroCam.attributes.createdBy) uroCamStyle = 'hidden'; } if(isChecked_cbShowOnlyCamsEditedBy === true) { if(filterNameID != uroCam.attributes.updatedBy) uroCamStyle = 'hidden'; } } if(isChecked_cbShowOnlyMyCams === true) { if((uroUserID != uroCam.attributes.createdBy)&&(uroUserID != uroCam.attributes.updatedBy)) uroCamStyle = 'hidden'; } if((isChecked_cbShowWorldCams === false) || (isChecked_cbShowUSACams === false) || (isChecked_cbShowNonWorldCams === false)) { let posWorld = uroCamCreator.indexOf('world_'); let posUSA = uroCamCreator.indexOf('usa_'); if((isChecked_cbShowWorldCams === false) && (posWorld === 0)) uroCamStyle = 'hidden'; if((isChecked_cbShowUSACams === false) && (posUSA === 0)) uroCamStyle = 'hidden'; if((isChecked_cbShowNonWorldCams === false) && (posWorld !== 0) && (posUSA !== 0)) uroCamStyle = 'hidden'; } if((isChecked_cbShowSpeedCams === false) || (isChecked_cbShowRedLightCams === false) || (isChecked_cbShowDummyCams === false)) { if((isChecked_cbShowSpeedCams === false) && (uroCamType == 2)) uroCamStyle = 'hidden'; if((isChecked_cbShowRedLightCams === false) && (uroCamType == 4)) uroCamStyle = 'hidden'; if((isChecked_cbShowDummyCams === false) && (uroCamType == 3)) uroCamStyle = 'hidden'; } uroCamSpeedString = uroGetLocalisedSpeedString(uroCam.attributes.speed, true); if((isChecked_cbShowSpeedCams === true) && (uroCamType == 2)) { if((isChecked_cbShowIfNoSpeedSet === false) && (uroCam.attributes.speed === null)) uroCamStyle = 'hidden'; if((isChecked_cbShowIfSpeedSet === false) && (uroCam.attributes.speed !== null)) uroCamStyle = 'hidden'; if(isChecked_cbShowIfInvalidSpeedSet === false) { let cPoint = uroCam.attributes.geometry.getCentroid(); let nSeg = uroGetClosestSegmentToPoint(cPoint); if(nSeg !== null) { let fwdSpeed = W.model.segments.getObjectById(nSeg).attributes.fwdMaxSpeed; let revSpeed = W.model.segments.getObjectById(nSeg).attributes.revMaxSpeed; let camSpeed = uroCam.attributes.speed; if(W.model.isImperial == true) { fwdSpeed = Math.round(fwdSpeed / 1.609); revSpeed = Math.round(revSpeed / 1.609); camSpeed = Math.round(camSpeed / 1.609); } if((camSpeed !== fwdSpeed) && (camSpeed !== revSpeed)) { uroCamStyle = 'hidden'; } } } } if((isChecked_cbShowRedLightCams === true) && (uroCamType == 4)) { if((isChecked_cbShowRLCIfNoSpeedSet === false) && (uroCam.attributes.speed === null)) uroCamStyle = 'hidden'; if((isChecked_cbShowRLCIfNonZeroSpeedSet === false) && (uroCam.attributes.speed > 0)) uroCamStyle = 'hidden'; if((isChecked_cbShowRLCIfZeroSpeedSet === false) && (uroCam.attributes.speed === 0)) uroCamStyle = 'hidden'; } if(isChecked_cbHideCreatedByMe === true) { if(uroUserID == uroCam.attributes.createdBy) uroCamStyle = 'hidden'; } if((isChecked_cbHideCreatedByRank0 === true) && (uroCamCreatorRank === 0)) uroCamStyle = 'hidden'; if((isChecked_cbHideCreatedByRank1 === true) && (uroCamCreatorRank == 1)) uroCamStyle = 'hidden'; if((isChecked_cbHideCreatedByRank2 === true) && (uroCamCreatorRank == 2)) uroCamStyle = 'hidden'; if((isChecked_cbHideCreatedByRank3 === true) && (uroCamCreatorRank == 3)) uroCamStyle = 'hidden'; if((isChecked_cbHideCreatedByRank4 === true) && (uroCamCreatorRank == 4)) uroCamStyle = 'hidden'; if((isChecked_cbHideCreatedByRank5 === true) && (uroCamCreatorRank == 5)) uroCamStyle = 'hidden'; if(isChecked_cbHideUpdatedByMe === true) { if(uroUserID == uroCam.attributes.updatedBy) uroCamStyle = 'hidden'; } if((isChecked_cbHideUpdatedByRank0 === true) && (uroCamUpdaterRank === 0)) uroCamStyle = 'hidden'; if((isChecked_cbHideUpdatedByRank1 === true) && (uroCamUpdaterRank == 1)) uroCamStyle = 'hidden'; if((isChecked_cbHideUpdatedByRank2 === true) && (uroCamUpdaterRank == 2)) uroCamStyle = 'hidden'; if((isChecked_cbHideUpdatedByRank3 === true) && (uroCamUpdaterRank == 3)) uroCamStyle = 'hidden'; if((isChecked_cbHideUpdatedByRank4 === true) && (uroCamUpdaterRank == 4)) uroCamStyle = 'hidden'; if((isChecked_cbHideUpdatedByRank5 === true) && (uroCamUpdaterRank == 5)) uroCamStyle = 'hidden'; if((isChecked_cbHideCWLCams === true) && (uroIsCamOnWatchList(uroCam.attributes.id) != -1)) uroCamStyle = 'hidden'; if(isChecked_InvertFiltere === true) { if(uroCamStyle == "hidden") { uroCamStyle = ""; } else { uroCamStyle = "hidden"; } } let uroCamGeometryID = uroCam.geometry.id; let svgElm = document.getElementById(uroCamGeometryID); if(svgElm !== null) { let origImage; if(uroCamStyle == "hidden") { if(isChecked_cbHighlightInsteadOfHideCams === true) { // set the "highlight" camera image here... let hrefImage = svgElm.getAttribute("xlink:href"); origImage = svgElm.getAttribute("origImage"); if((hrefImage === origImage) || (origImage === null)) { svgElm.setAttribute("origImage", hrefImage); svgElm.setAttribute("xlink:href", uroHighlightedCameraImages[(uroCamType-2)]); svgElm.addEventListener("mouseover", uroMarkerMouseOver, false); svgElm.addEventListener("mouseout", uroMarkerMouseOut, false); } } else { svgElm.remove(); } } else { // restore the original camera image here... if(svgElm.getAttribute("origImage") !== null) { origImage = svgElm.getAttribute("origImage"); svgElm.setAttribute("xlink:href", origImage); svgElm.removeAttribute("origImage"); } } } } } uroCamLayerChanged(); } uroPerformanceMonitoring(pmFunction, pmTStart); } function uroFilterMapComments() { var pmTStart = performance.now(); var pmFunction = "uroFilterMapComments"; if(uroFilterPreamble() === false) return; var uFURs_masterEnable = uroIsFilteringEnabled(false); var filterDescMustBePresent = uroGetCBChecked('_cbMCDescriptionMustBePresent'); var filterDescMustBeAbsent = uroGetCBChecked('_cbMCDescriptionMustBeAbsent'); var filterKeywordMustBePresent = uroGetCBChecked('_cbMCEnableKeywordMustBePresent'); var filterKeywordMustBeAbsent = uroGetCBChecked('_cbMCEnableKeywordMustBeAbsent'); var filterMyFollowed = uroGetCBChecked('_cbMCHideMyFollowed'); var filterMyUnfollowed = uroGetCBChecked('_cbMCHideMyUnfollowed'); var filterRoadworks = uroGetCBChecked('_cbMCFilterRoadworks'); var filterConstruction = uroGetCBChecked('_cbMCFilterConstruction'); var filterClosure = uroGetCBChecked('_cbMCFilterClosure'); var filterEvent = uroGetCBChecked('_cbMCFilterEvent'); var filterNote = uroGetCBChecked('_cbMCFilterNote'); var filterWSLM = uroGetCBChecked('_cbMCFilterWSLM'); var filterBOG = uroGetCBChecked('_cbMCFilterBOG'); var filterDifficult = uroGetCBChecked('_cbMCFilterDifficult'); var invertFilters = uroGetCBChecked('_cbInvertMCFilter'); var keywordPresent = uroGetElmValue('_textMCKeywordPresent'); var keywordAbsent = uroGetElmValue('_textMCKeywordAbsent'); var caseInsensitive = uroGetCBChecked('_cbMCCaseInsensitive'); var filterCommentsMustBePresent = uroGetCBChecked('_cbMCCommentsMustBePresent'); var filterCommentsMustBeAbsent = uroGetCBChecked('_cbMCCommentsMustBeAbsent'); var filterExpiryMustBePresent = uroGetCBChecked('_cbMCExpiryMustBePresent'); var filterExpiryMustBeAbsent = uroGetCBChecked('_cbMCExpiryMustBeAbsent'); var filterByCreatorEnable = uroGetCBChecked('_cbMCCreatorIDFilter'); var filterL1 = uroGetCBChecked('_cbHideMCRank0'); var filterL2 = uroGetCBChecked('_cbHideMCRank1'); var filterL3 = uroGetCBChecked('_cbHideMCRank2'); var filterL4 = uroGetCBChecked('_cbHideMCRank3'); var filterL5 = uroGetCBChecked('_cbHideMCRank4'); var filterL6 = uroGetCBChecked('_cbHideMCRank5'); var filterWRCMC = uroGetCBChecked('_cbHideWRCMCs'); var selectorCreator = document.getElementById('_selectMCCreatorID'); if(filterByCreatorEnable === false) { while(selectorCreator.options.length > 0) { selectorCreator.options.remove(0); } } var creatorUser = null; if(filterByCreatorEnable === true) { if(selectorCreator.options.length === 0) { uroUpdateEditorList(W.model.mapComments.objects, '_selectMCCreatorID', true, false, false, false); } if(selectorCreator.selectedOptions[0] != null) { creatorUser = parseInt(selectorCreator.selectedOptions[0].value); } } for (var mcIdx = 0; mcIdx < uroMCLayer.features.length; mcIdx++) { { var mcObj = uroMCLayer?.features[mcIdx]?.attributes?.wazeFeature?._wmeObject; if(mcObj !== undefined) { var desc = ''; if(mcObj.attributes.subject !== null) desc += mcObj.attributes.subject.replace(/<\/?[^>]+(>|$)/g, ""); if(mcObj.attributes.body !== null) desc += mcObj.attributes.body.replace(/<\/?[^>]+(>|$)/g, ""); var nComments = mcObj.attributes.conversation.length; if(nComments > 0) { for(var cIdx=0; cIdx < nComments; cIdx++) { desc += mcObj.attributes.conversation[cIdx].text.replace(/<\/?[^>]+(>|$)/g, ""); } } var mcStyle = 'visible'; if(uroIsOnIgnoreList(mcObj.attributes.id)) mcStyle = 'hidden'; if(uFURs_masterEnable === true) { var ukroadworks_ur = false; var construction_ur = false; var closure_ur = false; var event_ur = false; var note_ur = false; var wslm_ur = false; var bog_ur = false; var difficult_ur = false; var filterByNotIncludedKeyword = false; var filterByIncludedKeyword = true; var customType = uroGetCustomType(null, "mc", desc); if(customType === 0) ukroadworks_ur = true; else if(customType === 1) construction_ur = true; else if(customType === 2) closure_ur = true; else if(customType === 3) event_ur = true; else if(customType === 4) note_ur = true; else if(customType === 5) wslm_ur = true; else if(customType === 6) bog_ur = true; else if(customType === 7) difficult_ur = true; var rank = mcObj.attributes.lockRank; var expiry = mcObj.attributes.endDate; // keywords if(mcStyle == 'visible') { if(filterDescMustBePresent === true) { if(desc === '') mcStyle = 'hidden'; } if(filterDescMustBeAbsent === true) { if(desc !== '') mcStyle = 'hidden'; } if(filterCommentsMustBePresent === true) { if(nComments === 0) mcStyle = 'hidden'; } if(filterCommentsMustBeAbsent === true) { if(nComments > 0) mcStyle = 'hidden'; } if(filterKeywordMustBePresent === true) { var keywordIsPresentInDesc = uroKeywordPresent(desc,keywordPresent,caseInsensitive); filterByIncludedKeyword = (filterByIncludedKeyword && (!keywordIsPresentInDesc)); } if(filterKeywordMustBeAbsent === true) { var keywordIsAbsentInDesc = uroKeywordPresent(desc,keywordAbsent,caseInsensitive); filterByNotIncludedKeyword = (filterByNotIncludedKeyword || keywordIsAbsentInDesc); } filterByNotIncludedKeyword = (filterByNotIncludedKeyword && filterKeywordMustBeAbsent); filterByIncludedKeyword = (filterByIncludedKeyword && filterKeywordMustBePresent); if(filterByNotIncludedKeyword || filterByIncludedKeyword) { mcStyle = 'hidden'; } } //lock rank if(mcStyle == 'visible') { if((filterL1 === true) && (rank == 0)) mcStyle = 'hidden'; if((filterL2 === true) && (rank == 1)) mcStyle = 'hidden'; if((filterL3 === true) && (rank == 2)) mcStyle = 'hidden'; if((filterL4 === true) && (rank == 3)) mcStyle = 'hidden'; if((filterL5 === true) && (rank == 4)) mcStyle = 'hidden'; if((filterL6 === true) && (rank == 5)) mcStyle = 'hidden'; } // expiry if(mcStyle == 'visible') { if((filterExpiryMustBePresent === true) && (expiry === null)) mcStyle = 'hidden'; if((filterExpiryMustBeAbsent === true) && (expiry != null)) mcStyle = 'hidden'; } // is following? if(mcStyle == 'visible') { if(mcObj.attributes.isFollowing === true) { if(filterMyFollowed === true) mcStyle = 'hidden'; } else { if(filterMyUnfollowed === true) mcStyle = 'hidden'; } } if(mcStyle == 'visible') { if(creatorUser !== null) { if(mcObj.attributes.createdBy != creatorUser) mcStyle = 'hidden'; } if(filterWRCMC === true) { if(mcObj.attributes.createdBy == 304740435) mcStyle = 'hidden'; } } // custom tags if(mcStyle == 'visible') { if(ukroadworks_ur === true) { if(filterRoadworks === true) mcStyle = 'hidden'; } else if(construction_ur === true) { if(filterConstruction === true) mcStyle = 'hidden'; } else if(closure_ur === true) { if(filterClosure === true) mcStyle = 'hidden'; } else if(event_ur === true) { if(filterEvent === true) mcStyle = 'hidden'; } else if(note_ur === true) { if(filterNote === true) mcStyle = 'hidden'; } else if(wslm_ur === true) { if(filterWSLM === true) mcStyle = 'hidden'; } else if(bog_ur === true) { if(filterBOG === true) mcStyle = 'hidden'; } else if(difficult_ur === true) { if(filterDifficult === true) mcStyle = 'hidden'; } if(invertFilters === true) { if(mcStyle == 'hidden') mcStyle = 'visible'; else mcStyle = 'hidden'; } } } var geoID = uroMCLayer.features[mcIdx].geometry.id; if(document.getElementById(geoID) !== null) { document.getElementById(geoID).style.visibility = mcStyle; } } } } uroPerformanceMonitoring(pmFunction, pmTStart); } function uroFilterURs_onObjectsChanged() { if(uroFilterPreamble()) { if(uroBackfilling === false) { if(uroURDialogIsOpen === false) { uroURBackfill(); } else { uroFilterURs(); } } } } function uroFilterURs_onObjectsAdded() { if(uroFilterPreamble()) { if(uroBackfilling === false) { uroURBackfill(); } } } function uroFilterURs_onObjectsRemoved() { if(uroFilterPreamble()) { if(uroBackfilling === false) { uroURBackfill(); } } } function uroBackfillQueueObj(lon, lat, blockSize) { this.lon = lon; this.lat = lat; this.blockSize = blockSize; } function uroURBackfill_GetData() { if(uroBackfillQueue.length === 0) { uroBackfilling = false; uroFilterURs(); return; } 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 += W.Config.api_base; tURL += '/Features?language=en&mapUpdateRequestFilter=3'; tURL += '&bbox='+(lon)+','+(lat)+','+(lon + blockSize)+','+(lat + blockSize); backfillReq.open('GET',tURL,true); backfillReq.send(); } function uroURBackfill() { if((uroGetCBChecked('_cbURBackfill') === false) || (uroGetCBChecked('_cbMasterEnable') === false)) { uroFilterURs(); return; } 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 uroGetManagedAreas() { uroManagedAreas = []; uroIgnoreAreasUserID = null; for(var maObj in W.model.managedAreas.objects) { if(W.model.managedAreas.objects.hasOwnProperty(maObj)) { uroManagedAreas.push(W.model.managedAreas.objects[maObj]); } } return uroManagedAreas.length; } function uroCheckGeometryWithinManagedAreas(geo) { var retval = false; var ignoreUserMA = false; // If we're ignoring the user's managed area, then we first check to see if // the geopoint lies within that - if so then we can skip checking all the // other areas in the list... if(uroIgnoreAreasUserID !== null) { for(var uma = 0; uma < uroManagedAreas.length; ++uma) { if(uroIgnoreAreasUserID == uroManagedAreas[uma].userID) { if(uroManagedAreas[uma].geometry.containsPoint !== undefined) { if(uroManagedAreas[uma].geometry.containsPoint(geo) == true) { ignoreUserMA = true; } } break; } } } // Point either isn't within the user's area, or we're not ignoring it, so // check the rest of the areas in the list if(ignoreUserMA == false) { for(var ma = 0; ma < uroManagedAreas.length; ++ma) { if(uroIgnoreAreasUserID != uroManagedAreas[ma].userID) { if(uroManagedAreas[ma].geometry.containsPoint !== undefined) { if(uroManagedAreas[ma].geometry.containsPoint(geo) == true) { retval = true; break; } } } } } return retval; } function uroGetURDriveGeoms() { let retval = []; for (let urobj in W.model.mapUpdateRequests.objects) { if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj)) { let ureq = W.model.mapUpdateRequests.objects[urobj]; let ureqID = ureq.attributes.id; let hasGeo = false; let thisRet = []; thisRet.push(ureqID); thisRet.push(null); thisRet.push([]); let latMin = 9999; let latMax = -9999; let lonMin = 9999; let lonMax = -9999; let urs = W.model.updateRequestSessions.objects[ureqID]; if((urs !== undefined) && (urs.attributes.driveGeometry !== undefined)) { let cPairs = []; for(let i = 0; i < urs.attributes.driveGeometry.coordinates.length; ++i) { for(let j = 0; j < urs.attributes.driveGeometry.coordinates[i].length; ++j) { if((i === 0) || (j > 0)) { let coords = urs.attributes.driveGeometry.coordinates[i][j]; cPairs.push(coords); if(coords[0] > lonMax) { lonMax = coords[0]; } if(coords[0] < lonMin) { lonMin = coords[0]; } if(coords[1] > latMax) { latMax = coords[1]; } if(coords[1] < latMin) { latMin = coords[1]; } hasGeo = true; } } } let bbox = []; bbox.push(lonMin); bbox.push(lonMax); bbox.push(latMin); bbox.push(latMax); thisRet.push(bbox); thisRet.push(cPairs); } if(hasGeo === true) { retval.push(thisRet); } } } return retval; } function uroCompareDriveGeos(geoA, geoB) { const matchLength = 5; let retval = false; if((geoA.length >= matchLength) && (geoB.length >= matchLength)) { for(let i = 0; i < (geoA.length - matchLength); ++i) { for(let j = 0; j < (geoB.length - matchLength); ++j) { if((geoA[i][0] == geoB[j][0]) && (geoA[i][1] == geoB[j][1])) { retval = true; for(let k = 1; k < matchLength; ++k) { if((geoA[i+k][0] != geoB[j+k][0]) || (geoA[i+k][1] != geoB[j+k][1])) { retval = false; break; } } } if(retval === true) { break; } } if(retval === true) { break; } } } return retval; } function uroCompareDriveBBoxes(bbA, bbB) { let retval = true; if ( (bbA[0] > bbB[1]) || (bbA[1] < bbB[0]) || (bbA[2] > bbB[3]) || (bbA[3] < bbB[2]) ) { retval = false; } return retval; } function uroGetURDupes() { uroURDupes = []; // To determine which URs are duplicates of one another (i.e. have been raised by the same user // during the same section of a journey), we first compare the geometries of the drive tracks // attached to any URs which have them - as this is based on the users GPS position rather than // the WME map data, it makes it vanishingly unlikely that any two users would have identical // GPS positions (especially given the level of accuracy to which the track points are stored) // even if they were driving exactly the same route at the same speed, in the same lane etc. // // To accelerate this geometry comparison, we start by performing a simple bounding box overlap // check for the two geometries under consideration - if there's no overlap then there can't be // any geometry match, so no need to continue onto the more detailed comparision of the GPS // tracks themselves... let driveGeos = uroGetURDriveGeoms(); if(driveGeos.length > 1) { for(let i = 0; i < (driveGeos.length - 1); ++i) { if(driveGeos[i].length !== 5) { driveGeos[i][1] = false; } else { for(let j = (i + 1); j < driveGeos.length; ++j) { if(driveGeos[j].length === 5) { let geoMatch = uroCompareDriveBBoxes(driveGeos[i][3], driveGeos[j][3]); if(geoMatch === true) { geoMatch = uroCompareDriveGeos(driveGeos[i][4], driveGeos[j][4]); if(geoMatch === true) { driveGeos[i][2].push(driveGeos[j][0]); driveGeos[j][2].push(driveGeos[i][0]); } } } } } } for(let i = 0; i < driveGeos.length; ++i) { if(driveGeos[i][2].length > 0) { let res = []; res.push(driveGeos[i][0]); res.push(driveGeos[i][2]); uroURDupes.push(res); } } } // Once we've done the initial drive track comparision, uroURDupes will contain a list of // all the URs which were matched up based on that. However, as the track comparision is // inherently limited by the amount of track data included with each UR, this initial // comparison may mean some more widely spaced URs aren't flagged as duplicates simply // because there's insufficient overlap between their tracks, even though there may have // been further URs dropped inbetween to which they were matched. // // e.g. if a user drops 4 URs along a section of their journey, spaced such that each of // the GPS tracks generates a comparison match with the UR either side of it, but no // further than that, we would get: // // UR A......UR B.......UR C......UR D // // Matches: A to B, B to A & C, C to B & D, D to C // // Note how, although we know all of these URs are in fact duplicates, and can infer this // from seeing that e.g. A is flagged only as a duplicate of B, however as B is flagged as // a duplicate of A & C, and C is flagged as a duplicate of B & D, A MUST therefore be a // duplicate of B, C & D... // // To fix this, we now run a merging pass over each of the entries in uroURDupes - for // each entry we see if its ID appears in any of the other entries as a duplicate, and // if so we merge the duplicates for both entries. for(let i = 0; i < uroURDupes.length; ++i) { let urID = uroURDupes[i][0]; for(let j = 0; j < uroURDupes.length; ++j) { if(i != j) { if(uroURDupes[j][1].includes(urID) === true) { // https://stackoverflow.com/questions/1584370/how-to-merge-two-arrays-in-javascript-and-de-duplicate-items let mergedDupes = [...new Set([...uroURDupes[i][1], ...uroURDupes[j][1]])]; uroURDupes[i][1] = mergedDupes; uroURDupes[j][1] = mergedDupes; } } } } } function uroFilterURs() { var pmTStart = performance.now(); var pmFunction = "uroFilterURs"; if(uroUserID === -1) { return; } if(uroInhibitURFiltering === true) { return; } // compatibility fix for URComments - based on code supplied by RickZabel var hasActiveURFilters = false; if(uroIsFilteringEnabled(false) === true) { var urTabInputs = uroCtrlTabs[URO_TABS_ID.URS][URO_TABS_FIELD.TABBODY].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) { uroUpdateEditorList(W.model.updateRequestSessions.objects, '_selectURUserID', false, false, false, true); } if(selectorCommentUser.selectedOptions[0] != null) { commenterUser = parseInt(selectorCommentUser.selectedOptions[0].value); } } var resolverUser = null; if(uroGetCBChecked('_cbURResolverIDFilter') === true) { if(selectorResolver.options.length === 0) { uroUpdateEditorList(W.model.mapUpdateRequests.objects, '_selectURResolverID', false, false, true, false); } if(selectorResolver.selectedOptions[0] != null) { resolverUser = parseInt(selectorResolver.selectedOptions[0].value); } } uroCustomMarkerList = []; uroGetURDupes(); var uFURs_masterEnable = uroIsFilteringEnabled(false); var filterOutsideEditableArea = uroGetCBChecked('_cbURFilterOutsideArea'); var filterInsideManagedAreas = uroGetCBChecked('_cbURFilterInsideManagedAreas'); 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 showOnlyDupes = uroGetCBChecked('_cbURFilterDupes'); var keywordPresent = uroGetElmValue('_textKeywordPresent'); var keywordAbsent = uroGetElmValue('_textKeywordAbsent'); var caseInsensitive = uroGetCBChecked('_cbCaseInsensitive'); var thresholdMinAge = uroGetElmValue('_inputFilterMinDays'); var thresholdMaxAge = uroGetElmValue('_inputFilterMaxDays'); var thresholdMinComments = uroGetElmValue('_inputFilterMinComments'); var thresholdMaxComments = uroGetElmValue('_inputFilterMaxComments'); var thresholdMaxCommentAge = uroGetElmValue('_inputFilterCommentDays'); var thresholdMinCommentAge = uroGetElmValue('_inputFilterCommentDays2'); var ignoreOtherEditorComments = uroGetCBChecked('_cbIgnoreOtherEditorComments'); var urcFilteringIsActive = false; var urcCB = document.getElementById('URCommentsFilterEnabled'); if(urcCB !== null) { if(urcCB.checked) { urcFilteringIsActive = true; } } urcCB = document.getElementById('URCommentUROOnlyMyUR'); if(urcCB !== null) { if(urcCB.checked) { urcFilteringIsActive = true; } } urcCB = document.getElementById('URCommentUROHideTagged'); if(urcCB !== null) { if(urcCB.checked) { urcFilteringIsActive = true; } } filterInsideManagedAreas = filterInsideManagedAreas && (uroGetManagedAreas() !== 0); if(uroGetCBChecked('_cbURExcludeUserArea') == true) { uroIgnoreAreasUserID = W.loginManager.user.attributes.id; } for (var urobj in W.model.mapUpdateRequests.objects) { if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj)) { var ureq = W.model.mapUpdateRequests.objects[urobj]; var ureqID = ureq.attributes.id; var urStyle = 'visible'; var inhibitFiltering = ((ureqID == uroSelectedURID) && (noFilterURInURL)); var hasMyComments = false; var nComments = 0; var desc = ureq.attributes.description; var customType = uroGetCustomType(ureqID, URO_TMARKER.UR, desc); if(W.model.updateRequestSessions.objects[ureqID] != null) { nComments = W.model.updateRequestSessions.objects[ureqID].attributes.comments.length; if((uFURs_masterEnable === false) && (nComments === 0)) { // when master enable is turned off, we want to make sure that all URs, including ones that were previously hidden, are correctly // displayed in their native form - i.e. no comment count or custom conversation bubbles. The easiest way to achieve this is to // force the uroRenderCustomMarkers code to test for the presence of these bubbles on each UR, which we do by setting a non-zero // comment count for each UR... For URs which genuinely do have no comments we use -1 to indicate that we're not really setting // a comment count, but that we still need to do something that wouldn't be achieved by using 0. nComments = -1; } } // check UR against current session ignore list... if(uroIsOnIgnoreList(ureqID)) urStyle = 'hidden'; if((uFURs_masterEnable === true) && (inhibitFiltering === false)) { var wazeauto_ur = false; var ukroadworks_ur = false; var construction_ur = false; var closure_ur = false; var event_ur = false; var note_ur = false; var wslm_ur = false; var bog_ur = false; var difficult_ur = false; var filterByNotIncludedKeyword = false; var filterByIncludedKeyword = true; if(desc !== null) desc = desc.replace(/<\/?[^>]+(>|$)/g, ""); else desc = ''; if(customType === 0) ukroadworks_ur = true; else if(customType === 1) construction_ur = true; else if(customType === 2) closure_ur = true; else if(customType === 3) event_ur = true; else if(customType === 4) note_ur = true; else if(customType === 5) wslm_ur = true; else if(customType === 6) bog_ur = true; else if(customType === 7) difficult_ur = true; // check UR against editable area... if(filterOutsideEditableArea === true) { if(ureq.canEdit() === false) urStyle = 'hidden'; } if(filterInsideManagedAreas === true) { if(uroCheckGeometryWithinManagedAreas(ureq.attributes.geometry) === true) urStyle = 'hidden'; } if(showOnlyDupes === true) { let isDupe = false; for(let i = 0; i < uroURDupes.length; ++i) { if(uroURDupes[i][0] === ureqID) { isDupe = true; break; } } if(isDupe === false) urStyle = 'hidden'; } // state-age filtering if(urStyle == 'visible') { // check against closed/not identified filtering if enabled... if(filterSolved === true) { if(ureq.attributes.resolution === 0) urStyle = 'hidden'; } if(filterUnidentified === true) { if(ureq.attributes.resolution == 1) urStyle = 'hidden'; } if((ureq.attributes.resolvedOn !== null) && (filterClosed === true)) { urStyle = 'hidden'; } if((ureq.attributes.resolvedOn === null) && (filterOpen === true)) { urStyle = 'hidden'; } if(urStyle == 'visible') { // check UR against keyword filtering if enabled... if(filterDescMustBePresent === true) { if(desc === '') urStyle = 'hidden'; } if(filterDescMustBeAbsent === true) { if(desc !== '') urStyle = 'hidden'; } if(filterKeywordMustBePresent === true) { var keywordIsPresentInDesc = uroKeywordPresent(desc,keywordPresent,caseInsensitive); filterByIncludedKeyword = (filterByIncludedKeyword && (!keywordIsPresentInDesc)); } if(filterKeywordMustBeAbsent === true) { var keywordIsAbsentInDesc = uroKeywordPresent(desc,keywordAbsent,caseInsensitive); filterByNotIncludedKeyword = (filterByNotIncludedKeyword || keywordIsAbsentInDesc); } } if(urStyle == 'visible') { // do age-based filtering if enabled if(filterMinURAge === true) { if(uroGetURAge(ureq,0,false) < thresholdMinAge) urStyle = 'hidden'; } if(filterMaxURAge === true) { if(uroGetURAge(ureq,0,false) > thresholdMaxAge) urStyle = 'hidden'; } } if(urStyle == 'visible') { if(resolverUser !== null) { if(ureq.attributes.resolvedBy != resolverUser) urStyle = 'hidden'; } } if(urStyle == 'visible') { // do comments/following filtering if(W.model.updateRequestSessions.objects[ureqID] != null) { nComments = W.model.updateRequestSessions.objects[ureqID].attributes.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].attributes.comments[nComments-1].userID == -1) reporterIsLastCommenter = true; if(filterReporterLastCommenter === true) { if(reporterIsLastCommenter === true) urStyle = 'hidden'; } else if(filterReporterNotLastCommenter === true) { if(reporterIsLastCommenter === false) urStyle = 'hidden'; } hasMyComments = uroURHasMyComments(ureqID); if(hasMyComments === false) { if(filterHideAnyComments === true) urStyle = 'hidden'; if(filterHideNotLastCommenter === true) urStyle = 'hidden'; } else { if(filterHideMyComments === true) urStyle = 'hidden'; var userIsLastCommenter = false; if(W.model.updateRequestSessions.objects[ureqID].attributes.comments[nComments-1].userID == uroUserID) userIsLastCommenter = true; if(filterIfLastCommenter === true) { if(userIsLastCommenter === true) urStyle = 'hidden'; } else if(filterIfNotLastCommenter === true) { if(userIsLastCommenter === false) urStyle = 'hidden'; } } var cidx; if(ignoreOtherEditorComments === false) { commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].attributes.comments[nComments-1]); } else { for(cidx=0; cidx<nComments; cidx++) { var cObj = W.model.updateRequestSessions.objects[ureqID].attributes.comments[cidx]; if((cObj.userID == uroUserID) || (cObj.userID == -1)) { commentDaysOld = uroGetCommentAge(cObj); } } } if((filterCommentMinAge === true) && (commentDaysOld != -1)) { if(thresholdMinCommentAge > commentDaysOld) urStyle = 'hidden'; } if((filterCommentMaxAge === true) && (commentDaysOld != -1)) { if(thresholdMaxCommentAge < commentDaysOld) urStyle = 'hidden'; } if((commenterUser !== null) && (urStyle != 'hidden')) { urStyle = 'hidden'; for(cidx=0; cidx<nComments; cidx++) { if(W.model.updateRequestSessions.objects[ureqID].attributes.comments[cidx].userID == commenterUser) { urStyle = 'visible'; break; } } } var commentText = ''; for(cidx=0; cidx<nComments; cidx++) { commentText += W.model.updateRequestSessions.objects[ureqID].attributes.comments[cidx].text; } if(filterKeywordMustBePresent === true) { var keywordIsPresentInComments = uroKeywordPresent(commentText,keywordPresent,caseInsensitive); filterByIncludedKeyword = (filterByIncludedKeyword && (!keywordIsPresentInComments)); } if(filterKeywordMustBeAbsent === true) { var keywordIsAbsentInComments = uroKeywordPresent(commentText,keywordAbsent,caseInsensitive); filterByNotIncludedKeyword = (filterByNotIncludedKeyword || keywordIsAbsentInComments); } } else { if(filterUserID === true) { urStyle = 'hidden'; } } filterByNotIncludedKeyword = (filterByNotIncludedKeyword && filterKeywordMustBeAbsent); filterByIncludedKeyword = (filterByIncludedKeyword && filterKeywordMustBePresent); if(filterByNotIncludedKeyword || filterByIncludedKeyword) { urStyle = 'hidden'; } if(W.model.updateRequestSessions.objects[ureqID].attributes.isFollowing === true) { if(filterMyFollowed === true) urStyle = 'hidden'; } else { if(filterMyUnfollowed === true) urStyle = 'hidden'; } } } if(invertURStateFilters === true) { if(urStyle == 'hidden') urStyle = 'visible'; else urStyle = 'hidden'; } } // type filtering if(urStyle == 'visible') { // Test for Waze automatic URs before any others - these always (?) get inserted as General Error URs, // so we can't filter them by type... if(desc.indexOf('Waze Automatic:') != -1) { wazeauto_ur = true; } if(wazeauto_ur === true) { if(filterWazeAuto === true) urStyle = 'hidden'; } else if(ukroadworks_ur === true) { if(filterRoadworks === true) urStyle = 'hidden'; } else if(construction_ur === true) { if(filterConstruction === true) urStyle = 'hidden'; } else if(closure_ur === true) { if(filterClosure === true) urStyle = 'hidden'; } else if(event_ur === true) { if(filterEvent === true) urStyle = 'hidden'; } else if(note_ur === true) { if(filterNote === true) urStyle = 'hidden'; } else if(wslm_ur === true) { if(filterWSLM === true) urStyle = 'hidden'; } else if(bog_ur === true) { if(filterBOG === true) urStyle = 'hidden'; } else if(difficult_ur === true) { if(filterDifficult === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 6) { if(filterIncorrectTurn === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 7) { if (filterIncorrectAddress === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 8) { if(filterIncorrectRoute === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 9) { if(filterMissingRoundabout === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 10) { if(filterGeneralError === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 11) { if(filterTurnNotAllowed === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 12) { if(filterIncorrectJunction === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 13) { if(filterMissingBridgeOverpass === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 14) { if(filterWrongDrivingDirection === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 15) { if(filterMissingExit === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 16) { if(filterMissingRoad === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 18) { if(filterMissingLandmark === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 19) { if(filterBlockedRoad === true) urStyle = 'hidden'; } else if(ureq.attributes.type == 23) { if(filterNativeSpeedLimit === true) urStyle = 'hidden'; } else if(filterUndefined === true) urStyle = 'hidden'; if(invertURFilters === true) { if(urStyle == 'hidden') urStyle = 'visible'; else urStyle = 'hidden'; } } // stage-age filtering override for tagged URs if(noFilterTaggedURs === true) { if(ukroadworks_ur === true) { if(filterRoadworks === false) urStyle = 'visible'; } else if(construction_ur === true) { if(filterConstruction === false) urStyle = 'visible'; } else if(closure_ur === true) { if(filterClosure === false) urStyle = 'visible'; } else if(event_ur === true) { if(filterEvent === false) urStyle = 'visible'; } else if(note_ur === true) { if(filterNote === false) urStyle = 'visible'; } else if(wslm_ur === true) { if(filterWSLM === false) urStyle = 'visible'; } } } // only touch marker visibility if we've got active filter settings, or if URComments is not // doing any filtering of its own if((hasActiveURFilters === true) || (urcFilteringIsActive === false) || (uFURs_masterEnable === false)) { var urMarker = uroGetMarker(URO_TMARKER.UR,urobj); if(urMarker !== null) { urMarker.element.style.visibility = urStyle; } } if(urStyle != 'hidden') { uroAddCustomMarkers(ureqID,URO_TMARKER.UR,customType, hasMyComments, nComments); } } } uroRenderCustomMarkers(URO_TMARKER.UR); uroPerformanceMonitoring(pmFunction, pmTStart); } function uroGetProblemTypes() { uroKnownProblemTypeIDs = []; uroKnownProblemTypeNames = []; var tProblemList = I18n.lookup("problems.types"); for(var tObj in tProblemList) { if(tObj !== undefined) { uroKnownProblemTypeIDs.push(parseInt(tObj)); uroKnownProblemTypeNames.push(tProblemList[tObj].title); } } } function uroGetTS(day, month, year, hours, mins) { var retval = new Date(0); retval.setDate(day); retval.setMonth(month - 1); retval.setYear(year); retval.setHours(hours); retval.setMinutes(mins); return retval.getTime(); } function uroFilterProblems() { var pmTStart = performance.now(); var pmFunction = "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) { uroUpdateEditorList(W.model.problems.objects, '_selectMPUserID', false, false, true, false); } if(selector.selectedOptions[0] != null) { solverUser = parseInt(selector.selectedOptions[0].value); } } var urobj; var problem; var problemStyle; var problem_marker_img; var uro_uFP_masterEnable = uroIsFilteringEnabled(false); var filter_OutsideEditableArea = uroGetCBChecked('_cbMPFilterOutsideArea'); var filter_Solved = uroGetCBChecked('_cbMPFilterSolved'); var filter_Unidentified = uroGetCBChecked('_cbMPFilterUnidentified'); var filter_Closed = uroGetCBChecked('_cbMPFilterClosed'); var filter_NotClosedUserID = uroGetCBChecked('_cbMPNotClosedUserIDFilter'); var filter_ClosedUserID = uroGetCBChecked('_cbMPClosedUserIDFilter'); var filter_Reopened = uroGetCBChecked('_cbMPFilterReopenedProblem'); var filter_LowSeverity = uroGetCBChecked('_cbMPFilterLowSeverity'); var filter_MediumSeverity = uroGetCBChecked('_cbMPFilterMediumSeverity'); var filter_HighSeverity = uroGetCBChecked('_cbMPFilterHighSeverity'); var filter_TurnProblems = uroGetCBChecked('_cbMPFilter_T200'); var filterTypes = []; var i; for(i=0; i<uroKnownProblemTypeIDs.length; i++) { if(uroGetCBChecked('_cbMPFilter_T'+uroKnownProblemTypeIDs[i])) filterTypes.push(uroKnownProblemTypeIDs[i]); } var filter_TypeUnknown = uroGetCBChecked('_cbMPFilterUnknownProblem'); var filter_TaggedElgin = uroGetCBChecked('_cbFilterElgin'); var filter_TaggedTrafficCast = uroGetCBChecked('_cbFilterTrafficCast'); var filter_TaggedTrafficMaster = uroGetCBChecked('_cbFilterTrafficMaster'); var filter_TaggedCaltrans = uroGetCBChecked('_cbFilterCaltrans'); var filter_TaggedTFL = uroGetCBChecked('_cbFilterTFL'); var filter_Invert = uroGetCBChecked('_cbInvertMPFilter'); var filter_StartDateEnabled = uroGetCBChecked('_cbMPFilterStartDate'); var filter_EndDateEnabled = uroGetCBChecked('_cbMPFilterEndDate'); var filter_EndDatePassed = uroGetCBChecked('_cbMPFilterEndDatePassed'); var tsD = uroGetElmValue('_inputMPFilterStartDay'); var tsM = uroGetElmValue('_inputMPFilterStartMonth'); var tsY = uroGetElmValue('_inputMPFilterStartYear'); var startDate = uroGetTS(tsD, tsM, tsY, 0, 0); tsD = uroGetElmValue('_inputMPFilterEndDay'); tsM = uroGetElmValue('_inputMPFilterEndMonth'); tsY = uroGetElmValue('_inputMPFilterEndYear'); var endDate = uroGetTS(tsD, tsM, tsY, 0, 0); var nowTime = (new Date()).getTime(); var addCustomMarkers = uroIsFilteringEnabled(true); for (urobj in W.model.problems.objects) { if(W.model.problems.objects.hasOwnProperty(urobj)) { problem = W.model.problems.objects[urobj]; problemStyle = 'visible'; // check problem against current session ignore list... if(uroIsOnIgnoreList(ureqID)) problemStyle = 'hidden'; var ureqID = problem.attributes.id; var desc = ''; if(problem.attributes.description != null) { desc = problem.attributes.description; } var customType = null; var elgin_mp = false; var trafficcast_mp = false; var trafficmaster_mp = false; var caltrans_mp = false; var tfl_mp = false; if(addCustomMarkers === true) { customType = uroGetCustomType(ureqID, URO_TMARKER.MP, desc); if(customType === 100) elgin_mp = true; else if(customType === 101) trafficcast_mp = true; else if(customType === 102) trafficmaster_mp = true; else if(customType === 103) caltrans_mp = true; else if(customType === 104) tfl_mp = true; } if(uro_uFP_masterEnable === true) { if(filter_OutsideEditableArea === true) { if(problem.canEdit() === false) { problemStyle = 'hidden'; } } if(filter_EndDatePassed == true) { if(problem.attributes.endTime > nowTime) { problemStyle = 'hidden'; } } if(filter_StartDateEnabled == true) { var tStart = new Date(problem.attributes.startTime); tStart.setHours(0); tStart.setMinutes(0); tStart.setSeconds(0); tStart = tStart.getTime(); if(tStart != startDate) { problemStyle = 'hidden'; } } if(filter_EndDateEnabled == true) { var tEnd = new Date(problem.attributes.endTime); tEnd.setHours(0); tEnd.setMinutes(0); tEnd.setSeconds(0); tEnd = tEnd.getTime(); if(tEnd != endDate) { problemStyle = 'hidden'; } } // check against closed/not identified filtering if enabled... 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'; } } let marker = uroGetMarker(URO_TMARKER.MP, urobj); if(marker !== null) { marker.element.style.visibility = problemStyle; } if((problemStyle != 'hidden') && (ureqID !== null) && (customType !== null)) { uroAddCustomMarkers(ureqID,URO_TMARKER.MP,customType, false, 0); } } } if(uroDOMHasTurnProblems) { for (urobj in W.model.turnProblems.objects) { if(W.model.turnProblems.objects.hasOwnProperty(urobj)) { problem = W.model.turnProblems.objects[urobj]; problemStyle = 'visible'; // check problem against current session ignore list... if(uroIsOnIgnoreList(problem.attributes.id)) problemStyle = 'hidden'; if(uro_uFP_masterEnable === true) { // 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'; } } } let marker = uroGetMarker(URO_TMARKER.MP, urobj); if(marker !== null) { marker.element.style.visibility = problemStyle; } } } } uroRenderCustomMarkers(URO_TMARKER.MP); uroPerformanceMonitoring(pmFunction, pmTStart); } function uroToHex(decValue,digits) { 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 = uroCtrlTabs[URO_TABS_ID.MISC][URO_TABS_FIELD.TABBODY]; if(uiElms == null) { uroAddLog('caught missing UI'); return false; } if(uiElms.innerHTML.length === 0) { uroAddLog('caught empty UI'); return false; } uroNullMapViewport = false; return true; } function uroFilterItems_URsTabClick() { uroFilterURs(); } function uroFilterItems_MPsTabClick() { uroFilterProblems(); } function uroFilterItems_MCsTabClick() { uroFilterMapComments(); uroMCLayerChanged(); } function uroFilterItems_PlacesTabClick() { uroFilterPlaces(); } function uroFilterItems_CamsTabClick() { uroFilterCameras(); } function uroFilterItems_MiscTabClick() { uroFilterItems(); } function uroFilterItems_RTCsTabClick() { uroFilterRTCs(); } function uroFilterItems_RAsTabClick() { uroFilterRAs(); } function uroFilterItems_MasterEnableClick() { if(uroGetCBChecked('_cbMasterEnable') === false) { uroHidePopup('uroFilterItems_MasterEnableClick'); } uroFilterItems(); } function uroFilterItems() { uroFilterProblems(); uroFilterPlaces(); uroFilterCameras(); uroFilterURs(); uroFilterRTCs(); uroFilterRAs(); uroFilterMapComments(); } function uroFilterItemsOnMove() { W.map.events.unregister('mousemove',null,uroFilterItemsOnMove); uroFilterItems(); } function uroDeleteObject() { uroAddLog('delete camera ID '+uroShownFID); if(W.model.cameras.objects[uroShownFID] === null) { uroAddLog('camera object not found...'); return false; } uroRemoveCamFromWatchList(); var actionObj = require('Waze/Action/DeleteObject'); var deleteAction = new actionObj(W.model.cameras.objects[uroShownFID], null); W.model.actionManager.add(deleteAction); uroExitPopup(); uroHidePopup('uroDeleteObject'); return false; } function uroGetUserNameFromID(userID) { var userName; if(W.model.users.objects[userID] != null) { userName = W.model.users.objects[userID].attributes.userName; if(userName === undefined) { userName = userID; } } else { userName = userID; } return userName; } function uroGetUserNameAndRank(userID) { var userName; var userLevel; if(W.model.users.objects[userID] != null) { userName = W.model.users.objects[userID].attributes.userName; if(userName === undefined) { userName = userID; } else { userName = '<a href="' + (W.Config.user_profile.url + userName) + '" target="_blank">' + userName + '</a>'; } userLevel = W.model.users.objects[userID].attributes.rank + 1; } else { userName = userID; userLevel = '?'; } return userName + ' (' + userLevel + ')'; } function uroCheckCommentsForTag(idSrc, customText) { var ursObj = W.model.updateRequestSessions.objects[idSrc]; if(typeof(ursObj) == 'undefined') return -1; if(ursObj.attributes.comments.length === 0) return -1; for(var idx=ursObj.attributes.comments.length-1; idx>=0; idx--) { for(var tag=0; tag<uroCustomURTags.length; tag++) { var keyword = uroCustomURTags[tag]; if(ursObj.attributes.comments[idx].text.indexOf(keyword) != -1) { return tag; } } if(customText !== '') { if(ursObj.attributes.comments[idx].text.toLowerCase().indexOf(customText) != -1) { return 99; } } } return -1; } function uroGetCustomMarkerIdx(customType) { if(customType === 0) return 1; // ROADWORKS if(customType === 1) return 1; // CONSTRUCTION if(customType === 2) return 0; // CLOSURE if(customType === 3) return 4; // EVENT if(customType === 4) return 3; // NOTE if(customType === 5) return 5; // WSLM if(customType === 6) return 11; // BOG if(customType === 7) return 12; // DIFFICULT if(customType === 98) return 5; // Native speed limit URs if(customType === 99) return 2; // custom text if(customType === 100) return 6; // ELGIN if(customType === 101) return 7; // TRAFFICCAST if(customType === 102) return 8; // TRAFFICMASTER if(customType === 103) return 9; // CALTRANS if(customType === 104) return 10; // TFL return -1; } function uroGetCustomType(idSrc, markerType, desc) { var provider = ''; var customText = ''; if(uroGetCBChecked('_cbCustomKeywordMarkers')) customText = document.getElementById('_textCustomKeyword').value.toLowerCase(); if(desc === null) desc = ''; if(markerType == URO_TMARKER.UR) { var ureq = W.model.mapUpdateRequests.objects[idSrc]; // early test for native speed limit URs if(ureq.attributes.type == 23) return 98; } else if(markerType == URO_TMARKER.MP) { var mp = W.model.problems.objects[idSrc]; if(mp.attributes.provider != null) { provider = mp.attributes.provider; } } if(desc !== '') { if((markerType == URO_TMARKER.UR) || (markerType == 'mc')) { for(var tag=0; tag<uroCustomURTags.length; tag++) { var keyword = uroCustomURTags[tag]; if(desc.indexOf(keyword) != -1) { return tag; } } } if(markerType == URO_TMARKER.UR) { if((uroGetCBChecked('_cbCustomKeywordMarkers')) && (customText !== '')) { if(desc.toLowerCase().indexOf(customText) != -1) return 99; } } if(markerType == URO_TMARKER.MP) { if(desc.indexOf('[Elgin]') != -1) return 100; if(desc.indexOf('[ELGIN]') != -1) return 100; if(desc.indexOf('[elginroadworks]') != -1) return 100; if(desc.indexOf('[TrafficCast]') != -1) return 101; if(desc.indexOf('[TM]') != -1) return 102; if(desc.indexOf('[Caltrans]') != -1) return 103; if(desc.indexOf('[TfL Open Data]') != -1) return 104; if(provider.indexOf('London TFL Closures') != -1) return 104; } } if(markerType == URO_TMARKER.UR) { return uroCheckCommentsForTag(idSrc, customText); } return -1; } function uroGetRestrictionLanes(disposition) { var retval = ''; if(disposition == 1) retval += 'All lanes'; else if(disposition == 2) retval += 'Left lane'; else if(disposition == 3) retval += 'Middle lane'; else if(disposition == 4) retval += 'Right lane'; else retval += ' - '; return retval; } function uroGetRestrictionLaneType(laneType) { var retval = ''; if(laneType === null) retval += ' - '; else { if(laneType == 1) retval += 'HOV'; else if(laneType == 2) retval += 'HOT'; else if(laneType == 3) retval += 'Express'; else if(laneType == 4) retval += 'Bus lane'; else if(laneType == 5) retval += 'Fast lane'; else retval += ' - '; } return retval; } var uroVehicleTypes = [ [1280, 'fa-car'], [1024, 'fa-motorcycle'], [272, 'fa-taxi'], [1808, 'fa-bolt'] ]; function uroGetRestrictionVehicleTypes(restObj, allowInit, profileKey) { var i; var j; var k; var tVT; var retval = []; for(i = 0; i < uroVehicleTypes.length; ++i) { retval.push(allowInit); } var tRest = restObj._driveProfiles.get(profileKey); if(tRest !== undefined) { for(i = 0; i < tRest._driveProfiles.length; ++i) { tVT = tRest._driveProfiles[i].getVehicleTypes(); { if(tVT.length > 0) { for(j = 0; j < tVT.length; ++j) { for(k = 0; k < uroVehicleTypes.length; ++k) { if(tVT[j] == uroVehicleTypes[k][0]) { retval[k] = !allowInit; } } } } } } } return retval; } function uroFormatRestriction(restObj) { var retval = ''; if(restObj._defaultType == "DIFFICULT") { retval = '<tr><td colspan=13>Difficult Turn'; } else { var roDays = null; var roFromDate = null; var roToDate = null; var roFromTime = null; var roToTime = null; if(restObj._days !== undefined) { roDays = restObj._days; roFromDate = restObj._fromDate; roToDate = restObj._toDate; roFromTime = restObj._fromTime; roToTime = restObj._toTime; } else if(restObj._timeFrames.length > 0) { if(restObj._timeFrames[0]._weekdays !== undefined) { roDays = restObj._timeFrames[0]._weekdays; roFromDate = restObj._timeFrames[0]._startDate; roToDate = restObj._timeFrames[0]._endDate; roFromTime = restObj._timeFrames[0]._fromTime; roToTime = restObj._timeFrames[0]._toTime; } } var hasExpired = false; var isFuture = false; var tNow = Date.now(); if(roFromDate !== null) { var tFrom = new Date(roFromDate).getTime(); isFuture = (tFrom > tNow); } if(roToDate !== null) { var tTo = new Date(roToDate).getTime() + 86399999; hasExpired = (tTo < tNow); } if(isFuture === true) { retval = '<tr bgcolor="#8080FF">'; } else if(hasExpired === true) { retval = '<tr bgcolor="#FFFF80">'; } else { retval = '<tr>'; } if(roDays === null) { roDays = 127; } retval += '<td style="text-align:center;">'; if((roDays & 1) == 1) retval += 'M'; else retval += '-'; retval += '</td><td style="text-align:center;">'; if((roDays & 2) == 2) retval += 'Tu'; else retval += '-'; retval += '</td><td style="text-align:center;">'; if((roDays & 4) == 4) retval += 'W'; else retval += '-'; retval += '</td><td style="text-align:center;">'; if((roDays & 8) == 8) retval += 'Th'; else retval += '-'; retval += '</td><td style="text-align:center;">'; if((roDays & 16) == 16) retval += 'F'; else retval += '-'; retval += '</td><td style="text-align:center;">'; if((roDays & 32) == 32) retval += 'Sa'; else retval += '-'; retval += '</td><td style="text-align:center;">'; if((roDays & 64) == 64) retval += 'Su'; else retval += '-'; retval += '</td><td nowrap style="text-align:center;">'; if(roFromDate === null) retval += 'All dates'; else retval += roFromDate+' to '+roToDate; retval += '</td><td nowrap style="text-align:center;">'; if((restObj._allDay === true) || ((roFromTime === null) && (roToTime === null))) retval += 'All day'; else retval += roFromTime+' to '+roToTime; retval += '</td><td nowrap style="text-align:center;">'; retval += uroGetRestrictionLanes(restObj._disposition); retval += '</td><td nowrap style="text-align:center;">'; retval += uroGetRestrictionLaneType(restObj._laneType); retval += '</td><td nowrap style="text-align:center;">'; // for brevity, the popup only displays the allowed/prohibited restriction for the driveable vehicle types in the app... var typesAllowed = []; if((restObj._defaultType == "BLOCKED") || (restObj._defaultType == "TOLL")) { if(restObj._defaultType == "TOLL") { retval += I18n.lookup('restrictions.editing.segment.toll_road'); } typesAllowed = uroGetRestrictionVehicleTypes(restObj, false, "FREE"); } else { typesAllowed = uroGetRestrictionVehicleTypes(restObj, true, "BLOCKED"); } var i; for(i = 0; i < uroVehicleTypes.length; ++i) { if(typesAllowed[i] === true) { retval += '<i class="fa '+uroVehicleTypes[i][1]+'" style="color:#000000;"> </i> '; } else { retval += '<i class="fa '+uroVehicleTypes[i][1]+'" style="color:#d0d0d0;"> </i> '; } } retval += '</td><td>'; retval += uroClickify(restObj._description, ''); } retval += '</td></tr>'; return retval; } function uroHidePopup(caller) { if(uroPopupShown) { uroDiv.style.visibility = 'hidden'; uroPopupShown = false; uroPopupTimer = -2; uroShownFID = -1; } uroPopupSuppressed = false; } function uroSuppressPopup() { uroDiv.style.visibility = 'hidden'; window.getSelection().removeAllRanges(); uroPopupSuppressed = true; } function uroOpenURDialog(urID) { var t = {showNext: false, nextButtonString: I18n.lookup('problems.panel.done')}; var urObj = W.model.mapUpdateRequests.objects[urID]; W.reqres.request("problems:browse", _.extend(t, {problem: urObj})); } function uroRecentreSessionOnUR() { //uroGetMarker(URO_TMARKER.UR, uroShownFID).element.click(); uroOpenURDialog(uroShownFID); W.map.moveTo(uroGetMarker(URO_TMARKER.UR, uroShownFID).lonlat, 17); uroHidePopup('uroRecentreSessionOnUR'); return false; } function uroRecentreSessionOnMP() { uroGetMarker(URO_TMARKER.MP, uroShownFID).element.click(); W.map.moveTo(uroGetMarker(URO_TMARKER.MP, uroShownFID).lonlat, 17); uroHidePopup('uroRecentreSessionOnMP'); return false; } function uroRecentreSessionOnPUR() { uroGetMarker(URO_TMARKER.PUR, uroShownFID).element.click(); W.map.moveTo(uroGetMarker(URO_TMARKER.PUR, uroShownFID).lonlat, 17); uroHidePopup('uroRecentreSessionOnPUR'); return false; } function uroRecentreSessionOnPPUR() { uroGetMarker(URO_TMARKER.PPUR, uroShownFID).element.click(); W.map.moveTo(uroGetMarker(URO_TMARKER.PPUR, uroShownFID).lonlat, 17); uroHidePopup('uroRecentreSessionOnPPUR'); return false; } function uroRecentreSessionOnVenueNavPoint() { W.map.moveTo(uroGetVenueNavPoint(uroShownFID), 17); uroHidePopup('uroRecentreSessionOnVenueNavPoint'); return false; } function uroGetDateTimeString(ts) { 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; if(uroMarkerLayers[uroStackType] !== null) { uroAddLog('restacking markers...'); // strip off the .realX/realY attributes from any UR object we've previously added it to, to allow // the native recentering to work again... var idList = uroGetMarkerIDs(uroStackType); for(var marker of idList) { var testMarkerAttributes = uroGetAttributes(uroStackType, marker); if(testMarkerAttributes.geometry.realX != null) { testMarkerAttributes.geometry.x = testMarkerAttributes.geometry.realX; testMarkerAttributes.geometry.y = testMarkerAttributes.geometry.realY; delete(testMarkerAttributes.geometry.realX); delete(testMarkerAttributes.geometry.realY); } } // now restack any markers that were repositioned... for(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(uroGetMarker(uroStackType, fid) != null) { uroGetMarker(uroStackType, fid).element.style.left = orig_x; uroGetMarker(uroStackType, fid).element.style.top = orig_y; } } uroStackList = []; uroUnstackedMasterID = null; uroStackType = null; uroAddLog('...stacked!'); } } function uroIsIDAlreadyUnstacked(idSrc) { if(uroStackList.length === 0) return false; for(var idx=0; idx<uroStackList.length; idx++) { if(uroStackList[idx].fid == idSrc) return true; } return false; } function uroCheckStacking(stackType, masterID, unstackedX, unstackedY) { if(typeof(masterID) === 'number') { masterID = masterID.toString(); } if(uroIsIDAlreadyUnstacked(masterID) === true) return; if(uroStackType !== null) return; if(uroPopupDwellTimer > 0) return; uroAddLog('checking for marker stack, masterID: '+masterID+', stackType: '+stackType); let stackList = []; stackList.push(masterID); let threshSquared = uroGetElmValue('_inputUnstackSensitivity'); threshSquared *= threshSquared; let marker; let offset = 0.000000001; if(uroMarkerLayers[stackType] !== null) { let idList = uroGetMarkerIDs(stackType); let showOpen = true; let showClosed = false; let showTypes = null; if(stackType === URO_TMARKER.UR) { showTypes = W?.issueTrackerController?.app?.attributes?.issueTrackerFilter?.attributes?.mapUpdateRequestsFilter?.attributes?.status; } else if(stackType === URO_TMARKER.MP) { showTypes = W?.issueTrackerController?.app?.attributes?.issueTrackerFilter?.attributes?.mapProblemsFilter?.attributes?.status; } if(showTypes !== null) { showOpen = ((showTypes == 'OPEN') || (showTypes == 'BOTH')); showClosed = ((showTypes == 'CLOSED') || (showTypes == 'BOTH')); } for(marker of idList) { let testMarkerObj = uroGetMarker(stackType, marker); let testMarkerAttributes = uroGetAttributes(stackType, marker); if((testMarkerAttributes !== null) && (testMarkerObj !== null)) { // if multiple markers are stacked exactly on top of one another, WME will always open up the one which it would have rendered on the // top of the stack in the absence of any URO+ filtering, regardless of which UR pin actually receives the click event. To prevent // this, we give each pin in the stack a unique set of false coordinates, storing the original coordinates in newly created // properties so they can be restored later on. // // As of 3.169, the offset added to create the false coordinates has now been changed to a fractional value as opposed to the large integer // it previously was, as the latter method has now been seen to cause problems with displaying the "A" marker when a road closure request MP // is being viewed. By using a small fractional offset added to each stacked marker, the risk of accidentally setting the offset coords to // those of another marker is low if(testMarkerAttributes.geometry.realX === undefined) { testMarkerAttributes.geometry.realX = testMarkerAttributes.geometry.x; testMarkerAttributes.geometry.x += offset; testMarkerAttributes.geometry.realY = testMarkerAttributes.geometry.y; testMarkerAttributes.geometry.y += offset; offset += 0.000000001; } let includeInStack = (testMarkerObj.element.style.visibility != 'hidden'); let isClosed = testMarkerObj.element.classList.contains("recently-closed"); includeInStack = includeInStack && ((isClosed && showClosed) || (!isClosed && showOpen)); if(includeInStack) { if(testMarkerAttributes.id != masterID) { let xdiff = unstackedX - uroParsePxString(testMarkerObj.element.style.left); let ydiff = unstackedY - uroParsePxString(testMarkerObj.element.style.top); let distSquared = ((xdiff * xdiff) + (ydiff * ydiff)); if(distSquared < threshSquared) { stackList.push(testMarkerAttributes.id); } } } } } } let inhibitUnstacking = (W.map.getZoom() < uroGetElmValue('_inputUnstackZoomLevel')); inhibitUnstacking = inhibitUnstacking || (stackList.length < 2); if(inhibitUnstacking == false) { uroStackType = stackType; if(uroUnstackedMasterID != masterID) { uroAddLog('unstacked ID mismatch, relocating markers...'); uroRestackMarkers(); uroUnstackedMasterID = masterID; uroStackList = []; // push the highlighted marker onto the stacklist so uroIsIDAlreadyUnstacked() will return true uroStackList.push(new uroStackListObj(masterID,unstackedX,unstackedY)); for(let shoveIdx=0; shoveIdx < stackList.length; shoveIdx++) { let fid = stackList[shoveIdx]; let stackMarker = uroGetMarker(stackType, fid); if(stackMarker !== null) { let x = uroParsePxString(stackMarker.element.style.left); let y = uroParsePxString(stackMarker.element.style.top); // store the unstacked marker positions so they can be reinstated later uroStackList.push(new uroStackListObj(fid,x,y)); stackMarker.element.style.left = unstackedX + 'px'; stackMarker.element.style.top = unstackedY + 'px'; unstackedX += 5; unstackedY -= 20; } } // hide other markers to prevent confusion with the unstacked markers let listIDs = uroGetMarkerIDs(stackType); for(marker in listIDs) { if(listIDs.hasOwnProperty(marker)) { let toHideMarker = uroGetMarker(stackType, marker); if(toHideMarker !== null) { let toHideID = toHideMarker.id; if(uroIsIDAlreadyUnstacked(toHideID) === false) { toHideMarker = uroGetMarker(stackType, toHideID); if(toHideMarker !== null) { toHideMarker.element.style.visibility = 'hidden'; } } } } } } } else { uroRestackMarkers(); } } function uroGetVenueNavPoint(uroFID) { let retval = W.map.getCenter(); // allow the function to return a safe value in case we can't find the requested venue object... let vObj = W.model.venues.objects[uroFID]; if(vObj !== undefined) { if(vObj.attributes.entryExitPoints.length > 0) { // if the venue has any navpoints defined, use the position of the first one let tPoint = vObj.attributes.entryExitPoints[0].getPoint(); retval.lon = tPoint.coordinates[0]; retval.lat = tPoint.coordinates[1]; } else { // otherwise use the centrepoint of the venue point or polygon let tPoint = vObj.attributes.geometry.getCentroid(); var tLL = new OpenLayers.LonLat(); tLL.lon = tPoint.x; tLL.lat = tPoint.y; tLL.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326")); retval.lon = tLL.lon; retval.lat = tLL.lat; } } return retval; } function uroOpenNewTab() { // flush the current settings into localStorage before the new tab opens, so that when its instance of // URO+ fires up it'll have the same settings as this one uroSaveSettings(); return true; } function uroEditTBR() { if(uroTBRObj === null) { return; } uroTBRObj.getElementsByClassName('waze-icon-clock')[0].click(); return false; } function uroGetRTCDuration(rcObj) { var duration = new Date(rcObj.attributes.endDate) - new Date(rcObj.attributes.startDate); return Math.floor(duration / 86400000); } function uroGetRTCOffset(rcDate) { let dateObj = new Date(rcDate); return (0 - uroDateToDays(dateObj)); } function uroGetRTCOrigin(rcObj) { let reval = URO_TRTC.UNKNOWN; if(rcObj !== undefined) { if(rcObj.attributes.createdBy == -5) { retval = URO_TRTC.WAZEFEED; } else if((W.model.users.objects[rcObj.attributes.createdBy] !== undefined) && (W.model.users.objects[rcObj.attributes.createdBy].attributes.rank == 6)) { retval = URO_TRTC.WAZEOTHER; } else { retval = URO_TRTC.WME; } } return retval; } function uroGetRTCState(rcObj) { let retval = URO_SRTC.UNKNOWN; let startOffset = uroGetRTCOffset(rcObj.attributes.startDate); let endOffset = uroGetRTCOffset(rcObj.attributes.endDate); if(endOffset < 0) { retval = URO_SRTC.EXPIRED; } else if(startOffset < 0) { retval = URO_SRTC.ACTIVE; } else { retval = URO_SRTC.FUTURE; } return retval; } function uroAddClosureRowToTable(rcObj) { let result = ''; let rtcType = uroGetRTCOrigin(rcObj); if(rcObj.attributes.active === true) { result += '<tr>'; } else { result += '<tr bgcolor="#C0C0C0">'; } let startDate = rcObj.attributes.startDate; let endDate = "unknown"; if(rcObj.attributes.endDate !== null) { endDate = rcObj.attributes.endDate; } let provider = "---"; if(rcObj.attributes.provider !== null) { provider = rcObj.attributes.provider; } else if(rcObj.attributes.createdBy !== null) { provider = uroGetUserNameAndRank(rcObj.attributes.createdBy); } let reason = "---"; if(rcObj.attributes.reason !== null) { reason = rcObj.attributes.reason; } let mte = "---"; if(rcObj.attributes.eventId !== null) { try { mte = W.model.majorTrafficEvents.objects[rcObj.attributes.eventId].attributes.names[0].value; } catch(err) { } } let startOffset = uroGetRTCOffset(rcObj.attributes.startDate); let endOffset = uroGetRTCOffset(rcObj.attributes.endDate); let duration = uroGetRTCDuration(rcObj); if(duration === 0) duration = '<1 day'; else if(duration === 1) duration = '1 day'; else duration = duration + ' days'; let state = uroGetRTCState(rcObj); let status = ''; if(state === URO_SRTC.EXPIRED) { status = "Expired"; } else if(state === URO_SRTC.ACTIVE) { status = 'Active'; } else if(state === URO_SRTC.FUTURE) { if(startOffset == 0) { status = 'Today'; } else if(startOffset == 1) { status = 'In 1 day'; } else { status = 'In ' + startOffset + ' days'; } } result += '<td nowrap>' + status + '</td>'; result += '<td nowrap>' + startDate + ' to ' + endDate + ' (' + duration + ')</td>'; result += '<td nowrap>' + provider + '</td>'; result += '<td nowrap>' + reason + '</td>'; result += '<td nowrap>' + mte + '</td>'; result += '</td></tr>'; return result; } function uroGetAddress(streetID, houseNumber, formatForSegmentPopup, formatForNodePopup, showAsToll) { 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].attributes.isEmpty === false)) { streetName = W.model.streets.objects[streetID].attributes.name; } } if(formatForSegmentPopup === true) { if(showAsToll == true) { result += '<i class="fa fa-credit-card"></i> '; } result += '<b>'+streetName+'</b><br>'; } else { result += streetName + ', '; } if(doesStreetIDExist === true) { var cityName = I18n.lookup('edit.address.no_city'); var doesCityIDExist = true; var cityID = W.model.streets.objects[streetID].attributes.cityID; if(W.model.cities.objects[cityID] === undefined) { cityName = 'non-existent cityID'; doesCityIDExist = false; } else { if(W.model.cities.objects[cityID].attributes.name !== "") { cityName = W.model.cities.objects[cityID].attributes.name; } } result += cityName + ', '; if(doesCityIDExist === true) { 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].attributes.name; } } } } result += '<br>'; return result; } function uroGetSelectedSegmentRTCs(segID) { var closureTypes = 0; uroRTCObjs = []; var selectedSegs = []; if(segID === null) { // segID should always be set to a valid segment ID if we're being called from the segment mouseover // handler, so if it's null it implies we've instead been called from the closure panel handler where // we might therefore be dealing with a multi-segment selection... selectedSegs = W.selectionManager.getSegmentSelection().segments; } if((selectedSegs.length > 0) || (segID !== null)) { var i; for(var roadClosure in W.model.roadClosures.objects) { if(W.model.roadClosures.objects.hasOwnProperty(roadClosure)) { var rcObj = W.model.roadClosures.objects[roadClosure]; rcObj.segIDs = [rcObj.attributes.segID]; // copy the segID property into an array so we can push extra segIDs into it later... // set a direction value corresponding to the A-B or B-A setting - if we later end up combining an A-B and B-A closure // into a two-way closure, we can then change the direction value to indicate this as well if(rcObj.attributes.forward === true) { rcObj.direction = 1; } else { rcObj.direction = 2; } // for each of the selected or moused-over segments, find all the closures which have matching segIDs if(segID !== null) { if(rcObj.attributes.segID == segID) { uroRTCObjs.push(rcObj); } } else { for(i = 0; i < selectedSegs.length; ++i) { if(rcObj.attributes.segID == selectedSegs[i].attributes.id) { uroRTCObjs.push(rcObj); break; } } } } } // uroRTCObjs now contains all of the closures relating to all of the segments of interest, so we can // begin to organise them such that by the time we exit this function, the array will then contain an // optimised list of closures that matches up to the list shown in the closure sidepanel, taking into // account closures applying to all segments vs some, closures that can be merged into two-ways etc. // first sort the closure by their start date, with a secondary sort by direction for those closures // that have the same start date uroRTCObjs = uroRTCObjs.sort(function(a,b) { if(a.attributes.startDate === b.attributes.startDate) { if(a.direction == 1) return -1; return 1; } if(a.attributes.startDate > b.attributes.startDate) return 1; return -1; }); // if we've got at least two closures in the sorted list, we then test adjacent list entries // to see if they contain closure details which are identical except for their segment IDs, and // combine them if so if(uroRTCObjs.length > 1) { i = 0; while(i < (uroRTCObjs.length - 1)) { if( (uroRTCObjs[i].attributes.createdBy == uroRTCObjs[i+1].attributes.createdBy) && (uroRTCObjs[i].attributes.endDate == uroRTCObjs[i+1].attributes.endDate) && (uroRTCObjs[i].attributes.eventId == uroRTCObjs[i+1].attributes.eventId) && (uroRTCObjs[i].attributes.location == uroRTCObjs[i+1].attributes.location) && (uroRTCObjs[i].attributes.reason == uroRTCObjs[i+1].attributes.reason) && (uroRTCObjs[i].attributes.startDate == uroRTCObjs[i+1].attributes.startDate) && (uroRTCObjs[i].direction == uroRTCObjs[i+1].direction) ) { uroRTCObjs[i].segIDs.push(uroRTCObjs[i+1].attributes.segID); uroRTCObjs.splice(i+1, 1); } else { ++i; } } } // after that first trimming of the list, if there are still two or more entries then // we perform a second pass, this time merging any adjacent entries which have the same // segment IDs in their segIDs arrays - these are two-way closures applying to all those // segments, and so we also change the direction value to indicate two-way vs A-B or B-A if(uroRTCObjs.length > 1) { i = 0; while(i < (uroRTCObjs.length - 1)) { if ( (uroRTCObjs[i].segIDs.sort().join(',') == uroRTCObjs[i+1].segIDs.sort().join(',')) && (uroRTCObjs[i].attributes.createdBy == uroRTCObjs[i+1].attributes.createdBy) && (uroRTCObjs[i].attributes.endDate == uroRTCObjs[i+1].attributes.endDate) && (uroRTCObjs[i].attributes.eventId == uroRTCObjs[i+1].attributes.eventId) && (uroRTCObjs[i].attributes.location == uroRTCObjs[i+1].attributes.location) && (uroRTCObjs[i].attributes.reason == uroRTCObjs[i+1].attributes.reason) && (uroRTCObjs[i].attributes.startDate == uroRTCObjs[i+1].attributes.startDate) ) { uroRTCObjs[i].direction = 4; uroRTCObjs.splice(i+1, 1); } ++i; } } for(i = 0; i < uroRTCObjs.length; ++i) { closureTypes |= uroRTCObjs[i].direction; } } // the closure list ordering at this point doesn't always match up to the order used by the closures panel when // a mixture of "all segment" and "some segment" closures are present - need to work out what ordering rules // WME is using here... return closureTypes; } function uroGetLengthString(length) { var retval = ''; if(length == null) { retval = "Default"; } else if(W.model.isImperial == true) { retval = (length / (12 * 2.54)).toFixed(1) + "ft"; } else { retval = (length / 100).toFixed(1) + "m"; } return retval; } function uroNewLookHighlightedItemsCheck(e) { if ( (W.selectionManager._mouseInFeature === null) && (uroMousedOverMarkerType === null) && (uroPopupTimer === 0) ) { if(uroPopupShown === true) { uroHidePopup('uroNewLookHighlightedItemsCheck'); } uroMousedOverMapComment = null; return; } var moObj = null; var renderIntent = null; if(W.selectionManager._mouseInFeature !== null) { renderIntent = W.selectionManager._mouseInFeature.renderIntent; moObj = W.selectionManager._mouseInFeature.attributes?.wazeFeature?._wmeObject; if(moObj === undefined) { moObj = W.selectionManager._mouseInFeature.attributes?.repositoryObject; } if(moObj === undefined) moObj = null; } 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 ureq = null; var idx; var hovered = false; var targetTab = ''; var unstackedX; var unstackedY; var ureqID = null; var isUR = false; var isProblem = false; var isTurnProb = false; var isPlaceUpdate = false; var mouseX; var mouseY; var uroDaysResolved; var deltaX; var deltaY; var userLock; // function preamble... { if(uroMTEMode) return; if(!uroInitialised) return; if(e == 'dwellTimeout') { } else { if((uroMouseIsDown) && (e.buttons === 0)) { uroAddLog('trapped erroneous mousedown state'); uroMouseIsDown = false; } } if(uroMouseIsDown) { return; } if(OpenLayers === null) { if(uroNullOpenLayers === false) { uroAddLog('caught null OpenLayers'); uroNullOpenLayers = true; } return; } uroNullOpenLayers = false; if(W.map.getLayerByName("update_requests") === null) { if(uroNullURLayer === false) { uroAddLog('caught null UR layer'); uroNullURLayer = true; } return; } uroNullURLayer = false; if(W.map.getLayerByName("mapProblems") === null) { if(uroNullProblemLayer === false) { uroAddLog('caught null problem layer'); uroNullProblemLayer = true; } return; } uroNullProblemLayer = false; if(uroGetCBChecked('_cbMasterEnable') === false) { return; } if(e == 'dwellTimeout') { mouseX = uroPrevMouseX; mouseY = uroPrevMouseY; deltaX = 0; deltaY = 0; if(uroPointerWithinMap === false) { return; } } else { if(uroTestPointerOutsideMap(e.clientX, e.clientY)) { return; } mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left; mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top; var maxJitter = uroGetElmValue('_inputMaxJitter'); if((Math.abs(uroPrevMouseX - mouseX) > maxJitter) || (Math.abs(uroPrevMouseY - mouseY) > maxJitter)) { uroPopupDwellTimer = uroGetElmValue('_inputPopupDwellTimeout'); } deltaX = mouseX - uroPrevMouseX; deltaY = mouseY - uroPrevMouseY; uroPrevMouseX = mouseX; uroPrevMouseY = mouseY; } } uroWazeBits(); var mouseLonLat = W.map.getLonLatFromViewPortPx(new OpenLayers.Pixel(mouseX,mouseY)); var mousePoint = new OpenLayers.Geometry.Point(mouseLonLat.lon, mouseLonLat.lat); if(uroMousedOverMapComment !== null) { if(W.model.mapComments.objects[uroMousedOverMapComment] === undefined) { uroAddLog('clearing uroMousedOverMapComment: object no longer exists in current map view'); uroMousedOverMapComment = null; } else if(W.model.mapComments.objects[uroMousedOverMapComment].attributes.geometry.containsPoint(mousePoint) === false) { uroAddLog('clearing uroMousedOverMapComment: pointer no longer within comment boundary'); uroMousedOverMapComment = null; } } let popupXOffset = document.getElementById('editor-container').getBoundingClientRect().x; let popupYOffset = $(document.getElementById("WazeMap")).offset().top; let uroPopupX = mouseX + popupXOffset + 10; let uroPopupY = mouseY + popupYOffset; // popup for segments if((uroMousedOverMarkerType === null) && (uroGetCBChecked('_cbInhibitSegPopup') === false)) { if((moObj !== null) && (moObj.type === 'segment')) { if(W.map.getExtent().intersectsBounds(moObj.attributes.geometry.getBounds())) { let doPopUp = false; let restObj; if(uroMousedOverMapComment !== null) { uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for segment highlight'); uroMousedOverOtherObjectWithinMapComment = true; } let streetID = moObj.attributes.primaryStreetID; if(streetID !== null) { // generic segment data if(uroGetCBChecked('_cbInhibitSegGenericPopup') === false) { doPopUp = true; uroAddLog('building popup for segment '+streetID); let isToll = ((moObj.attributes.fwdToll == true) || (moObj.attributes.revToll == true)); result += uroGetAddress(streetID, null, true, false, isToll); if(moObj.attributes.streetIDs.length > 0) { // list any alternate names result += '<br>Alternate names:<br>'; for(let i = 0; i < moObj.attributes.streetIDs.length; ++i) { result += ' <i>' + W.model.streets.objects[moObj.attributes.streetIDs[i]].attributes.name + ', '; let cityName = ""; if(W.model.cities.objects[W.model.streets.objects[moObj.attributes.streetIDs[i]].attributes.cityID] != undefined) { cityName = W.model.cities.objects[W.model.streets.objects[moObj.attributes.streetIDs[i]].attributes.cityID].attributes.name; } if(cityName != "") { result += cityName; } else { result += ' no city'; } result += '</i><br>'; } result += '<br>'; } result += '<b>ID: </b>'+moObj.attributes.id+' - '; let autoLock = moObj.attributes.rank; userLock = moObj.attributes.lockRank; result += '<b>' + I18n.lookup("edit.segment.fields.lock") + ': </b>'; if(userLock !== null) { result += 'M' + (userLock+1) + ' / '; } result += 'A' + (autoLock+1) + ' - '; let level = moObj.attributes.level; result += '<b>' + I18n.lookup("edit.segment.fields.level") +': </b>'; if(level == 0) { result += I18n.lookup("edit.segment.levels")[0]; } else { result += level; } result += '<br>'; result += "<b>Last edit by</b> " + uroGetUserNameAndRank(moObj.attributes.updatedBy) + " <b>on</b> " + uroGetDateTimeString(moObj.attributes.updatedOn) + "<br><br>"; let fwdSpeed = moObj.attributes.fwdMaxSpeed; let revSpeed = moObj.attributes.revMaxSpeed; let fwdLanes = moObj.attributes.fwdLaneCount; let revLanes = moObj.attributes.revLaneCount; let fwdWidth = 'Not set'; let revWidth = 'Not set'; if(moObj.attributes.fromLanesInfo != null) { fwdWidth = uroGetLengthString(moObj.attributes.fromLanesInfo.laneWidth); } if(moObj.attributes.toLanesInfo != null) { revWidth = uroGetLengthString(moObj.attributes.toLanesInfo.laneWidth); } let fwdUnverified = moObj.attributes.fwdMaxSpeedUnverified; let revUnverified = moObj.attributes.revMaxSpeedUnverified; let fwdASC = ((moObj.attributes.fwdFlags & 1) === 1); let revASC = ((moObj.attributes.revFlags & 1) === 1); let roadType = moObj.attributes.roadType; let verifyLimits = true; if((roadType === 17) || (roadType === 20)) { verifyLimits = false; } result += '<table border=1><tr><th>Dir</th><th>Speed</th><th>ASC</th><th>Lanes</th><th>Width</th></tr>'; if(moObj.attributes.fwdDirection) { result += '<tr><td><b>A-B</b></td><td>'+uroGetLocalisedSpeedString(fwdSpeed)+'</td><td>'; if(fwdASC == true) { result += 'Yes'; } else { result += 'No'; } result += '</td><td>'; if(fwdLanes > 0) { result += fwdLanes; } else { result += 'Not set'; } result += '</td><td>'+fwdWidth+'</td></tr>'; } if(moObj.attributes.revDirection) { result += '<tr><td><b>B-A</b></td><td>'+uroGetLocalisedSpeedString(revSpeed)+'</td><td>'; if(revASC == true) { result += 'Yes'; } else { result += 'No'; } result += '</td><td>'; if(revLanes > 0) { result += revLanes; } else { result += 'Not set'; } result += '</td><td>'+revWidth+'</td></tr>'; } result += '</table>'; if((moObj.attributes.fwdDirection) && (moObj.attributes.revDirection) && (fwdSpeed != revSpeed) && (!fwdUnverified) && (!revUnverified)) { result += '<br>Two-way segment has different verified speed limits...'; } } // segment restrictions if(moObj.attributes.restrictions.length > 0) { result += '<br><table border=1>'; doPopUp = true; let fwdResult = '<tr><td colspan=13><b>A-B restrictions:</b></td></tr>'; let revResult = '<tr><td colspan=13><b>B-A restrictions:</b></td></tr>'; let bothResult = '<tr><td colspan=13><b>Two-way restrictions:</b></td></tr>'; let nABRestrictions = 0; let nBARestrictions = 0; let nBothRestrictions = 0; for(idx = 0; idx < moObj.attributes.restrictions.length; idx++) { restObj = moObj.attributes.restrictions[idx]; if(restObj._direction === "FWD") { nABRestrictions++; fwdResult += uroFormatRestriction(restObj); } else if(restObj._direction === "REV") { nBARestrictions++; revResult += uroFormatRestriction(restObj); } else if(restObj._direction === "BOTH") { nBothRestrictions++; bothResult += uroFormatRestriction(restObj); } else { uroAddLog("unknown restriction direction..."); } } if(nABRestrictions > 0) { result += fwdResult; } if(nBARestrictions > 0) { result += revResult; } if(nBothRestrictions > 0) { result += bothResult; } result += '</table>'; } if(W.map.getLayerByName("closures").getVisibility() === true) { let closureTypes = uroGetSelectedSegmentRTCs(moObj.attributes.id); if(closureTypes !== 0) { result += '<br><table border=1 width="100%">'; let rcObj; let roadClosure; if((closureTypes & 1) === 1) { result += '<tr><td colspan=4><b>A-B closures:</b></td></tr>'; for(roadClosure in uroRTCObjs) { if(uroRTCObjs.hasOwnProperty(roadClosure)) { rcObj = uroRTCObjs[roadClosure]; if(rcObj.direction === 1) { result += uroAddClosureRowToTable(rcObj); } } } } if((closureTypes & 2) === 2) { result += '<tr><td colspan=4><b>B-A closures:</b></td></tr>'; for(roadClosure in uroRTCObjs) { if(uroRTCObjs.hasOwnProperty(roadClosure)) { rcObj = uroRTCObjs[roadClosure]; if(rcObj.direction === 2) { result += uroAddClosureRowToTable(rcObj); } } } } if((closureTypes & 4) === 4) { result += '<tr><td colspan=4><b>Two-way closures:</b></td></tr>'; for(roadClosure in uroRTCObjs) { if(uroRTCObjs.hasOwnProperty(roadClosure)) { rcObj = uroRTCObjs[roadClosure]; if(rcObj.direction === 4) { result += uroAddClosureRowToTable(rcObj); } } } } if(closureTypes > 0) { doPopUp = true; } result += '</table>'; } } if(doPopUp === true) { uroFID = moObj.attributes.id; newPopupType = 'segment_restriction'; } } } else { uroAddLog('segment '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup'); } } } // popup for landmarks if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitLandmarkPopup') === false)) { var navpointPos=new OpenLayers.LonLat(); if((moObj !== null) && (moObj.type === 'venue')) { if(renderIntent === 'highlight') { if(W.map.getExtent().intersectsBounds(moObj.attributes.geometry.getBounds())) { if(uroMousedOverMapComment !== null) { uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for place highlight'); uroMousedOverOtherObjectWithinMapComment = true; } if(newPopupType === null) { uroFID = moObj.attributes.id; uroAddLog('building popup for place '+uroFID); navpointPos = uroGetVenueNavPoint(uroFID); result += '<b>'; if(moObj.attributes.name === '') { if(moObj.attributes.residential === true) result += '<i>Residential</i>'; else result += '<i>Unnamed</i>'; } else result += uroClickify(moObj.attributes.name, ''); if(moObj.attributes.externalProviderIDs.length > 0) { result += ' <i>(linked)</i>'; } if(moObj.attributes.adLocked) { result += ' <i>(AdLocked)</i>'; } result += '</b><br>'; if(moObj.attributes.brand !== null) { result += '<i>Brand: ' + moObj.attributes.brand + '</i><br>'; } var vDesc = moObj.attributes.description; if(vDesc !== '') { result += '"<i>' + uroClickify(vDesc, '') + '</i>"<br>'; } userLock = moObj.attributes.lockRank; result += '<b>Lock: </b>' + (userLock+1); result += '<hr>'; result += uroGetAddress(moObj.attributes.streetID, moObj.attributes.houseNumber, false, false, false); result += '<ul>'; for(idx = 0; idx < moObj.attributes.categories.length; idx++) { result += '<li>' + I18n.lookup("venues.categories." + moObj.attributes.categories[idx]); } result += '</ul>'; if(moObj.attributes.residential === true) { if(moObj.geometry.CLASS_NAME.indexOf('Geometry.Point') !== -1) { result += '<a href="#" id="_cloneRP">Clone place</a>'; objHasCloneLink = true; } } var npLink = document.location.href; var npLayers = ''; npLink = npLink.substr(0,npLink.indexOf('?zoomLevel')); npLink += '?zoomLevel=17&lat='+navpointPos.lat+'&lon='+navpointPos.lon+npLayers; targetTab = "_uroTab_" + Math.round(Math.random()*1000000); result += '<hr>Jump to nav point: <a href="'+npLink+'" id="_openInNewTab" target="'+targetTab+'">in new tab</a> - '; objHasOpenInNewTabLink = true; result += '<a href="#" id="_recentreSession">in this tab</a>'; objHasRecentreSessionLink = true; newPopupType = 'venue'; isVenue = true; } else { var otherID = moObj.attributes.id; uroAddLog('venue '+otherID+' is also highlighted'); } } else { uroAddLog('landmark '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup'); } } } } // popup for map comments if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitMapCommentPopup') === false)) { if(uroMCLayer.name !== 'mapComments') { uroWazeBits(); } if(uroMCLayer !== null) { uroMCSelected = false; if((moObj !== null) && (moObj.type === 'mapComment')) { if(renderIntent == 'highlight') { if(W.map.getExtent().intersectsBounds(moObj.attributes.geometry.getBounds())) { if(newPopupType === null) { if((uroMousedOverMapComment === moObj.attributes.id) && (uroMousedOverOtherObjectWithinMapComment === true)) { uroAddLog('inhibit popup for map comment '+uroMousedOverMapComment); } else { uroMousedOverOtherObjectWithinMapComment = false; if(moObj.attributes.geometry.id.indexOf('Polygon') !== -1) { // only capture ID for area comments... uroMousedOverMapComment = moObj.attributes.id; } uroFID = moObj.attributes.id; uroAddLog('building popup for map comment '+uroFID); result += '<b>'; if(moObj.attributes.subject === '') { result += '<i>No subject</i>'; } else result += uroClickify(moObj.attributes.subject, ''); result += '</b><br>'; result += uroClickify(moObj.attributes.body, '<br>'); var mcDaysOld = uroGetMCAge(moObj,0,false); var mcSubmittedTS = uroGetMCAge(moObj,0,true); if(mcSubmittedTS != -1) { mcSubmittedTS = uroGetDateTimeString(mcSubmittedTS); } if(mcDaysOld != -1) { result += '<i>Submitted ' + uroParseDaysAgo(mcDaysOld) + ' '; if(mcSubmittedTS != -1) result += '(' + mcSubmittedTS + ') '; if(moObj.attributes.createdBy != null) { result += ' by '+uroGetUserNameAndRank(moObj.attributes.createdBy); } result += '</i><br>'; } mcDaysOld = uroGetMCAge(moObj,1,false); mcSubmittedTS = uroGetMCAge(moObj,1,true); if(mcSubmittedTS != -1) { mcSubmittedTS = uroGetDateTimeString(mcSubmittedTS); } if(mcDaysOld != -1) { result += '<i>Updated ' + uroParseDaysAgo(mcDaysOld) + ' '; if(mcSubmittedTS != -1) result += '(' + mcSubmittedTS + ') '; if(moObj.attributes.createdBy != null) { result += ' by '+uroGetUserNameAndRank(moObj.attributes.updatedBy); } result += '</i><br>'; } mcDaysOld = uroGetMCAge(moObj,2,false); mcSubmittedTS = uroGetMCAge(moObj,2,true); if(mcDaysOld != -1) { result += '<i>Expires ' + uroParseDaysToGo(mcDaysOld) + ' '; result += '(' + uroGetDateTimeString(mcSubmittedTS) +')</i><br>'; } var mcHasMyComments = false; var mcNComments = moObj.attributes.conversation.length; if(mcNComments > 0) { for(var j=0; j<mcNComments; j++) { if(moObj.attributes.conversation[j].userID == uroUserID) { mcHasMyComments = true; break; } } } result += '<br>' + mcNComments +' comment'; if(mcNComments != 1) result += 's'; if((mcHasMyComments === false) && (mcNComments > 0)) result += ' (none by me)'; // add "ignore for this session" link result += '<br><a href="#" id="_addtoignore">Hide for this session</a>'; objHasIgnoreLink = true; newPopupType = 'map_comment'; isMapComment = true; } } else { var mcOtherID = moObj.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 !== null) && (uroMousedOverMarkerID !== null)) { var mousedMarker = uroGetMarker(uroMousedOverMarkerType, uroMousedOverMarkerID); if((uroMousedOverMarkerType == URO_TMARKER.UR) && (newPopupType === null) && (mousedMarker !== null)) { unstackedX = uroParsePxString(mousedMarker.element.style.left); unstackedY = uroParsePxString(mousedMarker.element.style.top); // check for stacking... if(uroShownFID != uroMousedOverMarkerID) { uroCheckStacking(URO_TMARKER.UR,uroMousedOverMarkerID, unstackedX, unstackedY); } hovered = true; if(uroGetCBChecked('_cbInhibitURPopup') === false) { if(uroMousedOverMapComment !== null) { uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for UR highlight'); uroMousedOverOtherObjectWithinMapComment = true; } isUR = true; newPopupType = uroMousedOverMarkerType; ureq = W.model.mapUpdateRequests.objects[uroMousedOverMarkerID]; let markerBCR = mousedMarker.element.getBoundingClientRect(); uroPopupX = markerBCR.left + markerBCR.width; uroPopupY = markerBCR.top + (markerBCR.height / 2); uroFID = uroMousedOverMarkerID; uroAddLog('building popup for UR '+uroMousedOverMarkerID); result = '<b>Update Request ('+uroMousedOverMarkerID+'): ' + I18n.lookup("update_requests.types." + ureq.attributes.type) + '</b><br>'; result += uroClickify(ureq.attributes.description, '<br>'); 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].attributes.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].attributes.comments[nComments-1]); if(commentDaysOld != -1) { result += ', last update '+uroParseDaysAgo(commentDaysOld); } } } if(uroURDupes.length > 0) { let thisID = parseInt(uroMousedOverMarkerID); for(let i = 0; i < uroURDupes.length; ++i) { if(uroURDupes[i][0] === thisID) { result += '<br><br>Duplicate of: '; let dupes = 0; for(let j = 0; j < uroURDupes[i][1].length; ++j) { if(uroURDupes[i][1][j] !== thisID) { if(dupes > 0) { result += ', '; } result += uroURDupes[i][1][j]; ++dupes; } } } } } } } if(((uroMousedOverMarkerType == URO_TMARKER.PUR) || (uroMousedOverMarkerType == URO_TMARKER.PPUR) || (uroMousedOverMarkerType == URO_TMARKER.RPUR)) && (newPopupType === null) && (mousedMarker !== null)) { ureq = W.model.venues.objects[uroMousedOverMarkerID]; unstackedX = uroParsePxString(mousedMarker.element.style.left); unstackedY = uroParsePxString(mousedMarker.element.style.top); if(uroShownFID != uroMousedOverMarkerID) { // check for stacking... uroCheckStacking(uroMousedOverMarkerType, uroMousedOverMarkerID, unstackedX, unstackedY); } hovered = true; if(uroGetCBChecked('_cbInhibitPUPopup') === false) { newPopupType = uroMousedOverMarkerType; if(uroMousedOverMapComment !== null) { uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for PUR highlight'); uroMousedOverOtherObjectWithinMapComment = true; } isPlaceUpdate = true; let markerBCR = mousedMarker.element.getBoundingClientRect(); uroPopupX = markerBCR.left + markerBCR.width; uroPopupY = markerBCR.top + (markerBCR.height / 2); uroFID = uroMousedOverMarkerID; if(uroMousedOverMarkerType == URO_TMARKER.PUR) { uroAddLog('building popup for placeUpdate '+uroMousedOverMarkerID); } else if(uroMousedOverMarkerType == URO_TMARKER.RPUR) { uroAddLog('building popup for residentialPlaceUpdate '+uroMousedOverMarkerID); } else { uroAddLog('building popup for parkingPlaceUpdate '+uroMousedOverMarkerID); } result = '<b>'; if(ureq.attributes.name === '') result += 'Unnamed landmark'; else result += ureq.attributes.name; result += '</b><br>'; result += '<ul>'; for(idx = 0; idx < ureq.attributes.categories.length; idx++) { result += '<li>' + I18n.lookup("venues.categories." + ureq.attributes.categories[idx]); } result += '</ul>'; if(ureq.attributes.residential === true) { result += '<i>Residential</i>'; } var daysOld = uroGetPURAge(ureq); if(daysOld != -1) { result += '<br><i>Submitted '+uroParseDaysAgo(daysOld)+'</i>'; } } } if((uroMousedOverMarkerType == URO_TMARKER.MP) && (newPopupType === null) && (mousedMarker !== null)) { unstackedX = uroParsePxString(mousedMarker.element.style.left); unstackedY = uroParsePxString(mousedMarker.element.style.top); // check for stacking... if(uroShownFID != uroMousedOverMarkerID) { uroCheckStacking(URO_TMARKER.MP,uroMousedOverMarkerID, unstackedX, unstackedY); } hovered = true; if(uroGetCBChecked('_cbInhibitMPPopup') === false) { if(uroMousedOverMapComment !== null) { uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for MP highlight'); uroMousedOverOtherObjectWithinMapComment = true; } isProblem = true; newPopupType = uroMousedOverMarkerType; ureq = W.model.problems.objects[uroMousedOverMarkerID]; if(ureq === undefined) { if(uroDOMHasTurnProblems) { ureq = W.model.turnProblems.objects[uroMousedOverMarkerID]; if(ureq != null) isTurnProb = true; } } let markerBCR = mousedMarker.element.getBoundingClientRect(); uroPopupX = markerBCR.left + markerBCR.width; uroPopupY = markerBCR.top + (markerBCR.height / 2); uroFID = uroMousedOverMarkerID; uroAddLog('building popup for problem '+uroMousedOverMarkerID); if(isTurnProb) result = '<b>Turn Problem ('+uroMousedOverMarkerID+'): ' + I18n.lookup("problems.types.turn.title"); else { result = '<b>Map Problem ('+uroMousedOverMarkerID+'): '; 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: ' + uroClickify(ureq.attributes.extraInfo, '<br>'); } if(ureq.attributes.provider != null) { result += 'Provider: ' + ureq.attributes.provider + '<br>'; } if(ureq.attributes.startTime != null) { result += 'From: ' + uroGetDateTimeString(ureq.attributes.startTime) + '<br>'; } if(ureq.attributes.endTime != null) { result += 'To: ' + uroGetDateTimeString(ureq.attributes.endTime) + '<br>'; } if(ureq.attributes.resolvedOn != null) { uroDaysResolved = uroGetURAge(ureq,1,false); if(uroDaysResolved != -1) { result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' '; if(ureq.attributes.resolvedBy != null) { result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy); } if((ureq.attributes.open === true) && (ureq.attributes.resolvedOn != null)) { result += '<br>Reopened by Waze'; } result += '</i>'; } } } } } if(hovered === false) { uroFID = -1; if(uroStackType !== null) { var tStackType = uroStackType; uroRestackMarkers(); if(tStackType == 1) { uroFilterURs(); } else if(tStackType == 2) { uroFilterProblems(); } else if(tStackType == 3) { uroFilterPlaces(); } } } else if(newPopupType !== null) { // add "open new WME tab" link var urPos=new OpenLayers.LonLat(); if(isPlaceUpdate) { urPos.lon = ureq.getOLGeometry().getCentroid().x; urPos.lat = ureq.getOLGeometry().getCentroid().y; } else { if(ureq.attributes.geometry.realX === undefined) { urPos.lon = ureq.attributes.geometry.x; urPos.lat = ureq.attributes.geometry.y; } else { urPos.lon = ureq.attributes.geometry.realX; urPos.lat = ureq.attributes.geometry.realY; } } urPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326")); var urLink = document.location.href; var urLayers = ''; urLink = urLink.substr(0,urLink.indexOf('?zoomLevel')); urLink += '?zoomLevel=17&lat='+urPos.lat+'&lon='+urPos.lon+urLayers; if(isUR) urLink += '&mapUpdateRequest='+uroMousedOverMarkerID; else if(isTurnProb) urLink += '&showturn='+uroMousedOverMarkerID+'&endshow'; else if(isProblem) urLink += '&mapProblem='+uroMousedOverMarkerID; else if(isPlaceUpdate) { if(uroMousedOverMarkerType == URO_TMARKER.PUR) { urLink += '&showpur='+uroMousedOverMarkerID+'&endshow'; } else { urLink += '&showppur='+uroMousedOverMarkerID+'&endshow'; } } targetTab = "_uroTab_" + Math.round(Math.random()*1000000); result += '<hr><ul><li><a href="'+urLink+'" id="_openInNewTab" target="'+targetTab+'">Open in new tab</a> - '; objHasOpenInNewTabLink = true; result += '<a href="#" id="_recentreSession">centre in current tab</a>'; objHasRecentreSessionLink = true; // add "open new livemap tab" link 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; } } } // look for nodes if((newPopupType === null) && (uroGetCBChecked('_cbInhibitNodesPopup') === false)) { if(uroMousedOverMarkerType == 'node') { ureq = W.model.nodes.objects[uroMousedOverMarkerID]; if(ureq === undefined) { uroMousedOverMarkerID = null; } else { ureqID = uroMousedOverMarkerID; if(uroMousedOverMapComment !== null) { uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for node highlight'); uroMousedOverOtherObjectWithinMapComment = true; } uroPopupY += 40; newPopupType = 'node'; uroFID = ureqID; uroAddLog('building popup for node '+uroFID); result += '<b>Node: ' + uroFID + '</b><br>'; for(var k=0; k<ureq.attributes.segIDs.length; k++) { var nodeSegID = ureq.attributes.segIDs[k]; var nodeStreetID = W.model.segments.objects[nodeSegID].attributes.primaryStreetID; result += uroGetAddress(nodeStreetID, null, false, true, false); } } } } // look for cameras if((newPopupType === null) && (uroGetCBChecked('_cbInhibitCamPopup') === false)) { if(uroMousedOverMarkerType == 'cam') { ureq = W.model.cameras.objects[uroMousedOverMarkerID]; ureqID = uroMousedOverMarkerID; let isSelected = false; let selFeats = W.selectionManager.getSelectedFeatures(); if(selFeats.length == 1) { isSelected = (W.selectionManager.getSelectedFeatures()[0].attributes.wazeFeature.type == 2); isSelected = isSelected && (W.selectionManager.getSelectedFeatures()[0].attributes.wazeFeature.id == ureqID); } if(isSelected === false) { if(uroMousedOverMapComment !== null) { uroAddLog('setting uroMousedOverOtherObjectWithinMapComment for camera highlight'); uroMousedOverOtherObjectWithinMapComment = true; } uroPopupY -= 20; newPopupType = 'camera'; uroFID = ureqID; uroAddLog('building popup for camera '+uroFID); if(I18n.lookup("edit.camera.fields.type") === undefined) { result += '<b>Camera: ' + ureq.TYPES[ureq.attributes.type] + '</b>'; } else { result += '<b>Camera: ' + I18n.lookup("edit.camera.fields.type." + ureq.attributes.type) + '</b>'; } result += '<br>'; result += 'ID: '+uroFID+'<br>'; result += 'Created by '; var userID; if(W.model.users.getByIds([ureq.attributes.createdBy])[0] != 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.getByIds([ureq.attributes.updatedBy])[0] != null) { userID = ureq.attributes.updatedBy; result += uroGetUserNameAndRank(userID); } else result += 'unknown'; result += ', '; camAge = uroGetCameraAge(ureq,0); if(camAge != -1) { result += uroParseDaysAgo(camAge); } else result += 'unknown days ago'; 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>'; } } } if((newPopupType !== null) && (uroPopupDwellTimer === 0) && (uroPopupSuppressed === false)) { if((uroFID != uroShownFID) || (newPopupType != uroShownPopupType)) { if(uroFID != uroShownFID) uroAddLog('FID mismatch, show popup: '+uroFID+'/'+uroShownFID); else uroAddLog('Popup type mismatch: '+newPopupType+'/'+uroShownPopupType); uroShownFID = uroFID; uroShownPopupType = newPopupType; uroPopupShown = false; } if(uroPopupShown === false) { uroAddLog('display popup at '+uroPopupX+','+uroPopupY); uroPopupShown = true; uroDiv.style.height = "auto"; uroDiv.style.width = "auto"; uroDiv.style.overflow = "auto"; uroDiv.innerHTML = uroModifyHTML(result); if((uroFID != -1) && (objHasIgnoreLink === true)) { uroAddEventListener('_addtoignore','click', uroAddToIgnoreList, true); } if(objHasDeleteLink === true) { uroAddEventListener('_deleteobject','click', uroDeleteObject, true); } if(objHasRemoveWatchLink === true) { uroAddEventListener('_removefromwatchlist','click', uroRemoveCamFromWatchList, true); } if(objHasAddWatchLink === true) { uroAddEventListener('_addtowatchlist','click', uroAddCamToWatchList, true); } if(objHasUpdateWatchLink === true) { uroAddEventListener('_updatewatchlist','click', uroUpdateCamWatchList, true); } if(objHasOpenInNewTabLink === true) { uroAddEventListener('_openInNewTab','mouseup', uroOpenNewTab, true); } if(objHasRecentreSessionLink === true) { if(isUR) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnUR, true); else if((isProblem)||(isTurnProb)) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnMP, true); else if(isPlaceUpdate) { if(newPopupType == URO_TMARKER.PUR) { uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnPUR, true); } else { uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnPPUR, true); } } else if(isVenue) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnVenueNavPoint, true); } if(objHasCloneLink === true) { uroAddEventListener('_cloneRP', 'click', uroCloneResidentialPlace, true); } if(newPopupType == 'turn_restriction') { uroAddEventListener('_editTBR','click', uroEditTBR, true); } // restrict the popup width to be no wider than just under half the window width to avoid it // completely overlapping the marker it's associated with - by keeping it to just below half // the window width we guarantee that it'll fit either to the left or the right of the marker // no matter how far across the screen the marker is located... rw = parseInt(uroDiv.clientWidth); uroAddLog('popup width is '+rw); if(rw > (window.innerWidth * 0.45)) { rw = (window.innerWidth * 0.45); uroDiv.style.width = rw+'px'; uroAddLog('restricted popup width to 45% of window width...'); } // get the div height after any adjustment of the width above, to account for whatever content // reflow may have occurred as a result of reducing the width... rh = parseInt(uroDiv.clientHeight); uroAddLog('popup height is '+rh); // similarly restrict the popup height to avoid it dropping of the bottom of the screen if a // segment has a bunch of closures/restrictions rh = parseInt(uroDiv.clientHeight); uroAddLog('popup height is '+rh); if(rh > (window.innerHeight * 0.80)) { rh = (window.innerHeight * 0.80); uroDiv.style.height = rh+'px'; uroDiv.style.overflow = 'scroll'; uroAddLog('restricted popup height to 80% of window height...'); } var origPopupX = uroPopupX; var movedLeft = false; if((uroPopupX + rw) > window.innerWidth) { // where the popup would be off the right hand side of the screen, move it completely over to the // other side of the mouse pointer uroPopupX -= (rw + 20); if(uroPopupX < 0) uroPopupX = 0; movedLeft = true; uroAddLog('popup would fall off RHS of screen, reposition to other side of mouse pointer...'); uroAddLog('uroPopupX is now '+uroPopupX); } if((uroPopupY + rh) > window.innerHeight) { // where the popup would be off the bottom of the screen, shift it up just far enough to be // fully visible uroPopupY -= (((uroPopupY + rh) - window.innerHeight) + 30); uroAddLog('popup would fall off bottom of screen, shift up to keep it all visible...'); } if(uroPopupY < 0) uroPopupY = 0; uroDiv.style.top = uroPopupY+'px'; uroDiv.style.left = uroPopupX+'px'; if(movedLeft === true) { // after relocating the popup to the left of the pointer, it may end up resizing itself // which may then cause it to completely overlap the UR marker, so perform one more check // of the div width and nudge to the left if required... rw = parseInt(uroDiv.clientWidth); uroAddLog('popup width after moving to left is '+rw); if(rw > (window.innerWidth * 0.45)) { rw = (window.innerWidth * 0.45); uroDiv.style.width = rw+'px'; uroAddLog('restricted popup width to 45% of window width...'); } var nudgeDist = parseInt(20 + (uroPopupX + rw) - origPopupX); if((uroPopupX + rw + 30) >= origPopupX) { uroDiv.style.left = parseInt(uroPopupX - nudgeDist)+'px'; } } uroDiv.style.visibility = 'visible'; uroPopupAutoHideTimer = (uroGetElmValue('_inputPopupAutoHideTimeout') * 10); } uroPopupTimer = -1; } else { if((uroPopupTimer == -1) && (uroFID != uroShownFID)) { uroPopupTimer = uroGetElmValue('_inputPopupEntryTimeout'); } } } function uroExclusiveCB() { 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; if(document.getElementById("uroAMList") == null) return; if(document.getElementsByClassName('topbar') == null) return; if(uroGetCBChecked("_cbMoveAMList") === false) { document.getElementsByClassName('area-managers-region')[0].style.display = "block"; uroAMList.innerHTML = uroModifyHTML(""); document.getElementsByClassName('topbar')[0].style.backgroundColor=null; return; } document.getElementsByClassName('topbar')[0].style.backgroundColor="#000000"; document.getElementsByClassName('area-managers-region')[0].style.display = "none"; var amList = ''; var tName = ''; if(W.map.managedAreasLayer.getVisibility() === true) { var mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left; var mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top; var mousePixel = W.map.getLonLatFromPixel(new OpenLayers.Pixel(mouseX, mouseY)); var mousePoint = new OpenLayers.Geometry.Point(mousePixel.lon, mousePixel.lat); for(var amObj in W.model.managedAreas.objects) { if(W.model.managedAreas.objects[amObj].geometry.containsPoint !== undefined) { if(W.model.managedAreas.objects[amObj].geometry.containsPoint(mousePoint)) { var amName = uroGetUserNameFromID(W.model.managedAreas.objects[amObj].attributes.userID); if(amList.indexOf(amName) === -1) { if(amList !== '') amList += ', '; tName = uroGetUserNameAndRank(W.model.managedAreas.objects[amObj].attributes.userID); if(tName.indexOf('a href') !== -1) { tName = tName.replace('a href', 'a style="color:#c0c0ff;" href'); } amList += tName; } } } } if(amList === '') { amList = 'none'; } amList = " - <b>Area Managers:</b> "+amList; } document.getElementById("uroAMList").innerHTML = uroModifyHTML(amList); } function uroNewTabAtMouseLoc(x, y) { var tPix = new OpenLayers.Pixel(x,y); var mPos = W.map.getLonLatFromPixel(tPix).transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326")); var nZoom = W.map.getZoom(); if(nZoom < 17) nZoom = 17; var nHref = window.location.origin + window.location.pathname; nHref += '?lon=' + mPos.lon; nHref += '&lat=' + mPos.lat; nHref += '&zoomLevel=' + nZoom; window.open(nHref); } function uroMouseDown(e) { uroMouseIsDown = true; if((e.altKey === true) && (e.ctrlKey === true)) { uroNewTabAtMouseLoc(e.offsetX, e.offsetY); } } function uroMouseUp() { uroMouseIsDown = false; } function uroTestPointerOutsideMap(mX, mY) { let mapElm = document.getElementById("map"); if(mapElm === undefined) return; let mapBCR = mapElm.getBoundingClientRect(); if ( (mX < mapBCR.left) || (mX > mapBCR.right) || (mY < mapBCR.top) || (mY > mapBCR.bottom) ) { uroPointerWithinMap = false; if(uroGetCBChecked('_cbKillInertialPanning') === true) { var controller = null; if (W.map.navigationControl) { controller = W.map.navigationControl; } else if(W.map.controls.find(control => control.CLASS_NAME == 'OpenLayers.Control.Navigation')) { controller = W.map.controls.find(control => control.CLASS_NAME == 'OpenLayers.Control.Navigation'); } if (controller !== null) { controller.dragPan.panMapStart(); } } return true; } else { uroPointerWithinMap = true; return false; } } function uroMouseOut(e) { if(uroTestPointerOutsideMap(e.clientX, e.clientY)) { uroHidePopup('uroMouseOut'); } } function uroUREvent_onObjectsAdded() { if(uroGetCBChecked('_cbURResolverIDFilter') === true) { uroUpdateEditorList(W.model.mapUpdateRequests.objects, '_selectURResolverID', false, false, true, false); } if((uroBackfilling === false) && (uroPopulatingRequestSessions === false)) { uroFilterURs(); } } function uroGetSelectedURCommentCount() { if(W.model.updateRequestSessions.objects[uroSelectedURID] != null) { var cachedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].attributes.comments.length; uroAddLog(uroSelectedURID+':'+cachedCommentCount+' '+uroExpectedCommentCount); // if there aren't the same number of cached comments as there are comments in the UR dialog list, initiate // a refresh of the comment data... if(cachedCommentCount != uroExpectedCommentCount) { if(uroPendingCommentDataRefresh === true) { if(cachedCommentCount > 0) { uroCachedLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].attributes.comments[cachedCommentCount-1].id; } else { uroCachedLastCommentID = null; } uroAddLog('updateRequestSessions refresh required for UR '+uroSelectedURID); if(uroCachedLastCommentID !== null) { uroAddLog('last comment ID for this UR is '+uroCachedLastCommentID); } else { uroAddLog('first comment for this UR, no previous comment to ID'); } 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.getAsync(idList); // the call to .get() initiates a XMLHttpRequest for the data, so we now need to switch modes - the // refresh process has started so we're no longer pending, but we are now waiting for the XMLHttpRequest // to return something... uroPendingCommentDataRefresh = false; uroWaitingCommentDataRefresh = true; } else { if(cachedCommentCount > 0) { var currentLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].attributes.comments[cachedCommentCount-1].id; if(currentLastCommentID == uroCachedLastCommentID) { // most recent comment loaded for this UR is the same one that was present at the start of this // refresh process, so kick back into pending mode so we can retry the .get()... uroAddLog('latest comment ID still the same, reverting to pending mode...'); uroPendingCommentDataRefresh = true; } else { // something may have gone awry here - the most recent comment loaded for this UR doesn't have the // same ID as the one present at the start of the refresh process, yet the comment counts still don't // match up, which suggests either a comment got lost along the way or someone else has commented on // the same UR at almost the same time. To get out of the loop this would create, assume that a // mismatch in the IDs means the .get() has completed successfully no matter what the new comment // count is, and take this new count to be the count we were expecting all along... uroAddLog('latest comment ID different, but expected count not correct...'); uroExpectedCommentCount = cachedCommentCount; } } else { uroAddLog('first comment on this UR not received yet, reverting to pending mode...'); uroPendingCommentDataRefresh = true; } } } else { // if the WME session is loaded with a UR already selected, such that WME has opened the UR dialog as part // of the session startup process, adding new comments to the UR cause the cached data to be updated immediately. // This prevents URO+ from switching into waiting mode in the above block of code, so we have to instead do // it here by comparing the cached count against the expected count following the Send click event. if(cachedCommentCount >= uroExpectedCommentCount) { uroPendingCommentDataRefresh = false; uroWaitingCommentDataRefresh = true; uroExpectedCommentCount = null; } // once the cached data has been updated, refilter the URs so that the new comment count is taken into account // immediately for filtering and display purposes if(uroWaitingCommentDataRefresh === true) { uroWaitingCommentDataRefresh = false; uroFilterURs(); uroAddLog('refresh complete'); } } } } function uroAddedComment() { // when the user clicks the Send button to submit a new UR comment, this event handler fires before the new comment is // posted to the server and thus also before the comment list gets updated in the UR dialog. So we take the current // comment count and, if the new comment edit box isn't empty, increment it by 1 to get the expected count. Then we // set the pending flag true to initiate a session refresh on the next 100ms tick uroExpectedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].attributes.comments.length; if(document.getElementsByClassName('new-comment-text')[0].value !== '') { uroExpectedCommentCount++; uroAddLog('new comment added to UR '+uroSelectedURID+', cache refresh required...'); uroPendingCommentDataRefresh = true; } else { uroPendingCommentDataRefresh = false; } } function uroInhibitNextUpdateRequestButton(e) { e.stopPropagation(); var doClick = true; if(document.getElementsByClassName('form-control new-comment-text').length > 0) { if(document.getElementsByClassName('form-control new-comment-text')[0].textLength > 0) { uroShowAlertBox("fa-warning", "URO+ Warning", "Comment not sent, close report panel anyway?", true, "Yes", "No", uroCloseReportPanel, null); // set doClick to false here, as uroCloseReportPanel will be called by the alert box handler if required... doClick = false; } } // no alert box has been generated, so close the panel if(doClick) { uroCloseReportPanel(); } } function uroCloseReportPanel() { document.getElementsByClassName('close-panel')[0].click(); } function uroIncrementClosureDate(oldDate, incByDays) { // Thanks to WME no longer using consistent ISO8601 date formatting when displaying // closure details, parsing the date string is now somewhat more involved. Thanks devs... // Default to returning the existing date, just in case we can't increment it var retval = oldDate; var sepChar = null; // Search through oldDate for a non-digit character, so we know what's // being used to seperate the year, month and date values by this locale... for(var i = 0; i < oldDate.length; ++i) { if((oldDate[i] < '0') || (oldDate[i] > '9')) { sepChar = oldDate[i]; break; } } if(sepChar != null) { // First build a "probe" date object we can use to determine what the user's WME locale // does with dates. var incDate = new Date(); incDate.setFullYear(3000); incDate.setMonth(10); incDate.setDate(22); // With these three carefully chosen date elements set, we now generate a localised datestring // using the WME locale setting - the first character of which then tells us the date // format - 1 = MDY (ack ptui), 2 = DMY (good), 3 = YMD (sweet!) var localeDate = incDate.toLocaleDateString(I18n.locale); var dateFormat = localeDate[0]; // Now we know the seperator character and the date format, so we can finally start to parse // the existing date string... var oldDateBits = oldDate.split(sepChar); var datePos; var monthPos; var yearPos; if(dateFormat == '1') { datePos = 1; monthPos = 0; yearPos = 2; } else if(dateFormat == '2') { datePos = 0; monthPos = 1; yearPos = 2; } else { datePos = 2; monthPos = 1; yearPos = 0; } incDate.setFullYear(parseInt(oldDateBits[yearPos])); incDate.setMonth(parseInt(oldDateBits[monthPos]) - 1); incDate.setDate(parseInt(oldDateBits[datePos])); var tDate = new Date(incDate.getTime()); incDate.setDate(tDate.getDate() + incByDays); retval = incDate.toLocaleDateString(I18n.locale); // Except for those pesky locales where toLocaleDateString() doesn't *quite* // return what WME is expecting. Because, you know, why make it easy for // scripters when you can chuck in a few subtle curveballs like this, eh... if(I18n.locale == 'bg') { // Remove the "r." suffix retval = retval.split(" ")[0] + " "; } } return retval; } function uroGetElementProperty(elmName, elmOffset, elmProperty) { var retval = null; if(document.getElementsByName(elmName).length > elmOffset) { retval = document.getElementsByName(elmName)[elmOffset][elmProperty]; } else if(document.getElementById(elmName) !== null) { retval = document.getElementById(elmName)[elmProperty]; } return retval; } function uroGetShadowElementProperty(elmName, shadowID, property) { var retval = null; var tObj = document.getElementById(elmName); if(tObj !== null) { var sObj = tObj.shadowRoot.querySelector('#'+shadowID); if(sObj === null) { sObj = tObj.shadowRoot.querySelector('.'+shadowID); } if(sObj !== null) { retval = sObj[property]; } } 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.getByIds([uroCRPStreetID])[0]; if(streetObj !== undefined) { document.getElementsByClassName('street-name')[0].value = streetObj.name; document.getElementsByClassName('street-name')[0].dispatchEvent(new Event('change', { 'bubbles': true })); // city name var cityObj = W.model.cities.getByIds([streetObj.cityID])[0]; if(cityObj !== undefined) { if(cityObj.attributes.isEmpty === true) { // The donor point place we create to take the cloned RPP properties may have been automatically given // a city name by WME, and thus the city name field will already be filled in and activated... If our // RPP doesn't however have a city name, we need to deactivate the city name field again, so that WME // doesn't complain about the user trying to save the new RPP with an empty city name if(document.getElementsByClassName("empty-city")[0].checked === false) { var snelm = document.getElementsByClassName('empty-city')[0]; snelm.checked = true; snelm.dispatchEvent(new Event('change', { 'bubbles': true })); } } document.getElementsByClassName('city-name')[0].value = cityObj.attributes.name; document.getElementsByClassName('city-name')[0].dispatchEvent(new Event('change', { 'bubbles': true })); // county document.getElementsByClassName('state-id')[0].value = cityObj.attributes.stateID; document.getElementsByClassName('state-id')[0].dispatchEvent(new Event('change', { 'bubbles': true })); // country document.getElementsByClassName('country-id')[0].value = cityObj.attributes.countryID; document.getElementsByClassName('country-id')[0].dispatchEvent(new Event('change', { 'bubbles': true })); } } // house number document.getElementsByClassName('house-number')[0].value = uroCRPHouseNumber; document.getElementsByClassName('house-number')[0].dispatchEvent(new Event('change', { 'bubbles': true })); // now wait for the user to confirm everything and click Apply... window.setTimeout(uroFinaliseCloneRP, 100); } function uroFinaliseCloneRP() { if(document.getElementsByClassName('address-form')[0].style.display != 'none') { window.setTimeout(uroFinaliseCloneRP, 100); return; } // once the user has applied the address changes and closed the address edit panel, WME will then // allow the place to be converted to residential... document.getElementsByClassName("toggle-residential")[0].click(); } function uroConvertToRP() { // panel isn't open yet, which means the user either hasn't clicked yet or WME is still processing the // placement of the venue, so wait a while and then check again... if(document.getElementById('edit-panel').getElementsByClassName('landmark').length === 0) { window.setTimeout(uroConvertToRP, 100); return; } // panel is open, so move to the next step of the cloning procedure by clicking on the address edit icon... document.getElementsByClassName('full-address-container')[0].children[0].click(); // now click on the "none" checkbox for the street name edit field so we can enter the street name var snelm = document.getElementsByClassName('empty-street')[0]; snelm.checked = false; snelm.dispatchEvent(new Event('change', { 'bubbles': true })); // WME automatically clears the checkbox associated with the city name edit field if we set the street // name to be one that has a city associated with it, which is nice :-) // the click event seems to take a while to execute, and if we call dispatchEvent on the edit field whilst // it's still tagged as disabled then it gets ignored, causing the value in that field to be dropped when // we apply the changes to the place. Trying to programatically detect when the field has been activated // doesn't seem to be reliable, however a fixed delay of 1s seems to work nicely window.setTimeout(uroCompleteRPClone, 1000); } function uroCloneResidentialPlace() { // trying to clone a RPP when one is already selected causes the selected one to be changed back to // a non-residential, as uroConvertToRP() thinks the user has already clicked to place the new RPP... if(document.getElementById('edit-panel').getElementsByClassName('landmark').length === 0) { 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 uroCReason; var uroCEvent; var uroCDirection; var uroCStartDate; var uroCStartTime; var uroCEndDate; var uroCEndTime; var uroCIgnoreTraffic; var uroCClosedNodes; function uroCompleteClosureCloning() { var loop; if(document.getElementsByClassName('edit-closure').length === 0) { window.setTimeout(uroCompleteClosureCloning,100); return; } if(uroTempFixMTEDropDown() == false) { window.setTimeout(uroCompleteClosureCloning,100); return; } // need to generate a change event on each of the form fields, because WME appears to be silently populating some hidden // closure object with the details as they're entered manually, and if we just set the form values without then forcing // the change event as well then WME will end up using its default values instead of the ones we've so lovingly copied... let form = $('#edit-panel .closures .edit-closure form'); if(uroCReason !== null) { let fObj = form.find('wz-text-input#closure_reason'); fObj.val(uroCReason); fObj.change(); } if(uroCDirection !== null) { let fObj = form.find('wz-select#closure_direction'); fObj[0].value = uroCDirection; fObj.change(); } if(uroCStartDate !== null) { let fObj = form.find('input#closure_startDate'); fObj.val(uroCStartDate); fObj.change(); } if(uroCStartTime !== null) { let fObj = form.find('div.form-group.start-date-form-group input.time-picker-input'); fObj.focus(); fObj.val(uroCStartTime); fObj.change(); } if(uroCIgnoreTraffic !== null) { let fObj = form.find('wz-checkbox#closure_permanent'); fObj.val(uroCIgnoreTraffic); fObj.change(); } if(uroCEndTime !== null) { let fObj = form.find('div.form-group.end-date-form-group input.time-picker-input'); fObj.focus(); fObj.val(uroCEndTime); fObj.change(); } // the current version of WME wipes any existing end date as soon as the end time is altered, so we now need // to set the date after the time instead of before as in earlier versions of this function... if(uroCEndDate !== null) { let fObj = form.find('input#closure_endDate'); fObj.val(uroCEndDate); fObj.change(); } // the old method of setting the MTE just by changing the value attribute on closure_eventId no longer // seems to work as expected (it runs OK from the dev console, but not within the scope of the userscript), // so just as we do for setting the event to None, we set the event to the desired value here by finding // the appropriate menu entry and clicking on it... var cEvents = document.getElementById('closure_eventId').getElementsByTagName('wz-option'); for(var i of cEvents) { if(i.value == uroCEvent) { i.click(); break; } } var nNodes = uroCClosedNodes.length; if(nNodes > 0) { let fObj = form.find('wz-toggle-switch'); for(loop = 0; loop < nNodes; ++loop) { if(uroCClosedNodes[loop] === true) { fObj[loop].click(); } } } if(uroGetCBChecked('_cbAutoApplyClonedClosure') == true) { window.setTimeout(uroClickClosureSave,100); } uroPendingClosureClone = -1; } function uroClickClosureSave() { document.getElementsByClassName('closures')[0].getElementsByClassName('save-button')[0].click(); } var uroPendingClosureClone = -1; var uroPendingClosureCloneIncrement = 0; function uroCopyClosure(closureOffset) { // grab the current closure details from the UI... uroCReason = uroGetShadowElementProperty('closure_reason', 'id', 'value'); uroCDirection = uroGetElementProperty('closure_direction', 0, 'value'); uroCStartDate = uroGetElementProperty('closure_startDate', 0, 'value'); uroCStartTime = document.querySelector('.start-date-form-group').querySelector('.time-picker-input').value; uroCEndDate = uroGetElementProperty('closure_endDate', 0, 'value'); uroCEndTime = document.querySelector('.end-date-form-group').querySelector('.time-picker-input').value; uroCEvent = uroGetElementProperty('closure_eventId', 0, 'value'); uroCIgnoreTraffic = uroGetElementProperty('closure_permanent', 0, 'checked'); uroCClosedNodes = []; var nNodes = document.getElementsByClassName('fromNodeClosed').length; if(nNodes > 0) { for(var loop = 0; loop < nNodes; ++loop) { uroCClosedNodes.push(document.getElementsByClassName('fromNodeClosed')[loop].checked); } } document.getElementsByClassName('closures')[0].getElementsByClassName('cancel-button')[0].click(); // auto-increment the start and end dates uroCStartDate = uroIncrementClosureDate(uroCStartDate,uroPendingClosureCloneIncrement); uroCEndDate = uroIncrementClosureDate(uroCEndDate,uroPendingClosureCloneIncrement); uroPendingClosureClone = -2; } function uroCloneClosure() { uroPendingClosureCloneIncrement = parseInt(this.id.split('-')[1]); uroPendingClosureClone = parseInt(this.id.split('-')[2]); } function uroDeleteNextClosureOnList() { var nClosures = document.querySelectorAll('.closure-item.is-editable').length; if(nClosures > 0) { if (nClosures != uroClosuresToDelete) { uroClosuresToDelete = nClosures; var ctObj = document.querySelector('.closure-item.is-editable'); var deleteMenuEntry = ctObj.querySelector('wz-menu-item.delete'); if(deleteMenuEntry !== null) { deleteMenuEntry.click(); } } window.setTimeout(uroDeleteNextClosureOnList,100); } else { uroConfirmClosureDelete = true; } } function uroDeleteAllClosures() { uroConfirmClosureDelete = true; uroShowAlertBox("fa-warning", "URO+ Warning", I18n.lookup("closures.delete_confirm_no_reason")+' ('+I18n.lookup("closures.apply_to_all")+')', true, "Yes", "No", uroDeleteAllClosuresAction, null); } function uroDeleteAllClosuresAction() { uroConfirmClosureDelete = false; var nClosures = document.getElementsByClassName('closure-item').length; if(nClosures > 0) { uroClosuresToDelete = -1; uroDeleteNextClosureOnList(); } else { uroConfirmClosureDelete = true; } } //} function uroGetMarkerType(marker) { var markerType = null; if(marker.tagName == "image") { markerType = 'cam'; } else if(marker.tagName == "circle") { markerType = 'node'; } else { if(marker.className.indexOf('user-generated') !== -1) markerType = URO_TMARKER.UR; else if(marker.className.indexOf('map-problem') !== -1) markerType = URO_TMARKER.MP; else if(marker.className.indexOf('place-update') !== -1) { if(marker.parentNode.id === W.map.getLayerByName("PARKING_PLACE_UPDATES").div.id) { markerType = URO_TMARKER.PPUR; } else if(marker.parentNode.id === W.map.getLayerByName("RESIDENTIAL_PLACE_UPDATES").div.id) { markerType = URO_TMARKER.RPUR; } else if(marker.parentNode.id === W.map.getLayerByName("place_updates").div.id) { markerType = URO_TMARKER.PUR; } } } return markerType; } function uroGetCameraIDFromGeoID(geoID) { var camID = null; var i=W.map.camerasLayer.features.length; var camObj; while(i > 0) { camObj = W.map.camerasLayer.features[i-1]; if(camObj.attributes.wazeFeature._wmeObject !== undefined) { if(camObj.attributes.wazeFeature._wmeObject.geometry.id === geoID) { camID = camObj.attributes.wazeFeature._wmeObject.attributes.id; break; } } i--; } return camID; } function uroGetNodeIDFromGeoID(geoID) { var nodeID = null; var i=W.map.nodeLayer.features.length; var nodeObj; while(i > 0) { nodeObj = W.map.nodeLayer.features[i-1]; if(nodeObj.geometry.id === geoID) { nodeID = nodeObj?.attributes?.wazeFeature?._wmeObject?.attributes?.id; break; } i--; } return nodeID; } var uroLastMarkerMousedOver = null; function uroMarkerMouseOver(e) { var markerType; markerType = uroGetMarkerType(this); if(markerType !== null) { var markerID = null; if(markerType === 'cam') { markerID = uroGetCameraIDFromGeoID(this.id); if(uroGetCBChecked('_cbHighlightInsteadOfHideCams') === true) { if(uroLastMarkerMousedOver !== markerID) { window.setTimeout(uroFilterCameras, 50); } } } else if(markerType === 'node') { markerID = uroGetNodeIDFromGeoID(this.id); } else { markerID = this.attributes["data-id"].value; } uroAddLog('hover over marker (Type '+markerType+' / ID '+markerID+')'); uroMousedOverMarkerID = markerID; uroMousedOverMarkerType = markerType; if(markerType == URO_TMARKER.UR) uroHoveredURID = markerID; if((markerType == URO_TMARKER.UR) || (markerType == URO_TMARKER.MP)) { uroChangeCustomMarkers(markerID,true,markerType); } uroLastMarkerMousedOver = markerID; } else { uroAddLog('hover over unknown object...'); } } function uroMarkerMouseOut(e) { var markerType; markerType = uroGetMarkerType(this); if(markerType !== null) { var markerID = null; if(markerType === 'cam') { markerID = uroGetCameraIDFromGeoID(this.id); if(uroGetCBChecked('_cbHighlightInsteadOfHideCams') === true) { window.setTimeout(uroFilterCameras, 50); } } else if(markerType === 'node') { markerID = uroGetNodeIDFromGeoID(this.id); } else { markerID = this.attributes["data-id"].value; } uroAddLog('hover off '+markerType+' ID '+markerID); uroMousedOverMarkerID = null; uroMousedOverMarkerType = null; uroHoveredURID = null; if((markerType == URO_TMARKER.UR) || (markerType == URO_TMARKER.MP)) { uroChangeCustomMarkers(markerID,false,markerType); } } else { uroAddLog('hover off unknown object...'); } uroLastMarkerMousedOver = null; } let uroInhibitSetCenter = false; function uroMarkerMouseDown() { // Do this stuff in the mousedown event rather than the click event so we fire before any of the native // click events - we need to ensure this happens for inhibiting marker centering, as we need to capture // the markerType ahead of our interceptor function being called to deal with the centering... var markerType = uroGetMarkerType(this); if(markerType !== null) { var markerID = this.attributes["data-id"].value; uroAddLog('clicked on '+markerType+' marker '+markerID); uroClickedOnMarkerID = markerID; uroClickedOnMarkerType = markerType; uroClickedOnMarkerForCenterInterceptor = markerType; uroClickedOnMarkerMapCenter = W.map.getCenter(); uroInhibitURFiltering = true; /* if(uroInhibitSetCenter === false) { if(uroDecentre() === true) { console.debug('Inhibiting setCenter...'); uroInhibitSetCenter = true; window.setTimeout(uroReenableSetCenter, 2000); } } */ } } function uroDecentre() { let inhibit = false; inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.UR) && (uroGetCBChecked("_cbInhibitURCentering"))); inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.MP) && (uroGetCBChecked("_cbInhibitMPCentering"))); inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.PUR) && (uroGetCBChecked("_cbInhibitPURCentering"))); inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.PPUR) && (uroGetCBChecked("_cbInhibitPPURCentering"))); inhibit = inhibit || ((uroClickedOnMarkerForCenterInterceptor == URO_TMARKER.RPUR) && (uroGetCBChecked("_cbInhibitRPURCentering"))); return inhibit; } function uroReenableSetCenter() { console.debug('Re-enabling setCenter...'); uroInhibitSetCenter = false; } function uroBlobMouseOver(e) { var blobType = this.attributes.uroBlobType; if(blobType !== undefined) { var blobID = this.attributes.uroBlobID; uroAddLog('hover over '+blobType+' ID '+blobID); } 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); if(blobType == 'map_comment') { if(W.model.mapComments.objects[blobID] != undefined) { var geoID = W.model.mapComments.objects[blobID].attributes.geometry.id; if(geoID.indexOf('Point') != -1) { // reapply visibility mods var svgElm = document.getElementById(uroMCLayer.div.id+'_vroot'); for(var svgIdx = 0; svgIdx < svgElm.children.length; svgIdx++) { if(svgElm.children[svgIdx].id === geoID) { window.setTimeout(uroReapplyPointMCVisibilityMods,10); } } } } } } else { uroAddLog('hover off unknown blob...'); } } function uroBlobClick() { var blobType = this.attributes.uroBlobType; if(blobType !== undefined) { var blobID = this.attributes.uroBlobID; uroAddLog('clicked on '+blobType+' blob '+blobID); } } function uroMCLayerChanged_changed() { uroMCLayerChanged(); } function uroMCLayerChanged_added() { uroMCLayerChanged(); } function uroMCLayerChanged_removed() { uroMCLayerChanged(); } function uroReapplyPointMCVisibilityMods() { if(uroApplyPointMCVisibilityMods() === false) { window.setTimeout(uroReapplyPointMCVisibilityMods,100); } } function uroApplyPointMCVisibilityMods() { var retval = true; if(uroHasSelectedMCs() === true) { retval = false; } else { var svgElm = document.getElementById(uroMCLayer.div.id+'_vroot'); for(var svgIdx = 0; svgIdx < svgElm.children.length; svgIdx++) { var svgChild = svgElm.children[svgIdx]; if(svgChild.id.indexOf('Point') != -1) { if(uroGetCBChecked('_cbMCEnhancePointMCVisibility') === true) { if(svgChild.getAttribute('r') == 6) { svgChild.setAttribute('fill','#ffff00'); svgChild.setAttribute('fill-opacity',0.75); svgChild.setAttribute('r',12); svgChild.setAttribute('touchedByURO',true); } else if((svgChild.getAttribute('touchedByURO') === "true")&&(svgChild.getAttribute('fill') === '#ffff00')) { // do nothing... } else { retval = false; break; } } else { if((svgChild.getAttribute('touchedByURO') === "true")&&(svgChild.getAttribute('fill') === '#ffff00')) { svgChild.setAttribute('fill','#ffffff'); svgChild.setAttribute('fill-opacity',1); svgChild.setAttribute('r',6); svgChild.setAttribute('touchedByURO',false); } } } } } return retval; } function uroHasSelectedMCs() { var retval = false; for(var mcObj in W.model.mapComments.objects) { if(W.model.mapComments.objects[mcObj].isSelected() === true) { retval = true; break; } } return retval; } function uroMCLayerChanged() { uroWazeBits(); if(uroMCLayer != null) { if(uroHasSelectedMCs() === false) { uroAddLog('adding MC blob event handlers'); var mcModel = null; for(var mObj=0; mObj<uroMCLayer.features.length; mObj++) { if(uroMCLayer.features[mObj].attributes.wazeFeature._wmeObject !== undefined) { mcModel = uroMCLayer.features[mObj].attributes.wazeFeature._wmeObject; { if(mcModel.selected !== true) { var mcBlobID = mcModel.attributes.geometry.id; var mcID = mcModel.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"; uroAddLog('added handlers to MC '+mcID); } } } } } uroApplyPointMCVisibilityMods(); } else { uroAddLog('MC selected, handlers not added yet...'); } uroFilterMapComments(); } } function uroPlaceLayerChanged() { uroAddLog('adding place blob event handlers'); for(var mObj=0; mObj<uroVenueLayer.features.length; mObj++) { // clicking on an area place now adds the polygon drag handles into the features[] array, so we need to test that // the current array entry isn't referring to one of these handles before trying to access the attributes... if(uroVenueLayer.features[mObj]?.attributes?.wazeFeature?._wmeObject !== undefined) { var mcBlobID = uroVenueLayer.features[mObj].attributes.wazeFeature._wmeObject.attributes.geometry.id; var mcID = uroVenueLayer.features[mObj].attributes.wazeFeature._wmeObject.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 uroAddMarkerEventHandlers(layerType) { var idList = uroGetMarkerIDs(layerType); for(var mObj of idList) { var mMarker = uroGetMarker(layerType, mObj); if(mMarker !== null) { var mIcon = mMarker.element; if(mIcon !== undefined) { mIcon.addEventListener("mouseover",uroMarkerMouseOver, false); mIcon.addEventListener("mouseout",uroMarkerMouseOut, false); mIcon.addEventListener("mousedown",uroMarkerMouseDown, false); ////mIcon.addEventListener("click",uroMarkerClick, false); } } } } function uroURLayerChanged() { uroAddLog('UR layer change detected'); uroAddMarkerEventHandlers(URO_TMARKER.UR); uroMO_URLayer.disconnect(); uroFilterURs(); uroMO_URLayer.observe(W.map.getLayerByName("update_requests").div,{childList: true, attributes : true, characterData : true, subtree: true}); } function uroMPLayerChanged() { uroAddLog('adding MP marker event handlers'); uroAddMarkerEventHandlers(URO_TMARKER.MP); } function uroPURLayerChanged() { uroAddLog('adding PUR marker event handlers'); uroAddMarkerEventHandlers(URO_TMARKER.PUR); } function uroPPULayerChanged() { uroAddLog('adding PPU marker event handlers'); uroAddMarkerEventHandlers(URO_TMARKER.PPUR); } function uroRPULayerChanged() { uroAddLog('adding RPU marker event handlers'); uroAddMarkerEventHandlers(URO_TMARKER.RPUR); } function uroCamLayerChanged() { uroAddLog('adding camera marker event handlers'); var i = W.map.camerasLayer.features.length; var svgElm = null; while(i > 0) { svgElm = document.getElementById(W.map.camerasLayer.features[i-1].geometry.id); if(svgElm !== null) { svgElm.addEventListener("mouseover", uroMarkerMouseOver, false); svgElm.addEventListener("mouseout", uroMarkerMouseOut, false); } i--; } } var uroDelayNodeLayerUpdate = true; function uroNodeLayerChanged() { if(uroDelayNodeLayerUpdate === true) { // When the layer change event fires, WME hasn't yet updated the SVG content and so the node markers visible at this point // in time will get nuked once the update takes place. We must therefore wait a short time to allow the SVG update to occur // before attaching event listeners to the node markers... uroDelayNodeLayerUpdate = false; window.setTimeout(uroNodeLayerChanged, 1000); } else { uroDelayNodeLayerUpdate = true; uroAddLog('adding node event handlers'); var i = W.map.nodeLayer.features.length; if(i > 0) { var svgElm = null; while(i > 0) { svgElm = document.getElementById(W.map.nodeLayer.features[i-1].geometry.id); if(svgElm !== null) { svgElm.addEventListener("mouseover", uroMarkerMouseOver, false); svgElm.addEventListener("mouseout", uroMarkerMouseOut, false); } i--; } uroNodeLayerScanAttempts = 0; } } } function uroClosuresLayerChanged() { uroAddLog('reapplying closures filter'); uroFilterRTCs(); } function uroSidePanelChanged() { if(document.querySelector('wz-tab.closures-tab') !== null) { var uroMO_ClosureUI = new MutationObserver(uroClosureEditUIChanged); uroMO_ClosureUI.disconnect(); uroMO_ClosureUI.observe(document.querySelector('wz-tab.closures-tab'),{subtree: true, attributes: true}); uroClosureEditUIChanged(); } } function uroClosureEditUIChanged() { if(document.getElementsByClassName('edit-closure').length === 1) { // note: this also fires when the UI is closed, due to the change events triggered as its elements are removed // prior to the tab itself closing... var notReady = 0; if(document.getElementById('closure_reason').shadowRoot.querySelector('#id') === null) notReady += 1; if(document.getElementById('closure_direction').shadowRoot.querySelector('.selected-value') === null) notReady += 2; else if(document.getElementById('closure_direction').shadowRoot.querySelector('.selected-value').innerText === "") notReady += 4; if(document.getElementById('closure_startDate') === null) notReady += 8; if(document.getElementById('closure_endDate') === null) notReady += 16; if(document.getElementById('closure_eventId').shadowRoot.querySelector('.selected-value') === null) notReady += 32; else if(document.getElementById('closure_eventId').shadowRoot.querySelector('.selected-value').innerText === "") notReady += 64; if(document.getElementById('closure_permanent').shadowRoot.querySelector('.wz-checkbox') === null) notReady += 128; if(notReady === 0) { if(uroPendingClosureClone === -3) { uroCompleteClosureCloning(); } else if(uroPendingClosureClone !== -1) { uroCopyClosure(uroPendingClosureClone); } else { uroTempFixMTEDropDown(); } } } else { if(uroPendingClosureClone === -2) { // generate a click event on the Add a closure button to open up the closure editing UI, then // wait for the UI to finish opening... document.getElementsByClassName('add-closure-button')[0].click(); uroPendingClosureClone = -3; } uroTSTClosureCloningHandler(); } } function uroFinalizeListenerSetup() { uroFinalisingListenerSetup = true; // filter markers when the marker objects are modified (this happens whenever WME needs to load fresh marker data // due to having panned/zoomed the map beyond the extents of the previously loaded data) W.model.mapUpdateRequests.on("objectschanged", uroFilterURs_onObjectsChanged); W.model.mapUpdateRequests.on("objectsadded", uroFilterURs_onObjectsAdded); W.model.mapUpdateRequests.on("objectsremoved", uroFilterURs_onObjectsRemoved); W.model.updateRequestSessions.on("objectsadded", uroUREvent_onObjectsAdded); W.model.cameras.on("objectschanged", uroCamLayerChanged); W.model.cameras.on("objectsadded", uroCamLayerChanged); W.model.cameras.on("objectsremoved", uroCamLayerChanged); W.model.cameras.on("objectschanged", uroFilterCameras); W.model.cameras.on("objectsadded", uroFilterCameras); W.model.cameras.on("objectsremoved", uroFilterCameras); W.model.problems.on("objectschanged", uroFilterProblems); W.model.problems.on("objectsadded", uroFilterProblems); W.model.problems.on("objectsremoved", uroFilterProblems); W.model.venues.on("objectschanged", uroFilterPlaces); W.model.venues.on("objectsadded", uroFilterPlaces); W.model.venues.on("objectsremoved", uroFilterPlaces); W.model.mapComments.on("objectschanged", uroMCLayerChanged_changed); W.model.mapComments.on("objectsadded", uroMCLayerChanged_added); W.model.mapComments.on("objectsremoved", uroMCLayerChanged_removed); var uroMO_PlaceLayer = new MutationObserver(uroPlaceLayerChanged); uroMO_PlaceLayer.observe(uroVenueLayer.div,{childList: true, attributes : true, characterData : true, subtree: true}); uroMO_URLayer = new MutationObserver(uroURLayerChanged); uroMO_URLayer.observe(W.map.getLayerByName("update_requests").div,{childList: true, attributes : true, characterData : true, subtree: true}); var uroMO_MPLayer = new MutationObserver(uroMPLayerChanged); uroMO_MPLayer.observe(W.map.getLayerByName("mapProblems").div,{childList: true, attributes : true, characterData : true, subtree: true}); var uroMO_PURLayer = new MutationObserver(uroPURLayerChanged); uroMO_PURLayer.observe(W.map.getLayerByName("place_updates").div,{childList: true, attributes : true, characterData : true, subtree: true}); var uroMO_PPULayer = new MutationObserver(uroPPULayerChanged); uroMO_PPULayer.observe(W.map.getLayerByName("PARKING_PLACE_UPDATES").div,{childList: true, attributes : true, characterData : true, subtree: true}); var uroMO_RPULayer = new MutationObserver(uroRPULayerChanged); uroMO_RPULayer.observe(W.map.getLayerByName("RESIDENTIAL_PLACE_UPDATES").div,{childList: true, attributes : true, characterData : true, subtree: true}); var uroMO_ClosuresLayer = new MutationObserver(uroClosuresLayerChanged); uroMO_ClosuresLayer.observe(W.map.getLayerByName("closures").div,{childList: true, attributes : true, characterData : true, subtree: true}); var uroMO_NodeLayer = new MutationObserver(uroNodeLayerChanged); uroMO_NodeLayer.observe(W.map.nodeLayer.div,{childList: true, attributes : true, characterData : true, subtree: true}); var uroMO_SidePanel = new MutationObserver(uroSidePanelChanged); uroMO_SidePanel.observe(document.getElementById('edit-panel'),{childList: true, subtree: true}); uroAddEventListener('_btnUndoLastHide',"click", uroRemoveLastAddedIgnore, true); uroAddEventListener('_btnClearSessionHides',"click", uroRemoveAllIgnores, true); uroEnableIgnoreListControls(); uroAddEventListener('_btnClearCamWatchList',"click", uroClearCamWatchList, true); uroAddEventListener('_btnSettingsToText',"click", uroSettingsToText, true); uroAddEventListener('_btnTextToSettings',"click", uroTextToSettings, true); uroAddEventListener('_btnResetSettings',"click", uroDefaultSettings, true); uroAddEventListener('_btnClearSettingsText',"click", uroClearSettingsText, true); uroAddEventListener('_cbMasterEnable',"click", uroFilterItems_MasterEnableClick, true); /* uroAddEventListener('_btnDebugToScreen',"click", uroDumpDebug, true); */ uroAddEventListener('uroDiv',"dblclick",uroSuppressPopup,true); uroAddEventListener('_selectCameraUserID',"change", uroCamEditorSelected, true); uroAddEventListener('_selectPlacesUserID',"change", uroPlacesEditorSelected, true); uroAddEventListener('_selectHidePlacesUserID',"change", uroHidePlacesEditorSelected, true); uroAddEventListener('uroAlertTickBtn','click',uroCloseAlertBoxWithTick,true); uroAddEventListener('uroAlertCrossBtn','click',uroCloseAlertBoxWithCross,true); for(var i = 0; i < uroCtrlTabs.length; ++i) { uroSetOnClick(uroCtrlTabs[i][URO_TABS_FIELD.LINKID], uroCtrlTabs[i][URO_TABS_FIELD.SHOWFN]); } for(var idx=0;idx<W.Config.venues.categories.length;idx++) { uroSetOnClick('_uroPlacesGroupState-'+idx,uroPlacesGroupCollapseExpand); } uroAddLog('finalise onload'); uroNewLookCheckDetailsRequest(); // filter markers as and when the map is moved W.map.events.register("moveend", null, uroFilterItems); W.map.events.register("moveend", null, uroGetAMs); W.map.events.register("mousemove", null, uroGetAMs); W.map.events.register("mousemove", null, uroNewLookHighlightedItemsCheck); W.map.events.registerPriority("mousedown", null, uroMouseDown); // trap mousedown on Streetview marker drag if(document.getElementsByClassName('street-view-control').length === 0) return; document.getElementsByClassName('street-view-control')[0].onmousedown = uroMouseDown; W.map.events.register("mouseup", null, uroMouseUp); W.map.events.register("mouseout", null, uroMouseOut); uroSetSectionTabStyles(); uroLoadSettings(); uroAddLog('getting user ID...'); uroUserID = W.loginManager.user.attributes.id; uroAddLog('...ID is '+uroUserID); uroAddLog('filtering...'); uroFilterItems(); uroAddLog('...done'); uroShowDebugOutput = uroPersistentDebugOutput; 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 uroAddLog('adding exclusiveCB handlers...'); 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); } } uroAddLog('...done'); // manually call the layer-change handlers on startup, since there's a good chance WME will already have // completed its own startup layer changes before our handlers get registered, preventing the marker handlers // from being set up as expected on any markers which are visible in the startup map view before the user forces // a layer update by panning/zooming/etc... uroMCLayerChanged(); uroPlaceLayerChanged(); uroURLayerChanged(); uroMPLayerChanged(); uroPURLayerChanged(); uroCamLayerChanged(); uroNodeLayerChanged(); uroPPULayerChanged(); uroRPULayerChanged(); uroClosuresLayerChanged(); uroSetupListeners = false; uroMainTickStage = 0; window.clearInterval(uroMainTickHandlerID); window.setInterval(uroMainTick, 10); uroInitialised = true; } function uroTSTPopupHandler() { if(document.getElementsByClassName('panel')[0] === undefined) { uroHidePopupOnPanelOpen = true; } if(uroPopupShown === true) { var hidePopup = false; if(document.getElementsByClassName('panel')[0] != null) { if(uroHidePopupOnPanelOpen === true) { hidePopup = true; uroHidePopupOnPanelOpen = false; } } if(hidePopup === true) { uroHidePopup('uroTSTPopupHandler 1'); } } if((uroAreaNameHoverObj !== null) && (uroAreaNameHoverTime != -1) && (uroAreaNameOverlayShown === false)) { if(++uroAreaNameHoverTime > 5) { uroAreaNameOverlaySetup(); } } uroReplaceAreaNames(false); if(uroPopupAutoHideTimer > 0) { if(--uroPopupAutoHideTimer === 0) { uroHidePopup('uroTSTPopupHandler 2'); } } if(uroPopupTimer > 0) { if(uroMouseInPopup === false) { uroPopupTimer--; } } if(uroPopupTimer === 0) { uroHidePopup('uroTSTPopupHandler 3'); } if(uroPopupDwellTimer > 0) { uroPopupDwellTimer--; if(uroPopupDwellTimer === 0) { uroNewLookHighlightedItemsCheck('dwellTimeout'); } } } function uroTSTNextBtnHandler() { // replaces the "next xxx" button on UR, MP and PUR editing UIs // Correctly determining what WME is displaying for the "next" button in the UR/MP/(P)PUR panel is not trivial due to // inconsistencies in the panel behaviour depending on whether it was opened by clicking directly on the relevant // marker, or by clicking on the associated feed entry... For PURs, there's also the added complication of multi-part // update requests, where the same marker/panel are used to access more than one request and where, therefore, we need // to enable access to all requests contained within the PUR, but still inhibit the "next" button once the last // request in the multi-part sequence has been viewed. // // For directly-accesed markers, the "next" button caption is: // // URs = "Next update request" (update_requests.panel.next) // MPs = "Next map problem" (problems.panel.next) // PURs = "Next place" for single-part PURs or for the last part of a multi-part PUR (venues.update_requests.panel.next_venue) // = "Next" for all but the last part of a multi-part PUR (venues.update_requests.panel.next) // PPURs = "Next place" (venues.update_requests.panel.next_venue) // // For markers accessed via the feed, the "next" button caption always appears to be "Next issue" (feed.issues.next) if(W.map.panelRegion.hasView() === true) { var nurButton = W.map.panelRegion.$el[0].getElementsByClassName('next')[0]; if(nurButton === undefined) { nurButton = W.map.panelRegion.$el[0].getElementsByClassName('next-venue')[0]; } if(nurButton !== undefined) { var doneString = I18n.lookup('problems.panel.done'); var btnCaptionIsNextPlace = (nurButton.innerHTML.indexOf(I18n.lookup('venues.update_requests.panel.next_venue')) !== -1); var btnCaptionIsDefaultUR = (nurButton.innerHTML.indexOf(I18n.lookup('update_requests.panel.next')) !== -1); var btnCaptionIsDefaultMP = (nurButton.innerHTML.indexOf(I18n.lookup('problems.panel.next')) !== -1); var btnCaptionIsNextIssue = (nurButton.innerHTML.indexOf(I18n.lookup('feed.issues.next')) !== -1); var updateButton = false; var panelClass = W.map.panelRegion.$el[0].childNodes[0].childNodes[0].className; var isURorMPPanel = (panelClass.indexOf('problem-edit') !== -1); var isPURPanel = (panelClass.indexOf('place-update') !== -1); if(isURorMPPanel === true) { // user has enabled UR button mod? if(uroGetCBChecked('_cbInhibitNURButton') === true) { // the native UR panel button will always either be "Next update request" or "Next issue" updateButton = ((btnCaptionIsDefaultUR) || (btnCaptionIsNextIssue)); } // user has enabled MP button mod? if(uroGetCBChecked('_cbInhibitNMPButton') === true) { // there's no way to determine if the edit panel has been opened for a UR or a MP, however as MPs // don't currently appear in the feed, the native button only uses "Next map problem" as its caption updateButton = (updateButton || btnCaptionIsDefaultMP); } } else if(isPURPanel === true) { if(uroGetCBChecked('_cbInhibitNPURButton') === true) { // for a (P)PUR, only modify the button if it's showing the "Next place" or "Next issue" caption, to // avoid messing up the "Next" button used to move to the next part of a multi-part PUR... updateButton = ((btnCaptionIsNextPlace === true) || (btnCaptionIsNextIssue)); } } if(updateButton === true) { uroAddLog('inhibit Next UR/MP/PUR button'); // alter the button caption nurButton.innerHTML = uroModifyHTML(doneString); // Add a new click handler to override the native one - this acts both to prevent the normal action of the "Next UR/MP/PUR" button in // moving to the next UR/MP/PUR, and also allows us to warn about closing the UR panel if there's an unsent comment... nurButton.addEventListener("click", uroInhibitNextUpdateRequestButton, false); } uroInhibitURFiltering = false; } } } function uroTSTCommentAddedHandler() { // test for the opening or closing of the UR editing dialog so we can detect when a new comment is added var URDialogIsOpen = false; var panelOpen = (document.getElementById('panel-container').firstChild !== null); if(panelOpen) { URDialogIsOpen = (document.getElementById('panel-container').getElementsByClassName('conversation').length > 0); } if(URDialogIsOpen) { var thisSelectedURID = document.getElementsByClassName('permalink')[0].href.split('&mapUpdateRequest='); if(thisSelectedURID.length > 1) { thisSelectedURID = thisSelectedURID[1].split('&')[0]; } else { thisSelectedURID = null; } if((thisSelectedURID != uroSelectedURID) || ((thisSelectedURID != uroClickedOnMarkerID) && (uroClickedOnMarkerID != null))) { // if the user selects a new UR whilst the editing dialog is still open, treat it in the // same way as if the user had selected that UR with the dialog closed uroURDialogIsOpen = false; uroSelectedURID = null; } if(((uroURDialogIsOpen === false) && (uroSelectedURID === null)) || (uroURReclickAttempts > 0)) { // user is editing a new UR // add our own click event handler to the Send button, so we can do stuff whenever a new comment is added if(document.getElementsByClassName('new-comment-form').length > 0) { if(document.getElementsByClassName('new-comment-form')[0].getElementsByClassName('send-button').length > 0) { document.getElementsByClassName('new-comment-form')[0].getElementsByClassName('send-button')[0].addEventListener("click", uroAddedComment, false); uroSelectedURID = thisSelectedURID; uroAddLog('user is editing UR '+uroSelectedURID); uroExpectedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].attributes.comments.length; if((uroHoveredURID !== null) && (uroSelectedURID !== null) && (parseInt(uroHoveredURID) !== parseInt(uroSelectedURID))) { if(uroURReclickAttempts === 0) { uroAddLog('DANGER, WILL ROBINSON! You clicked on UR ID '+uroHoveredURID+' but WME has loaded the details for UR ID '+uroSelectedURID+' instead, attempting to fix...'); } if(++uroURReclickAttempts < 3) { //uroRestackMarkers(); var urMarker = uroGetMarker(URO_TMARKER.UR,uroHoveredURID); if(urMarker !== null) { var urMarkerAttributes = uroGetAttributes(URO_TMARKER.UR, uroHoveredURID); if(urMarkerAttributes !== null) { urMarkerAttributes.geometry.x = urMarkerAttributes.geometry.realX; urMarkerAttributes.geometry.y = urMarkerAttributes.geometry.realY; uroOpenURDialog(uroHoveredURID); } } return; } else { uroAddLog('Woe is me, attempting to open UR ID '+uroHoveredURID+' has failed...'); uroShowAlertBox('fa-warning', 'URO+ Warning', 'WME may have opened the details panel for a different UR to the one you selected, proceed with caution', false, "OK", "", null, null); } } uroURReclickAttempts = 0; uroFilterURs(); } } } } else if(uroURDialogIsOpen === true) { // dialog was open and has now been closed uroSelectedURID = null; uroClickedOnMarkerID = null; uroFilterURs(); } uroURDialogIsOpen = URDialogIsOpen; if(((uroPendingCommentDataRefresh === true) || (uroWaitingCommentDataRefresh === true)) && (uroSelectedURID !== null)) { uroAddLog('check completion of comment data refresh for UR '+uroSelectedURID+' ('+uroPendingCommentDataRefresh+','+uroWaitingCommentDataRefresh+')'); uroGetSelectedURCommentCount(); } } function uroTSTOWLHandler() { /* var selectedTotal = uroSelectedItems.length; if((selectedTotal > 0) && (document.getElementById('_uroDivOWLBtns') === null)) { var selectedClass = uroSelectedItems[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.indexOf("Feature.Vector.Segment") != -1) { selectedSegments = true; for(loop=0; loop<selectedTotal; loop++) { fid = uroSelectedItems[loop].model.attributes.id; var segIdx = uroIsSegOnWatchList(fid); if(segIdx == -1) { displayAddToOWLBtn = true; } else { if(uroSegDataChanged(segIdx)) { displayUpdateOWLBtn = true; } displayRemoveFromOWLBtn = true; } } } else if(selectedClass.indexOf("Feature.Vector.Landmark") != -1) { selectedLandmarks = true; for(loop=0; loop<selectedTotal; loop++) { fid = uroSelectedItems[loop].model.attributes.id; 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... // if(uroGetCBChecked('_cbMasterEnable') === false) { return; } if ( (document.querySelectorAll('.closures-list').length > 0) && (document.querySelector('.closures-list').getAttribute('touchedbyuro') === null) ) { var nClosures; var cLoop; var btnElm; // Cloning doesn't work with certain locales due to the way the date strings are formatted... if ( (I18n.locale == "fa-IR") || (I18n.locale == 'ar') || (I18n.locale == 'zh') || (I18n.locale == 'ko') ) { // Sorry :-( } else { // for the others, are there any closures defined for all of the selected segment(s)... if(document.getElementsByClassName('full-closures').length > 0) { nClosures = document.getElementsByClassName('full-closures')[0].querySelectorAll('.closure-item.is-editable').length; if(nClosures > 0) { // and if so, have we already added the clone icon? for(cLoop = 0; cLoop < nClosures; cLoop++) { btnElm = document.getElementsByClassName('full-closures')[0].querySelectorAll('.closure-item.is-editable')[cLoop].getElementsByClassName('closure-title')[0]; if((btnElm.innerHTML.indexOf('_uroCloneClosure-') == -1) && (uroGetRTCOrigin(uroRTCObjs[cLoop]) !== URO_TRTC.UNKNOWN)) { var newNode = document.createElement("div"); var anchorID1 = '_uroCloneClosure-1-'+cLoop; var anchorID2 = '_uroCloneClosure-7-'+cLoop; var newAnchor = '<a id="'+anchorID1+'" href="#">'; newAnchor += "<i style='font-size: 150%; cursor: copy' class='fa fa-copy'></i>"; newAnchor += "</a><sup>+1</sup> "; newAnchor += '<a id="'+anchorID2+'" href="#">'; newAnchor += "<i style='font-size: 150%; cursor: copy' class='fa fa-copy'></i>"; newAnchor += "</a><sup>+7</sup>"; newNode.innerHTML = uroModifyHTML(newAnchor); btnElm.prepend(newNode); uroAddEventListener(anchorID1,"click",uroCloneClosure,false); uroAddEventListener(anchorID2,"click",uroCloneClosure,false); } } } } } // if there's more than one closure (full or partial) listed, also add the delete all button if not already present nClosures = document.querySelectorAll('.closure-item.is-editable').length; if(nClosures > 0) { if(document.getElementById('_btnDeleteAllClosures') === null) { var daDiv = document.createElement('wz-button'); daDiv.className = 'delete-all-button btn is-expanded'; //btn-primary'; daDiv.id = '_btnDeleteAllClosures'; let tHTML = '<i class="fa fa-trash"></i> '+I18n.lookup("closures.delete_confirm_no_reason"); if(nClosures > 1) { tHTML += ' ('+I18n.lookup("closures.apply_to_all")+')'; } daDiv.innerHTML = uroModifyHTML(tHTML); daDiv.style.width = '100%'; daDiv.style.marginBottom = '10px'; var acBtn = document.getElementsByClassName('add-closure-button')[0]; if(acBtn !== undefined) { acBtn.parentNode.insertBefore(daDiv, acBtn.nextSibling); uroAddEventListener('_btnDeleteAllClosures',"click", uroDeleteAllClosures, false); } } } let deferTouching = false; if(uroGetCBChecked('_cbAutoScrollClosureList') == true) { // Scroll to the end of the closure tab, as that's where the closure you're most likely to be cloning // is located... var cItems = document.getElementsByClassName('closure-item'); if(cItems.length > 0) { let lastOne = cItems[cItems.length - 1]; if(lastOne.getBoundingClientRect().height === 0) { deferTouching = true; } else { lastOne.scrollIntoView(); } } } if(deferTouching === false) { document.getElementsByClassName('closures-list')[0].setAttribute('touchedbyuro','true'); } } } function uroMiscUITweaksHandler() { if(uroFilterPreamble()) { // give user the option of setting their own background colour... { var mapviewport = document.getElementById("WazeMap").getElementsByClassName("olMapViewport")[0]; if((uroGetCBChecked('_cbWhiteBackground') === true) && (uroGetCBChecked('_cbMasterEnable') === true)) { var customColour = '#' + uroToHex(uroGetElmValue('_inputCustomBackgroundRed'),2); customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundGreen'),2); customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundBlue'),2); mapviewport.style.setProperty('background',customColour,'important'); } else { mapviewport.style.setProperty('background',"#354148",'important'); } } // allows user to hide the area managers layer without switching off the layer completely... { // ...if this sounds like a weird option - why not just switch off the layer from the layers menu? - then // remember that in order for URO+ to be able to display in its own tab the list of AMs under the current // mouse pointer location, which is somewhat more useful than the list given in the topbar, it needs the // AM layer to be activated so that the AM areas data is loaded into WME. It doesn't however need the layer // to then be visible, and since having a bunch of purple polygons covering the map can make for a rather // difficult editing experience, being able to hide the polys whilst retaining the area information is // of real benefit... if((uroGetCBChecked('_cbHideAMLayer')) && (uroGetCBChecked('_cbMasterEnable'))) { W.map.managedAreasLayer.setOpacity(0); } else { W.map.managedAreasLayer.setOpacity(1); } } // gives user the option of hiding the somewhat unnecessary editor info panel at the top of the sidebar - disabled // in current beta due to changes in sidebar layout, code can be removed entirely once this change hits production { if(document.getElementById("user-details") !== null) { var panelDisplay = ''; if(uroGetCBChecked('_cbHideEditorInfo')) { panelDisplay = "none"; } document.getElementById("user-details").style.display = panelDisplay; } } // gives user the option of hiding the vector segments when the raster segment layer is hidden { if(uroGetCBChecked('_cbHideSegmentsWhenRoadsHidden')) { W.map.segmentLayer.drawn = W.map.roadLayer.visibility; W.map.nodeLayer.drawn = W.map.roadLayer.visibility; } else { W.map.segmentLayer.drawn = true; W.map.nodeLayer.drawn = true; } } var tDesc; // reformats the comment body text whenever the map comments sidepanel is opened, so that any linebreaks // added to the body text when the comment was created, and which are available in the W.model. data for // WME itself to include if only it could be arsed to do so, are included in the sidepanel rendering of // the text... { if(document.getElementsByClassName('map-comment-feature-editor').length > 0) { if(document.getElementsByClassName('map-comment-feature-editor')[0].touchedByURO === undefined) { if(document.getElementsByClassName('body-preview').length > 0) { var mcID = -1; if(uroFID == -1) { // if uroFID is -1, this implies the map comment panel has been opened as the result of WME // being started up with a map comment included in the URL... mcID = document.location.href.split('&mapComments=')[1]; if(mcID === undefined) { mcID = -1; } } else { mcID = uroFID; } if(mcID !== -1) { if(W.model.mapComments.objects[mcID] !== undefined) { if(W.model.mapComments.objects[mcID].attributes !== undefined) { tDesc = W.model.mapComments.objects[mcID].attributes.body; tDesc = uroClickify(tDesc, ''); document.getElementsByClassName('body-preview')[0].innerHTML = uroModifyHTML(tDesc); document.getElementsByClassName('map-comment-feature-editor')[0].touchedByURO = true; } } } } } } } // clickifies the ExtraInfo URL present in some MPs { if(document.getElementById('panel-container').getElementsByClassName('extraInfo').length > 0) { if(document.getElementById('panel-container').getElementsByClassName('extraInfo')[0].touchedByURO === undefined) { tDesc = document.getElementById('panel-container').getElementsByClassName('extraInfo')[0].innerHTML; tDesc = uroClickify(tDesc, ''); document.getElementById('panel-container').getElementsByClassName('extraInfo')[0].innerHTML = uroModifyHTML(tDesc); document.getElementById('panel-container').getElementsByClassName('extraInfo')[0].touchedByURO = true; } } } if(W.map.getZoom() != uroLastZoom) { uroLastZoom = W.map.getZoom(); uroMCLayerChanged(); } } } function uroMainTick() { if(uroMTEMode) return; if(uroSetupListeners) { if(uroFinalisingListenerSetup === false) { if(W.loginManager.isLoggedIn()) { if(document.getElementsByClassName('topbar').length === 0) return; uroFinalizeListenerSetup(); document.getElementsByClassName('topbar')[0].appendChild(uroAMList); } } } else { if(uroGetCBChecked('_cbMasterEnable') === true) { // grab a copy of the currently selected items list here, so that we only need to handle the production/beta API // differences in one place... if(W.selectionManager.selectedItems === undefined) { uroSelectedItems = W.selectionManager._selectedFeatures; } else { uroSelectedItems = W.selectionManager.selectedItems; } // do one maintick handler call in each 10ms cycle to minimise the time stuck within the maintick handler without // unduly affecting the overall response time for each individual handler if(uroMainTickStage === 0) uroTSTPopupHandler(); if(uroMainTickStage == 1) uroTSTNextBtnHandler(); if(uroMainTickStage == 2) uroTSTCommentAddedHandler(); if(uroMainTickStage == 3) uroTSTOWLHandler(); if(uroMainTickStage == 4) uroMiscUITweaksHandler(); if(++uroMainTickStage == 6) uroMainTickStage = 0; } } } function uroActiveTab(_id) { var e = document.getElementById(_id); e.style.backgroundColor = "greenyellow"; e.style.borderTop = "1px solid"; e.style.borderLeft = "1px solid"; e.style.borderRight = "1px solid"; e.style.borderBottom = "0px solid"; } function uroInactiveTab(_id) { var e = document.getElementById(_id); e.style.backgroundColor = "gainsboro"; e.style.borderTop = "0px solid"; e.style.borderLeft = "0px solid"; e.style.borderRight = "0px solid"; e.style.borderBottom = "1px solid"; } function uroInactiveAllTabs() { for(var i = 0; i < uroCtrlTabs.length; ++i) { uroInactiveTab(uroCtrlTabs[i][URO_TABS_FIELD.TABHEADER]); if(!uroCtrlsHidden) { uroSetStyleDisplay(uroCtrlTabs[i][URO_TABS_FIELD.TABBODY], 'none'); } } } function uroShowTab(tabID) { uroInactiveAllTabs(); uroActiveTab(uroCtrlTabs[tabID][URO_TABS_FIELD.TABHEADER]); if(!uroCtrlsHidden) uroSetStyleDisplay(uroCtrlTabs[tabID][URO_TABS_FIELD.TABBODY], 'block'); return false; } function uroShowURsTab() { uroShowTab(URO_TABS_ID.URS); return false; } function uroShowMPsTab() { uroShowTab(URO_TABS_ID.MPS); return false; } function uroShowMCsTab() { uroShowTab(URO_TABS_ID.MCS); return false; } function uroShowPlacesTab() { uroShowTab(URO_TABS_ID.PLACES); for(var idx=0;idx<uroPlacesGroupsCollapsed.length;idx++) { uroPlacesGroupCEHandler(idx); } return false; } function uroShowCamsTab() { uroShowTab(URO_TABS_ID.CAMS); return false; } function uroShowOWLTab() { uroShowTab(URO_TABS_ID.OWL); uroOWLUpdateHTML(); return false; } function uroShowMiscTab() { uroShowTab(URO_TABS_ID.MISC); return false; } function uroShowRTCsTab() { uroShowTab(URO_TABS_ID.RTCS); return false; } function uroShowRAsTab() { uroShowTab(URO_TABS_ID.RAS); return false; } function uroNewLookCheckDetailsRequest() { 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 { uroGetMarker(URO_TMARKER.MP,urID).element.click(); doRetry = false; } catch(err) { uroAddLog('problems not fully loaded, retrying...'); } if(doRetry) window.setTimeout(uroNewLookCheckDetailsRequest,500); } else { showmarkerpos = thisurl.indexOf('&showpur='); if((endmarkerpos != -1) && (showmarkerpos != -1)) { showmarkerpos += 9; uroAddLog('showPUR tab opened'); urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos); uroAddLog(' PUR ID = '+urID); try { uroGetMarker(URO_TMARKER.PUR, urID).element.click(); doRetry = false; } catch(err) { uroAddLog('PURs not fully loaded, retrying...'); } if(doRetry) window.setTimeout(uroNewLookCheckDetailsRequest,500); } else { showmarkerpos = thisurl.indexOf('&showppur='); if((endmarkerpos != -1) && (showmarkerpos != -1)) { showmarkerpos += 10; uroAddLog('showPPUR tab opened'); urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos); uroAddLog(' PPUR ID = '+urID); try { uroGetMarker(URO_TMARKER.PPUR, urID).element.click(); doRetry = false; } catch(err) { uroAddLog('PPURs not fully loaded, retrying...'); } if(doRetry) window.setTimeout(uroNewLookCheckDetailsRequest,500); } } } } function uroUpdateVenueEditorLists() { if(Object.keys(W.model.venues.objects).length === 0) return; // build the list of all userIDs contained in the currently loaded venue objects 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); } } } // check for any previously selected userIDs in the two selector lists, then clear both lists // and repopulate using the newly gathered ID collection from above, and finally reselect the // previously selected user if they're still present in the new list... var selector; var selectedUser; var users = W.model.users.getByIds(listedIDs); var selectorEntry; for(var i=0; i<2; i++) { if(i === 0) selector = document.getElementById('_selectPlacesUserID'); else selector = document.getElementById('_selectHidePlacesUserID'); selectedUser = null; if(selector.selectedOptions[0] != null) { selectedUser = parseInt(selector.selectedOptions[0].value); } while(selector.options.length > 0) { selector.options.remove(0); } selector.options.add(new Option('<select a user>', null)); if(listedIDs.length > 0) { selectorEntry = ''; for(idx=0; idx<users.length; idx++) { if(users[idx].attributes.userName === undefined) { selectorEntry = users[idx].attributes.id; } else { selectorEntry = users[idx].attributes.userName; } selector.options.add(new Option(selectorEntry, users[idx].id)); if(users[idx].attributes.id == selectedUser) { selectedIdx = idx+1; } } } if(selectedIdx !== null) { selector.selectedIndex = selectedIdx; } } } function uroPlacesEditorSelected() { var selector = document.getElementById('_selectPlacesUserID'); if(selector.selectedIndex > 0) { document.getElementById('_textPlacesEditor').value = document.getElementById('_selectPlacesUserID').selectedOptions[0].innerHTML; } } function uroHidePlacesEditorSelected() { var selector = document.getElementById('_selectHidePlacesUserID'); if(selector.selectedIndex > 0) { document.getElementById('_textHidePlacesEditor').value = document.getElementById('_selectHidePlacesUserID').selectedOptions[0].innerHTML; } } 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.flex = '1'; obj.style.overflowY = 'auto'; } function uroSetSectionTabStyles() { for(var i =0; i < uroCtrlTabs.length; ++i) { uroSetStyles(uroCtrlTabs[i][URO_TABS_FIELD.TABBODY]); } } function uroPlacesGroupCEHandler(groupidx) { if(uroPlacesGroupsCollapsed[groupidx] === false) { document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "block"; document.getElementById('_uroPlacesGroupState-'+groupidx).className = "fa fa-minus-square-o"; } else { document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "none"; document.getElementById('_uroPlacesGroupState-'+groupidx).className = "fa fa-plus-square-o"; } } function uroPlacesGroupCollapseExpand() { var groupidx = this.id.substr(21); if(uroPlacesGroupsCollapsed[groupidx] === true) uroPlacesGroupsCollapsed[groupidx] = false; else uroPlacesGroupsCollapsed[groupidx] = true; uroPlacesGroupCEHandler(groupidx); return false; } function uroWazeBits() { var i; uroMCLayer = null; for(i=0;i<W.map.layers.length;i++) { if(W.map.layers[i].CLASS_NAME.indexOf('Layer.Vector.RootContainer') !== -1) uroRootContainer = W.map.layers[i].div.id; if(W.map.layers[i].name == 'mapComments') uroMCLayer = W.map.layers[i]; if(W.map.layers[i].name == 'venues') uroVenueLayer = W.map.layers[i]; } uroPlacesRoot = uroVenueLayer.id + '_vroot'; } function uroGetMarkerIDs(markerType) { let idList = []; for(let i = 0; i < uroMarkerLayers[markerType].length; ++i) { let dID = uroMarkerLayers[markerType][i]?.element?.attributes['data-id']?.value; if(dID !== undefined) { idList.push(dID); } } return idList; } function uroGetAttributes(markerType, markerID) { if(markerType == URO_TMARKER.UR) return W.model.mapUpdateRequests.objects[markerID].attributes; if(markerType == URO_TMARKER.MP) return W.model.problems.objects[markerID].attributes; return null; } function uroGetMarker(markerType, markerID) { if(typeof(markerID) === 'number') { markerID = markerID.toString(); } let retval = null; let validMarkerType = false; validMarkerType = validMarkerType || (markerType === URO_TMARKER.UR); validMarkerType = validMarkerType || (markerType === URO_TMARKER.MP); validMarkerType = validMarkerType || (markerType === URO_TMARKER.PUR); validMarkerType = validMarkerType || (markerType === URO_TMARKER.PPUR); validMarkerType = validMarkerType || (markerType === URO_TMARKER.RPUR); validMarkerType = validMarkerType || (markerType === URO_TMARKER.RTC); if((validMarkerType === true) && (markerID !== null)) { for(let i = 0; i < uroMarkerLayers[markerType].length; ++i) { if(uroMarkerLayers[markerType][i]?.element?.attributes['data-id']?.value === markerID) { retval = uroMarkerLayers[markerType][i]; } } } return retval; } function uroGetMarkerLayers() { uroMarkerLayers = []; // do NOT alter the order in which the marker objects are pushed into uroMarkerLayers[], unless the corresponding // URO_TMARKER.x values are also changed to reflect the new order... // URO_TMARKER.UR uroMarkerLayers.push(W.map.getLayerByName("update_requests").markers); // URO_TMARKER.MP uroMarkerLayers.push(W.map.getLayerByName("mapProblems").markers); // URO_TMARKER.PUR uroMarkerLayers.push(W.map.getLayerByName("place_updates").markers); // URO_TMARKER.PPUR uroMarkerLayers.push(W.map.getLayerByName("PARKING_PLACE_UPDATES").markers); // URO_TMARKER.RPUR uroMarkerLayers.push(W.map.getLayerByName("RESIDENTIAL_PLACE_UPDATES").markers); // URO_TMARKER.RTC uroMarkerLayers.push(W.map.getLayerByName("closures").markers); } function uroWazeBitsAvailable() { uroAddLog('All WazeBits present and correct...'); W.prefs.on('change:isImperial', uroInitialise); if(W.map.layers === undefined) { W.map.layers = W.map.getLayers(); } if(W.map.events === undefined) { W.map.events = W.map.getMapEventsListener(); } if(W.model.problems === undefined) { W.model.problems = W.model.mapProblems; } uroGetMarkerLayers(); // To avoid creating multiple URO tabs in the event that the initialise code // gets called more than once, we now test for the presence of our tab and skip // if it's already present if(document.getElementById('uroTabHeader') === null) { uroSetupUI(); } } function uroWaitForW() { if(document.getElementsByClassName("sandbox").length > 0) { uroAddLog('WME practice mode detected, script is disabled...'); return; } if(document.location.href.indexOf('user') !== -1) { uroAddLog('User profile page detected, script is disabled...'); return; } if(window.W === undefined) { window.setTimeout(uroWaitForW, 100); return; } if (W.userscripts?.state?.isReady) { uroWazeBitsAvailable(); } else { document.addEventListener("wme-ready", uroWazeBitsAvailable, {once: true}); } } function uroAddInterceptor() { uroAddLog('Adding interceptor functions...'); // add interceptor function for window.confirm(), to inhibit the closure deletion confirmation that would // pop up for each individual closure when we're using the delete all button - the user has already // confirmed the delete action using our own requester var _confirm = window.confirm; window.confirm = function(msg) { var cm_delete_confirm = I18n.lookup("closures.delete_confirm").split('"')[0].trimRight(1); if(msg.indexOf(cm_delete_confirm) != -1) { uroAddLog('intercepted closure delete confirmation...'); if(uroConfirmClosureDelete) { return _confirm(msg); } else { return true; } } else if(typeof(msg) == 'undefined') { uroAddLog('Intercepted blank confirmation...'); return true; } else { return _confirm(msg); } }; var _setCenter = W.map.setProjectedCenter; W.map.setProjectedCenter = function(e,t,n,s) { if(uroInhibitSetCenter === true) { console.debug('Inhibited setCenter'); return true; } else { return _setCenter.call(W.map, e,t,n,s); } } uroConfirmIntercepted = true; } function uroEnterPopup() { uroMouseInPopup = true; } function uroExitPopup() { uroMouseInPopup = false; } function uroToggleDebug() { uroShowDebugOutput = !uroShowDebugOutput; var dbgMode = "none"; if(uroShowDebugOutput) { dbgMode = "inline"; } document.getElementById('_uroDebugMode').style.display = dbgMode; } function uroInitialise() { uroInitialised = false; uroSetupListeners = true; uroFinalisingListenerSetup = false; if(document.URL.indexOf('beta') != -1) uroBetaEditor = true; uroWaitForW(); } function uroWaitForControlsContainer() { if(document.getElementById('uroControlsContainer') === null) { window.setTimeout(uroWaitForControlsContainer,500); } else { var updateURL = 'https://greasyfork.org/scripts/1952-uroverview-plus-uro'; uroAddLog('adding controls to sidebar container...'); var tabbyHTML = '<div id="uroTabHeader"><b><a href="'+updateURL+'" target="_blank">UROverview Plus</a></b> <label id="_uroVersion">'+uroVersion+'</label>'; tabbyHTML += '<label id="_uroDebugMode">(dbg)</label>'; tabbyHTML += ' <input type="checkbox" id="_cbMasterEnable" checked>Enabled</input>'; var i; var tabsTotal = uroCtrlTabs.length; var tabsPerRow = Math.ceil(tabsTotal / Math.ceil(tabsTotal / URO_MAX_TABS_PER_ROW)); var tabCount = 0; for(i = 0; i < tabsTotal; ++i) { if(tabCount == 0) { tabbyHTML += '<table border=0 width="100%"><tr>'; } tabbyHTML += '<td valign="center" align="center" id="'+uroCtrlTabs[i][URO_TABS_FIELD.TABHEADER]+'">'; tabbyHTML += '<a href="#" id="'+uroCtrlTabs[i][URO_TABS_FIELD.LINKID]+'" style="text-decoration:none;font-size:12px">'+uroCtrlTabs[i][URO_TABS_FIELD.TABTITLE]+'</a></td>'; if(((++tabCount == tabsPerRow) && (i < (tabsTotal - 1))) || (i == (tabsTotal - 1))) { tabbyHTML += '</tr></table>'; tabCount = 0; } } tabbyHTML += '</div>'; document.getElementById('uroControlsContainer').innerHTML = uroModifyHTML(tabbyHTML); // tab elements for(i = 0; i < uroCtrlTabs.length; ++i) { uroCtrlTabs[i][URO_TABS_FIELD.TABBODY] = document.createElement('div'); } // other sidebar elements uroAMList = document.createElement('div'); uroAMList.style.color = "#ffff00"; uroCtrlHides = document.createElement('div'); // Object watchlist tab { uroCWLGroups = []; uroOWLUpdateHTML(); } // footer for tabs container uroCtrlHides.id = 'uroCtrlHides'; let tHTML = '<input type="button" id="_btnUndoLastHide" value="Undo last hide" /> '; tHTML += '<input type="button" id="_btnClearSessionHides" value="Undo all hides" /><p>'; uroCtrlHides.innerHTML = uroModifyHTML(tHTML); // footer for AM list uroAMList.id = 'uroAMList'; uroCtrlTabs[0][URO_TABS_FIELD.SHOWFN](); window.addEventListener("beforeunload", uroSaveSettings, false); } } async function uroAddTab_API() { let {tabLabel, tabPane} = W.userscripts.registerSidebarTab("URO+"); tabLabel.innerText = "URO+"; tabPane.innerHTML = uroModifyHTML(uroControls.innerHTML); await W.userscripts.waitForElementConnected(tabPane); uroCompleteUISetup(); } function uroCompleteUISetup() { uroAddLog('waiting for controls container...'); uroWaitForControlsContainer(); uroDOMHasTurnProblems = (W.model.turnProblems != null); uroGetProblemTypes(); var i; for(i = 0; i < uroCtrlTabs.length; ++i) { if(uroCtrlTabs[i][URO_TABS_FIELD.POPULATEFN] != null) { uroCtrlTabs[i][URO_TABS_FIELD.TABBODY].innerHTML = uroCtrlTabs[i][URO_TABS_FIELD.POPULATEFN](); } document.getElementById('uroControlsContainer').appendChild(uroCtrlTabs[i][URO_TABS_FIELD.TABBODY]); uroCtrlTabs[i][URO_TABS_FIELD.TABBODY].onclick = uroCtrlTabs[i][URO_TABS_FIELD.CLICKFN]; } document.getElementById('uroControlsContainer').appendChild(uroCtrlHides); uroWazeBits(); uroDiv.addEventListener("mouseover", uroEnterPopup, false); uroDiv.addEventListener("mouseout", uroExitPopup, false); if(sessionStorage.UROverview_FID_IgnoreList === undefined) sessionStorage.UROverview_FID_IgnoreList = ''; if(sessionStorage.UROverview_FID_WatchList === undefined) sessionStorage.UROverview_FID_WatchList = ''; if(uroConfirmIntercepted === false) uroAddInterceptor(); uroAddStyle('urostyle_UnstackedMarkers', '.map-marker.marker-selected { transform: scale(1) !important; }'); uroMainTickHandlerID = window.setInterval(uroMainTick,1000); } function uroSetupUI() { // create a new div to display the UR details floaty-box uroDiv = document.createElement('div'); uroDiv.id = "uroDiv"; uroDiv.style.position = 'absolute'; uroDiv.style.visibility = 'hidden'; uroDiv.style.top = '0'; uroDiv.style.left = '0'; uroDiv.style.zIndex = 10000; uroDiv.style.backgroundColor = 'aliceblue'; uroDiv.style.borderWidth = '3px'; uroDiv.style.borderStyle = 'solid'; uroDiv.style.borderRadius = '10px'; uroDiv.style.boxShadow = '5px 5px 10px Silver'; uroDiv.style.padding = '4px'; document.body.appendChild(uroDiv); // create a new div to display script alerts uroAlerts = document.createElement('div'); uroAlerts.id = "uroAlerts"; uroAlerts.style.position = 'fixed'; uroAlerts.style.visibility = 'hidden'; uroAlerts.style.top = '50%'; uroAlerts.style.left = '50%'; uroAlerts.style.zIndex = 10000; uroAlerts.style.backgroundColor = 'aliceblue'; uroAlerts.style.borderWidth = '3px'; uroAlerts.style.borderStyle = 'solid'; uroAlerts.style.borderRadius = '10px'; uroAlerts.style.boxShadow = '5px 5px 10px Silver'; uroAlerts.style.padding = '4px'; uroAlerts.style.webkitTransform = "translate(-50%, -50%)"; uroAlerts.style.transform = "translate(-50%, -50%)"; var alertsHTML = '<div id="header" style="padding: 4px; background-color:LightGreen; font-weight: bold;">Alert title goes here...</div>'; alertsHTML += '<div id="content" style="padding: 4px; background-color:White; overflow:auto;max-height:500px">Alert content goes here...</div>'; alertsHTML += '<div id="controls" align="center" style="padding: 4px;">'; alertsHTML += '<span id="uroAlertTickBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px 10px 2px 10px;">'; alertsHTML += '<i class="fa fa-check"> </i>'; alertsHTML += '<span id="uroAlertTickBtnCaption" style="font-weight: bold;"></span>'; alertsHTML += '</span>'; alertsHTML += ' '; alertsHTML += '<span id="uroAlertCrossBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px 10px 2px 10px;">'; alertsHTML += '<i class="fa fa-times"> </i>'; alertsHTML += '<span id="uroAlertCrossBtnCaption" style="font-weight: bold;"></span>'; alertsHTML += '</span>'; alertsHTML += '</div>'; uroAlerts.innerHTML = uroModifyHTML(alertsHTML); document.body.appendChild(uroAlerts); uroControls = document.createElement('section'); uroControls.style.fontSize = '12px'; uroControls.style.height = '100%'; uroControls.id = "sidepanel-uroverview"; uroControls.className = "tab-pane"; uroControls.innerHTML = uroModifyHTML('<div id="uroControlsContainer" style="display:flex;flex-direction:column;height:80vh;"></div>'); uroAddTab_API(); } function uroAddStyle(ID, css) { let head, style; head = document.getElementsByTagName('head')[0]; if (!head) { return; } uroRemoveStyle(ID); // in case it is already there style = document.createElement('style'); style.type = 'text/css'; style.innerHTML = uroModifyHTML(css); style.id = ID; head.appendChild(style); } function uroRemoveStyle(ID) { let style = document.getElementById(ID); if (style) { style.parentNode.removeChild(style); } } uroInitialise();