UROverview Plus (URO+)

Adds filtering and pop-up infobox for UR, MP and camera markers

当前为 2017-03-09 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name UROverview Plus (URO+)
  3. // @namespace http://greasemonkey.chizzum.com
  4. // @description Adds filtering and pop-up infobox for UR, MP and camera markers
  5. // @include https://www.waze.com/*/editor/*
  6. // @include https://www.waze.com/editor/*
  7. // @include https://editor-beta.waze.com/*
  8. // @include https://beta.waze.com/*
  9. // @exclude https://www.waze.com/user/*editor/*
  10. // @exclude https://www.waze.com/*/user/*editor/*
  11. // @grant none
  12. // @version 3.98
  13. // ==/UserScript==
  14.  
  15. /*
  16.  
  17. TO-DO ITEMS
  18. =======================================================================================================================
  19. Bug fixes - MUST BE CLEARED BEFORE RELEASE
  20. =======================================================================================================================
  21.  
  22.  
  23.  
  24. =======================================================================================================================
  25. Things to be checked
  26. =======================================================================================================================
  27.  
  28.  
  29.  
  30. =======================================================================================================================
  31. Proposed functionality
  32. =======================================================================================================================
  33.  
  34. Adjust default settings for places & MP tabs to account for dynamic nature of tab building
  35.  
  36. Restore custom icon functionality
  37.  
  38. Convert camera XHR code to async operation
  39.  
  40. "Hide All" button for feed - automatically clicks the trashcan on all visible feed entries
  41.  
  42. Allow custom alert box to be used as a confirm() replacement as well
  43.  
  44. Improve reliability of yellow/green comment marker choice
  45.  
  46. Implement some sort of UR "hotspot" marking to highlight areas of the map with significant clustering of URs
  47.  
  48. Flush settings to localStorage whenever a change is made, or at least before opening a new tab via a popup
  49.  
  50. User-defined setting presets
  51.  
  52. Extend unstacking to cameras
  53.  
  54. Place filtering
  55. - by last user to edit
  56.  
  57. More localisation
  58.  
  59. First-run information
  60. - show quickstart guide to URO features if no existing settings are present (i.e. new installation)
  61.  
  62. =======================================================================================================================
  63. New functionality in progress
  64. =======================================================================================================================
  65.  
  66. Addition of segment and place watchlist functionality
  67. */
  68.  
  69. /* JSHint Directives */
  70. /* globals $: */
  71. /* globals W: true */
  72. /* globals I18n: */
  73. /* globals OL: true */
  74. /* globals OpenLayers: true */
  75. /* globals Waze: true */
  76. /* globals require: */
  77. /* jshint bitwise: false */
  78. /* jshint eqnull: true */
  79.  
  80.  
  81. var uroVersion = "3.98";
  82. var uroReleaseDate = "20170309";
  83.  
  84. // list of changes affecting all users
  85. var uroChanges =
  86. [
  87. "Fix for missing segment popup in Firefox when certain segment address details were missing",
  88. "Fix for console errors when something other than a UR editing panel is opened",
  89. "Popups auto-hide when the map comments dropdown menu is accessed",
  90. "Inhibits speed limit validation on private and parking segments",
  91. "MP filter list now auto-generated based on whatever MP types are present in WME",
  92. "Popups now enabled for map comments"
  93. ];
  94. // list of changes affecting only WME Beta users (at least until the next production release including these parts of the beta code...)
  95. var uroBetaChanges =
  96. [
  97. "Compatibility fixes"
  98. ];
  99.  
  100. // true enables debug output during script startup
  101. var uroShowDebugOutput = true;
  102. // true keeps debug output enabled after script startup
  103. var uroPersistentDebugOutput = false;
  104. /*
  105. var uroRecentDebug = [];
  106. */
  107. var uroCtrlsHidden = false;
  108. var uroCurrentTab = 1;
  109. var uroFID = -1;
  110. var uroShownFID = -1;
  111. var uroShownPopupType = null;
  112. var uroInhibitSave = true;
  113. var uroPopupTimer = -2;
  114. var uroPopupDwellTimer = -1;
  115. var uroPopupShown = false;
  116. var uroPopupSuppressed = false;
  117. var uroSetupListeners = true;
  118. var uroRootContainer = null;
  119. var uroPlacesRoot = null;
  120. var uroConfirmIntercepted = false;
  121. var uroCustomMarkerList = [];
  122. var uroPendingURSessionIDs = [];
  123. var uroRequestedURSessionIDs = [];
  124. var uroPlacesGroupsCollapsed = [];
  125. var uroKnownProblemTypeIDs = [];
  126. var uroKnownProblemTypeNames = [];
  127.  
  128. var uroMouseInPopup = false;
  129. var uroURControlsIdx = null;
  130. var uroProblemControlsIdx = null;
  131. var uroTurnsLayerIdx = null;
  132. var uroMCLayerIdx = null;
  133.  
  134. var uroNullCamLayer = false;
  135. var uroNullOpenLayers = false;
  136. var uroNullURLayer = false;
  137. var uroNullProblemLayer = false;
  138. var uroNullMapViewport = false;
  139.  
  140. var uroURDialogIsOpen = false;
  141. var uroHoveredURID = null;
  142. var uroSelectedURID = null;
  143. var uroURReclickAttempts = 0;
  144. var uroPendingCommentDataRefresh = false;
  145. var uroWaitingCommentDataRefresh = false;
  146. var uroExpectedCommentCount = null;
  147. var uroCachedLastCommentID = null;
  148.  
  149. var uroMCSelected = false;
  150. var uroPlaceSelected = false;
  151. var uroAutoCentreDisabledOn = [];
  152. var uroMouseIsDown = false;
  153. var uroBackfilling = false;
  154. var uroHidePopupOnPanelOpen = false;
  155.  
  156. var uroUserID = -1;
  157. var uroURIDInURL = null;
  158.  
  159. var uroDOMHasTurnProblems = false;
  160. var uroBetaEditor = false;
  161. var uroWazeBitsPresent = 0;
  162. var uroMTEMode = false;
  163. var uroFinalisingListenerSetup = false;
  164. var uroInitialised = false;
  165.  
  166. var uroOWL = null;
  167. var uroDiv = null;
  168. var uroAlerts = null;
  169. var uroControls = null;
  170. var uroCtrlURs = null;
  171. var uroCtrlMPs = null;
  172. var uroCtrlPlaces = null;
  173. var uroCtrlCameras = null;
  174. var uroCtrlMisc = null;
  175. var uroCtrlHides = null;
  176. var uroAMList = [];
  177.  
  178. var uroCWLGroups = [];
  179. var uroCamWatchObjects = [];
  180. var uroSegWatchObjects = [];
  181. var uroPlaceWatchObjects = [];
  182.  
  183. var uroFriendlyAreaNames = [];
  184. var uroAreaNameHoverTime = -1;
  185. var uroAreaNameHoverObj = null;
  186. var uroAreaNameOverlayShown = false;
  187. var uroANEditHovered = false;
  188. var uroANEditBox = null;
  189.  
  190. var uroPrevMouseX = -1;
  191. var uroPrevMouseY = -1;
  192.  
  193. var dteControlsIdx = -1;
  194. var dteOldestFullDrive = new Date(0);
  195. var dteEpoch = new Date(0);
  196. var dteTopID = '';
  197. var dteClearHighlightsOnPanelClose = false;
  198. var dteArmClearHighlightsOnPanelClose = false;
  199. var dteOffset = 0;
  200.  
  201. var uroUserTabId = '';
  202. var uroShowFeedFilter = false;
  203.  
  204. var uroTBRObj = null;
  205.  
  206. var uroBackfillQueue = [];
  207.  
  208. var uroUnstackedMasterID = null;
  209. var uroStackList = [];
  210. var uroStackType = null;
  211.  
  212. var uroMousedOverMarkerID = null;
  213. var uroMousedOverMarkerType = null;
  214. var uroClickedOnMarkerID = null;
  215. var uroClickedOnMarkerType = null;
  216.  
  217.  
  218. var uroAlertBoxStack = [];
  219. var uroAlertBoxTickAction = null;
  220. var uroAlertBoxCrossAction = null;
  221. var uroAlertBoxInUse = false;
  222.  
  223. var uroNativeMarkerImage = 'problems-sc703e224cf.png';
  224.  
  225. var uroCustomURTags = ['[ROADWORKS]','[CONSTRUCTION]','[CLOSURE]','[EVENT]','[NOTE]','[WSLM]','[BOG]','[DIFFICULT]'];
  226. var uroAltMarkers =
  227. [
  228. // 0 = closure
  229. "url('data:image/png;base64,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')",
  230. // 1 = cone
  231. "url('data:image/png;base64,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')",
  232. // 2 = custom text
  233. "url('data:image/png;base64,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')",
  234. // 3 = editor note
  235. "url('data:image/png;base64,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')",
  236. // 4 = event
  237. "url('data:image/png;base64,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')",
  238. // 5 = wslm & native SL URs
  239. "url('data:image/png;base64,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')",
  240. // 6 = Elgin
  241. "url('data:image/png;base64,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')",
  242. // 7 = TrafficCast
  243. "url('data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACEAAAKrCAYAAAB/UOT4AAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH4AQYFhMQ3k7mpgAAAB1pVFh0Q29tbWVudAAAAAAAQ3JlYXRlZCB3aXRoIEdJTVBkLmUHAAAABmJLR0QAAACwAAA8HUVvAAAmIUlEQVR42u1cB1SUR7umK11FxWUBKctSlqXsUqT3oqIiCLaA2DW22GIsUaPGRKPGbmI0MYqxJpoYjbF3UFQI2DUW1AiCFKUv7NxnPj5gVUzQm//ee86dxzMHdtlv5pm3z7tzVFNjYGBgYGBgYGBgYGBgaBYaGJrBwcFacrlcu2HQ1/R9/u//OtTpxHQRDL3+/fsbDxgwwLRnz55mPXr0EEZERJjHxcUJY2Njzfr06dNh4MCBxuHh4Xo8KfV/hUC3bt20fX19jZKTkwWTJk2SzJo1K3j27Nnx06dPT/7www8H473B06ZNS8HrAXg/FiMIryWjR48WiEQiPUhJ853JYCeakZGReikpKYIPPvjAbc6cOb2/+eabuceOHTt1/Pjxgl9//bXmp59+Uu7evZv88ssvysOHD1enpaU9PXPm5JmNGzfMmjJlSvSoUaMce/fubeLj46MzfPjwt1MT2GvFxMQYg4Bo6tSpUdu3b1904sSJ7HPnzhZUVlZWKZVKQqGoqyYZj7eRLVlDyVeXepPV57uSZemR5PKjXZUXMy+c37Zt2zRIKxiqMofqDDCvRksJaPbq1avNuHHjnGfMmNH/999/P4KdF5eUFJdVVFTU1igqiaK2mtwoOELWnO9Glpz1J8vOBZIvzwWT5WnhZGVaFMl8spe8qCgqxTO3zp07lwopxg4ePFgMm2kzYsQIzX9SgQZswBBidF64cOEwLH7l4MGDlWVlZTUKRQ2pxeIlVY/I/ptzycIzMrLonC9ZkR5KdlwZT07eX0tuPjtCnpbdJOU1haRGWUPwTFlxcfF9zHP0k08+SR45cqQLDNcI67zRRtRNTU11YenWkydPTjx16lTmoUOHKsrLy5V1dXWkTllLiqsektSs4WRpmj9ZdNaPbM8ZSx6UXCQ1deWkAaUlz0lFRSWhz1BUV1dXFRYW5n3//feXYFtJgwYNsvPy8mrVLIGuXbtqYwjgBaFQwX6wp+Kv4wjgn6KuimzPHgfxB5Il6cHk0O3FHDGKBw8eEBgeMTIyIhoaGmTFihVEFZin5smTJ8WfffbZCaila2JiYnuspfGaGmALRvHx8dLVq1fPOXnyZPHz588VtbW1jRMdu/clWXjak3xxNoCcvPcV915hYQGBixJM0ThMTEwIvOglEtSQoZaq9evXlyxevHg24oxDaGjoa9LQlkgkpklJSeHwgoz09PQS2IGyYZIHpRlk6dlgEPAjB29/TmqVChAoJAEBAY2LY3dk586dBM+Sp0+fkleB+erwTOmFCxcyIbVgPGukGj/U/fz8dP39/UXz588fB2t+lJ+fXwU0ktgJw1t2LohsuJhICsrvcu/B5RoJHDlyhDS47ZtApVpSUlJz8eLFAhj9SDzfCXFIo0EV6jAUI4RhOSbbe+nSpXyooo5OSv8VVNwi6y725lzw8J1l3IRbt27lFtfV1eV23iDyvwP9+4sXL+pAohABL7Vv37623bt3125MSLCHdohsoZjwHkRWVlNTU9fw8NXCg/CGUMSAEPLk+VXuPXNzc44EQniLCDQAnlL37Nmz4itXrqTDVT2gEt3OnTurqyGkaiI3mMJ14uARkFhJXYN7UZx+uB62EEhWpUdxr2G0RFtbm8CdyR9//EHeBlCJElIuw3PXYRdRSHTtsL6GGvSiCXUIQSIZrlkKK65T3dnBOwsQE3zI1qzR3OuVK1cSTU1N4uTkRN4WdN7S0tLqy5cv3xs2bNhAeIgF1tehNqEFo7RAnhj622+/Fb1KYt/NWeSLM13I7qtTudeIfFwsgPTeiQS1Nxj/E0hieHR0tD0tD7hsicBhjvdGQRKFRUVFL6lj/+25IOFLdlwdz71GwOFIuLm5vRMJ6qoZGRnFWG8EEpsj1jeg6tAACTMkrOGw2lwYTrUqiaP3v4RN+JOvLsZxr2nqpjZhbW3dbDxoCQl4SPGYMWNS4KZirK9PDVMDWbMDjesgcQPh9TliRCOLS3m7uFD9ZXo4qVA8I8+eFREDAwOip6dHduzY8VYkaOyBusthmHeQJBOgDmusr8sFK4il7fjx43vu3bv3+Pnz559RD6mPE3XkwfMMsvJCN7IsLYxcfryLm2zIkCGci3p7exMkOKIquX+wByUiZjES4wGE7rDAwEAh5qkP3yCh9/777/t9/vnnqxCw8v7666/qqqpKPKZAhiwj32UmkyUIVtty3idVeI3ChhgaGnJEkHVbJAUkMao+BWJREeqLmchTckihLV+Lqqn169dPGyTEqILG7t+/P5uyLS4uUihJLf7VkPSH39UXL2nBJCf/QGO8oFmTEqFGit2RvLw8opr0GkAlhWipRHnwguYOZNJ4REtrENBNSEioD90dO3ZURxhtO3bs2IhFixatRb2Ym5v7oKy8vKyuDsWJUllHNl7uSxbDS9ZciCH5Zbe5yeFNRCAQcERo7KBkMDnBRl4iQSWHSFyF+FD4EQCD9JDJZG35I0ITEDxaofKxmzBhwhDUhgehlsK8/DwFQi1H4ll5LpJYMLJpAFlxPrKRCLUJFEEciYaE9uWXX9IQ3agGeJwCUijZtWvXTuy8e1hYmAWSpk6zhQ30ZDx06FBfuOukH3/8MQNkymGkStR2KGCqybWCQ2R5ehhXU9JaMifvAKlSlNYXvTUKTgKoFwiCEbc4JQgV1R04cKAcmzr73nvvJSFUOyNXGb7xkBQVFaWDykqID4fARiajPri4Z8+eFwUFBQoYKmfhV/J/IytBZDFC+dJzAagvx5LLf+0izyruN9kAJFdW/pzQ544ePfr8u+++e4oEOR2q6oKYZEpPbH9b6MJ39fGAFS3z5s6dO3Hz5s2HUREVIX3X1rsaas3Kh2Rz1jCy+HQXEAkiy8+FcuX+1xm9Uf4P5yrx8vIXtampqeWIJekzZ84cDjX4IjBZ0jBN1/nbExcN43BZWupZIU8EfPrppyPXrl27DkQeQadkw4YNjTvOzPuRbLz0HuJINOqNIM5elmAcOrSDoLAtWLVq1RJ4XCIMkbqjGQ3R1CXhov94IqNEtIKCgvQRwMxwknLH0S4GPyfMmzdvHVCKRAe3q+XEXoMCOLfkErnw6Ady4MBS8u23n1Rt2rRpC3LMBJSLkTBCR9QNHaCClp1PsTl1lUMwPXm3hghNENat4Tm+INUfbjwTbvwMIbgWeYC8KEOJX1nO2Quqsorly5cvhk31QxDrAgJWcNs2dB46HySg8TancI4ENSAa2+FObWAjlghqMhS0sfCeBciEJaqxAOFeuWzZsk04PiZi5zKEdHNKgD6PIlpHpW3wz6qAAdEPt0LlpI+JjDBJO9QOHemk+N0JhhuKhYZhx7th+RV89FTggJwGqY0CgTB83hmjs7OzsykKJhO4pTE2YqhC6I1k1PnGhx59kF9U1KVLF6pTFxQ+ciQc35CQkGhURCkQ93yk9evZ2dnkzJkzBSiU52OhIRjdMAJAwhPPuHp6ekownxjzWIFAJ8xvDDdt9aY4Qd9sRXeOh23wsBsW96M7wwRdMXFPEIvH3wbgvY9A+Ev4/fcITI9Rle3C31fg7x/j+aH4ex+Q6IXRHe9F4r0gSopuCJ8zo1LhY8Vr0tCk4qKix4fs8XAXLBiO0YMujon7QRIDMVEKXs/BZ5eC6Fr8XI+fX2HS5R4eHgvw9/GYIwnv98ezCXgulm4CzwbSjVE1YQNt3kSCkwQebIthScVI2fOiDaU7opPR3VGp4Pc4fpF++D2REqW7x+Ix9HP4GUWliBEEdfhQAoAdtROoxQBrNeuunFFSd6J6ox8GeQsXFxdrXqeOlBgdWEDK69sNu3enP+lrajvUKOln8LsTfQ5z2dLdUzVQW6MEVIzzzU0yKiorK6vWfIChHRZjKkIqJWozdNAJXx0Nf6ODuid9juqfelqDZ7xNQ60hVnDtQl5CL7UM/2k0tBR5l9doaYx4qz4mQrIWEp02Hcg12oiS//k+Jl0ErqiHjGiMXGKKXELzihBh3BzFknDq1Klm06ZN6zBjxgzacKP9zn+vjzlixAhtVMZGyKgCZFMJklMwUnQ8smQyfh+MrDoY9UIKfh+AtB+Ln0FfffWVBEWzAMap161bt3fvYyJPaKLk05s/f75gzZo1bqgneqM6mnvt2rVT169fL8DZoQYJS4kcQrKyspRXr16tvn379tObN6+fOXLk8CwQiUaic0S5aAKV6UBab6cmREItVFbGICD65ptvos6ePbsIC2ffuXOnQKFQNPYxac8qtySDpD9eT07kLsZp7RPy25+zyP3i85V37905j5wyDak/GNmXHjMNIiIiWkaEig9Zsg2KVWeItj92fAQEiisrK2jforauTsEt/rTsFjmVu4gcuTeLHOXGbHIEJI7dnUcePs8kVdXlpRUV5bdu3ryZ+u2339J2sxj20wZFzt/3MekZADZgCDE6b9++fRhEfCUnJ6cSlXMNPTvQ0q6ytpRcLzxADt2fQo4+mEVOPfyUZOXvJHeLz5CC8lukrKaAVNeVcwUPCh8cG8rvY46jsJtk2JUL1GOE+uTNfUwEE91JkyZZQ4SJ2H0mSFSAgJI7EmJU1paQS0+2kBMP55JjuXPIH1i8tOoxoWeTpvNmNVEoahu7N1R9OPjk4SxzaenSpUnwIDuUA833MQcNGqQNvQnANhSGth8kuLZRQ++KqiArbyd0P4+cfLiQ3Hl2onGh0tJS8sMPPxCkeO4wRE9nr7SJanB0KIZxn5g1a1ZXqKU9XPhl+4D1asAYjXCIkf78889z6m2gUqHaLLlbdBI6nwES88n9kvONBxscol/qY6LgoVXWa4dhqKUK54+SLVu2zIa0Hfr06fN6HxOWa4oaMPzKlSsZ8IISqoaGSUqqH5GTuQtA4BNyu+g4dyKjBFCJNy5Oq3A8Sx4+fMj9rbmmGU7lpTdu3MjEOsHY+Mt9TBiKLuKCCMFmHKz5ET5cRZtcDRNk5++CJ3xKMv7aQCpqirn3Vq9e3Ujg3r17LWohQho1t27dKti4ceNIHIo7QRr1KkFZRr/dMRozZow8MzNz7927d/OhikZbKFMUkrTHa8jp3IXkz+LT9aRQ0tHFkRnJ48ePW9zHpPMioBUiyKXCLmxhF019TNR87eA6oRDVPVhyGaTQ2PV4Wn4ThvgZOfPwc1JWXcC9ByPmSNBT+dv0MeEpdZi/GBtNnzhxogfsR9fe3l6dnj/picv0448/jsMOS2hnX3XS+yVp5Dg84tzDpY0dfalUSvr27Uvy8/PftmelxPxlsLnrSHRRUEc7FM4aavhFE+oQIjsmQ0yltLGlSuLms98REz4mf+TV96foidvW1pZrB7xL9w4kqmF395BLBmLzFjAHHWoTWvQFSAy9cOFC0askrhbuI8cQGa88/Zl7jaBDLCwsCI6E70qiDhJ/AhLDcZCyR+DSU4NeaIFiDrcZhYxY+CqJ64X7ybH7s0j20/qm2fHjx7neNiZ5JxLUOJGJi/H8CNQojjAHAzUcYjQGDhxotmDBguFQRy4Mp1qVxO2io8iOszkPoaCxAOGdIM03Gw9aSgKST4F3iCEEfTUcWjVg7R1AIunixYs3ioqKnqt2+R++uIxMORv5YiFR1FVgkiqCKpxzTySmtyKBeZXYZDls4g6yaQL9vo12+rlgBbG0hTp6QtTHwfIZgkpjnCiqfADDXAhpzCePS+u7+qiqOBfFbujELXJR3h6UkGQxjo0HkK3DcHZt6mOifNODjvy+/vrrVTDOPNpapqxpM7VWWY0YsZocujuLy6C1yhquTUilQYmsWrWqpd1c+j2YAvVJ0ZIlS2bCDuVBQUFtkfTq+5goTLUhHvHChQvHnjp1KpvqDGJT1MuilvxZdIr8dncKiMwgf724xk36559/NhJBoCMIdLRX2axUaC1CpQCbewESmSia4gFrnNZ1EW/qQ7eZmRlN5W2nT58eAWmsTU9Pz83PzyurD9/1tcGJ+4vIvtsfQC0LyIvq+sY6sm3jd2GYg4waNYrMnTuX1pyvJi/6/VcVNlcI2/sIxZMHDkuv9zFRU7ZCKLVD5Bzy66+/HkQ6LnxW9AyRVlHfna8uJL/cmkD2YRy8O508r85vXACJj/smqCGhoRygIbpRDag3FLSpsn///p3YbHekCQsYZPN9THo3AqLyhcFNQl7IoP1H2g5WcH1MBXlYmkn23hpPfr45huy//SF5WJJJauoq60WOdAPv4r6WREXGLc53cutQ5JTDGM/CGJMQnZ1xeH5zHxOFjQ78X4gPh6AMmwzmFxEhX1CDqqmp5iRyv/gC+en6aLL92iCy+/owcvLBcs5mnlflv9THrKgs4wwRYf75jh07niIWTUcN0QXqM8WB+c19TFiqBg0eMBYrEAldvHjxxN27dx9G0Vu0b9++xj7mC6iCfiO0JSuRbM1JIttyUsiOq8PJ7mtjua+r7hanQQrltT/99FM5VJOOo8NwkPCFFCyxxj/3MfEhTQQRI6jG6rPPPgtAcToSlfI61IePoB5O3E0h/RD58eoksjk7mXyXGU82XO5N1l/qSU6d2ke/kCnAmWUJaspE+rUCbQ1g7pb3ManvIsXqoxY0Q/xwh0XHYLIJ8O91CFSlNH+o9jH/ep5Dsp7sIceOrUPB+0UVit4tID9h6NChkcgNjogHHaCCd+tjgkxrZDkT6tOovHxxAO4PYjNxMGq2j4nKrAJl32LEnX7IkF1Qt1rJZLJ/p48Jd2qDiS0RWWWoRWPhygsQeF7qY8IFlcuXL9+E0iARO5fhuf9sHxMLDVu5ciXrY7I+Jutjsj4mu4/J7mOy+5jsPia7j8nuY7L7mOw+JruPye5jsvuY7D4mu4/J7mOy+5jsPia7j8n6mKyPyfqY7D4mu4/J7mOy+5jsPia7j8nuY7L7mOw+JruPye5jsvuY7D4mu4/J7mOy+5jsPia7j8n6mKyPyfqY7D4mu4/J7mOy+5jsPia7j8nuY7L7mOw+JruPye5jsvuY7D4mu4/J7mOy+5jsPibrY7I+JutjsvuY7D4mu4/J7mOy+5jsPia7j8nuY7L7mOw+JruPye5jsvuY7D4mu4/J7mOy+5jsPibrY7I+Jutj/p/tY1r5WWkJXAXaDYO+/o/3MekiGHpOvZ2MneKcTEVRIjNRpEhoFWRlLu4uFtp1tTOzj7HvIImXGNsE2ujxpP6dPqZduJ220FNoJE2UCuSj5BLPDzyDZZNk8S7jXJKl70sHS0ZIBkvHSFNcx7kOkE+Sx3pO8gzyGuslkaXIBO2s2+lBSu/ex8RONG2DbfVc+rkIZCNkbvLJ8t7yxfK5olTRKcPNhgWa6zRr1FaqKdWWo6Rbo6bU2qhVbbTD6Kloh+iM5xLPWW6j3KJdk10d7bvZm5h7mOvIkmRvpyaw1xJFiIyd+zqL3Ea7RXms8VhkvtU822i7UYFOlk6V2lUsfA3jCsYJjD0YuzF2YGxXIzontCtt99me91njM002RhYMVZlDdQaYV6OlBDTF0eI27kPdnd3GufUXfy8+ov+9frHeBb0yjcsatWo5WIiOs/yiWzF+4Mc2fpxSIwZZBqWWlyxvefzkkQopxjr3cxY7xDi0kSfLNf9JBRq2YbaGLskuzrKPZMOEqcIr2uu1K7UuadU0Lp6FcQAjlV8YO1f7BeMIRhrGJYxsDEhL74pemeCS4L79D/ZH3ae4J7skubjAcI2wzpv7mPod9HVh6dbSUdJEy22WmTobdSrULkPvV+on5Qjs4XdLJfAzxnleLdf4QV9fbnpPJ1unyuS8SZ7ZF2aXJMMkSc6JznZCd2HzfUxRmEjbJtRG4JDgEOr4veP+1t+3LlO/rF7XuAD9uZdfnO7+IE+M/u0oRgKGAYYGxgwVUhhal7VqDM8YFptNNTvh0Nehq0NPh/ZYT+M1NXRy6WRk191O6r7AfY7+Fv1izQxNBSf+hskO8yrYyv9O3zuHkaL2Uh9TrQ3GvJdJULI6F3SqWs9rXeIy02W2Y5yjg7W/9et9zPb27U3t+9iHW/1glaG/Q78EulU2TnKBX3yrigQoAbnK4tEYX/JSOvMKCQz1S+p1BmkGpVZ7rDIlAyXBFt4WL/cxLbwsdIVeQpHrVNdx7Xa0e6RzFm6YpULiF94Id/H6pu+FqBD4VkU1bxqQqtZ5rRqjH40K3D9yHwmX7YQ4pNGgCnUzdzMjm0gbufNW571tfmqTr54BW2iY9DK/OCXxO//eYn7x1vzOr7WABP6uflG9DiQKZdtkqQ69HGztIuya+piwh3Y2XW1CO+7qeE8vTa9MPRskrqnovcH/M/j3THkSo1pIoEElf6jXGZ43LLY7aJfuEOfgYdnFUtfY3FhdDSFVE7nBVJwojtP7Xq9E/byKR9BxXMUj6OvNGFoYJry7XnuLkaOm1LuoV2a+z/y600CnKOtA63ZYX0MNetEUuAuEogRRsu4G3VKOhOrOfudJNCw4E0MTw/YtCfAS076gXd1xb8d79gPsB3b272yB9XWoTWiZe5lbiBJFQ/W/0S96jURDdNzHvx7LxwL3dyNB7a3t9rZPxAPFw61DrO1pecBlSwQpc7sBdqMMNxoWqqe/oo6Dr5CYyJNweEcScFX9XfrFWG+EKFrkaOlraUDVoQESZo6DHYebbDbJ1U7Trn6JxBFeHbv418t5mzBvPh60hITBboNixxTHFNtIWzHW16eGqWEXbdeBxnXBFsENvdN6z9WzVLzjtIph5vBJSo93z2VvSQKxR/eCbrlgn+COY5JjAtTRdB9T3FXcVjpU2tP2a9vjhrsMn71kF9Qtd/IueoJ/L453UZeXk9U/SQFzKakqHDc7HhD3FodZdLFo6mMipehJU6R+Dh85rDLeZJynfRIqaYiYdIGf+GD1M/86E0OfJxLRQimArOYZTYXBToMiyUTJTFF3kVzoIWy6j+nc21nbLcVNLBktGSvcIMymbDXPI4G9Giu2qdjBZj5rqvFGuoFXXU4zBCjxi2rK1qmtX9DcYd/XPt423NYaTtF0H1O/vb66Uy+nttIh0giH6Q5rTTaZ5Ooe161P5Vd5Uf7Ie8lOvnChk3+D0YEnosmTCcL46hUSkJxOmk6V0R6jQscxjh9ZR1p7dJJ2er2PKXtP1so1ydXOaajTENEK0UHDTYaFVHxqf/AkLvMq2cqXdZdUcstgnkRDQvsI4w8VNaRpKvRS9Uqc1jrttI2x7d45oPOb72OiBjSW9Jf4igeLJ1msscjQWapTjkpJyYn4Kl9XblepJ1XFn81LYAr/98v1Q/20el2r9a3KbbbYnLWNs02yDLB0Rq76+/uYKL+E9vH2IXaD7CabrzK/2Gp1qxeaZyGRLJ7IGZ5IQ4FL0/xJlRR/rUly9LnWm1o/11+o/9Q82ny6dbh1F8Sklt/HpGWedJJ0os1Sm8O683SL1Ber1zYukMnnklSVSnsHby976iVGK3OdL3TKLVZapLuMdxlu18PO1yrY6u3vY3pP9Q5wn+Y+UrJAss5ovtEjLlrOV9nxaf68sVPFXrbWe4r+5/oF0nnSJa7vuybSWsXMw8yMhui3vo8pH4ZnR3u443gX4zTSaYLdZLt1rea2KlVbz7vdVf7ned6Nv0b19KlmldMSpy1uH7lNQLkYaelv6Wjubd4B55l3v49pE2xjgkRjLU2S+kqGSvqLUkQzdafoPtPIwkHoEu8lmfWEWv3YqkI6V7pYFCfqJ+om6gIvsDKVmP579zElCRJL+1h7mW0P21jbFNsFCGgljSQucTahFH8s3mQZbpko9BbKBDKBOSXwr/cxBR4CJ8sQy1DzcPNhtrNtd+tswuEIBDQ3ayqsNlilmQWZjRJ4C8LMPM3+5/qYpqtMr9MDkeEPhv+7fUxKgvUxWR+T9TFV+5ie8Z5a0iipNh3Okc7aHr09/vN9TLqod4K3XtSEKOOoiVGmISNDzMJGhQkDUgLMw8aECUNGhZiFvh/aIXxcuLF3oreerJfs3+tj+r7nqy2LlRl1m9pN0OezPpIBqwYE91/bPz5xVWJywvKEwfFL4wf3WdEnJXFF4oB+q/vF4mdQwqIEScyMGIGtl62eQ4jDu/cxsRNN3wG+et0/7C5I+CzBbcC6Ab2TdyfP7Xmy5ymXEy4FHQ51qFH7DVXXfu6UphQcFVS7nHV5GnM65sx7u96bFb8oPrrHrB6OQcOCTKTRUp3QcaFvpyanMCctv0F+xiAgSliSEJX0a9Ki8NPh2bI0WYF1rnWV2mMs/ASD/rzJV1UZfErH4cjymnll9wvdzw/6edC03ot6B/sP9Tf3fs/bQBwk1mgpAc3AIYFtYufFOvdd0bd/7PHYI5KTkmLJXUlZh9wOtWp/YSE67vHZM0NlXODHLTUizZWWRtyNuDXw7MDUuOVxsRGTIsR+KX5tQsaG/H0f062Hm4Zvkq9hj497OPdf339Y6MnQKxZHLSotHljUNC5Od3+dO0fUS4CvJejCavcxHvKfwXB65FQWdDfofsyxmKOxK2KTI6dGugQMDzBy7ur85j5mJ3En3ZDRIdaxX8QmRpyKyLQ6ZlWhlgu908XzeBI5/MINi9NF8zGe8iMX4xH/WQyrXKsqz9ueeb7bfS9FfxKdBEnYwVib72N69fXShicIoj+MDo09GrtfdFJUZpxrXMdNls+TaChy6ZnitsrCd/ly3x9DwB8X85uIC+4LahyvORZ7r/I+ETIxpKvvEN/2br3dXrYPl+4uGpIwiVHw6GBp4pbEOQ4nHIpN75kqGgk85cV9me/q3sEo4He95ZU+ZgTf03iqQgSqsbxjWWW52bKk+5ruswNHBTq4xri+3se0D7I3hc7CI05EZEjSJCVqD6CGPH6ih7wEsngJFPAin6my+HK++qaHoAevkIAnGdwzqBPfEJcGnArIDJ8SHgy7eLmP6dLNRVceJxfFrYkb53HG45HVLasqLKxslMI1XgVX+EkLMRapELjMEyvkfz5VGQ1EsBHT26Y19qftC2JWxIz0GuDVybWHa71KqKVKIiRGKMzkvY722uue7p5PWXNxIJ+39BwVOyis/yqBW9yLJ1aoQuJVIioq0b+rXwcShT1+75HqN9TPVhYva+pjOoY5tgsYGRDqecrznsNthzLjRzDIPH6SBzyBP3iLp4sk8iR2NUOgOWnwBmqQa1Bnf9O+ODg9ON1/lL+HU6STbgdRB3U1x1BHTRimadiUsDibwzYlBncghcf8g3SCP3lbyOYXyOJbALF8vChsZryBBFWxzW2bMq8zXteDJgZFufRwaSfyF2moufdy13SKchIGTw5OtjpoVfoaiVv8wtf4BQ7yPcz330Dg70jAmE1vmla7nHS55/+B/0Dn7s4WkmhUWbAJLWk3qUXQpKChNvttivTv6DeRoPq8wXvGDX4Beva04D2jJSTyXybR+lbrOvFh8ROfCT7Dpb2k9g5hDnqUhKZHood54NTAUeLfxIX6t0Di0SskLvM/C/gvXiiJ8S2UQjMkzI+YF3uP9x7hlujmKAoVGahJY6Qa8j5ys9CZocNdfnfJFdwQVHMkGryjIUjl8AvQ/oQ9RhIfP14l0ByJJzyJ3CYSvpN8U6RxUrFjpKO+mihQpOExwKMDSCTJD8pv2FyxeW7wwKCuMV/cVYmUDYHLB8ODD07/REBFCphLaXbdrNz5pPMdn3E+CU7dnaxx+q/vY8r7ytuGTAvp6f+j/3HRCdEz/du8XTzhQ3NDwrrLT76Od9HJ/AL5ryz+SrRskAKkqqRS8N/rf0A+SB4mChY19TFlCTK9gA8C/AKXBq6y22+XZ5pjWo20rGxM3Vl87XBVRbw+PJFFLSDwqN7VjbKNFMLfhUV+n/jNdOnrIrf2tW66j+kx0EM7cHygOHBW4FjZHlm21Qmr4ra32ioaU/INvmi5yBczT/nf/VS+fEnjJfWkGQIPuApM2fFAxxfuh90zPYd7xkt6SqxFQaKm+5jGnYzVQaRtwKSAiIClAWsd9znmWmRZlOn/yXvKY14S6bx9NNQQtJvXlSci5FuJ0/muTQOBh/VZt31O+yqLIxaFXWZ2+UgSJ/GwkFu83sf0Hebbym+0n53fVL8hPpt9Dnbe17mwTU4bBbfzR3zVlMaPDH53fGLibKSTSkLbrFJlId8Y5hgqTPeZlvil+u107efa3S7CzgIG2Xwf062Pm7H3SG9fr8lek9x+cMton9q+HKFZyU2Yyxe1Z/lxjnffxyqGdwhjI99S/LOeQKusVnXtdrYrh5rPuqa4JokjxM7mcvM39zF79Oih4zHIQygfLA/x/MBzskuqy0WTbSYvDLMNFdyk91W6dif5nxm8wd7ld/6A/xwI62fpK0z2mDzvuLrjU/tE++mOPR27iKPF/3wfs2fPnvr0GiSIhPrP9Z8o3yA/3HFtx6JW37Wq5VRzn6+szvBfyBzjv6Y+xtvCifqM2/p661rjb4zLJd9K0n1n+A6XJkh9RZGilt/HTExMNBo4cKBVyJyQgMD5gSN9V/quE64WPlL7nq8l7/JeksUToF9b/4ZBD0K/1vc1269pX+CzzGeJzxSfRIeeDnLLLpZmVv5Wb3cfMywsTN/nfR8z/4n+7gGzAmJgJxPkn8jXtVnWppTra9/ii5xbfK2RXt9I1V3RusprrdcW33m+E1zfc41EbnC08rPqIHQXvvt9THG42MSpp5M1fNtXPlbe3+V9l5nGc4yfaV/VbrrQcaXeBgz2GVR4fu65WDJA0s8p1qmLbYitlUAqaEPn+W/1Men/a0TvY7r2d7WU9JHI7OPtYx1HOC4w3mP8ch/zRzWl8zznTTbRNomd/TrLhB5Cc0rgX+1j0kueQi+hk1WYVahlpOUwu7l2u1tvbl3fx0zVVNh8a5NmHmI+SugjDDPzZn1M1sdkfcz/531MBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgYGBgaGJvwXGTqQx1I7tAQAAAAASUVORK5CYII=')",
  244. // 8 = TrafficMaster
  245. "url('data:image/png;base64,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')",
  246. // 9 = CalTrans
  247. "url('data:image/png;base64,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')",
  248. // 10 = TfL
  249. "url('data:image/png;base64,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')",
  250. // 11 = boots on ground
  251. "url('data:image/png;base64,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')",
  252. // 12 = difficult turn
  253. "url('data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACEAAAKrCAYAAAB/UOT4AAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH4AgVAAgkef0zlwAAAB1pVFh0Q29tbWVudAAAAAAAQ3JlYXRlZCB3aXRoIEdJTVBkLmUHAAAABmJLR0QAAAAAAAD5Q7t/AAAhYUlEQVR42u1cC1SU1fYfFAExAYnXDCAEogjMMMMMDG98EC3ykTdzEsVEMLyKwkCimVYr7SKaENdSs7y2WvrP1VXTbqm9rHvNtJXdrmbZWt3U7HGT8qZp+cbvv39nzuCA8w2i3fVf63/3nrXXzPd95+zf/vbeZ59zNmeh0TAxMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTExMTH9B6gHcc8hQ4Z4m83mXk7GNe7L5785eUEwQGJjY/3o+5bc3Nx+BBySnZ0dlpKSEo5vXOM+nst23lIhr5sGx1uGh4f3ycrKCiaOpOv49PT0ZIvFYsrJybHQvQx84xr38Rzt0B790P9GlfHCm5Cg3hBGb9if3jbFbrcX79u3r7m1tXXXr7/++l1bW9sFhQjfuMZ9PEc7tEc/qXxvaZnrVsQL2tPb9aVvbWZm5uBFixaN+frrr3cQ2CXFhS5evKhkZmYpw4cXiuuzF35RpFKX0H7RwoVj0B9ypLxe16OIUMBqtQbk5+dHk4C0PXv2NF++fPlc25U2pTPNm/eQ0ru3hthLWbJkaYdnbW2XFfTb8+67zZADeZDblSLCBdAYZiwqKrIePXp0m6JCBw8eVAICtEpNjUZwaGiscuTIYbdtj2zYsK1o+HAr5EK+J9f0lAEYiUAjBbarKXDp0iWlrGyqotVqlOPHHRwZ2UOpqqpxNLhy5VpFdu7cnmk0WqxmcyRw5HDuaIXi4mJfDLW0tDTD+++/v1zxQG+++abSs2eQ8qc/aejKwWvWaBR//whl9+7dqv3WLVq03JCUZAAO8DpYQyaeQLLCgPnz55eQL8+rCfrpp5+UkSPHKjk5GuXECY2ybp1GefZZx+/sbI0yYcIU5fTp06qKzJ09uwQ4wJNuac8HvsnJyRF00/LVV1/tVBNwhcz8l7+8qlB7Zds2jbJ4sUb8Bs+f77jn5aVR3nnnXVUlIB84wAOu0xo95fCJr6urG+fJDSdPnlQSE5OVUaMcLhg50gEKHjHCce/OOzVKVlaucubMGbcyaPheBg7wgCtiAybJyMi4NS8vD7Gw2pMSDQ0NFAsa5a23HIDV1RpxDZ4+3XHvzTcd188995yqHOAAD7hOl/gimdCNrG+++eZDtY5ffvklBV5v5b77rgZjU5NG6dVLo3h7a5THH796v7RUowQH91Mog14j55dzpxXgAA+4wiVIqRStMQUFBUPJhD+oKVFYeLvSp49G+f77q2CbNmkUX1+N4uOjUdauvXr/u+80IomNHz/+mpgCAQd4wAW+hpJSHznp3EH+uuhOgZdeeon87kdv7tUOBP7wQwcYFHn9dU2HZ0uXwkL+ys6dO93FxUXgARf4iIlbKJ0mUGotdqfA2bNnleTkVCU+viMIuLVVI6wDRT799NrncXEaxWrNdmtZ4AEX+EIJ4gG4SfnhGkssXtxIVvBV/vrXa0HAQUEORX766dpnO3ciZvoozzzzTAeZwAEecIUSMAfl9DiaYIp+/vnn466NP//8cyUmJkm5914N+dO9En/4g0aZM8f9M/QZO1ZDlsxAMLbLBQ7wgCvcgcDAxEI85MiRIx+cv3i2fZqeO/chJTBQo+zf7x4EvHGjI2UjY7p7/ve/O6y1ePFSxIIC+TQv7QWeXG/0bh+iZB7rjh072u124MABivoAkQkvX1ZXIDhYo/j5aZQZM9y3uXTJYangYJ0Y5iDgAK99iMoVFFY/+pKSklLMG5gl777bpgwcqFEOHVK3AizgTNvFxertELQI7KqqaszA54EDPOA6k1UPuXiNoyw2fP/+/a/v2rVLCIa/nb7tzE6AujqH348e9dwWsiAT8oEDPBGUcnUuJjCsmrECuu+++8pJw9OIhUcf1ShPPaVR/vhHdcYMCougXUuL+zbLl2so5WsUna7P6dGjR5cDB3iuE1j7JCatMayysrIlMLDfFX//UEpEoeTzm2PI6N079Epubl4L4QwFTvvk1Wl5jwANwbKd0umIbdu2bV2+/Ora5ty5c8qNEq3Alffee28z5MptQUhnK7TvrOihP/kpghoaqcNdGzdu3PTxxx8rb7/9tnIztHDhwtbExMSpJN8E+cBR26mJhS6CRW5eTBMnThy5ZMmS5smTJ59bt26dcuHCBY8WudJpbUlTfxtly9doNIyCPMiVGdq7yxU3GmIM33///fry8vLC6urqigceeGBvd95+3rx5B2fNmlVZWlpaiOEIedejQAdFYDISEGqz2RImTZpkveOOO0bT4rRq7Nixhz2Bz50799TMmTPt1GcMTVC0wDYmULCHSld3bxcmd95+JCSIhEXdfvvtesrzhUTls2fP/pfK8u/kggULajE5Ia6Io9EfcuRI6JYCQgnslpDbsTImK+hIiZShQ4cWkY/ttKI+hRigDCvmBOK2pUuXNtPwGwUFoDgUQH9a1Pq4lA26dsW4cePQ2BcbFGzbkFqxT4BQUiaRQHLpevwTTzyxgpRoO/nLCWEFWry8QRaYTM+GYTOMVRMSEtaRZL1A5AUXhVSVaY8FdAQo9gfY0GJRSthmmn6zMQXTfewd7G+99ZZY2x87duwo3Z8LJQjsTuI8UiKd+qQiL5C8gfQsFst8aVVftSGKm76yFBAHkxJ4DvI8fEyCR5NiY+nZBGpTT4Ib6fdKyiPHyD3r6bqJnj9Mzyqozz3IM0hOdA9KF0ApvBDJ0XnaofeUi94wajQIkU0ghfAxwEnweHrbiSSojK4fIV5GglbQ92qLxbKKfrfQ9+P0vJpkTIK1qO846ofyQDH1zZdJMIYsHqSmhLAESj5YbMj0nS5NOwxvBGF4O1iFft8tQcbTbxsUxdsT+Ei0w0IWViQuwPIeCmC4Ik7ILZg93Q5XEZQYTvAbGqOmYDAYbpM+HQzFwEg+0t9GWSYy4hqxg6BEG/qdhH5YUePt4QbEGhRwCU7PdSoUv2SCuUVuXoNgJbkACobAzux8BsbwRD/4X5YcxMjoTsJy5gpRLpQW6lAy7IqdJUU55Hv8FpU8V4JwH0rffpS8/MGUTf1/97vfISv6yOe/fR2T3qYHLXS8aS7xq6mpuYUmtH7Tp08PqaqqCps6dWo4vnGN+3iOdmiPfjddx4QQmgl73XPPPX1mzJgRTGCRBBZPE1QyXZvsdruFnmfgG9e4j+doh/boh/43qowX3oQE9YYwesP+xCmrVq0q/uKLL5ppntp1/vz5DnVMXOM+nqMd2qMf+kMO5HVrAoP29EZ9qaOW3m7w+vXrx/zwww/X1DGxNZg/fz6tnBaJ6wuXzrXXMdEe/dAfciAPcq9rAkNDMmcAmTiaBKQdOnRItY754osvKhMn2gRv3fpK5523qGMeOnCgGXIgD3K7UkS4ABrDjA8++KD1+PHjqnVMmrSUyZMrleeftwmuqKiiXfpxt22P/+1v2x6cM8cKudLC7l1TUFDQE4GEoEKgkQKqdUysIVaseEaprLTRQsbB06bdS/uP51XrmMcPHNhe/fvfW2ZMmxYJHOBdY4Xa2lpfOdQM5AKPdUzsU++9t0x55x0bXTl4506bUlp6v9jJq9HODRuWV1ZUGIADvA7WgHmIA8l3A55//vkS8qdqHRNVucbGJuXhh23K6dM2ZdcuG20JHL8XLLDRjmulKKyo0Zpnny0BDvCkWxxWoMnJd8qUKRG0drS0trZ6rGPu2/eRQotf2vLblC1bbOI3eMMGxz38PnjwU1UlIB84wAOusAZ8g2BBklm9erXHOuYvv/yi2O11ypIlDhc0Nl5VYvFixz18z5//sOr+BHVM4AAPuCI2pCtuJR8hFjzWMV9++WWKBRvFhANw7VqbuAY/95zjHp7h2tOuDTjAA65wCUyCZELBkvXjjz+q1jG///57CryJytNPXw3GV1+1KSUlNmX8eJuyefPV+8uX2xQyt3Lq1Klr5Jy7cFYBDvCAK1yClEo3YmiHNZQCSrWOuWjRImXSJMdwdILt3WtTJkxwKOI6Uv79b0cCa2lpcbtNBA7wgAt8De2Y+sA/NPup1jH37NlDfp8o3twJBP7nPx1gUOQf/+j47JVXYKFS8ccZd3VM4AEX+BpMvaRNAmUzt3VMbILr6uqVWbM6goBPnbIJ60CRr7++9vnMmTbloYcWuLUs8IALfKEEBckA3HRniS1btpIVJiiffXYtCLiszKHImTPXPjt4EK6aJP5I09kSwAOuUALmoOQRRz4q+vXXXzsk/2+//VaZMaNWefJJG/nTvRIvv2xT1q93/wx9mppsZMl5yokTJ9rlAgd4wBXuQGBgYiGthtB88cHFSxfap+n161+kScqmHD3qHsQZnEjZyJjunh8+7LDWli2viJkV8ilh7QWeXG/0bh+idMP60UcfPePyFxoy5WSRCdva1BWYMsURE2vWuG9z+bLDUlOmTBPDHAQc4LUPUbmCCqZ5Xt/Q0FCKeQOz5LJlzUpNjU355ht1K8ACzozZ0KDeDkGLwF6zZi1m4PPAAR5wRbLC+g/BAf/U1dUNP3LkyOuU0YRg+Nvp287sBHjhBYffW1s9t4UsyIR84AAPuHL96ZjAsGrGCoi2/OW0ZDuNWPjzn23Kjh02Zft2dcYMCoug3bZt6u2gRGVl6elHH320HDjAa5/AnAsaTCbQjgJm2FNPPdVSVjblSmlpBSWiCvL5zTFklJZOvVJfP6eFFkxDgdM+ebkuaqAV9g5YtlPQjNi3b9/W7du3d0haN0oorH/22WebIVduC0I6WMFJ8E15ebk/WSKCGhlJ47t27969iXyofPLJJzdVx9y4cWMrmX8q9iaQDxxnLLhd6CJYsM4krU2NjY0jSUDz008/fQ5Fd+QOTxbpXMekqb9tx44dr9FoGAV5kAv5Xe1B2hXBGG5qatIvW7ascOXKlRVr167tVh1z3bp1B1esWFFJL1KI4Qh516NAB0VgMnqD0MceeyyBBFlpCzB6wYIFVUuXLj3cBfgpUtq+ZMkSbHwyyQuYpEIhr9u7MEQurQX9aHESRDESVV9frydFConLX3jhhX+pLP9O0rNaTE6IK4qBaPSHHDkSulfHRCfsluQ+MnDevHk6mmxSSJkisoydFiaijilrmKKOuWnTpmYKvFFQAIpDAfQvKyvzgbzr3Rh7UTLpieEjN0IBcjeO/UjUtGnTEmnU5BKP37x5s6hj4s/Oci/yBpl9MllhGDbDWDUhIWEdSRkyEHnBqZAnZdpjAR0Biv0BNrRYlBKAmYRlYwqm3yV0z75//35Rx6TN71Gy0FwoQQreSZxHiqTTdyryAik0kL5jscyHVbHxcTtEcRMWwJvLjGakxjnI8/AxCRxN7hlL9ybQ73rixjlz5qw8fPjwMQrW9XS/ie49TIpU0O97kGeQnIiLqF8BlMIL0bdOdYcOn8lFbxg1GITIJkGF8DHASZnxZIWJJKSM7j9C95cR4Aq6v5quV9HvFuLH6Xc18SRYC+ckqN0YuYrKx4vBTSQvyK0STkug5IPFhkzf6dK08HMRhOHtYBUCuRsgUI7u2aAo3p7uj0Q7LGRhRbpXgOU9FMBwRZxQXKAi6L6OiaDEcILf0Bg1hYqKitukTwdDMTCSD/wth6FJpvlUxA6CEm1IiST0k+WjGLgBsQYFXILTc50KxS8EqcxygTAhrCTLP8Fy99SBnc/AGJ7oB//LmpcYGd1JWF5OhRArsBA6Q7nrZbR39oWc36KS939OfB6Tz2PyeUw+j8nnMfk8Jp/H5POYfB6Tz2PyeUw+j8nnMfk8Jp/H5POYfB6Tz2PyeUw+j9nd85i0OPEbNmyYP5gSmD+u+Twmn8fk85h8HpPPY/J5TD6Pyecx+Twmn8fk85h8HpPPY/J5TD6Pyecx+Twmn8fk85j/FecxnecnYTKKFR+ykA8sBKWul2WCEn0hB+zp7fn/Y/J5TD6Pyecx+Twmn8fk85h8HpPPY/J5TD6Pyecx+Twmn8fk85h8HpPPY/J5TD6Pyecx/1+dx3QWWF2LqxUVFf719fWieCpr1nwek89j8nlMPo/J5zH5PCafx+TzmHwe87/4PObw4cMDFi5caGxoaCj/PzuP+eSTT/rxeUw+j8nnMfk8Jp/H5POYfB6Tz2Pyecz/5v+PKQqssUNivbVmbS8n4/o/fh4TIEGxQX79h/S/JTo3ul+EOSIkMjsyLDQlNBzfuMZ9PEc7qdRvcx4Tb9knvE+fyKzIYF2WLjLcHB4fkR6RHG4JN+lydBZtljYD37gW9+k52qE9+qH/DZ/HxJtEZUX1hrDI3Mj+umxdisFuKM7fl9+c0pqyK+zXsO96tfW6oFE0Cr5xjft4jnZoj37oDznSMtc/gUH7qJyovmRirTZTOzh9UfqYtK/TdvRs63kJoO2fi8SZxMPl9QXHN9qhffpCyxj0hxzIk1bxui4FdFZdQHR+dHREZkRa7p7cZt/Lvuc0VzrAOz7ziHtLXtLpWZtGQb/cd7ObIQfyILcrRRwuII1hxv5F/a1pR9O2uYF2fA4SBxDXSA4lPuK+bdqGhG39h0dbIRfyPbmmJwIJQYVAMx01bVdVAE4pI9YSH5ccSVwln7uxmmln0vYIY6hFa46IBI7b85gDigf4YqiFpYUZst/PXq6qAD5vEvck/pPLPRxt8Sferd7PuGjQ8pCkWw3AAV4Ha8A8FDyBZIUBxvnGEp/LPudVRf1EPJI4h/gE8TriZ+XvbOIJxKfVFTHM1pcAB3jSLVcPcIQkh0REDYmyGL4y7FQVATP/RR7mQbS4nMfUzJf3vIjf8aAEyQcO8Dqcx0SwIMno6/TjPLrhJHEi8Sh5PVKCgkfIe3cSZxGfcS+jR1uPy8ABHnBFbMAkugzdrZF5kYbM9zNXe1SiQcbCW/K6Wl6Dp3eKl+fU5QAHeMB1usQXySQ8Izxr0DeDPlTt+qUMvPtc7jUR9yL2Jn7c5X4pcTBxqxs55zQKcIAHXOESkZqzI2N0BbqhwWeCf1BVopC4D/H3Lvc2EfsS+xCvdbn/nUxg493EFH2AAzzgAl8TXxTfB/4Jzwq/w7vN+6JbBV6Sfm/qdP9DCQZFXu/0bKm0kJswBw7wgAt8DaZerVWbEJ4ZXuxWgbPEycTxbp61SutAkU/dPI8jtrq3LPCAC3yhRPSQ6AG46X3ZjSUWSyv8VcVNQVKRn9w82ylj5plOliAc4AFXKAFzUE6P0+XrikJ+DjneofnnxDHE97pPxeKDM3VzPOSVsdKS31y9DxzgAVe4QwQmTSzaXO2QQUcGfaC56DJNzyUOJN7vYdhulCn7hMrzv0trLXbMrJCbeDRxL/DEhIbAdA5Rmm6tlh2Wq4Y7IKMemfCyBwUwFP2IZ3iY7ObIdl867gEHeO1DFMkCqx9apunjSuJKxbyBjncTDyQ+5MEKLucxNcUe2n0qA5tmWZ9LPueBAzzgOpNVDwQH/KPN0w7X79e/rtklBf/Bxbed2fmpk34/2kVbeR4T8oEDPBGUrucxsWrGCij+vvjyHlk9TotYeJT4KeI/euBnpUXQrkWlzXJHyvfSaU5Hj44uBw7wrjmPKVdUsMawmMqYFlLiikjTvtLnN8O+IpdcCc4NbtHmaIcCp33y6nweE3sHLNsjsiNGJG9L3qpZ3jHn3/Bnn0ZJfS91M+QK+YSjeh6TFqH+5KeI8PRwY3h2+F1JG5M2aT4mIW/fhAL08V/o3xqYGDg1zBxmgnzgqJ7HRKQiWLDOpOg1JUxMGJm0JKm5x+Qe58QK6kIXFumU0LwavNrinol7bUDJgFGQB7mQ39UepF0RjGH9/Xr94PLBhYOrB1eEPBCytztv329ev4OJsxIrB5YOLMRwhLzrUaCDIjBZcmly6CDboITESYnW6DuiR+uKdVW9x/Y+7AncZ67PqYEzB9oTJiWMoXSUGWYMS4jKiwqFvG7vwuTO24+iI4hmu6iY22P00UXRhdpCbbn/bP9/uVMg4GTAyZQFKbWYnBBX2nRtNPpDzg2fxxTbQcrtWBnHFcfpYopiUiKHRhbFlsTa+57ue0rEwGXHnODV5tWWtDSpOSI3YpRQgBSHAugfmhzavfOYSeOSxHlMsRGibZvYENM+QQg1hyfqcnW5NHLGJz6RuMLrileb5heHFRJ3Jr4Rmhk6mYbhMLEZplUTEpJYRxbGBiIvuCikXsd0xgI6AhT7A2xoxaI0V2eOzI/MxhQclhVWEpIVYk94K+FdKKA9pj0amh86VyiRE3EnJaQ8bbY2nfqkiryQETFQl6mLxTIfVpUbH/XzmLIUEAeT6vJ0EDccPibho8OsYWPDc8Mn3Jp1a32/jH6N/XL7rQz9OPRYeEn4+qCMoKZga/DDoVmhFRF5EfcgzyA5hVvDiyKyIgqglHihnEidpx16T7noDaOGgxDZNOcXwscAJ8HjI/IjJtLblhHYI/2s/ZYFmANWBFoDVwdaAlcFmgNbgixBj4dkhlSHZYdNgrWo7zjqNwYvQRbMx4vBTTRUg9SUEJZAyQeLDZhRaE+2gJ/xRsIiSL2wSmbY3RJkPP22CSvR29MCdqQYHbSQhRXJmgVY3kMBDFfECblF/TwmghLDCX5DY9QUQg2ht8GnotgB/0I5Sj7wt0jvlnCTcB1di9ihoEQb+p2EflhR4+3hBsQaFHAJTs91KhS/5FyC7BkIEworIWZQvyKBndn5DIzhiX7wP0aac2R0J2F1OI8pLdShZNgVO0uKcsj/tucxSWiPmIKYnuZKs3fGrIxeTsY17uP5f6SOCcEAMZQa/NJr0m9Jr07vZ5puCkmrSgtLnZoajm9c4z6eox3aS4W8buqNyazepnKTf9I9SX3MM8zBBBZpnG6MN800JafOSDWZ7CaLcZYxA9/imu7jOdqhPfrBSjekDDpZ6ix+humGYP1MfXxKdUqmqdqUkr8qv3jYF8Oa9Sf1u8LOd6pj0jXu4znaob252twfyqTPSO8Ny1y3IlAgZUqKv2GGQZs0LcmYODNxhOF/DIvNP5ivrWNekvuShS7Xso6J9sPWDxtjnGkcbKw0atOmp/WFVbpUxKmAucocYLFboklA2tBDQ9XrmC8ST5S81X0dc+iBIc2QA3mQ25UiXjAZNIYZLQ9arJbjFvU65jHiycTPS65QKYpgx/W35G2WOWYr5Ar5aq4Rw4sCCUGFQDMfN6vXMbGGWEFcKetY4Gly/6FSxzQfMGw3/l5vMU0zRgIHeNdYIbM201cMuemphoJDBZ7rmAfkTv2dTmWAUrmTV6tVbTAtN1SkGIADvA7WgHlMlaZAGmID8p/PL/Fp81DHRFWukfhhWa/cJbcE+L1A7rjOqiuS92xuCXCAJ93isEJUZpSvYYohwjzbbEltTfVcx6SNjMYmt/xb5G/wBnnPJuveKh/IBw7wgCusAd8gWJBk8lbnea5jYilnd6noN7oosdilsjNffX+COiZwRFIjXBEb0hW3mmvNhrxDeZ7rmC/LWDggr9fK63td6pbOePGwawMO8AQuXAKTIJkYqgxZiT8mqtcxv5eB97TLvVeJS2SpcLPLfYT1FOJTbuRQjgUO8IArXIKUaqoyxRgfMA4NPuuhjrmIeJIcjs7PXllQL+k0Uv4tE1iLSh2TcIAHXOBrcubm9IF/DDUG9TrmHun3Vzvd/6cEgyL/6PTsFWmhg+7rmMADLvA1mHpNM0wJ+mq9+4LPBVmNmeXm2SlpHSjytZvnM4kfcm9Z4AEX+EIJS61lAG66tYRzGH6m4qYyqcgZlT9RlciieydLAA+4QgmYg5JHHPmoKOTXTnXMb2VV7kkPdUyMmPUe8kqTtKRLiRE4wAOuwx2Y62liMdYahww+PvgD53QsvtfLSeqoh2G7V6Zstb/2HJbW2iLrmCQ3qTVpL/CAKwLTOUSN1UZr9kfZV+uYX0lTbpCd1RSYImNijYfJbr1sJ/9CABzgtQ9RJAusfkyzTHpLg6VUzBvouEz+qfEbD1bY6ZIxGzy0+1oGNimKOilwgAdckaywkBEjBHFRZxxuPGJ8XRRQbdLfXdUxX5B+b+2i7csOmZAPHOAB17k6FxMYVs1iBfSEpbzX/F6nRSz8mXgH8XYP/La0yA75hzi1dqSEd6X3aeOjxnLgAK99AnMuaDCZCGvUGoelPJXSQsF0RaTpCS5LuBvlCSLlX4mvj29JtacOBU775OW6qIFW2Dtg2U5jeIRxn3GrZnunpHWjny9pifeZZTPkQj5wOljBdZGLPUZGbUYEra6Merv+LuNu4ybx9+9Pbq6O2Xdj39aEqQlTsTeBfLGXUdmpiYUugkWsM6tNJmujdaRlo6XZ+2nvc2IFdakLi3SuY77s1TZ4x+DXrA3WUZAHuZDf1R6kXRGM4YKmAn3OspzCrJVZFVFro7pVxwxfF37QusJamdWYVYjhCHnXo0AHRWCygoaC0JzHchJyGnOspgdNo/UL9FVBS4M81jH91vmdyliZYbcusY5JnZmaaZhqSLDUWEIhr1u7MDRE5GbPzvajtWAQBVKUpd6iT3swrVD/oL484IUA93XMXwJOZr2QVYvJCXGVNiMtGv0hR46E7tUx0Qm7JbHooZVx5rxMXfrc9BRTvanI2Gi09z0r65htV+uYlk2WZkOtYRQUgOJQAP2NZUYfl7JB164oeLSgJ4aP2InTtk3uxrH1j0qdlppotBtzU+wp49M2pznqmHJBqz+gfyOpJmmyodowDJthrJqQkLCOtNZZA5EXnAp5UqY9FtARoNgfYEMrFqU1JrOpzpSNKTi5JrkksTbRPnj/YFHH1P2gO5pYnzhXKGE33EmK5tEElZ5mT0sV5YIq48C0mWmxWOYLq9LGx+0QxU1YQExmlNFgUsqeOcjz8DHljdHJs5LHptSmTBhkH1Qfb49vjJsTtzLscNixAUsHrI+rjWtKsCc8TIpU6Gv19yDPIDmRZYoMswwFUEq8ULVZp7pDh89gLmH6WaZBiGxKsYXwMcD1dfrx+gf0E5Oqk8oI7JH42vhlsTWxK2LrYlfH2mNX3VZzW8uAmgGPJ9oTq5PtyZNgLZyToP5j8BL0QvkiVshNllmWILdKOC2Bkg8WG8KM0J5MCz/jjYRF6O1gFX2N/m6AQLnk6mSbUBRvX6MfiXZiIUtWpBcpwPIeCmC4Ik70ZfpbUAVyW8dEUGI4wW9ojJqCocJwG3wKU4q8D6bkA39DMFKxGI50LWIHQUlt0mrSkkQ/lJfo7eEGxBoUcAlO9SIZTIXiF4JUrDVIKZhQWAkLEcQNgrcTO5+BRX6hfvA/RppzZHQnYXk5FUKswEKdS4ZdsbOkiL6Q42lY/i9/ofIO/jaWEgAAAABJRU5ErkJggg==')",
  254. ];
  255.  
  256. var uroMarkers =
  257. [
  258. // 0 = comment count circle
  259. ["data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACEAAAAaCAYAAAA5WTUBAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB90LCxYGJyle3m4AAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAv1JREFUSMfFls1r3EYYxn+zo9XsqjuSxpBQU5OCDy0Y3JODs2Z36W6P/YBCk159s+/5F9Z/QsHkUAruZZ344Jx7KPTSa2/NwSdDE1ISf8SWI63k6UVLN8ZuZFnQ56IZDYwevR/P+wguh7jieXF9Gewla3vJWeELAfA8b9ZxnFWl1P0kST7NsswDkFJGrus+i+P4cZqmP0VR9JyKIWq12owx5kff95PBYBCPRiN7EaPRyA4Gg9j3/cQY80hK6VdFwFFK3TXGvOx0OoktiE6nExtjXjSbzYUb/T3QqNfrvSAITofDob0uhsOhDcPwTVkiAmgCC8aYV2UITBMxxrwok5oGcEdr/XR5eXlsb4g8NY+uVQPAh0KIr7XW6f7+vq0Cvu8nnufNvu/jtak03FZKfbe4uGjn5uYqqe6lpSXrOM5qERIS8IEZz/O66+vrTlUttra25iql7hfSImAB+DIIgrd7e3u2KhwcHNggCE6LREIBGlBZlqn5+fnK1C4MQybqWiQdKi/O/wWTwpSAkFLGWZZVdnkURUgpoyIk/t3Uas93dnYqI7G9vY3rus+KkLBABtgoin7f3NxMqyKxtbWVxHH8uIhUG+AT4CMhxK1Wq/XD8fGxrIJEEATjNE0/ft+IrwEx8AaIrbWH1tpfV1ZWbhyNbrc7llL+XMRjyDwdTt6mzSRJ/jw5Ofkiy7JGr9crRWBjY4Pd3d3XR0dH31hrz4pOTw18BnwFPBBCPPR9/6zsKA+C4FQpdbeoc3tngAErwLfA90KIh2EYHrbb7cITtd1uj40xr+r1eq+s7jSAO0A3J/IAWG21Wr9ordN+vz++yt71+/2x1jrVWj/NR0DjukbmHVcF3ALmgJl8L4UQoeu6n3ued+/8/Hw2yzKVG91YSvlXFEW/xXH8xFr7B/A38PYqZ13EbYtcwkPgdk7kgylZF1PWPc076xR4DbwEDvN3tmwkLtZIMx/xwWTA5d1ELm6T1j4CjoGznFgpX/lfZxJwpwhMR2JCJJkobllN+QeQNNGuneopoQAAAABJRU5ErkJggg=="],
  260. // 1 = green comment marker
  261. ["data:image/png;base64,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"],
  262. // 2 = yellow (own) comment marker
  263. ["data:image/png;base64,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"]
  264. ];
  265.  
  266. var uroFeedFilterReloads = 0;
  267. var uroFeedFilterFilters =
  268. [
  269. ['feed.issues.motivations.CAN_BE_SOLVED_BY_RANK', 'motivation'],
  270. ['feed.issues.motivations.CLOSE_TO_FAVORITES', 'motivation'],
  271. ['feed.issues.motivations.ISSUE_AGE', 'motivation'],
  272. ['feed.issues.motivations.ISSUE_REOPENED', 'motivation'],
  273. ['feed.issues.motivations.NEAR_DRIVES', 'motivation'],
  274. ['feed.issues.motivations.REPORTED_BY_USER', 'motivation'],
  275. ['feed.issues.motivations.USER_FOLLOWS_ISSUE', 'motivation'],
  276. ['feed.issues.motivations.USER_FOLLOWS_ISSUE_LAST_COMMENT', 'motivation'],
  277. ['venues.update_requests.panel.flag_title.IMAGE', 'title'],
  278. ['venues.update_requests.panel.flag_title.VENUE', 'title'],
  279. ['venues.update_requests.panel.title.ADD_IMAGE', 'title'],
  280. ['venues.update_requests.panel.title.ADD_VENUE', 'title'],
  281. ['venues.update_requests.panel.title.DELETE_VENUE', 'title'],
  282. ['venues.update_requests.panel.title.UPDATE_VENUE', 'title']
  283. ];
  284.  
  285.  
  286. /*
  287. function uroAddDebug(debugtext)
  288. {
  289. var ts = Math.round(performance.now());
  290. if(uroRecentDebug.length == 100)
  291. {
  292. uroRecentDebug.shift();
  293. }
  294. uroRecentDebug.push(ts+': '+debugtext);
  295. console.debug('URO+DBG '+ts+':'+debugtext);
  296. }
  297.  
  298. function uroDumpDebug()
  299. {
  300. if(uroRecentDebug.length > 0)
  301. {
  302. document.getElementById('WazeMap').innerHTML = '<textarea id="uroDbgOutput" style="width:100%;height:100%">';
  303. var dbgOutput = '';
  304. for(var i=0; i<uroRecentDebug.length; i++)
  305. {
  306. dbgOutput += uroRecentDebug[i]+'\n';
  307. }
  308. document.getElementById('uroDbgOutput').textContent = dbgOutput;
  309. }
  310. }
  311. */
  312.  
  313. function uroTempFixMTEDropDown()
  314. {
  315. // temporary fix for that bloody annoying bug in the closure event dropdown... sort it out devs!
  316. // also removes the "Choose Event" non-option from the list, so that it now always starts with "None"
  317. if(document.getElementsByName('closure_eventId').length > 0)
  318. {
  319. if(document.getElementsByName('closure_eventId')[0].selectedOptions.length > 0)
  320. {
  321. document.getElementsByName('closure_eventId')[0].required = false;
  322. if(document.getElementsByName('closure_eventId')[0].selectedOptions[0].text == I18n.lookup('closures.choose_event'))
  323. {
  324. document.getElementsByName('closure_eventId')[0].selectedOptions[0].remove();
  325. }
  326. }
  327. }
  328. }
  329.  
  330. function uroAddLog(logtext)
  331. {
  332. if(uroShowDebugOutput) console.log('URO+: '+logtext);
  333. }
  334. function uroGetCBChecked(cbID)
  335. {
  336. try
  337. {
  338. return(document.getElementById(cbID).checked);
  339. }
  340. catch(err)
  341. {
  342. return null;
  343. }
  344. }
  345. function uroSetCBChecked(cbID, state)
  346. {
  347. try
  348. {
  349. document.getElementById(cbID).checked = state;
  350. }
  351. catch(err)
  352. {
  353. }
  354. }
  355. function uroGetElmValue(elmID)
  356. {
  357. try
  358. {
  359. return(document.getElementById(elmID).value);
  360. }
  361. catch(err)
  362. {
  363. return null;
  364. }
  365. }
  366. function uroSetStyleDisplay(elm,style)
  367. {
  368. try
  369. {
  370. document.getElementById(elm).style.display = style;
  371. }
  372. catch(err)
  373. {
  374. }
  375. }
  376. function uroSetOnClick(elm,fn)
  377. {
  378. try
  379. {
  380. document.getElementById(elm).onclick = fn;
  381. }
  382. catch(err)
  383. {
  384. }
  385. }
  386. function uroAddEventListener(elm,eventType,eventFn,eventBool)
  387. {
  388. try
  389. {
  390. document.getElementById(elm).addEventListener(eventType, eventFn, eventBool);
  391. }
  392. catch(err)
  393. {
  394. }
  395. }
  396.  
  397.  
  398. function uroAlertBoxObj(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction)
  399. {
  400. this.headericon = headericon;
  401. this.title = title;
  402. this.content = content;
  403. this.hasCross = hasCross;
  404. this.tickText = tickText;
  405. this.crossText = crossText;
  406. this.tickAction = tickAction;
  407. this.crossAction = crossAction;
  408. }
  409. function uroCloseAlertBox()
  410. {
  411. document.getElementById('uroAlerts').childNodes[0].innerHTML = '';
  412. document.getElementById('uroAlerts').childNodes[1].innerHTML = '';
  413. document.getElementById('uroAlertTickBtnCaption').innerHTML = '';
  414. document.getElementById('uroAlertCrossBtnCaption').innerHTML = '';
  415. uroAlertBoxTickAction = null;
  416. uroAlertBoxCrossAction = null;
  417. document.getElementById('uroAlerts').style.visibility = "hidden";
  418. document.getElementById('uroAlertCrossBtn').style.visibility = "hidden";
  419. uroAlertBoxInUse = false;
  420. if(uroAlertBoxStack.length > 0)
  421. {
  422. uroBuildAlertBoxFromStack();
  423. }
  424. }
  425. function uroCloseAlertBoxWithTick()
  426. {
  427. if(typeof uroAlertBoxTickAction === 'function')
  428. {
  429. uroAlertBoxTickAction();
  430. }
  431. uroCloseAlertBox();
  432. }
  433. function uroCloseAlertBoxWithCross()
  434. {
  435. if(typeof uroAlertBoxCrossAction === 'function')
  436. {
  437. uroAlertBoxCrossAction();
  438. }
  439. uroCloseAlertBox();
  440. }
  441. function uroShowAlertBox(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction)
  442. {
  443. uroAlertBoxStack.push(new uroAlertBoxObj(headericon, title, content, hasCross, tickText, crossText, tickAction, crossAction));
  444. if(uroAlertBoxInUse === false)
  445. {
  446. uroBuildAlertBoxFromStack();
  447. }
  448. }
  449. function uroBuildAlertBoxFromStack()
  450. {
  451. uroAlertBoxInUse = true;
  452. uroAlertBoxTickAction = null;
  453. uroAlertBoxCrossAction = null;
  454. var titleContent = '<span style="font-size:14px;padding:2px;">';
  455. titleContent += '<i class="fa '+uroAlertBoxStack[0].headericon+'"> </i>&nbsp;';
  456. titleContent += uroAlertBoxStack[0].title;
  457. titleContent += '</span>';
  458. document.getElementById('uroAlerts').childNodes[0].innerHTML = titleContent;
  459. document.getElementById('uroAlerts').childNodes[1].innerHTML = uroAlertBoxStack[0].content;
  460. document.getElementById('uroAlertTickBtnCaption').innerHTML = uroAlertBoxStack[0].tickText;
  461. if(uroAlertBoxStack[0].hasCross)
  462. {
  463. document.getElementById('uroAlertCrossBtnCaption').innerHTML = uroAlertBoxStack[0].crossText;
  464. document.getElementById('uroAlertCrossBtn').style.visibility = "visible";
  465. if(typeof uroAlertBoxStack[0].crossAction === "function")
  466. {
  467. uroAlertBoxCrossAction = uroAlertBoxStack[0].crossAction;
  468. }
  469. }
  470. else
  471. {
  472. document.getElementById('uroAlertCrossBtn').style.visibility = "hidden";
  473. }
  474. if(typeof uroAlertBoxStack[0].tickAction === "function")
  475. {
  476. uroAlertBoxTickAction = uroAlertBoxStack[0].tickAction;
  477. }
  478. document.getElementById('uroAlerts').style.visibility = "";
  479. uroAlertBoxStack.shift();
  480. }
  481.  
  482.  
  483. function uroFirstTimerWelcomePack()
  484. {
  485. uroAddLog('welcome new users to Club URO...');
  486.  
  487. // for now, just show the update notes...
  488. uroShowUpdateNotes();
  489. }
  490. function uroShowUpdateNotes()
  491. {
  492. uroAddLog('let existing users know what\'s new in this release');
  493.  
  494. var releaseNotes = '';
  495. releaseNotes += '<p>Thanks for upgrading to URO+ '+uroVersion+' ('+uroReleaseDate+'). What\'s changed?</p>';
  496.  
  497. var loop;
  498. if(uroChanges.length > 0)
  499. {
  500. releaseNotes += '<ul>';
  501. for(loop=0; loop < uroChanges.length; loop++)
  502. {
  503. releaseNotes += '<li>'+uroChanges[loop];
  504. }
  505. releaseNotes += '</ul>';
  506. }
  507. if((uroBetaEditor) && (uroBetaChanges.length > 0))
  508. {
  509. releaseNotes += '<p>For WME Beta:<p>';
  510. releaseNotes += '<ul>';
  511. for(loop=0; loop < uroBetaChanges.length; loop++)
  512. {
  513. releaseNotes += '<li>'+uroBetaChanges[loop];
  514. }
  515. releaseNotes += '</ul>';
  516. }
  517. uroShowAlertBox('fa-info-circle', 'URO+ Release Notes', releaseNotes, false, "OK", "", null, null);
  518. }
  519. function uroAdvertiseCustomIcons()
  520. {
  521. /*
  522. //// TEMPORARY REMOVAL UNTIL CUSTOM ICON SUPPORT IS FIXED IN NEW WME
  523. uroAddLog('advertise the benefits of custom UR icons...');
  524.  
  525. var confirmMsg = '';
  526. 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>';
  527. 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>';
  528. confirmMsg += '<div style="background-image:'+uroAltMarkers[1]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  529. confirmMsg += '<div style="background-image:'+uroAltMarkers[0]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  530. confirmMsg += '<div style="background-image:'+uroAltMarkers[4]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  531. confirmMsg += '<div style="background-image:'+uroAltMarkers[3]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  532. confirmMsg += '<div style="background-image:'+uroAltMarkers[5]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  533. confirmMsg += '<div style="background-image:'+uroAltMarkers[11]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  534. confirmMsg += '<div style="background-image:'+uroAltMarkers[12]+';background-position:0 -523px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  535. confirmMsg += '<div style="background-image:'+uroAltMarkers[10]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  536. confirmMsg += '<div style="background-image:'+uroAltMarkers[6]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  537. confirmMsg += '<div style="background-image:'+uroAltMarkers[8]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  538. confirmMsg += '<div style="background-image:'+uroAltMarkers[7]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  539. confirmMsg += '<div style="background-image:'+uroAltMarkers[9]+';background-position:0 -25px;visibility:visible;height:40px;width:33px;float:left;"></div>';
  540. confirmMsg += '<p style="clear:left;">Would you like me to automatically enable these custom markers?</p>';
  541. confirmMsg += '<p>If you change your mind later on, they can be enabled/disabled via the Misc tab within the URO+ settings</p>';
  542. uroShowAlertBox('fa-info-circle', 'URO+ Message to Users', confirmMsg, true, 'Yes please', 'No thanks', uroSetMarkerCBs, null);
  543. */
  544. }
  545.  
  546. function uroSetMarkerCBs()
  547. {
  548. uroSetCBChecked('_cbCustomRoadworksMarkers', true);
  549. uroSetCBChecked('_cbCustomConstructionMarkers', true);
  550. uroSetCBChecked('_cbCustomClosuresMarkers', true);
  551. uroSetCBChecked('_cbCustomEventsMarkers', true);
  552. uroSetCBChecked('_cbCustomNotesMarkers', true);
  553. uroSetCBChecked('_cbCustomBOGMarkers', true);
  554. uroSetCBChecked('_cbCustomDifficultMarkers', true);
  555. uroSetCBChecked('_cbCustomWSLMMarkers', true);
  556. uroSetCBChecked('_cbCustomNativeSLMarkers', true);
  557. uroSetCBChecked('_cbCustomElginMarkers', true);
  558. uroSetCBChecked('_cbCustomTrafficMasterMarkers', true);
  559. uroSetCBChecked('_cbCustomTrafficCastMarkers', true);
  560. uroSetCBChecked('_cbCustomCaltransMarkers', true);
  561. uroSetCBChecked('_cbCustomTFLMarkers', true);
  562. }
  563.  
  564. function uroGatherSettings(container)
  565. {
  566. var options = '';
  567. var urOptions = document.getElementById(container).getElementsByTagName('input');
  568. for (var optIdx=0;optIdx<urOptions.length;optIdx++)
  569. {
  570. var id = urOptions[optIdx].id;
  571. if((id.indexOf('_cb') === 0)||(id.indexOf('_text') === 0)||(id.indexOf('_input') === 0))
  572. {
  573. options += ':' + id;
  574. if(urOptions[optIdx].type == 'checkbox') options += ',' + urOptions[optIdx].checked.toString();
  575. else if((urOptions[optIdx].type == 'text')||(urOptions[optIdx].type == 'number')) options += ',' + urOptions[optIdx].value.toString();
  576. }
  577. }
  578. return options;
  579. }
  580. function uroGatherCamWatchList()
  581. {
  582. var liststr = '';
  583. for(var loop=0;loop<uroCamWatchObjects.length;loop++)
  584. {
  585. var camObj = uroCamWatchObjects[loop];
  586. if((camObj.fid != null) && (camObj.persistent === true))
  587. {
  588. if(loop > 0) liststr += ':';
  589.  
  590. liststr += camObj.fid+',';
  591. liststr += camObj.watch.lon+',';
  592. liststr += camObj.watch.lat+',';
  593. liststr += camObj.watch.type+',';
  594. liststr += camObj.watch.azymuth+',';
  595. liststr += camObj.watch.speed+',';
  596. liststr += camObj.watch.validated+',';
  597. liststr += camObj.groupID+',';
  598. liststr += camObj.server;
  599. }
  600. }
  601. return liststr;
  602. }
  603. function uroGatherSegWatchList()
  604. {
  605. var liststr = '';
  606. for(var loop=0;loop<uroSegWatchObjects.length;loop++)
  607. {
  608. var segObj = uroSegWatchObjects[loop];
  609. if((segObj.fid != null) && (segObj.persistent === true))
  610. {
  611. if(loop > 0) liststr += ':';
  612.  
  613. liststr += segObj.fid+',';
  614. liststr += segObj.watch.left+',';
  615. liststr += segObj.watch.right+',';
  616. liststr += segObj.watch.bottom+',';
  617. liststr += segObj.watch.top+',';
  618. liststr += segObj.watch.fromNode+',';
  619. liststr += segObj.watch.toNode+',';
  620. liststr += segObj.watch.fwdDir+',';
  621. liststr += segObj.watch.revDir+',';
  622. liststr += segObj.watch.length+',';
  623. liststr += segObj.watch.level+',';
  624. liststr += segObj.watch.rank+',';
  625. liststr += segObj.watch.roadType+',';
  626. liststr += segObj.watch.updatedOn+',';
  627. liststr += segObj.groupID+',';
  628. liststr += segObj.server;
  629. }
  630. }
  631. return liststr;
  632. }
  633. function uroGatherPlaceWatchList()
  634. {
  635. var liststr = '';
  636. for(var loop=0;loop<uroPlaceWatchObjects.length;loop++)
  637. {
  638. var placeObj = uroPlaceWatchObjects[loop];
  639. if((placeObj.fid != null) && (placeObj.persistent === true))
  640. {
  641. if(loop > 0) liststr += ':';
  642.  
  643. liststr += placeObj.fid+',';
  644. liststr += placeObj.watch.left+',';
  645. liststr += placeObj.watch.right+',';
  646. liststr += placeObj.watch.bottom+',';
  647. liststr += placeObj.watch.top+',';
  648. liststr += placeObj.watch.name+',';
  649. liststr += placeObj.watch.imageCount+',';
  650. liststr += placeObj.watch.residential+',';
  651. liststr += placeObj.watch.updatedOn+',';
  652. liststr += placeObj.groupID+',';
  653. liststr += placeObj.server;
  654. }
  655. }
  656. return liststr;
  657. }
  658. function uroGatherCWLGroups()
  659. {
  660. var liststr = '';
  661. for(var loop=0;loop<uroCWLGroups.length;loop++)
  662. {
  663. var groupObj = uroCWLGroups[loop];
  664. if(groupObj.groupID != -1)
  665. {
  666. if(loop > 0) liststr += ':';
  667.  
  668. liststr += groupObj.groupID+',';
  669. liststr += groupObj.groupName+',';
  670. liststr += groupObj.groupCollapsed;
  671. }
  672. }
  673. return liststr;
  674. }
  675. function uroGatherPlacesGroups()
  676. {
  677. var liststr = '';
  678. for(var loop=0;loop<uroPlacesGroupsCollapsed.length;loop++)
  679. {
  680. if(loop > 0) liststr += ':';
  681. liststr += uroPlacesGroupsCollapsed[loop];
  682. }
  683. return liststr;
  684. }
  685. function uroGatherFriendlyAreaNames()
  686. {
  687. var liststr = '';
  688. for(var loop=0;loop<uroFriendlyAreaNames.length;loop++)
  689. {
  690. var fnObj = uroFriendlyAreaNames[loop];
  691. if(loop > 0) liststr += ':';
  692.  
  693. liststr += fnObj.fName+',';
  694. liststr += fnObj.area+',';
  695. liststr += fnObj.server;
  696. }
  697. return liststr;
  698. }
  699. function uroSaveSettings()
  700. {
  701. if(uroInhibitSave)
  702. {
  703. uroAddLog('save inhibited');
  704. return;
  705. }
  706.  
  707. if (localStorage)
  708. {
  709. localStorage.UROverviewUROptions = uroGatherSettings('uroCtrlURs');
  710. localStorage.UROverviewMPOptions = uroGatherSettings('uroCtrlMPs');
  711. localStorage.UROverviewCameraOptions = uroGatherSettings('uroCtrlCameras');
  712. localStorage.UROverviewMiscOptions = uroGatherSettings('uroCtrlMisc');
  713. localStorage.UROverviewPlacesOptions = uroGatherSettings('uroCtrlPlaces');
  714. localStorage.UROverviewCamWatchList = uroGatherCamWatchList();
  715. localStorage.UROverviewSegWatchList = uroGatherSegWatchList();
  716. localStorage.UROverviewPlaceWatchList = uroGatherPlaceWatchList();
  717. localStorage.UROverviewCWLGroups = uroGatherCWLGroups();
  718. localStorage.UROverviewFriendlyAreaNames = uroGatherFriendlyAreaNames();
  719. localStorage.UROverviewPlacesGroups = uroGatherPlacesGroups();
  720.  
  721. localStorage.UROverviewMasterEnable = uroGetCBChecked('_cbMasterEnable');
  722. localStorage.UROverviewCurrentVersion = uroVersion;
  723.  
  724. uroAddLog('save complete');
  725. }
  726. else
  727. {
  728. uroAddLog('no localStorage, save blocked');
  729. }
  730. }
  731. function uroApplySettings(settings)
  732. {
  733. var options = settings.split(':');
  734. for(var optIdx=0;optIdx<options.length;optIdx++)
  735. {
  736. var fields = options[optIdx].split(',');
  737. if(fields[0].indexOf('_cb') === 0)
  738. {
  739. if(document.getElementById(fields[0]) !== null)
  740. {
  741. uroSetCBChecked(fields[0], (fields[1] == 'true'));
  742. }
  743. }
  744. else if((fields[0].indexOf('_input') === 0)||(fields[0].indexOf('_text') === 0))
  745. {
  746. if(document.getElementById(fields[0]) !== null) document.getElementById(fields[0]).value = fields[1];
  747. }
  748. }
  749. }
  750. function uroApplyCamWatchList()
  751. {
  752. var objects = localStorage.UROverviewCamWatchList.split(':');
  753. uroCamWatchObjects = [];
  754. if(objects.length > 0)
  755. {
  756. for(var objIdx=0;objIdx<objects.length;objIdx++)
  757. {
  758. var fields = objects[objIdx].split(',');
  759. if(fields.length >= 7)
  760. {
  761. // following two bits of code add in blank fields if the user has updated their copy of URO+ from an
  762. // older version which didn't include support for either of these field types
  763.  
  764. // add default groupID field
  765. if(fields.length == 7)
  766. {
  767. fields.push(0);
  768. }
  769. // set default groupID value to 0 (no group)
  770. if(fields[7] == -1)
  771. {
  772. fields[7] = 0;
  773. }
  774.  
  775. // add default server field
  776. if(fields.length == 8)
  777. {
  778. fields.push('??');
  779. }
  780. // set default server value to unknown
  781. if(fields[8] === 0)
  782. {
  783. fields[8] = '??';
  784. }
  785.  
  786. uroCamWatchObjects.push(new uroCamWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7],fields[8]));
  787. }
  788. }
  789. }
  790. }
  791. /*
  792. function uroApplySegWatchList()
  793. {
  794. var objects = localStorage.UROverviewSegWatchList.split(':');
  795. uroSegWatchObjects = [];
  796.  
  797. for(var objIdx=0;objIdx<objects.length;objIdx++)
  798. {
  799. var fields = objects[objIdx].split(',');
  800. 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]));
  801. }
  802. }
  803. function uroApplyPlaceWatchList()
  804. {
  805. var objects = localStorage.UROverviewPlaceWatchList.split(':');
  806. uroPlaceWatchObjects = [];
  807.  
  808. for(var objIdx=0;objIdx<objects.length;objIdx++)
  809. {
  810. var fields = objects[objIdx].split(',');
  811. 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]));
  812. }
  813. }
  814. */
  815. function uroApplyCWLGroups()
  816. {
  817. var objects = localStorage.UROverviewCWLGroups.split(':');
  818. uroCWLGroups = [];
  819. if(objects.length === 0)
  820. {
  821. uroCWLGroups.push(new uroOWLGroupObj(0,'No group',false));
  822. }
  823. else
  824. {
  825. for(var objIdx=0;objIdx<objects.length;objIdx++)
  826. {
  827. var fields = objects[objIdx].split(',');
  828. if(fields.length < 2)
  829. {
  830. fields.push(false);
  831. }
  832. uroCWLGroups.push(new uroOWLGroupObj(fields[0],fields[1],(fields[2] == 'true')));
  833. }
  834. }
  835. }
  836. /*
  837. function uroApplyPlacesGroups()
  838. {
  839. var t = localStorage.UROverviewPlacesGroups.split(':');
  840. for(var i=0;i<t.length;i++)
  841. {
  842. uroPlacesGroupsCollapsed[i] = (t[i] == "true");
  843. }
  844. }
  845. */
  846. function uroApplyFriendlyAreaNames()
  847. {
  848. var objects = localStorage.UROverviewFriendlyAreaNames.split(':');
  849. uroFriendlyAreaNames = [];
  850.  
  851. for(var objIdx=0;objIdx<objects.length;objIdx++)
  852. {
  853. var fields = objects[objIdx].split(',');
  854. uroFriendlyAreaNames.push(new uroAFNObj(fields[0],parseFloat(fields[1]),fields[2]));
  855. }
  856.  
  857. uroReplaceAreaNames(true);
  858. }
  859. function uroTranslateLegacyMPTabSettings()
  860. {
  861. var options = localStorage.UROverviewMPOptions.split(':');
  862. for(var optIdx=0;optIdx<options.length;optIdx++)
  863. {
  864. var fields = options[optIdx].split(',');
  865. if(fields[0].indexOf('_cb') === 0)
  866. {
  867. if(fields[0] == '_cbMPFilterParkingLotInputAsPoint') uroSetCBChecked('_cbMPFilter_T50', (fields[1] == 'true'));
  868. if(fields[0] == '_cbMPMissingPLP_T70') uroSetCBChecked('_cbMPFilter_T70', (fields[1] == 'true'));
  869. if(fields[0] == '_cbMPMissingPLP_T71') uroSetCBChecked('_cbMPFilter_T71', (fields[1] == 'true'));
  870. if(fields[0] == '_cbMPFilterDrivingDirectionMismatch') uroSetCBChecked('_cbMPFilter_T101', (fields[1] == 'true'));
  871. if(fields[0] == '_cbMPFilterMissingJunction') uroSetCBChecked('_cbMPFilter_T102', (fields[1] == 'true'));
  872. if(fields[0] == '_cbMPFilterMissingRoad') uroSetCBChecked('_cbMPFilter_T103', (fields[1] == 'true'));
  873. if(fields[0] == '_cbMPFilterCrossroadsJunctionMissing') uroSetCBChecked('_cbMPFilter_T104', (fields[1] == 'true'));
  874. if(fields[0] == '_cbMPFilterRoadTypeMismatch') uroSetCBChecked('_cbMPFilter_T105', (fields[1] == 'true'));
  875. if(fields[0] == '_cbMPFilterRestrictedTurn') uroSetCBChecked('_cbMPFilter_T106', (fields[1] == 'true'));
  876. if(fields[0] == '_cbMPFilterTurnProblem') uroSetCBChecked('_cbMPFilter_T200', (fields[1] == 'true'));
  877. if(fields[0] == '_cbMPFilterRoadClosureProblem') uroSetCBChecked('_cbMPFilter_T300', (fields[1] == 'true'));
  878. }
  879. }
  880. }
  881. function uroLoadSettings()
  882. {
  883. var isNewInstall = true;
  884. var isUpgradeInstall = true;
  885. var notifyAboutCustomIcons = true;
  886.  
  887. uroAddLog('loadSettings()');
  888. if (localStorage.UROverviewUROptions != null)
  889. {
  890. uroAddLog('recover UR tab settings');
  891. uroApplySettings(localStorage.UROverviewUROptions);
  892. isNewInstall = false;
  893. }
  894.  
  895. if (localStorage.UROverviewCameraOptions != null)
  896. {
  897. uroAddLog('recover camera tab settings');
  898. uroApplySettings(localStorage.UROverviewCameraOptions);
  899. isNewInstall = false;
  900. }
  901.  
  902. if (localStorage.UROverviewMPOptions != null)
  903. {
  904. uroAddLog('recover MP tab settings');
  905. uroTranslateLegacyMPTabSettings();
  906. uroApplySettings(localStorage.UROverviewMPOptions);
  907. isNewInstall = false;
  908. }
  909.  
  910. if (localStorage.UROverviewPlacesOptions != null)
  911. {
  912. uroAddLog('recover Places tab settings');
  913. uroApplySettings(localStorage.UROverviewPlacesOptions);
  914. isNewInstall = false;
  915. }
  916.  
  917. if (localStorage.UROverviewMiscOptions != null)
  918. {
  919. uroAddLog('recover misc tab settings');
  920. uroApplySettings(localStorage.UROverviewMiscOptions);
  921. isNewInstall = false;
  922.  
  923. if(localStorage.UROverviewCurrentVersion != null)
  924. {
  925. notifyAboutCustomIcons = false;
  926. }
  927. else
  928. {
  929. if(uroGetCBChecked('_cbCustomRoadworksMarkers') === true) notifyAboutCustomIcons = false;
  930. if(uroGetCBChecked('_cbCustomConstructionMarkers')=== true) notifyAboutCustomIcons = false;
  931. if(uroGetCBChecked('_cbCustomClosuresMarkers') === true) notifyAboutCustomIcons = false;
  932. if(uroGetCBChecked('_cbCustomEventsMarkers') === true) notifyAboutCustomIcons = false;
  933. if(uroGetCBChecked('_cbCustomNotesMarkers') === true) notifyAboutCustomIcons = false;
  934. if(uroGetCBChecked('_cbCustomWSLMMarkers') === true) notifyAboutCustomIcons = false;
  935. if(uroGetCBChecked('_cbCustomBOGMarkers') === true) notifyAboutCustomIcons = false;
  936. if(uroGetCBChecked('_cbCustomDifficultMarkers') === true) notifyAboutCustomIcons = false;
  937. if(uroGetCBChecked('_cbCustomNativeSLMarkers') === true) notifyAboutCustomIcons = false;
  938. }
  939. }
  940.  
  941. if(localStorage.UROverviewCWLGroups != null)
  942. {
  943. uroAddLog('recover CWL groups');
  944. uroApplyCWLGroups();
  945. isNewInstall = false;
  946. }
  947. else
  948. {
  949. uroAddLog('set default CWL group');
  950. uroCWLGroups.push(new uroOWLGroupObj(0,'No group',false));
  951. }
  952.  
  953. if(localStorage.UROverviewCamWatchList != null)
  954. {
  955. uroAddLog('recover camera watchlist');
  956. uroApplyCamWatchList();
  957. uroGetCurrentCamWatchListObjects();
  958. isNewInstall = false;
  959. }
  960. /*
  961. if(localStorage.UROverviewSegWatchList != null)
  962. {
  963. uroAddLog('recover segment watchlist');
  964. uroApplySegWatchList();
  965. uroGetCurrentSegWatchListObjects();
  966. isNewInstall = false;
  967. }
  968.  
  969. if(localStorage.UROverviewPlaceWatchList != null)
  970. {
  971. uroAddLog('recover places watchlist');
  972. uroApplyPlaceWatchList();
  973. //uroGetCurrentPlaceWatchListObjects();
  974. isNewInstall = false;
  975. }
  976.  
  977. if(localStorage.UROverviewPlacesGroups != null)
  978. {
  979. uroAddLog('recover places groups');
  980. uroApplyPlacesGroups();
  981. isNewInstall = false;
  982. }
  983. */
  984. if(localStorage.UROverviewCurrentVersion != null)
  985. {
  986. uroAddLog('comparing install versions');
  987. if(localStorage.UROverviewCurrentVersion == uroVersion)
  988. {
  989. isUpgradeInstall = false;
  990. }
  991. }
  992.  
  993. if(localStorage.UROverviewFriendlyAreaNames != null)
  994. {
  995. uroAddLog('recover friendly area names');
  996. uroApplyFriendlyAreaNames();
  997. isNewInstall = false;
  998. }
  999.  
  1000. if(localStorage.UROverviewMasterEnable != null)
  1001. {
  1002. uroAddLog('recover master enable state');
  1003. document.getElementById('_cbMasterEnable').checked = (localStorage.UROverviewMasterEnable == "true");
  1004. }
  1005.  
  1006. if(isNewInstall)
  1007. {
  1008. uroFirstTimerWelcomePack();
  1009. }
  1010. else if(isUpgradeInstall)
  1011. {
  1012. uroShowUpdateNotes();
  1013. }
  1014.  
  1015. if(notifyAboutCustomIcons)
  1016. {
  1017. uroAdvertiseCustomIcons();
  1018. }
  1019.  
  1020. uroInhibitSave = false;
  1021. }
  1022. function uroDefaultSettings()
  1023. {
  1024. 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);
  1025. }
  1026. function uroDefaultSettingsAction()
  1027. {
  1028. var defaultSettings = '';
  1029. defaultSettings += '[UROverviewMPOptions][len=788]:_cbMPFilterParkingLotInputAsPoint,false:_cbMPFilterMissingJunction,false:_cbMPFilterMissingRoad,false:_cbMPFilterCrossroadsJunctionMissing,false:_cbMPFilterDrivingDirectionMismatch,false:_cbMPFilterRoadTypeMismatch,false:_cbMPFilterRestrictedTurn,false:_cbMPFilterRoadClosureProblem,false:_cbMPFilterUnknownProblem,false:_cbMPFilterTurnProblem,false:_cbFilterElgin,false:_cbFilterTrafficCast,false:_cbFilterTrafficMaster,false:_cbFilterCaltrans,false:_cbFilterTFL,false:_cbMPFilterReopenedProblem,false:_cbInvertMPFilter,false:_cbMPFilterOutsideArea,false:_cbMPFilterClosed,false:_cbMPFilterSolved,false:_cbMPFilterUnidentified,false:_cbMPClosedUserIDFilter,false:_cbMPNotClosedUserIDFilter,false:_cbMPFilterLowSeverity,false:_cbMPFilterMediumSeverity,false:_cbMPFilterHighSeverity,false[END]';
  1030. defaultSettings += '[UROverviewPlaceWatchList][len=0][END]';
  1031. defaultSettings += '[UROverviewSegWatchList][len=0][END]';
  1032. defaultSettings += '[UROverviewPlacesGroups][len=65]false:false:false:false:false:false:false:false:false:false:false[END]';
  1033. defaultSettings += '[UROverviewMasterEnable][len=4]true[END]';
  1034. defaultSettings += '[UROverviewFriendlyAreaNames][len=0][END]';
  1035. defaultSettings += '[UROverviewMiscOptions][len=1337]:_cbNativeConvoMarkers,true:_cbNativeBetaConvoMarkers,true:_cbCommentCount,false:_cbURBackfill,false:_inputUnstackSensitivity,15:_inputUnstackZoomLevel,3:_cbCustomRoadworksMarkers,false:_cbCustomConstructionMarkers,false:_cbCustomClosuresMarkers,false:_cbCustomEventsMarkers,false:_cbCustomNotesMarkers,false:_cbCustomBOGMarkers,false:_cbCustomDifficultMarkers,false:_cbCustomWSLMMarkers,false:_cbCustomNativeSLMarkers,false:_cbCustomKeywordMarkers,false:_textCustomKeyword,:_cbCustomElginMarkers,false:_cbCustomTrafficMasterMarkers,false:_cbCustomTrafficCastMarkers,false:_cbCustomCaltransMarkers,false:_cbCustomTFLMarkers,false:_inputPopupDwellTimeout,2:_inputPopupEntryTimeout,2:_inputMaxJitter,2:_cbInhibitURPopup,false:_cbInhibitMPPopup,false:_cbInhibitCamPopup,false:_cbInhibitSegPopup,false:_cbInhibitSegGenericPopup,false:_cbInhibitTurnsPopup,false:_cbInhibitLandmarkPopup,false:_cbInhibitPUPopup,false:_cbDateFmtDDMMYY,true:_cbDateFmtMMDDYY,false:_cbDateFmtYYMMDD,false:_cbTimeFmt24H,true:_cbTimeFmt12H,false:_cbWhiteBackground,false:_inputCustomBackgroundRed,255:_inputCustomBackgroundGreen,255:_inputCustomBackgroundBlue,255:_cbInhibitNURButton,false:_cbInhibitNMPButton,false:_cbInhibitNPURButton,false:_cbHideAMLayer,false:_cbDisablePlacesFiltering,false:_cbDisableTabStyling,false:_cbHideEditorInfo,false:_cbEnableDTE,false[END]';
  1036. defaultSettings += '[UROverviewUROptions][len=1756]:_cbURFilterOutsideArea,false:_cbNoFilterForURInURL,false:_cbFilterWazeAuto,false:_cbFilterIncorrectTurn,false:_cbFilterIncorrectAddress,false:_cbFilterIncorrectRoute,false:_cbFilterMissingRoundabout,false:_cbFilterGeneralError,false:_cbFilterTurnNotAllowed,false:_cbFilterIncorrectJunction,false:_cbFilterMissingBridgeOverpass,false:_cbFilterWrongDrivingDirection,false:_cbFilterMissingExit,false:_cbFilterMissingRoad,false:_cbFilterBlockedRoad,false:_cbFilterMissingLandmark,false:_cbFilterSpeedLimits,false:_cbFilterUndefined,false:_cbFilterRoadworks,false:_cbFilterConstruction,false:_cbFilterClosure,false:_cbFilterEvent,false:_cbFilterNote,false:_cbFilterBOG,false:_cbFilterDifficult,false:_cbFilterWSLM,false:_cbInvertURFilter,false:_cbFilterOpenUR,false:_cbFilterClosedUR,false:_cbFilterSolved,false:_cbFilterUnidentified,false:_cbEnableMinAgeFilter,false:_inputFilterMinDays,:_cbEnableMaxAgeFilter,false:_inputFilterMaxDays,:_cbHideMyFollowed,false:_cbHideMyUnfollowed,false:_cbURDescriptionMustBePresent,false:_cbURDescriptionMustBeAbsent,false:_cbEnableKeywordMustBePresent,false:_textKeywordPresent,:_cbEnableKeywordMustBeAbsent,false:_textKeywordAbsent,:_cbCaseInsensitive,false:_cbHideMyComments,false:_cbHideAnyComments,false:_cbHideIfLastCommenter,false:_cbHideIfNotLastCommenter,false:_cbHideIfReporterLastCommenter,false:_cbHideIfReporterNotLastCommenter,false:_cbEnableMinCommentsFilter,false:_inputFilterMinComments,:_cbEnableMaxCommentsFilter,false:_inputFilterMaxComments,:_cbEnableCommentAgeFilter2,false:_inputFilterCommentDays2,:_cbEnableCommentAgeFilter,false:_inputFilterCommentDays,:_cbIgnoreOtherEditorComments,false:_cbURUserIDFilter,false:_cbURResolverIDFilter,false:_cbInvertURStateFilter,false:_cbNoFilterForTaggedURs,false[END]';
  1037. defaultSettings += '[UROverviewCameraOptions][len=878]:_cbShowWorldCams,true:_cbShowUSACams,true:_cbShowNonWorldCams,true:_cbShowOnlyCamsCreatedBy,false:_cbShowOnlyCamsEditedBy,false:_textCameraEditor,:_cbShowOnlyMyCams,false:_cbShowApprovedCams,true:_cbShowNonApprovedCams,true:_cbShowOlderCreatedNonApproved,false:_inputCameraMinCreatedDays,:_cbShowOlderUpdatedNonApproved,false:_inputCameraMinUpdatedDays,:_cbShowSpeedCams,true:_cbShowIfSpeedSet,true:_cbShowIfNoSpeedSet,true:_cbShowRedLightCams,true:_cbShowDummyCams,true:_cbHideCreatedByMe,false:_cbHideCreatedByRank0,false:_cbHideCreatedByRank1,false:_cbHideCreatedByRank2,false:_cbHideCreatedByRank3,false:_cbHideCreatedByRank4,false:_cbHideCreatedByRank5,false:_cbHideUpdatedByMe,false:_cbHideUpdatedByRank0,false:_cbHideUpdatedByRank1,false:_cbHideUpdatedByRank2,false:_cbHideUpdatedByRank3,false:_cbHideUpdatedByRank4,false:_cbHideUpdatedByRank5,false:_cbHideCWLCams,false[END]';
  1038. defaultSettings += '[UROverviewCamWatchList][len=0][END]';
  1039. defaultSettings += '[UROverviewPlacesOptions][len=5421]:_cbFilterUneditablePlaceUpdates,false:_cbFilterLockRankedPlaceUpdates,false:_cbFilterNewPlacePUR,false:_cbFilterUpdatedDetailsPUR,false:_cbFilterNewPhotoPUR,false:_cbFilterFlaggedPUR,false:_cbLeavePURGeos,false:_cbInvertPURFilters,false:_cbPURFilterLowSeverity,false:_cbPURFilterMediumSeverity,false:_cbPURFilterHighSeverity,false:_cbEnablePURMinAgeFilter,false:_inputPURFilterMinDays,:_cbEnablePURMaxAgeFilter,false:_inputPURFilterMaxDays,:_cbPlaceFilterEditedLessThan,false:_inputFilterPlaceEditMinDays,:_cbPlaceFilterEditedMoreThan,false:_inputFilterPlaceEditMaxDays,:_cbHidePlacesL0,false:_cbHidePlacesL1,false:_cbHidePlacesL2,false:_cbHidePlacesL3,false:_cbHidePlacesL4,false:_cbHidePlacesL5,false:_cbHidePlacesStaff,false:_cbHidePlacesAdLocked,false:_cbHideAreaPlaces,false:_cbHidePointPlaces,false:_cbHidePhotoPlaces,false:_cbHideNoPhotoPlaces,false:_cbHideLinkedPlaces,false:_cbHideNoLinkedPlaces,false:_cbHideKeywordPlaces,false:_cbHideNoKeywordPlaces,false:_textKeywordPlace,:_cbShowOnlyPlacesCreatedBy,false:_cbShowOnlyPlacesEditedBy,false:_textPlacesEditor,:_cbPlacesFilter-CAR_SERVICES,false:_cbPlacesFilter-GAS_STATION,false:_cbPlacesFilter-GARAGE_AUTOMOTIVE_SHOP,false:_cbPlacesFilter-CAR_WASH,false:_cbPlacesFilter-CHARGING_STATION,false:_cbPlacesFilter-TRANSPORTATION,false:_cbPlacesFilter-AIRPORT,false:_cbPlacesFilter-BUS_STATION,false:_cbPlacesFilter-FERRY_PIER,false:_cbPlacesFilter-SEAPORT_MARINA_HARBOR,false:_cbPlacesFilter-SUBWAY_STATION,false:_cbPlacesFilter-TRAIN_STATION,false:_cbPlacesFilter-BRIDGE,false:_cbPlacesFilter-TUNNEL,false:_cbPlacesFilter-TAXI_STATION,false:_cbPlacesFilter-JUNCTION_INTERCHANGE,false:_cbPlacesFilter-PROFESSIONAL_AND_PUBLIC,false:_cbPlacesFilter-COLLEGE_UNIVERSITY,false:_cbPlacesFilter-SCHOOL,false:_cbPlacesFilter-CONVENTIONS_EVENT_CENTER,false:_cbPlacesFilter-GOVERNMENT,false:_cbPlacesFilter-LIBRARY,false:_cbPlacesFilter-CITY_HALL,false:_cbPlacesFilter-ORGANIZATION_OR_ASSOCIATION,false:_cbPlacesFilter-PRISON_CORRECTIONAL_FACILITY,false:_cbPlacesFilter-COURTHOUSE,false:_cbPlacesFilter-CEMETERY,false:_cbPlacesFilter-FIRE_DEPARTMENT,false:_cbPlacesFilter-POLICE_STATION,false:_cbPlacesFilter-MILITARY,false:_cbPlacesFilter-HOSPITAL_MEDICAL_CARE,false:_cbPlacesFilter-OFFICES,false:_cbPlacesFilter-POST_OFFICE,false:_cbPlacesFilter-RELIGIOUS_CENTER,false:_cbPlacesFilter-KINDERGARDEN,false:_cbPlacesFilter-FACTORY_INDUSTRIAL,false:_cbPlacesFilter-EMBASSY_CONSULATE,false:_cbPlacesFilter-INFORMATION_POINT,false:_cbPlacesFilter-SHOPPING_AND_SERVICES,false:_cbPlacesFilter-ARTS_AND_CRAFTS,false:_cbPlacesFilter-BANK_FINANCIAL,false:_cbPlacesFilter-SPORTING_GOODS,false:_cbPlacesFilter-BOOKSTORE,false:_cbPlacesFilter-PHOTOGRAPHY,false:_cbPlacesFilter-CAR_DEALERSHIP,false:_cbPlacesFilter-FASHION_AND_CLOTHING,false:_cbPlacesFilter-CONVENIENCE_STORE,false:_cbPlacesFilter-PERSONAL_CARE,false:_cbPlacesFilter-DEPARTMENT_STORE,false:_cbPlacesFilter-PHARMACY,false:_cbPlacesFilter-ELECTRONICS,false:_cbPlacesFilter-FLOWERS,false:_cbPlacesFilter-FURNITURE_HOME_STORE,false:_cbPlacesFilter-GIFTS,false:_cbPlacesFilter-GYM_FITNESS,false:_cbPlacesFilter-SWIMMING_POOL,false:_cbPlacesFilter-HARDWARE_STORE,false:_cbPlacesFilter-MARKET,false:_cbPlacesFilter-SUPERMARKET_GROCERY,false:_cbPlacesFilter-JEWELRY,false:_cbPlacesFilter-LAUNDRY_DRY_CLEAN,false:_cbPlacesFilter-SHOPPING_CENTER,false:_cbPlacesFilter-MUSIC_STORE,false:_cbPlacesFilter-PET_STORE_VETERINARIAN_SERVICES,false:_cbPlacesFilter-TOY_STORE,false:_cbPlacesFilter-TRAVEL_AGENCY,false:_cbPlacesFilter-ATM,false:_cbPlacesFilter-CURRENCY_EXCHANGE,false:_cbPlacesFilter-CAR_RENTAL,false:_cbPlacesFilter-FOOD_AND_DRINK,false:_cbPlacesFilter-RESTAURANT,false:_cbPlacesFilter-BAKERY,false:_cbPlacesFilter-DESSERT,false:_cbPlacesFilter-CAFE,false:_cbPlacesFilter-FAST_FOOD,false:_cbPlacesFilter-FOOD_COURT,false:_cbPlacesFilter-BAR,false:_cbPlacesFilter-ICE_CREAM,false:_cbPlacesFilter-CULTURE_AND_ENTERTAINEMENT,false:_cbPlacesFilter-ART_GALLERY,false:_cbPlacesFilter-CASINO,false:_cbPlacesFilter-CLUB,false:_cbPlacesFilter-TOURIST_ATTRACTION_HISTORIC_SITE,false:_cbPlacesFilter-MOVIE_THEATER,false:_cbPlacesFilter-MUSEUM,false:_cbPlacesFilter-MUSIC_VENUE,false:_cbPlacesFilter-PERFORMING_ARTS_VENUE,false:_cbPlacesFilter-GAME_CLUB,false:_cbPlacesFilter-STADIUM_ARENA,false:_cbPlacesFilter-THEME_PARK,false:_cbPlacesFilter-ZOO_AQUARIUM,false:_cbPlacesFilter-RACING_TRACK,false:_cbPlacesFilter-THEATER,false:_cbPlacesFilter-OTHER,false:_cbPlacesFilter-CONSTRUCTION_SITE,false:_cbPlacesFilter-LODGING,false:_cbPlacesFilter-HOTEL,false:_cbPlacesFilter-HOSTEL,false:_cbPlacesFilter-CAMPING_TRAILER_PARK,false:_cbPlacesFilter-COTTAGE_CABIN,false:_cbPlacesFilter-BED_AND_BREAKFAST,false:_cbPlacesFilter-OUTDOORS,false:_cbPlacesFilter-PARK,false:_cbPlacesFilter-PLAYGROUND,false:_cbPlacesFilter-BEACH,false:_cbPlacesFilter-SPORTS_COURT,false:_cbPlacesFilter-GOLF_COURSE,false:_cbPlacesFilter-PLAZA,false:_cbPlacesFilter-PROMENADE,false:_cbPlacesFilter-POOL,false:_cbPlacesFilter-SCENIC_LOOKOUT_VIEWPOINT,false:_cbPlacesFilter-SKI_AREA,false:_cbPlacesFilter-NATURAL_FEATURES,false:_cbPlacesFilter-ISLAND,false:_cbPlacesFilter-SEA_LAKE_POOL,false:_cbPlacesFilter-RIVER_STREAM,false:_cbPlacesFilter-FOREST_GROVE,false:_cbPlacesFilter-FARM,false:_cbPlacesFilter-CANAL,false:_cbPlacesFilter-SWAMP_MARSH,false:_cbPlacesFilter-DAM,false:_cbPlacesFilter-PARKING_LOT,false:_cbFilterPrivatePlaces,false:_cbInvertPlacesFilter,false[END]';
  1040. defaultSettings += '[UROverviewCurrentVersion][len=0][END]';
  1041. defaultSettings += '[UROverviewCWLGroups][len=16]0,No group,false[END]';
  1042. document.getElementById('_txtSettings').value = defaultSettings;
  1043. uroTextToSettings();
  1044. document.getElementById('_txtSettings').value = '';
  1045. }
  1046. function uroSettingsToText()
  1047. {
  1048. var txtSettings = '';
  1049.  
  1050. uroSaveSettings();
  1051.  
  1052. for(var lsEntry in localStorage)
  1053. {
  1054. if(lsEntry.indexOf('UROverview') === 0)
  1055. {
  1056. txtSettings += '['+lsEntry+'][len=' + localStorage[lsEntry].length + ']' + localStorage[lsEntry] + '[END]\n';
  1057. }
  1058. }
  1059.  
  1060. document.getElementById('_txtSettings').value = txtSettings;
  1061. document.getElementById('_txtSettings').focus();
  1062. document.getElementById('_txtSettings').select();
  1063. }
  1064. function uroTextToSettings()
  1065. {
  1066. var txtSettings = '';
  1067. txtSettings = uroGetElmValue('_txtSettings');
  1068. if(txtSettings.indexOf('[END]') == -1) return;
  1069.  
  1070. var subText = txtSettings.split('[END]');
  1071. for(var i=0;i<subText.length;i++)
  1072. {
  1073. var aPos = subText[i].indexOf('[');
  1074. var bPos = subText[i].indexOf(']');
  1075. if((aPos != -1) && (bPos != -1))
  1076. {
  1077. var settingID = subText[i].substr(aPos+1,bPos-1-aPos);
  1078. subText[i] = subText[i].substr(bPos+1);
  1079. bPos = subText[i].indexOf(']');
  1080. if(bPos != -1)
  1081. {
  1082. var settingLength = subText[i].substr(5,bPos-5);
  1083. subText[i] = subText[i].substr(bPos+1);
  1084. if(subText[i].length == settingLength)
  1085. {
  1086. localStorage[settingID] = subText[i];
  1087. }
  1088. }
  1089. }
  1090. }
  1091. uroLoadSettings();
  1092. }
  1093. function uroClearSettingsText()
  1094. {
  1095. document.getElementById('_txtSettings').value = '';
  1096. }
  1097.  
  1098.  
  1099. function uroDateToDays(dateToConvert)
  1100. {
  1101. var dateNow = new Date();
  1102.  
  1103. var elapsedSinceEpoch = dateNow.getTime();
  1104. var elapsedSinceEvent = elapsedSinceEpoch - dateToConvert;
  1105.  
  1106. dateNow.setHours(0);
  1107. dateNow.setMinutes(0);
  1108. dateNow.setSeconds(0);
  1109. dateNow.setMilliseconds(0);
  1110.  
  1111. var elapsedSinceMidnight = elapsedSinceEpoch - dateNow.getTime();
  1112.  
  1113. if(elapsedSinceEvent < elapsedSinceMidnight)
  1114. {
  1115. // event occurred today...
  1116. return 0;
  1117. }
  1118. else
  1119. {
  1120. // event occurred at some point prior to midnight this morning, so return a minimum value of 1...
  1121. return 1 + Math.floor((elapsedSinceEvent - elapsedSinceMidnight) / 86400000);
  1122. }
  1123. }
  1124. function uroGetURAge(urObj,ageType,getRaw)
  1125. {
  1126. if(ageType === 0)
  1127. {
  1128. if((urObj.attributes.driveDate === null)||(urObj.attributes.driveDate === 0)) return -1;
  1129. if(getRaw) return urObj.attributes.driveDate;
  1130. else return uroDateToDays(urObj.attributes.driveDate);
  1131. }
  1132. else if(ageType === 1)
  1133. {
  1134. if((urObj.attributes.resolvedOn === null)||(urObj.attributes.resolvedOn === 0)) return -1;
  1135. if(getRaw) return urObj.attributes.resolvedOn;
  1136. else return uroDateToDays(urObj.attributes.resolvedOn);
  1137. }
  1138. else
  1139. {
  1140. return -1;
  1141. }
  1142. }
  1143. function uroGetMCAge(mcObj,ageType,getRaw)
  1144. {
  1145. if(ageType === 0)
  1146. {
  1147. if((mcObj.attributes.createdOn === null)||(mcObj.attributes.createdOn === 0)) return -1;
  1148. if(getRaw) return mcObj.attributes.createdOn;
  1149. else return uroDateToDays(mcObj.attributes.createdOn);
  1150. }
  1151. else if(ageType === 1)
  1152. {
  1153. if((mcObj.attributes.updatedOn === null)||(mcObj.attributes.updatedOn === 0)) return -1;
  1154. if(getRaw) return mcObj.attributes.updatedOn;
  1155. else return uroDateToDays(mcObj.attributes.updatedOn);
  1156. }
  1157. else
  1158. {
  1159. return -1;
  1160. }
  1161. }
  1162. function uroGetPURAge(purObj)
  1163. {
  1164. if(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded !== null)
  1165. {
  1166. return uroDateToDays(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded);
  1167. }
  1168. else
  1169. {
  1170. return -1;
  1171. }
  1172. }
  1173. function uroGetCameraAge(camObj, mode)
  1174. {
  1175. if(mode === 0)
  1176. {
  1177. if(camObj.attributes.updatedOn === null) return -1;
  1178. return uroDateToDays(camObj.attributes.updatedOn);
  1179. }
  1180. if(mode === 1)
  1181. {
  1182. if(camObj.attributes.createdOn === null) return -1;
  1183. return uroDateToDays(camObj.attributes.createdOn);
  1184. }
  1185. }
  1186. function uroGetCommentAge(commentObj)
  1187. {
  1188. if(commentObj.createdOn === null) return -1;
  1189. return uroDateToDays(commentObj.createdOn);
  1190. }
  1191. function uroParseDaysAgo(days)
  1192. {
  1193. if(days === 0) return 'today';
  1194. else if(days === 1) return '1 day ago';
  1195. else return days+' days ago';
  1196. }
  1197. function uroGetLocalisedSpeedString(camSpeed, includeValidity)
  1198. {
  1199. if(camSpeed !== null)
  1200. {
  1201. var conversionFactor = 1; // default to metric
  1202. var multipleFactor = 10; // default to limits being set in multiples of 10
  1203.  
  1204. var country;
  1205. if(W.model.countries.top === undefined)
  1206. {
  1207. country = W.model.countries.additionalInfo[0].name;
  1208. }
  1209. else
  1210. {
  1211. country = W.model.countries.top.name;
  1212. }
  1213. if(country !== null)
  1214. {
  1215. // country-specific deviations from the above...
  1216. if
  1217. (
  1218. (country == "United Kingdom") ||
  1219. (country == "Jersey") ||
  1220. (country == "Guernsey") ||
  1221. (country == "United States")
  1222. )
  1223. {
  1224. // countries using MPH
  1225. conversionFactor = 1.609;
  1226. }
  1227. if
  1228. (
  1229. (country == "United States") ||
  1230. (country == "Guernsey")
  1231. )
  1232. {
  1233. // countries with speed limits set in multiples of 5
  1234. multipleFactor = 5;
  1235. }
  1236. }
  1237.  
  1238. var speed = Math.round(camSpeed / conversionFactor);
  1239. var retval = speed;
  1240. if(conversionFactor == 1) retval += "KM/H";
  1241. else retval += "MPH";
  1242. if(includeValidity === true)
  1243. {
  1244. // special handling for the 7KM/H spielstrasse found in Germany...
  1245. if(country == "Germany")
  1246. {
  1247. if(speed != 7)
  1248. {
  1249. if(speed % multipleFactor !== 0) retval += " (not valid?)";
  1250. }
  1251. }
  1252. else
  1253. {
  1254. if(speed % multipleFactor !== 0) retval += " (not valid?)";
  1255. }
  1256. }
  1257. return retval;
  1258. }
  1259. else return "not set";
  1260. }
  1261.  
  1262.  
  1263. // --------------------------------------------------------------------------------------------------------------------
  1264. // AREA FRIENDLYNAME STUFF
  1265. // --------------------------------------------------------------------------------------------------------------------
  1266. function uroAFNObj(fName, area, server)
  1267. {
  1268. this.fName = fName;
  1269. this.area = area;
  1270. this.server = server;
  1271. }
  1272. function uroUpdateAreaName(name, server, area)
  1273. {
  1274. var foundExisting = false;
  1275. for(var i=0; i<uroFriendlyAreaNames.length; i++)
  1276. {
  1277. if((uroFriendlyAreaNames[i].server == server) && (uroFriendlyAreaNames[i].area == area))
  1278. {
  1279. if(name === "")
  1280. {
  1281. uroFriendlyAreaNames.splice(i,1);
  1282. foundExisting = true;
  1283. }
  1284. else
  1285. {
  1286. uroFriendlyAreaNames[i].fName = name;
  1287. foundExisting = true;
  1288. }
  1289. }
  1290. }
  1291.  
  1292. if((foundExisting === false) && (name !== ""))
  1293. {
  1294. uroFriendlyAreaNames.push(new uroAFNObj(name,area,server));
  1295. }
  1296. uroReplaceAreaNames(true);
  1297. }
  1298. function uroAreaNameHover()
  1299. {
  1300. if((uroAreaNameHoverObj === null) || (uroAreaNameHoverObj != this))
  1301. {
  1302. uroAreaNameHoverTime = 0;
  1303. }
  1304. uroAreaNameHoverObj = this;
  1305. }
  1306. function uroAreaNameUnHover()
  1307. {
  1308. if(uroANEditHovered === true)
  1309. {
  1310. return false;
  1311. }
  1312. if(uroAreaNameOverlayShown)
  1313. {
  1314. uroAreaNameHoverObj.removeChild(uroANEditBox);
  1315. }
  1316. uroAreaNameHoverObj = null;
  1317. uroAreaNameHoverTime = -1;
  1318. uroAreaNameOverlayShown = false;
  1319. }
  1320. function uroANEditHover()
  1321. {
  1322. uroANEditHovered = true;
  1323. uroAddEventListener('uroANEditBox','mouseout',uroANEditUnHover,false);
  1324. uroAddEventListener('uroANEditBox','click',uroANEditClick,false);
  1325. }
  1326. function uroANEditUnHover()
  1327. {
  1328. var newName = document.getElementById('_textAreaName').value;
  1329. // sanitise name to avoid conflicts with config storage delimiters...
  1330. newName = newName.replace(',','');
  1331. newName = newName.replace(':','');
  1332. var server = W.location.code;
  1333. var area = uroGetAreaArea(uroAreaNameHoverObj.parentNode.children[1]);
  1334. uroAreaNameHoverObj.removeChild(uroANEditBox);
  1335. uroAreaNameOverlayShown = false;
  1336. uroANEditHovered = false;
  1337. uroUpdateAreaName(newName, server, area);
  1338. }
  1339. function uroANEditClick(e)
  1340. {
  1341. // this traps the click to prevent it falling through to the underlying area name element and
  1342. // potentially causing the map view to be relocated to that area...
  1343. e.stopPropagation();
  1344. }
  1345. function uroGetAreaArea(listObj)
  1346. {
  1347. var area = listObj.getElementsByTagName('span')[0].innerHTML;
  1348. area = parseFloat(area.split(' ')[0]);
  1349. return area;
  1350. }
  1351. function uroAreaNameOverlaySetup()
  1352. {
  1353. uroAreaNameOverlayShown = true;
  1354.  
  1355. uroANEditBox = document.createElement('div');
  1356. uroANEditBox.id = "uroANEditBox";
  1357. uroANEditBox.style.position = "absolute";
  1358. uroANEditBox.style.top = '7px';
  1359. uroANEditBox.style.left = '2px';
  1360. uroANEditBox.style.width = "99%";
  1361. uroAreaNameHoverObj.appendChild(uroANEditBox);
  1362. uroANEditBox.onmouseover = uroANEditHover();
  1363. var existingName = uroAreaNameHoverObj.innerHTML;
  1364. var italicTagPos = existingName.indexOf(' <i>');
  1365. if(italicTagPos == -1)
  1366. {
  1367. existingName = "";
  1368. }
  1369. else
  1370. {
  1371. existingName = existingName.substr(0,italicTagPos);
  1372. }
  1373. uroANEditBox.innerHTML = '<input type="text" style="font-size:14px; line-height:16px; height:22px; width:100%" id="_textAreaName" value="'+existingName+'">';
  1374. }
  1375.  
  1376. function uroReplaceAreaNames(replaceAfterNameChange)
  1377. {
  1378. if(document.getElementById('sidepanel-areas') === undefined)
  1379. {
  1380. return;
  1381. }
  1382.  
  1383. if(replaceAfterNameChange === false)
  1384. {
  1385. if(document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id == "friendlyNamed")
  1386. {
  1387. return;
  1388. }
  1389. }
  1390.  
  1391. var panelRootObj = document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0];
  1392. if(panelRootObj === undefined)
  1393. {
  1394. // we get here if the user doesn't have any areas defined...
  1395. return;
  1396. }
  1397. var areaCount = panelRootObj.children.length;
  1398. if(areaCount === 0)
  1399. {
  1400. return;
  1401. }
  1402.  
  1403. var localisedManagedArea = I18n.lookup("user.areas.managed_area");
  1404. for(var loop=0; loop < areaCount; loop++)
  1405. {
  1406. var childObjPElems = panelRootObj.children[loop].getElementsByTagName('p');
  1407. var title = childObjPElems[0].innerHTML;
  1408. if(title.indexOf(localisedManagedArea) > -1)
  1409. {
  1410. var area = uroGetAreaArea(childObjPElems[1]);
  1411. childObjPElems[0].innerHTML = localisedManagedArea;
  1412.  
  1413. for(var fnIdx=0; fnIdx < uroFriendlyAreaNames.length; fnIdx++)
  1414. {
  1415. var fnObj = uroFriendlyAreaNames[fnIdx];
  1416. if((fnObj.area == area) && (fnObj.server == W.location.code))
  1417. {
  1418. childObjPElems[0].innerHTML = fnObj.fName +' <i>('+localisedManagedArea+')</i>';
  1419. break;
  1420. }
  1421. }
  1422. var titleObj = panelRootObj.getElementsByClassName('title')[loop];
  1423. titleObj.addEventListener("mouseover", uroAreaNameHover, false);
  1424. titleObj.addEventListener("mouseout", uroAreaNameUnHover, false);
  1425. titleObj.style.cursor = "text";
  1426. }
  1427. }
  1428. document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id = "friendlyNamed";
  1429. }
  1430.  
  1431. // --------------------------------------------------------------------------------------------------------------------
  1432. // WATCHLIST STUFF
  1433. // --------------------------------------------------------------------------------------------------------------------
  1434.  
  1435. // Generic Functions
  1436. function uroTypeCast(varin)
  1437. {
  1438. if(varin == "null") return null;
  1439. if(typeof varin == "string") return parseInt(varin);
  1440. return varin;
  1441. }
  1442. function uroTruncate(val)
  1443. {
  1444. if(val === null) return val;
  1445. if(val < 0) return Math.ceil(val);
  1446. return Math.floor(val);
  1447. }
  1448. function uroOWLGroupObj(groupID, groupName, groupCollapsed)
  1449. {
  1450. groupID = uroTypeCast(groupID);
  1451. this.groupID = groupID;
  1452. this.groupName = groupName;
  1453. this.groupCount = 0;
  1454. this.groupCollapsed = groupCollapsed;
  1455. }
  1456.  
  1457. // Camera Functions
  1458. function uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon)
  1459. {
  1460. if(type !== null) type = uroTypeCast(type);
  1461. if(azymuth !== null) azymuth = uroTruncate(uroTypeCast(azymuth)%360);
  1462. if(speed !== null) speed = uroTruncate(uroTypeCast(speed));
  1463. if(typeof validated == "string") validated = (validated == "true");
  1464. if(lat !== null) lat = uroTruncate(uroTypeCast(lat));
  1465. if(lon !== null) lon = uroTruncate(uroTypeCast(lon));
  1466.  
  1467. this.type = type;
  1468. this.azymuth = azymuth;
  1469. this.speed = speed;
  1470. this.validated = validated;
  1471. this.lat = lat;
  1472. this.lon = lon;
  1473. }
  1474. function uroCamWatchObj(persistent, fid, lon, lat, type, azymuth, speed, validated, groupID, server)
  1475. {
  1476. fid = uroTypeCast(fid);
  1477. groupID = uroTypeCast(groupID);
  1478. if(typeof persistent == "string") persistent = (persistent == "true");
  1479.  
  1480. this.fid = fid;
  1481. this.persistent = persistent;
  1482. this.loaded = false;
  1483. this.server = server;
  1484. this.groupID = groupID;
  1485. this.watch = new uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon);
  1486. this.current = new uroCamWatchObjCheckProps(null, null, null, null, null, null);
  1487. }
  1488. function uroCamDataChanged(idx)
  1489. {
  1490. var camObj = uroCamWatchObjects[idx];
  1491. if(camObj.loaded === false) return false;
  1492. if(camObj.current.type != camObj.watch.type) return true;
  1493. if(camObj.current.azymuth != camObj.watch.azymuth) return true;
  1494. if(camObj.current.speed != camObj.watch.speed) return true;
  1495. if(camObj.current.validated != camObj.watch.validated) return true;
  1496. if(camObj.current.lat != camObj.watch.lat) return true;
  1497. if(camObj.current.lon != camObj.watch.lon) return true;
  1498. return false;
  1499. }
  1500. function uroFindCWLGroupByIdx(groupIdx)
  1501. {
  1502. var groupName = '';
  1503. for(var loop=0;loop<uroCWLGroups.length;loop++)
  1504. {
  1505. if(uroCWLGroups[loop].groupID == groupIdx)
  1506. {
  1507. groupName = uroCWLGroups[loop].groupName;
  1508. break;
  1509. }
  1510. }
  1511. return groupName;
  1512. }
  1513. function uroIsCamOnWatchList(fid)
  1514. {
  1515. for(var loop=0;loop<uroCamWatchObjects.length;loop++)
  1516. {
  1517. if(uroCamWatchObjects[loop].fid == fid) return loop;
  1518. }
  1519. return -1;
  1520. }
  1521. function uroAddCurrentCamWatchData(idx, lat, lon, type, azymuth, speed, validated, server)
  1522. {
  1523. var camObj = uroCamWatchObjects[idx];
  1524. camObj.loaded = true;
  1525. camObj.server = server;
  1526. camObj.current = new uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon);
  1527. return(uroCamDataChanged(idx));
  1528. }
  1529. function uroAddCamToWatchList()
  1530. {
  1531. if(uroIsCamOnWatchList(uroShownFID) == -1)
  1532. {
  1533. var camObj = W.model.cameras.objects[uroShownFID];
  1534. uroCamWatchObjects.push(new uroCamWatchObj(true, uroShownFID, camObj.geometry.x, camObj.geometry.y, camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, 0, W.location.code));
  1535. uroAddCurrentCamWatchData(uroCamWatchObjects.length-1, camObj.geometry.y, camObj.geometry.x, camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, W.location.code);
  1536. uroAddLog('added camera '+uroShownFID+' to watchlist');
  1537. uroOWLUpdateHTML();
  1538. }
  1539. }
  1540. function uroRemoveCamFromWatchList()
  1541. {
  1542. var camidx = uroIsCamOnWatchList(uroShownFID);
  1543. if(camidx != -1)
  1544. {
  1545. uroCamWatchObjects.splice(camidx,1);
  1546. uroAddLog('removed camera '+uroShownFID+' from watchlist');
  1547. uroOWLUpdateHTML();
  1548. }
  1549. }
  1550. function uroUpdateCamWatchList()
  1551. {
  1552. var camIdx = uroIsCamOnWatchList(uroShownFID);
  1553. if(camIdx != -1)
  1554. {
  1555. var camObj = W.model.cameras.objects[uroShownFID];
  1556. uroCamWatchObjects[camIdx].watch = new uroCamWatchObjCheckProps(camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, camObj.geometry.y, camObj.geometry.x);
  1557. }
  1558. }
  1559. function uroClearCamWatchList()
  1560. {
  1561. 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);
  1562. }
  1563. function uroClearCamWatchListAction()
  1564. {
  1565. uroCamWatchObjects = [];
  1566. uroOWLUpdateHTML();
  1567. }
  1568. function uroRetrieveCameras(lat, lon)
  1569. {
  1570. var camPos = new OpenLayers.LonLat();
  1571. var camChanged = false;
  1572.  
  1573. camPos.lon = lon;
  1574. camPos.lat = lat;
  1575. camPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  1576.  
  1577. var camURL = 'https://' + document.location.host;
  1578. camURL += Waze.Config.api_base;
  1579. camURL += '/Features?language=en&cameras=true&bbox=';
  1580. var latl = camPos.lat - 0.25;
  1581. var latu = camPos.lat + 0.25;
  1582. var lonl = camPos.lon - 0.25;
  1583. var lonr = camPos.lon + 0.25;
  1584. camURL += lonl+','+latl+','+lonr+','+latu;
  1585. uroAddLog('retrieving camera data around '+camPos.lon+','+camPos.lat);
  1586.  
  1587. var camReq = new XMLHttpRequest();
  1588. camReq.open('GET',camURL,false);
  1589. try
  1590. {
  1591. camReq.send();
  1592. uroAddLog('response '+camReq.status+' received');
  1593. if (camReq.status === 200)
  1594. {
  1595. var camData = JSON.parse(camReq.responseText);
  1596. for(var camIdx = 0; camIdx < camData.cameras.objects.length; camIdx++)
  1597. {
  1598. var camObj = camData.cameras.objects[camIdx];
  1599. var listIdx = uroIsCamOnWatchList(camObj.id);
  1600. if(listIdx != -1)
  1601. {
  1602. camPos.lon = camObj.geometry.coordinates[0];
  1603. camPos.lat = camObj.geometry.coordinates[1];
  1604. camPos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
  1605. camPos.lon = uroTruncate(camPos.lon);
  1606. camPos.lat = uroTruncate(camPos.lat);
  1607. camChanged |= uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, camObj.validated, W.location.code);
  1608. }
  1609. else if(camObj.validated === false)
  1610. {
  1611.  
  1612. }
  1613. }
  1614. }
  1615. else
  1616. {
  1617. uroAddLog('request failed (status != 200)');
  1618. }
  1619. }
  1620. catch(err)
  1621. {
  1622. uroAddLog('camera load request failed (exception '+err+' caught)');
  1623. }
  1624. return camChanged;
  1625. }
  1626. function uroGetCurrentCamWatchListObjects()
  1627. {
  1628. var camChanged = false;
  1629. var camsChanged = [];
  1630. var camsDeleted = [];
  1631. var camidx;
  1632. var camObj;
  1633. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  1634. {
  1635. camObj = uroCamWatchObjects[camidx];
  1636. if((camObj.loaded === false) && ((camObj.server == W.location.code) || (camObj.server == '??')))
  1637. {
  1638. if(typeof W.model.cameras.objects[camObj.fid] == 'object')
  1639. {
  1640. if(W.model.cameras.objects[camObj.fid].state != "Delete")
  1641. {
  1642. var wazeObj = W.model.cameras.objects[camObj.fid];
  1643. camChanged |= uroAddCurrentCamWatchData(camidx, wazeObj.geometry.y, wazeObj.geometry.x, wazeObj.attributes.type, wazeObj.attributes.azymuth, wazeObj.attributes.speed, wazeObj.attributes.validated);
  1644. }
  1645. else
  1646. {
  1647. camChanged |= uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon);
  1648. }
  1649. }
  1650. else
  1651. {
  1652. camChanged |= uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon);
  1653. }
  1654. }
  1655. }
  1656.  
  1657. if(camChanged)
  1658. {
  1659. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  1660. {
  1661. if(uroCamDataChanged(camidx))
  1662. {
  1663. camsChanged.push(uroCamWatchObjects[camidx]);
  1664. }
  1665. }
  1666. }
  1667.  
  1668. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  1669. {
  1670. camObj = uroCamWatchObjects[camidx];
  1671. if((camObj.loaded === false) && (camObj.server == W.location.code))
  1672. {
  1673. camsDeleted.push(camObj);
  1674. }
  1675. }
  1676.  
  1677. if((camsChanged.length > 0) || (camsDeleted.length > 0))
  1678. {
  1679. var alertStr = '';
  1680. for(camidx=0;camidx<camsChanged.length;camidx++)
  1681. {
  1682. alertStr += 'Camera ID '+camsChanged[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsChanged[camidx].groupID)+'" has been changed<br>';
  1683. }
  1684. alertStr += '<br>';
  1685. for(camidx=0;camidx<camsDeleted.length;camidx++)
  1686. {
  1687. alertStr += 'Camera ID '+camsDeleted[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsDeleted[camidx].groupID)+'" has been deleted<br>';
  1688. }
  1689. uroShowAlertBox("fa-info-circle", "URO+ Camera Watchlist Alert", alertStr, false, "OK", null, null, null);
  1690. }
  1691. }
  1692. function uroClearDeletedCameras()
  1693. {
  1694. for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
  1695. {
  1696. if(uroCamWatchObjects[camidx].loaded === false)
  1697. {
  1698. uroShownFID = uroCamWatchObjects[camidx].fid;
  1699. uroRemoveCamFromWatchList();
  1700. }
  1701. }
  1702. }
  1703. function uroClearUnknownServerCameras()
  1704. {
  1705. var confirmMsg = '<p>Cameras with an unknown server <i>cannot</i> be automatically verified by URO+</p>';
  1706. 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>';
  1707. confirmMsg += 'If the cameras then continue to show up as an unknown server, it is safe to delete them...<br><br>';
  1708. confirmMsg += 'Do you still wish to proceed with deleting all unknown server cameras?';
  1709.  
  1710. uroShowAlertBox("fa-warning", "URO+ Warning", confirmMsg, true, "Delete unknown cameras", "Keep unknown cameras", uroClearUnknownServerCamerasAction, null);
  1711. }
  1712. function uroClearUnknownServerCamerasAction()
  1713. {
  1714. for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
  1715. {
  1716. if(uroCamWatchObjects[camidx].server == '??')
  1717. {
  1718. uroShownFID = uroCamWatchObjects[camidx].fid;
  1719. uroRemoveCamFromWatchList();
  1720. }
  1721. }
  1722. }
  1723. function uroRescanCamWatchList()
  1724. {
  1725. for(var camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  1726. {
  1727. uroCamWatchObjects[camidx].loaded = false;
  1728. }
  1729. uroGetCurrentCamWatchListObjects();
  1730. uroOWLUpdateHTML();
  1731. }
  1732. function uroGotoCam()
  1733. {
  1734. var camidx = this.id.substr(13);
  1735. var camPos = new OpenLayers.LonLat();
  1736. camPos.lon = uroCamWatchObjects[camidx].watch.lon;
  1737. camPos.lat = uroCamWatchObjects[camidx].watch.lat;
  1738. W.map.setCenter(camPos,4);
  1739. W.map.camerasLayer.setVisibility(true);
  1740. return false;
  1741. }
  1742.  
  1743. // Segment Functions
  1744. /*
  1745. function uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn)
  1746. {
  1747. if(left !== null) left = uroTruncate(uroTypeCast(left));
  1748. if(right !== null) right = uroTruncate(uroTypeCast(right));
  1749. if(bottom !== null) bottom = uroTruncate(uroTypeCast(bottom));
  1750. if(top !== null) top = uroTruncate(uroTypeCast(top));
  1751. if(fromNode !== null) fromNode = uroTypeCast(fromNode);
  1752. if(toNode !== null) toNode = uroTypeCast(toNode);
  1753. if(fwdDir !== null) fwdDir = uroTypeCast(fwdDir);
  1754. if(revDir !== null) revDir = uroTypeCast(revDir);
  1755. if(length !== null) length = uroTypeCast(length);
  1756. if(level !== null) level = uroTypeCast(level);
  1757. if(rank !== null) rank = uroTypeCast(rank);
  1758. if(roadType !== null) roadType = uroTypeCast(roadType);
  1759. if(updatedOn !== null) updatedOn = uroTypeCast(updatedOn);
  1760.  
  1761. this.left = left;
  1762. this.right = right;
  1763. this.bottom = bottom;
  1764. this.top = top;
  1765. this.fromNode = fromNode;
  1766. this.toNode = toNode;
  1767. this.fwdDir = fwdDir;
  1768. this.revDir = revDir;
  1769. this.length = length;
  1770. this.level = level;
  1771. this.rank = rank;
  1772. this.roadType = roadType;
  1773. this.updatedOn = updatedOn;
  1774. }
  1775. function uroSegWatchObj(persistent, fid, left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn, groupID, server)
  1776. {
  1777. fid = uroTypeCast(fid);
  1778. groupID = uroTypeCast(groupID);
  1779. if(typeof persistent == "string") persistent = (persistent == "true");
  1780.  
  1781. this.fid = fid;
  1782. this.persistent = persistent;
  1783. this.loaded = false;
  1784. this.server = server;
  1785. this.groupID = groupID;
  1786.  
  1787. this.watch = new uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn);
  1788. this.current = new uroSegWatchObjCheckProps(null, null, null, null, null, null, null, null, null, null, null, null, null);
  1789. }
  1790. function uroSegDataChanged(idx)
  1791. {
  1792. var segObj = uroSegWatchObjects[idx];
  1793. if(segObj.loaded === false) return false;
  1794. if(segObj.current.left != segObj.watch.left) return true;
  1795. if(segObj.current.right != segObj.watch.right) return true;
  1796. if(segObj.current.bottom != segObj.watch.bottom) return true;
  1797. if(segObj.current.top != segObj.watch.top) return true;
  1798. if(segObj.current.fromNode != segObj.watch.fromNode) return true;
  1799. if(segObj.current.toNode != segObj.watch.toNode) return true;
  1800. if(segObj.current.fwdDir != segObj.watch.fwdDir) return true;
  1801. if(segObj.current.revDir != segObj.watch.revDir) return true;
  1802. if(segObj.current.length != segObj.watch.length) return true;
  1803. if(segObj.current.level != segObj.watch.level) return true;
  1804. if(segObj.current.rank != segObj.watch.rank) return true;
  1805. if(segObj.current.roadType != segObj.watch.roadType) return true;
  1806. if(segObj.current.updatedOn != segObj.watch.updatedOn) return true;
  1807. return false;
  1808. }
  1809. function uroIsSegOnWatchList(fid)
  1810. {
  1811. for(var loop=0;loop<uroSegWatchObjects.length;loop++)
  1812. {
  1813. if(uroSegWatchObjects[loop].fid == fid) return loop;
  1814. }
  1815. return -1;
  1816. }
  1817. function uroAddCurrentSegWatchData(idx, left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn, server)
  1818. {
  1819. var segObj = uroSegWatchObjects[idx];
  1820. segObj.loaded = true;
  1821. segObj.server = server;
  1822. segObj.current = new uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn);
  1823. return(uroSegDataChanged(idx));
  1824. }
  1825. function uroClearSegWatchList()
  1826. {
  1827. if(confirm('Removing all segments from the OWL cannot be undone\nAre you sure you want to do this?') === true)
  1828. {
  1829. uroSegWatchObjects = [];
  1830. uroOWLUpdateHTML();
  1831. }
  1832. }
  1833. function uroAddUpdateSegWatchList()
  1834. {
  1835. var selectedCount = W.selectionManager.selectedItems.length;
  1836. if(selectedCount === 0)
  1837. {
  1838. return;
  1839. }
  1840.  
  1841. for(var loop=0;loop < selectedCount; loop++)
  1842. {
  1843. var segObj = W.selectionManager.selectedItems[loop].model.attributes;
  1844. var fid = segObj.id;
  1845. var idx = uroIsSegOnWatchList(fid);
  1846. if(idx != -1)
  1847. {
  1848. 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);
  1849. uroAddLog('updated watchlist details for segment '+fid);
  1850. }
  1851. else
  1852. {
  1853. uroSegWatchObjects.push(new uroSegWatchObj(true, fid, segObj.geometry.bounds.left, segObj.geometry.bounds.right, segObj.geometry.bounds.bottom, segObj.geometry.bounds.top, segObj.fromNodeID, segObj.toNodeID, segObj.fwdDirection, segObj.revDirection, segObj.length, segObj.level, segObj.rank, segObj.roadType, segObj.updatedOn, 0, W.location.code));
  1854. uroAddCurrentSegWatchData(uroSegWatchObjects.length-1, segObj.geometry.bounds.left, segObj.geometry.bounds.right, segObj.geometry.bounds.bottom, segObj.geometry.bounds.top, segObj.fromNodeID, segObj.toNodeID, segObj.fwdDirection, segObj.revDirection, segObj.length, segObj.level, segObj.rank, segObj.roadType, segObj.updatedOn, W.location.code);
  1855. uroAddLog('added segment '+fid+' to watchlist');
  1856. }
  1857. }
  1858. //uroOWLUpdateHTML();
  1859. }
  1860. function uroRemoveSegFromWatchList()
  1861. {
  1862. var selectedCount = W.selectionManager.selectedItems.length;
  1863. if(selectedCount === 0)
  1864. {
  1865. return;
  1866. }
  1867.  
  1868. for(var loop=0;loop < selectedCount; loop++)
  1869. {
  1870. var fid = W.selectionManager.selectedItems[loop].model.attributes.id;
  1871. var idx = uroIsSegOnWatchList(fid);
  1872. if(idx != -1)
  1873. {
  1874. uroSegWatchObjects.splice(idx,1);
  1875. uroAddLog('removed segment '+fid+' from watchlist');
  1876. }
  1877. }
  1878. //uroOWLUpdateHTML();
  1879. }
  1880. function uroRetrieveSegments(lat, lon)
  1881. {
  1882. var pos = new OpenLayers.LonLat();
  1883. var changed = false;
  1884.  
  1885. pos.lon = lon;
  1886. pos.lat = lat;
  1887. pos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  1888.  
  1889. var URL = 'https://' + document.location.host;
  1890. URL += Waze.Config.api_base;
  1891. URL += '/Features?roadTypes=1%2C2%2C3%2C4%2C5%2C6%2C7%2C8%2C9%2C10%2C11%2C12%2C13%2C14%2C15%2C16%2C17%2C18%2C19%2C20%2C21';
  1892. URL += '&bbox=';
  1893. var latl = pos.lat - 0.25;
  1894. var latu = pos.lat + 0.25;
  1895. var lonl = pos.lon - 0.25;
  1896. var lonr = pos.lon + 0.25;
  1897. URL += lonl+','+latl+','+lonr+','+latu;
  1898. URL += '&language=en';
  1899. uroAddLog('retrieving segment data around '+pos.lon+','+pos.lat);
  1900.  
  1901. var req = new XMLHttpRequest();
  1902. req.open('GET',URL,false);
  1903. try
  1904. {
  1905. req.send();
  1906. uroAddLog('response '+req.status+' received');
  1907. if (req.status === 200)
  1908. {
  1909. var data = JSON.parse(req.responseText);
  1910. for(var idx = 0; idx < data.segments.objects.length; idx++)
  1911. {
  1912. var obj = data.segments.objects[idx];
  1913. var listIdx = uroIsSegOnWatchList(obj.id);
  1914. if(listIdx != -1)
  1915. {
  1916. //pos.lon = obj.geometry.coordinates[0];
  1917. //pos.lat = obj.geometry.coordinates[1];
  1918. //pos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
  1919. //camPos.lon = uroTruncate(camPos.lon);
  1920. //camPos.lat = uroTruncate(camPos.lat);
  1921. //camChanged |= uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, camObj.validated, W.location.code);
  1922. }
  1923. else if(obj.validated === false)
  1924. {
  1925.  
  1926. }
  1927. }
  1928. }
  1929. else
  1930. {
  1931. uroAddLog('request failed (status != 200)');
  1932. }
  1933. }
  1934. catch(err)
  1935. {
  1936. uroAddLog('segment load request failed (exception '+err+' caught)');
  1937. }
  1938. return changed;
  1939. }
  1940. function uroGetCurrentSegWatchListObjects()
  1941. {
  1942. var segChanged = false;
  1943. var segsChanged = [];
  1944. var segsDeleted = [];
  1945. var idx;
  1946. var segObj;
  1947.  
  1948. for(idx=0;idx<uroSegWatchObjects.length;idx++)
  1949. {
  1950. segObj = uroSegWatchObjects[idx];
  1951. if((segObj.loaded === false) && ((segObj.server == W.location.code) || (segObj.server == '??')))
  1952. {
  1953. var segLat = (segObj.watch.top + segObj.watch.bottom) / 2;
  1954. var segLon = (segObj.watch.right + segObj.watch.left) / 2;
  1955. if(typeof W.model.segments.objects[segObj.fid] == 'object')
  1956. {
  1957. if(W.model.segments.objects[segObj.fid].state != "Delete")
  1958. {
  1959. var wazeObj = W.model.segments.objects[segObj.fid];
  1960. segChanged |= uroAddCurrentSegWatchData(idx, wazeObj.geometry.bounds.left, wazeObj.geometry.bounds.right, wazeObj.geometry.bounds.bottom, wazeObj.geometry.bounds.top, wazeObj.fromNodeID, wazeObj.toNodeID, wazeObj.fwdDirection, wazeObj.revDirection, wazeObj.length, wazeObj.level, wazeObj.rank, wazeObj.roadType, wazeObj.updatedOn, W.location.code);
  1961. }
  1962. else
  1963. {
  1964. segChanged |= uroRetrieveSegments(segLat, segLon);
  1965. }
  1966. }
  1967. else
  1968. {
  1969. segChanged |= uroRetrieveSegments(segLat, segLon);
  1970. }
  1971. }
  1972. }
  1973.  
  1974. if(segChanged)
  1975. {
  1976. for(idx=0;idx<uroSegWatchObjects.length;idx++)
  1977. {
  1978. if(uroSegDataChanged(idx))
  1979. {
  1980. segsChanged.push(uroSegWatchObjects[idx]);
  1981. }
  1982. }
  1983. }
  1984.  
  1985. for(idx=0;idx<uroSegWatchObjects.length;idx++)
  1986. {
  1987. segObj = uroSegWatchObjects[idx];
  1988. if((segObj.loaded === false) && (segObj.server == W.location.code))
  1989. {
  1990. segsDeleted.push(segObj);
  1991. }
  1992. }
  1993.  
  1994. if((segsChanged.length > 0) || (segsDeleted.length > 0))
  1995. {
  1996. var alertStr = 'Segment WatchList Alert!!!\r\n';
  1997. for(idx=0;idx<segsChanged.length;idx++)
  1998. {
  1999. alertStr += 'Segment ID '+segsChanged[idx].fid+' in group "'+uroFindCWLGroupByIdx(segsChanged[idx].groupID)+'" has been changed\r\n';
  2000. }
  2001. for(idx=0;idx<segsDeleted.length;idx++)
  2002. {
  2003. alertStr += 'Segment ID '+segsDeleted[idx].fid+' in group "'+uroFindCWLGroupByIdx(segsDeleted[idx].groupID)+'" has been deleted\r\n';
  2004. }
  2005. alert(alertStr);
  2006. }
  2007. }
  2008.  
  2009. // Places Functions
  2010. function uroPlaceWatchObjCheckProps(left, right, bottom, top, name, imageCount, residential, updatedOn)
  2011. {
  2012. if(left !== null) left = uroTruncate(uroTypeCast(left));
  2013. if(right !== null) right = uroTruncate(uroTypeCast(right));
  2014. if(bottom !== null) bottom = uroTruncate(uroTypeCast(bottom));
  2015. if(top !== null) top = uroTruncate(uroTypeCast(top));
  2016. if(imageCount !== null) imageCount = uroTypeCast(imageCount);
  2017. if(typeof residential == "string") residential = (residential == "true");
  2018. if(updatedOn !== null) updatedOn = uroTypeCast(updatedOn);
  2019.  
  2020. this.left = left;
  2021. this.right = right;
  2022. this.bottom = bottom;
  2023. this.top = top;
  2024. this.name = name;
  2025. this.imageCount = imageCount;
  2026. this.residential = residential;
  2027. this.updatedOn = updatedOn;
  2028. }
  2029. function uroPlaceWatchObj(persistent, fid, left, right, bottom, top, imageCount, name, residential, updatedOn, groupID, server)
  2030. {
  2031. groupID = uroTypeCast(groupID);
  2032. if(typeof persistent == "string") persistent = (persistent == "true");
  2033.  
  2034. this.fid = fid;
  2035. this.persistent = persistent;
  2036. this.loaded = false;
  2037. this.server = server;
  2038. this.groupID = groupID;
  2039. this.watch = new uroPlaceWatchObjCheckProps(left, right, bottom, top, name, imageCount, residential, updatedOn);
  2040. this.current = new uroPlaceWatchObjCheckProps(null, null, null, null, null, null, null, null);
  2041. }
  2042. function uroPlaceDataChanged(idx)
  2043. {
  2044. var placeObj = uroPlaceWatchObjects[idx];
  2045. if(placeObj.loaded === false) return false;
  2046. if(placeObj.current.left != placeObj.watch.left) return true;
  2047. if(placeObj.current.right != placeObj.watch.right) return true;
  2048. if(placeObj.current.bottom != placeObj.watch.bottom) return true;
  2049. if(placeObj.current.top != placeObj.watch.top) return true;
  2050. if(placeObj.current.name != placeObj.watch.name) return true;
  2051. if(placeObj.current.imageCount != placeObj.watch.imageCount) return true;
  2052. if(placeObj.current.residential != placeObj.watch.residential) return true;
  2053. if(placeObj.current.updatedOn != placeObj.watch.updatedOn) return true;
  2054. return false;
  2055. }
  2056. function uroIsPlaceOnWatchList(fid)
  2057. {
  2058. for(var loop=0;loop<uroPlaceWatchObjects.length;loop++)
  2059. {
  2060. if(uroPlaceWatchObjects[loop].fid == fid) return loop;
  2061. }
  2062. return -1;
  2063. }
  2064. function uroClearPlaceWatchList()
  2065. {
  2066. if(confirm('Removing all places from the OWL cannot be undone\nAre you sure you want to do this?') === true)
  2067. {
  2068. uroPlaceWatchObjects = [];
  2069. uroOWLUpdateHTML();
  2070. }
  2071. }
  2072. */
  2073.  
  2074. function uroHighlightCWLEntry()
  2075. {
  2076. this.style.backgroundColor = '#FFFFAA';
  2077. return false;
  2078. }
  2079. function uroUnhighlightCWLEntry()
  2080. {
  2081. var camidx = this.id.substr(8);
  2082. var changed = uroCamDataChanged(camidx);
  2083. var deleted = (uroCamWatchObjects[camidx].loaded === false);
  2084.  
  2085. if(uroCamWatchObjects[camidx].server != W.location.code)
  2086. {
  2087. if(uroCamWatchObjects[camidx].server == '??') this.style.backgroundColor = '#A0A0A0';
  2088. else this.style.backgroundColor = '#AAFFAA';
  2089. }
  2090. else if(changed) this.style.backgroundColor = '#AAAAFF';
  2091. else if(deleted) this.style.backgroundColor = '#FFAAAA';
  2092. else this.style.backgroundColor = '#FFFFFF';
  2093. return false;
  2094. }
  2095. function uroCWLIconHighlight()
  2096. {
  2097. this.style.color="#0000ff";
  2098. return false;
  2099. }
  2100. function uroCWLIconLowlight()
  2101. {
  2102. this.style.color="#ccccff";
  2103. return false;
  2104. }
  2105. function uroPopulateCWLGroupSelect()
  2106. {
  2107. var selector = document.getElementById('_uroCWLGroupSelect');
  2108. while(selector.options.length > 0)
  2109. {
  2110. selector.options.remove(0);
  2111. }
  2112. for(var loop=0;loop<uroCWLGroups.length;loop++)
  2113. {
  2114. var groupObj = uroCWLGroups[loop];
  2115. if(groupObj.groupID != -1)
  2116. {
  2117. selector.options.add(new Option(groupObj.groupName,groupObj.groupID));
  2118. }
  2119. }
  2120. }
  2121. function uroGetNextCWLGroupID()
  2122. {
  2123. var nextID = 1;
  2124. for(var loop=0;loop<uroCWLGroups.length;loop++)
  2125. {
  2126. if(uroCWLGroups[loop].groupID >= nextID)
  2127. {
  2128. nextID = uroCWLGroups[loop].groupID + 1;
  2129. }
  2130. }
  2131. return nextID;
  2132. }
  2133. function uroFindCWLGroupByName(groupName)
  2134. {
  2135. var groupID = -1;
  2136. for(var loop=0;loop<uroCWLGroups.length;loop++)
  2137. {
  2138. if((uroCWLGroups[loop].groupName == groupName) && (uroCWLGroups[loop].groupID != -1))
  2139. {
  2140. groupID = uroCWLGroups[loop].groupID;
  2141. break;
  2142. }
  2143. }
  2144. return groupID;
  2145. }
  2146. function uroAddCWLGroup()
  2147. {
  2148. var groupID = uroGetNextCWLGroupID();
  2149. var groupName = uroGetElmValue('_uroCWLGroupEntry');
  2150. if(uroFindCWLGroupByName(groupName) == -1)
  2151. {
  2152. uroCWLGroups.push(new uroOWLGroupObj(groupID,groupName,false));
  2153. uroPopulateCWLGroupSelect();
  2154. }
  2155. }
  2156. function uroRemoveCWLGroup()
  2157. {
  2158. var loop;
  2159. var selector = document.getElementById('_uroCWLGroupSelect');
  2160. var groupID = parseInt(selector.selectedOptions[0].value);
  2161. if(groupID === 0) return false; // prevent deletion of the default group
  2162.  
  2163. for(loop=0;loop<uroCamWatchObjects.length;loop++)
  2164. {
  2165. var cwObj = uroCamWatchObjects[loop];
  2166. if(cwObj.groupID == groupID)
  2167. {
  2168. cwObj.groupID = 0;
  2169. }
  2170. }
  2171. for(loop=0;loop<uroCWLGroups.length;loop++)
  2172. {
  2173. var groupObj = uroCWLGroups[loop];
  2174. if(groupObj.groupID == groupID)
  2175. {
  2176. groupObj.groupID = -1;
  2177. }
  2178. }
  2179. uroOWLUpdateHTML();
  2180. }
  2181. function uroAssignCameraToGroup()
  2182. {
  2183. var camidx = this.id.substr(13);
  2184. var selector = document.getElementById('_uroCWLGroupSelect');
  2185. uroCamWatchObjects[camidx].groupID = parseInt(selector.selectedOptions[0].value);
  2186. uroOWLUpdateHTML();
  2187. return false;
  2188. }
  2189. function uroAddBtnEvl(btnID, evlType, evlFunction)
  2190. {
  2191. var btnObj = document.getElementById(btnID);
  2192. if(btnObj !== null)
  2193. {
  2194. btnObj.addEventListener(evlType, evlFunction, true);
  2195. }
  2196. }
  2197. function uroCWLGroupCollapseExpand()
  2198. {
  2199. var groupidx = this.id.substr(18);
  2200. if(uroCWLGroups[groupidx].groupCollapsed === true) uroCWLGroups[groupidx].groupCollapsed = false;
  2201. else uroCWLGroups[groupidx].groupCollapsed = true;
  2202. uroOWLUpdateHTML();
  2203. return false;
  2204. }
  2205.  
  2206. var uroSelectedOWLGroup = null;
  2207. function uroOWLUpdateHTML(doFullUpdate)
  2208. {
  2209. var camTypes = new Array("","","Speed", "Dummy", "Red Light");
  2210. var iHTML = '';
  2211.  
  2212. if(document.getElementById('_uroCWLGroupSelect') !== null)
  2213. {
  2214. uroSelectedOWLGroup = document.getElementById('_uroCWLGroupSelect').selectedIndex;
  2215. }
  2216. iHTML = '<br><b>Camera Watchlist:</b><br><br>';
  2217. iHTML += '<div id="_uroCWLCamList" style="height:65%;overflow:auto;">';
  2218. if(uroCWLGroups.length > 0)
  2219. {
  2220. var camidx;
  2221. for(var groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
  2222. {
  2223. var groupObj = uroCWLGroups[groupidx];
  2224. iHTML += '<div id="_uroCWLGroup-'+groupidx+'">';
  2225. if(groupObj.groupCollapsed === true)
  2226. {
  2227. iHTML += '<i class="fa fa-plus-square-o" style="cursor:pointer;font-size:14px;" id="_uroCWLGroupState-'+groupidx+'"></i>';
  2228. }
  2229. else
  2230. {
  2231. iHTML += '<i class="fa fa-minus-square-o" style="cursor:pointer;font-size:14px;" id="_uroCWLGroupState-'+groupidx+'"></i>';
  2232. }
  2233. iHTML += '<b>'+groupObj.groupName+'</b><br>';
  2234. groupObj.groupCount = 0;
  2235. if(uroCamWatchObjects.length > 0)
  2236. {
  2237. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  2238. {
  2239. var camObj = uroCamWatchObjects[camidx];
  2240. if(camObj.groupID == groupObj.groupID)
  2241. {
  2242. groupObj.groupCount++;
  2243. var changed = uroCamDataChanged(camidx);
  2244. var deleted = (camObj.loaded === false);
  2245. iHTML += '<div id="_uroCWL-'+camidx+'" style="padding:3px;border-width:2px;border-style:solid;border-color:#FFFFFF;background-color:';
  2246. if(camObj.server != W.location.code)
  2247. {
  2248. if(camObj.server == '??') iHTML += '#A0A0A0;';
  2249. else iHTML += '#AAFFAA;';
  2250. }
  2251. else if(changed) iHTML += '#AAAAFF;';
  2252. else if(deleted) iHTML += '#FFAAAA;';
  2253. else iHTML += '#FFFFFF;';
  2254.  
  2255. if(groupObj.groupCollapsed === true) iHTML += 'display:none;">';
  2256. else iHTML += 'display:block;">';
  2257.  
  2258. iHTML += 'ID: '+camObj.fid;
  2259. iHTML += ' ('+camObj.server+')';
  2260. iHTML += ' Type: '+camTypes[camObj.watch.type];
  2261. if(camObj.server != W.location.code)
  2262. {
  2263. if(camObj.server == '??')
  2264. {
  2265. iHTML += '<br><i>Unknown server</i>';
  2266. }
  2267. else
  2268. {
  2269. iHTML += '<br><i>Not on this server</i>';
  2270. }
  2271. }
  2272. else if(deleted)
  2273. {
  2274. iHTML += '<br>DELETED';
  2275. }
  2276. else if(changed)
  2277. {
  2278. if(camObj.current.type != camObj.watch.type)
  2279. {
  2280. iHTML += '<br>&nbsp;&nbsp;Type changed';
  2281. iHTML += ' ('+camObj.watch.type+' to '+camObj.current.type+')';
  2282. }
  2283. if(camObj.current.azymuth != camObj.watch.azymuth)
  2284. {
  2285. iHTML += '<br>&nbsp;&nbsp;Azimuth changed';
  2286. iHTML += ' ('+camObj.watch.azymuth+' to '+camObj.current.azymuth+')';
  2287. }
  2288. if(camObj.current.speed != camObj.watch.speed)
  2289. {
  2290. iHTML += '<br>&nbsp;&nbsp;Speed changed';
  2291. iHTML += ' ('+camObj.watch.speed+' to '+camObj.current.speed+')';
  2292. }
  2293. if(camObj.current.validated != camObj.watch.validated)
  2294. {
  2295. iHTML += '<br>&nbsp;&nbsp;Approval state changed';
  2296. iHTML += ' ('+camObj.watch.validated+' to '+camObj.current.validated+')';
  2297. }
  2298. if(camObj.current.lat != camObj.watch.lat)
  2299. {
  2300. iHTML += '<br>&nbsp;&nbsp;Latitude changed';
  2301. iHTML += ' ('+camObj.watch.lat+' to '+camObj.current.lat+')';
  2302. }
  2303. if(camObj.current.lon != camObj.watch.lon)
  2304. {
  2305. iHTML += '<br>&nbsp;&nbsp;Longitude changed';
  2306. iHTML += ' ('+camObj.watch.lon+' to '+camObj.current.lon+')';
  2307. }
  2308. }
  2309.  
  2310. if(camObj.server == W.location.code)
  2311. {
  2312. if(deleted === false)
  2313. {
  2314. iHTML += '&nbsp;<i class="fa fa-group" style="cursor:pointer;font-size:14px;color:#ccccff;" id="_uroCWLIcon1-'+camidx+'"></i>';
  2315. }
  2316. iHTML += '&nbsp;<i class="fa fa-arrow-circle-right" style="cursor:pointer;font-size:14px;color:#ccccff;" id="_uroCWLIcon2-'+camidx+'"></i>';
  2317. }
  2318. iHTML += '</div>';
  2319. }
  2320. }
  2321. }
  2322. iHTML += '</div>';
  2323. }
  2324. }
  2325. iHTML += '</div><div id="_uroCWLControls">';
  2326. iHTML += '<hr>Group control:<br>';
  2327. iHTML += '<select id="_uroCWLGroupSelect" style="width:40%;height:22px;"></select>&nbsp;<input type="button" id="_btnCWLGroupDel" value="Delete group"><br>';
  2328. iHTML += '<input type="text" id="_uroCWLGroupEntry" style="width:40%;height:22px;">&nbsp;<input type="button" id="_btnCWLGroupAdd" value="Add group">';
  2329. iHTML += '<br><input type="button" id="_btnRescanCamWatchList" value="Refresh Camera Data"><br><br>';
  2330. iHTML += '<b>Remove cameras from OWL:</b><br>';
  2331. iHTML += '<input type="button" id="_btnRemoveDeletedCameras" value="Deleted">&nbsp;&nbsp;';
  2332. iHTML += '<input type="button" id="_btnRemoveUnknownServerCameras" value="Unknown Server">&nbsp;&nbsp;';
  2333. iHTML += '<input type="button" id="_btnClearCamWatchList" value="ALL Cameras">';
  2334. iHTML += '</div>';
  2335. uroOWL.innerHTML = iHTML;
  2336.  
  2337. uroFinaliseOWLHTMLUpdate();
  2338. }
  2339. function uroFinaliseOWLHTMLUpdate()
  2340. {
  2341. if(uroCamWatchObjects.length > 0)
  2342. {
  2343. if(document.getElementById("_uroCWL-0") == null)
  2344. {
  2345. setTimeout(uroFinaliseOWLHTMLUpdate,100);
  2346. return;
  2347. }
  2348. for(var camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  2349. {
  2350. document.getElementById("_uroCWL-"+camidx).onmouseover = uroHighlightCWLEntry;
  2351. document.getElementById("_uroCWL-"+camidx).onmouseleave = uroUnhighlightCWLEntry;
  2352.  
  2353. if(uroCamWatchObjects[camidx].server == W.location.code)
  2354. {
  2355. var icon1 = document.getElementById("_uroCWLIcon1-"+camidx);
  2356. var icon2 = document.getElementById("_uroCWLIcon2-"+camidx);
  2357. if(icon1 !== null)
  2358. {
  2359. icon1.onmouseover = uroCWLIconHighlight;
  2360. icon1.onmouseleave = uroCWLIconLowlight;
  2361. icon1.onclick = uroAssignCameraToGroup;
  2362. }
  2363. if(icon2 !== null)
  2364. {
  2365. icon2.onmouseover = uroCWLIconHighlight;
  2366. icon2.onmouseleave = uroCWLIconLowlight;
  2367. icon2.onclick = uroGotoCam;
  2368. }
  2369. }
  2370. }
  2371. }
  2372.  
  2373. if(document.getElementById('_btnClearCamWatchList') == null)
  2374. {
  2375. setTimeout(uroFinaliseOWLHTMLUpdate,100);
  2376. return;
  2377. }
  2378. uroAddBtnEvl('_btnClearCamWatchList', 'click', uroClearCamWatchList);
  2379. uroAddBtnEvl('_btnRemoveDeletedCameras', 'click', uroClearDeletedCameras);
  2380. uroAddBtnEvl('_btnRemoveUnknownServerCameras', 'click', uroClearUnknownServerCameras);
  2381. uroAddBtnEvl('_btnRescanCamWatchList', 'click', uroRescanCamWatchList);
  2382. uroAddBtnEvl('_btnCWLGroupDel', 'click', uroRemoveCWLGroup);
  2383. uroAddBtnEvl('_btnCWLGroupAdd', 'click', uroAddCWLGroup);
  2384. if(document.getElementById('_uroCWLGroupSelect') !== null)
  2385. {
  2386. uroAddLog('populating CWL group list');
  2387. uroPopulateCWLGroupSelect();
  2388. var selector = document.getElementById('_uroCWLGroupSelect');
  2389. if(uroSelectedOWLGroup >= selector.length)
  2390. {
  2391. uroSelectedOWLGroup = 0;
  2392. }
  2393. selector.selectedIndex = uroSelectedOWLGroup;
  2394. }
  2395.  
  2396. if(uroCWLGroups.length > 0)
  2397. {
  2398. for(var groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
  2399. {
  2400. if(uroCWLGroups[groupidx].groupCount === 0)
  2401. {
  2402. uroSetStyleDisplay('_uroCWLGroup-'+groupidx,'none');
  2403. }
  2404. else
  2405. {
  2406. uroSetOnClick('_uroCWLGroupState-'+groupidx,uroCWLGroupCollapseExpand);
  2407. }
  2408. }
  2409. }
  2410. }
  2411.  
  2412. // --------------------------------------------------------------------------------------------------------------------
  2413. // END OF WATCHLIST STUFF
  2414. // --------------------------------------------------------------------------------------------------------------------
  2415.  
  2416.  
  2417. function uroIsOnIgnoreList(fid)
  2418. {
  2419. if(sessionStorage.UROverview_FID_IgnoreList.indexOf('fid:'+fid) == -1) return false;
  2420. else return true;
  2421. }
  2422. function uroEnableIgnoreListControls()
  2423. {
  2424. var btnState = "visible";
  2425. if(sessionStorage.UROverview_FID_IgnoreList === '')
  2426. {
  2427. btnState = "hidden";
  2428. }
  2429. document.getElementById('_btnUndoLastHide').style.visibility = btnState;
  2430. document.getElementById('_btnClearSessionHides').style.visibility = btnState;
  2431. uroFilterItems();
  2432. }
  2433. function uroAddToIgnoreList()
  2434. {
  2435. if(!uroIsOnIgnoreList(uroShownFID))
  2436. {
  2437. sessionStorage.UROverview_FID_IgnoreList += 'fid:'+uroShownFID;
  2438. uroAddLog('added fid '+uroShownFID+' to ignore list');
  2439. uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
  2440. uroDiv.style.visibility = 'hidden';
  2441. uroEnableIgnoreListControls();
  2442.  
  2443. W.map.events.register("mousemove", null, uroFilterItemsOnMove);
  2444. }
  2445. return false;
  2446. }
  2447. function uroRemoveLastAddedIgnore()
  2448. {
  2449. var ignorelist = sessionStorage.UROverview_FID_IgnoreList;
  2450. var fidpos = ignorelist.lastIndexOf('fid:');
  2451. if(fidpos != -1)
  2452. {
  2453. ignorelist = ignorelist.slice(0,fidpos);
  2454. sessionStorage.UROverview_FID_IgnoreList = ignorelist;
  2455. uroAddLog('removed last fid from ignore list');
  2456. uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
  2457. uroEnableIgnoreListControls();
  2458. }
  2459. }
  2460. function uroRemoveAllIgnores()
  2461. {
  2462. sessionStorage.UROverview_FID_IgnoreList = '';
  2463. uroEnableIgnoreListControls();
  2464. }
  2465. function uroKeywordPresent(desc, keyword)
  2466. {
  2467. var re;
  2468. if(uroGetCBChecked('_cbCaseInsensitive') === true) re = RegExp(keyword,'i');
  2469. else re = RegExp(keyword);
  2470.  
  2471. if(desc.search(re) != -1) return true;
  2472. else return false;
  2473. }
  2474. function uroClickify(desc)
  2475. {
  2476. var linkStartPos = desc.indexOf('http://');
  2477. if(linkStartPos == -1) linkStartPos = desc.indexOf('https://');
  2478. if(linkStartPos != -1)
  2479. {
  2480. var descPreLink = desc.slice(0,linkStartPos);
  2481. var descURL = desc.slice(linkStartPos);
  2482. var linkEndPos = descURL.indexOf(' ');
  2483. var descPostLink = '';
  2484. if(linkEndPos != -1)
  2485. {
  2486. descPostLink = descURL.slice(linkEndPos);
  2487. descURL = descURL.slice(0,linkEndPos);
  2488. }
  2489. var linkTarget = '';
  2490. if(descURL.indexOf('cryosphere') != -1) linkTarget = '_cryosphere';
  2491. else if(descURL.indexOf('waze.com') != -1) linkTarget = '_wazeUR';
  2492. desc = descPreLink + '<a target="'+linkTarget+'" href="'+descURL+'">here</a>' + descPostLink;
  2493. }
  2494. return desc;
  2495. }
  2496. function uroGetUpdateRequestSessions()
  2497. {
  2498. var idList = [];
  2499.  
  2500. while((idList.length < 50) && (uroPendingURSessionIDs.length))
  2501. {
  2502. var id = uroPendingURSessionIDs.shift();
  2503. idList.push(id);
  2504. }
  2505.  
  2506. if(idList.length > 0)
  2507. {
  2508. uroAddLog('grabbing '+idList.length+' updateRequestSessions, IDs: '+idList);
  2509. W.model.updateRequestSessions.get(idList);
  2510. }
  2511.  
  2512. if((uroPendingURSessionIDs.length) || (uroRequestedURSessionIDs.length))
  2513. {
  2514. setTimeout(uroGetUpdateRequestSessions,1000);
  2515. }
  2516. }
  2517. function uroRefreshUpdateRequestSessions()
  2518. {
  2519. var urcount = 0;
  2520. uroPendingURSessionIDs = [];
  2521. uroRequestedURSessionIDs = [];
  2522. for (var urID in W.model.mapUpdateRequests.objects)
  2523. {
  2524. if(W.model.mapUpdateRequests.objects.hasOwnProperty(urID))
  2525. {
  2526. if(W.model.updateRequestSessions.objects[urID] === undefined)
  2527. {
  2528. uroPendingURSessionIDs.push(urID);
  2529. }
  2530. urcount++;
  2531. }
  2532. }
  2533. uroGetUpdateRequestSessions();
  2534. }
  2535. function uroURHasMyComments(fid)
  2536. {
  2537. if(uroUserID === -1) return false;
  2538. var nComments = W.model.updateRequestSessions.objects[fid].comments.length;
  2539. if(nComments === 0) return false;
  2540.  
  2541. for(var cidx=0; cidx<nComments; cidx++)
  2542. {
  2543. if(W.model.updateRequestSessions.objects[fid].comments[cidx].userID == uroUserID) return true;
  2544. }
  2545.  
  2546. return false;
  2547. }
  2548. function uroACMObj(urID, markerType, customType, hasMyComments, nComments)
  2549. {
  2550. this.urID = urID;
  2551. this.markerType = markerType;
  2552. this.customType = customType;
  2553. this.hasMyComments = hasMyComments;
  2554. this.nComments = nComments;
  2555. }
  2556. function uroAddCustomMarkers(urID, markerType, customType, hasMyComments, nComments)
  2557. {
  2558. var useCustomMarker = false;
  2559. if(uroGetCBChecked('_cbMasterEnable') === true)
  2560. {
  2561. if(customType === 0) useCustomMarker = (uroGetCBChecked('_cbCustomRoadworksMarkers'));
  2562. else if(customType === 1) useCustomMarker = (uroGetCBChecked('_cbCustomConstructionMarkers'));
  2563. else if(customType === 2) useCustomMarker = (uroGetCBChecked('_cbCustomClosuresMarkers'));
  2564. else if(customType === 3) useCustomMarker = (uroGetCBChecked('_cbCustomEventsMarkers'));
  2565. else if(customType === 4) useCustomMarker = (uroGetCBChecked('_cbCustomNotesMarkers'));
  2566. else if(customType === 5) useCustomMarker = (uroGetCBChecked('_cbCustomWSLMMarkers'));
  2567. else if(customType === 6) useCustomMarker = (uroGetCBChecked('_cbCustomBOGMarkers'));
  2568. else if(customType === 7) useCustomMarker = (uroGetCBChecked('_cbCustomDifficultMarkers'));
  2569. else if(customType === 98) useCustomMarker = (uroGetCBChecked('_cbCustomNativeSLMarkers'));
  2570. else if(customType === 99) useCustomMarker = (uroGetCBChecked('_cbCustomKeywordMarkers'));
  2571. else if(customType === 100) useCustomMarker = (uroGetCBChecked('_cbCustomElginMarkers'));
  2572. else if(customType === 101) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficCastMarkers'));
  2573. else if(customType === 102) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficMasterMarkers'));
  2574. else if(customType === 103) useCustomMarker = (uroGetCBChecked('_cbCustomCaltransMarkers'));
  2575. else if(customType === 104) useCustomMarker = (uroGetCBChecked('_cbCustomTFLMarkers'));
  2576. }
  2577. if(!useCustomMarker) customType = -1;
  2578. uroCustomMarkerList.push(new uroACMObj(urID, markerType, customType, hasMyComments, nComments));
  2579. }
  2580. function uroRenderCustomMarkers(markerType)
  2581. {
  2582. var urID;
  2583. var elmID;
  2584. var newSpan;
  2585. var divElem;
  2586. var objIdx;
  2587. var customType;
  2588. var cmlObj;
  2589. var defaultMarkerURL = "url('"+document.location.origin + '/assets-editor/sprites/vectors/' + uroNativeMarkerImage+"')";
  2590. var touchedByURO = false;
  2591. if(markerType == 'ur')
  2592. {
  2593. var useDefaultConvoMarker = false;
  2594. var addCommentCount = false;
  2595.  
  2596. if(uroGetCBChecked('_cbMasterEnable') === true)
  2597. {
  2598. if((uroGetCBChecked('_cbNativeConvoMarkers')) && (uroBetaEditor === false)) useDefaultConvoMarker = true;
  2599. if((uroGetCBChecked('_cbNativeBetaConvoMarkers')) && (uroBetaEditor === true)) useDefaultConvoMarker = true;
  2600. if(uroGetCBChecked('_cbCommentCount')) addCommentCount = true;
  2601. }
  2602. else
  2603. {
  2604. useDefaultConvoMarker = true;
  2605. }
  2606.  
  2607. var uRCM_masterEnable = uroGetCBChecked('_cbMasterEnable');
  2608. divElem = document.getElementById(W.map.updateRequestLayer.id);
  2609. if(divElem.childNodes.length > 0)
  2610. {
  2611. for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
  2612. {
  2613. customType = -1;
  2614. cmlObj = uroCustomMarkerList[objIdx];
  2615. if(cmlObj.markerType == 'ur')
  2616. {
  2617. if(uRCM_masterEnable === true)
  2618. {
  2619. customType = cmlObj.customType;
  2620. }
  2621. if(customType < 100)
  2622. {
  2623. urID = cmlObj.urID;
  2624. var nComments = cmlObj.nComments;
  2625. var iconObj = W.map.updateRequestLayer.markers[urID].icon;
  2626. newSpan = '';
  2627.  
  2628. if(nComments !== 0)
  2629. {
  2630. var classList = iconObj.imageDiv.classList;
  2631. elmID = "commentCount_"+urID;
  2632.  
  2633. if(addCommentCount)
  2634. {
  2635. // add a new comment count bubble if the UR doesn't already have one
  2636. if(document.getElementById(elmID) === null)
  2637. {
  2638. newSpan += '<span id="'+elmID+'" style="position:absolute;top:-9px;left:-11px;pointer-events:none;z-index:1">';
  2639. // define the comment-count holding span within the span used to hold the empty bubble image, and before the image is
  2640. // added to the HTML, to avoid z-indexing issues when adjacent comment count bubbles are overlapped...
  2641. newSpan += '<span id="'+elmID+"_inner"+'" style="position:absolute;top:4px;left:11px;font-size:11px;;pointer-events:none"></span>';
  2642. newSpan += '<img src="'+uroMarkers[0]+'">';
  2643. newSpan += '</span>';
  2644. }
  2645. }
  2646. else
  2647. {
  2648. // remove comment count bubble from this UR marker if one has previously been
  2649. // added and the user has now disabled the option...
  2650. if(document.getElementById(elmID) !== null)
  2651. {
  2652. document.getElementById(elmID).remove();
  2653. }
  2654. if(document.getElementById(elmID+"_inner") !== null)
  2655. {
  2656. document.getElementById(elmID+"_inner").remove();
  2657. }
  2658. }
  2659.  
  2660. elmID = "convoMarker_"+urID;
  2661. if(useDefaultConvoMarker === false)
  2662. {
  2663. if(document.getElementById(elmID) === null)
  2664. {
  2665. var hasMyComments = cmlObj.hasMyComments;
  2666. // 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
  2667. // conversation marker, the custom marker remains on top...
  2668. newSpan += '<span id="'+elmID+'" style="position:absolute;top:-9px;left:18px;pointer-events:none;z-index:1">';
  2669. if(hasMyComments) newSpan += '<img src="'+uroMarkers[2]+'">';
  2670. else newSpan += '<img src="'+uroMarkers[1]+'">';
  2671. newSpan += '</span>';
  2672. classList.remove("has-comments");
  2673. }
  2674. }
  2675. else
  2676. {
  2677. // remove custom conversation marker from this UR if one has previously been
  2678. // added and the user has now disabled this option
  2679. if(document.getElementById(elmID) !== null)
  2680. {
  2681. document.getElementById(elmID).remove();
  2682. }
  2683. if(nComments > 0)
  2684. {
  2685. // only replace the native marker class if the UR has comments - if we're just clearing the custom
  2686. // marker following a master enable switchoff, we don't then want to add native markers to URs which
  2687. // didn't have them in the first place...
  2688. classList.add("has-comments");
  2689. }
  2690. }
  2691. }
  2692. // change main marker if required
  2693. touchedByURO = W.map.updateRequestLayer.markers[urID].touchedByURO;
  2694.  
  2695. /*
  2696. //// TEMPORARY REMOVAL UNTIL CUSTOM ICON SUPPORT IS FIXED IN NEW WME
  2697. if(customType != -1)
  2698. {
  2699. if((touchedByURO === undefined) || (touchedByURO === false))
  2700. {
  2701. customType = uroGetCustomMarkerIdx(customType);
  2702. W.map.updateRequestLayer.markers[urID].icon.imageDiv.style.backgroundImage = uroAltMarkers[customType];
  2703. W.map.updateRequestLayer.markers[urID].touchedByURO = true;
  2704. }
  2705. }
  2706. else
  2707. {
  2708. if((touchedByURO === undefined) || (touchedByURO === true))
  2709. {
  2710. W.map.updateRequestLayer.markers[urID].icon.imageDiv.style.backgroundImage = defaultMarkerURL;
  2711. W.map.updateRequestLayer.markers[urID].touchedByURO = false;
  2712. }
  2713. }
  2714. */
  2715. if(newSpan !== '')
  2716. {
  2717. iconObj.$div.prepend(newSpan);
  2718. if(addCommentCount)
  2719. {
  2720. var styleLeft;
  2721. if(nComments < 10) styleLeft = '11px';
  2722. else if(nComments < 100) styleLeft = '8px';
  2723. else styleLeft = '5px';
  2724. elmID = "commentCount_"+urID;
  2725. if(document.getElementById(elmID+"_inner") !== null)
  2726. {
  2727. document.getElementById(elmID+"_inner").innerHTML = nComments;
  2728. document.getElementById(elmID+"_inner").style.left = styleLeft;
  2729. }
  2730. }
  2731. }
  2732. }
  2733. }
  2734. }
  2735. }
  2736. }
  2737.  
  2738. else if(markerType == 'mp')
  2739. {
  2740. divElem = document.getElementById(W.map.problemLayer.id);
  2741. if(divElem.childNodes.length > 0)
  2742. {
  2743. for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
  2744. {
  2745. cmlObj = uroCustomMarkerList[objIdx];
  2746. if(cmlObj.markerType == 'mp')
  2747. {
  2748. customType = cmlObj.customType;
  2749. if((customType >= 100) || (customType == -1))
  2750. {
  2751. urID = cmlObj.urID;
  2752. // change main marker if required
  2753. touchedByURO = W.map.problemLayer.markers[urID].touchedByURO;
  2754. /*
  2755. //// TEMPORARY REMOVAL UNTIL CUSTOM ICON SUPPORT IS FIXED IN NEW WME
  2756. if(customType != -1)
  2757. {
  2758. if((touchedByURO === undefined) || (touchedByURO === false))
  2759. {
  2760. customType = uroGetCustomMarkerIdx(customType);
  2761. W.map.problemLayer.markers[urID].icon.imageDiv.style.backgroundImage = uroAltMarkers[customType];
  2762. W.map.problemLayer.markers[urID].touchedByURO = true;
  2763. }
  2764. }
  2765. else
  2766. {
  2767. if((touchedByURO === undefined) || (touchedByURO === true))
  2768. {
  2769. W.map.problemLayer.markers[urID].icon.imageDiv.style.backgroundImage = defaultMarkerURL;
  2770. W.map.problemLayer.markers[urID].touchedByURO = false;
  2771. }
  2772. }
  2773. */
  2774. }
  2775. }
  2776. }
  2777. }
  2778. }
  2779. }
  2780.  
  2781.  
  2782. function uroFilterPlaces()
  2783. {
  2784. if(uroFilterPreamble() === false) return;
  2785.  
  2786. if(uroPlaceSelected === true) return;
  2787.  
  2788. if(uroGetCBChecked('_cbDisablePlacesFiltering') === true) return;
  2789.  
  2790. uroUpdateVenueEditorList();
  2791.  
  2792. var filterNameID = null;
  2793. var tbUserName = uroGetElmValue('_textPlacesEditor');
  2794. var selector = document.getElementById('_selectPlacesUserID');
  2795. if(selector.selectedIndex > 0)
  2796. {
  2797. var selUserName = document.getElementById('_selectPlacesUserID').selectedOptions[0].innerHTML;
  2798. if(selUserName == tbUserName)
  2799. {
  2800. filterNameID = document.getElementById('_selectPlacesUserID').selectedOptions[0].value;
  2801. }
  2802. }
  2803. if(filterNameID === null)
  2804. {
  2805. var userObj = W.model.users.getByAttributes({userName:tbUserName})[0];
  2806. if(userObj !== undefined)
  2807. {
  2808. filterNameID = userObj.id;
  2809. }
  2810. }
  2811.  
  2812. var filterCats = [];
  2813. for(var i=0; i<W.Config.venues.categories.length; i++)
  2814. {
  2815. var parentCategory = W.Config.venues.categories[i];
  2816. var subCategory;
  2817.  
  2818. if(uroGetCBChecked('_cbPlacesFilter-'+parentCategory) === true)
  2819. {
  2820. filterCats.push(parentCategory);
  2821. for(var i1=0; i1<W.Config.venues.subcategories[parentCategory].length; i1++)
  2822. {
  2823. subCategory = W.Config.venues.subcategories[parentCategory][i1];
  2824. filterCats.push(subCategory);
  2825. }
  2826. }
  2827. else
  2828. {
  2829. for(var i2=0; i2<W.Config.venues.subcategories[parentCategory].length; i2++)
  2830. {
  2831. subCategory = W.Config.venues.subcategories[parentCategory][i2];
  2832. if(uroGetCBChecked('_cbPlacesFilter-'+subCategory) === true)
  2833. {
  2834. filterCats.push(subCategory);
  2835. }
  2836. }
  2837. }
  2838. }
  2839.  
  2840. var placeStyle;
  2841.  
  2842. var uFP_filterEditedLessThan = uroGetCBChecked('_cbPlaceFilterEditedLessThan');
  2843. var uFP_filterEditedMoreThan = uroGetCBChecked('_cbPlaceFilterEditedMoreThan');
  2844. var uFP_filterL0 = uroGetCBChecked('_cbHidePlacesL0');
  2845. var uFP_filterL1 = uroGetCBChecked('_cbHidePlacesL1');
  2846. var uFP_filterL2 = uroGetCBChecked('_cbHidePlacesL2');
  2847. var uFP_filterL3 = uroGetCBChecked('_cbHidePlacesL3');
  2848. var uFP_filterL4 = uroGetCBChecked('_cbHidePlacesL4');
  2849. var uFP_filterL5 = uroGetCBChecked('_cbHidePlacesL5');
  2850. var uFP_filterStaff = uroGetCBChecked('_cbHidePlacesStaff');
  2851. var uFP_filterAL = uroGetCBChecked('_cbHidePlacesAdLocked');
  2852. var uFP_filterOnLockLevel = (uFP_filterL0 || uFP_filterL1 || uFP_filterL2 || uFP_filterL3 || uFP_filterL4 || uFP_filterL5 || uFP_filterStaff);
  2853. var uFP_filterNoPhotos = uroGetCBChecked('_cbHideNoPhotoPlaces');
  2854. var uFP_filterWithPhotos = uroGetCBChecked('_cbHidePhotoPlaces');
  2855. var uFP_filterNoLinks = uroGetCBChecked('_cbHideNoLinkedPlaces');
  2856. var uFP_filterWithLinks = uroGetCBChecked('_cbHideLinkedPlaces');
  2857. var uFP_filterNoKeyword = uroGetCBChecked('_cbHideKeywordPlaces');
  2858. var uFP_filterKeyword = uroGetCBChecked('_cbHideNoKeyeordPlaces');
  2859. var uFP_filterPrivate = uroGetCBChecked('_cbFilterPrivatePlaces');
  2860. var uFP_invertFilters = uroGetCBChecked('_cbInvertPlacesFilter');
  2861. var uFP_masterEnable = uroGetCBChecked('_cbMasterEnable');
  2862. var uFP_filterAreaPlaces = uroGetCBChecked('_cbHideAreaPlaces');
  2863. var uFP_filterPointPlaces = uroGetCBChecked('_cbHidePointPlaces');
  2864. var uFP_filterCreatedBy = uroGetCBChecked('_cbShowOnlyPlacesCreatedBy');
  2865. var uFP_filterEditedBy = uroGetCBChecked('_cbShowOnlyPlacesEditedBy');
  2866. var uFP_NameKeyword = document.getElementById('_textKeywordPlace').value.toLowerCase();
  2867. var uFP_thresholdMinDays = document.getElementById('_inputFilterPlaceEditMinDays').value;
  2868. var uFP_thresholdMaxDays = document.getElementById('_inputFilterPlaceEditMaxDays').value;
  2869. for(var v=0; v<W.map.landmarkLayer.features.length; v++)
  2870. {
  2871. placeStyle = 'visible';
  2872. if(uFP_masterEnable === true)
  2873. {
  2874. var lmObj = W.map.landmarkLayer.features[v];
  2875.  
  2876. // when an area place is selected, the drag points for editing the place outline now get added as objects into W.map.landmarkLayer.features,
  2877. // however none of these objects have the .model property - we must therefore check each entry in features[] to see if it has .model before
  2878. // attempting to filter it...
  2879. if(lmObj.model != null)
  2880. {
  2881. if(lmObj.model.attributes.id < 0)
  2882. {
  2883. // don't apply filtering to newly-created places - this allows the user to leave their filtering settings unchanged whilst
  2884. // adding a new place which, once saved, would then be hidden...
  2885. break;
  2886. }
  2887. if(uFP_filterAreaPlaces)
  2888. {
  2889. if(lmObj.model.attributes.geometry.id.indexOf('Polygon') !== -1)
  2890. {
  2891. placeStyle = 'hidden';
  2892. }
  2893. }
  2894. if(uFP_filterPointPlaces)
  2895. {
  2896. if(lmObj.model.attributes.geometry.id.indexOf('Point') !== -1)
  2897. {
  2898. placeStyle = 'hidden';
  2899. }
  2900. }
  2901.  
  2902. if(placeStyle == 'visible')
  2903. {
  2904. if((uFP_filterEditedLessThan) || (uFP_filterEditedMoreThan))
  2905. {
  2906. var editDate = lmObj.model.attributes.updatedOn;
  2907. if(editDate === undefined)
  2908. {
  2909. // where a place has never been edited since its creation, use the creation date instead...
  2910. editDate = lmObj.model.attributes.createdOn;
  2911. }
  2912. if(editDate != null)
  2913. {
  2914. var editDaysAgo = uroDateToDays(editDate);
  2915. if(uFP_filterEditedLessThan)
  2916. {
  2917. if(editDaysAgo < uFP_thresholdMinDays)
  2918. {
  2919. placeStyle = 'hidden';
  2920. }
  2921. }
  2922. if(uFP_filterEditedMoreThan)
  2923. {
  2924. if(editDaysAgo > uFP_thresholdMaxDays)
  2925. {
  2926. placeStyle = 'hidden';
  2927. }
  2928. }
  2929. }
  2930. }
  2931. }
  2932.  
  2933. if(placeStyle == 'visible')
  2934. {
  2935. if(uFP_filterOnLockLevel)
  2936. {
  2937. var lockLevel = lmObj.model.attributes.lockRank;
  2938. if ((uFP_filterL0) && (lockLevel === 0)) placeStyle = 'hidden';
  2939. if ((uFP_filterL1) && (lockLevel === 1)) placeStyle = 'hidden';
  2940. if ((uFP_filterL2) && (lockLevel === 2)) placeStyle = 'hidden';
  2941. if ((uFP_filterL3) && (lockLevel === 3)) placeStyle = 'hidden';
  2942. if ((uFP_filterL4) && (lockLevel === 4)) placeStyle = 'hidden';
  2943. if ((uFP_filterL5) && (lockLevel === 5)) placeStyle = 'hidden';
  2944. if ((uFP_filterStaff) && (lockLevel === 6)) placeStyle = 'hidden';
  2945. }
  2946. }
  2947.  
  2948. if(placeStyle == 'visible')
  2949. {
  2950. if(uFP_filterAL)
  2951. {
  2952. if(lmObj.model.attributes.adLocked) placeStyle = 'hidden';
  2953. }
  2954. }
  2955.  
  2956. if(placeStyle == 'visible')
  2957. {
  2958. if(uFP_filterNoPhotos || uFP_filterWithPhotos)
  2959. {
  2960. var nPhotos = 0;
  2961. for(var loop=0; loop<lmObj.model.attributes.images.length; loop++)
  2962. {
  2963. if(lmObj.model.attributes.images[loop].attributes.approved) nPhotos++;
  2964. }
  2965. if((uFP_filterNoPhotos) && (nPhotos === 0)) placeStyle = 'hidden';
  2966. if((uFP_filterWithPhotos) && (nPhotos !== 0)) placeStyle = 'hidden';
  2967. }
  2968. }
  2969.  
  2970. if(placeStyle == 'visible')
  2971. {
  2972. if(uFP_filterNoLinks || uFP_filterWithLinks)
  2973. {
  2974. var nLinks = lmObj.model.attributes.externalProviderIDs.length;
  2975. if((uFP_filterNoLinks) && (nLinks === 0)) placeStyle = 'hidden';
  2976. if((uFP_filterWithLinks) && (nLinks !== 0)) placeStyle = 'hidden';
  2977. }
  2978. }
  2979.  
  2980. if(placeStyle == 'visible')
  2981. {
  2982. if((uFP_filterPrivate === true) && (lmObj.model.attributes.residential === true))
  2983. {
  2984. placeStyle = 'hidden';
  2985. }
  2986. else
  2987. {
  2988. for(var cat=0; cat<filterCats.length; cat++)
  2989. {
  2990. if(lmObj.model.attributes.categories.contains(filterCats[cat]))
  2991. {
  2992. placeStyle = 'hidden';
  2993. break;
  2994. }
  2995. }
  2996. }
  2997. }
  2998. if(placeStyle == 'visible')
  2999. {
  3000. if(uFP_filterNoKeyword || uFP_filterKeyword)
  3001. {
  3002. var venueName = lmObj.model.attributes.name.toLowerCase();
  3003. var noKeywordMatch = true;
  3004. if(uFP_NameKeyword === '')
  3005. {
  3006. noKeywordMatch = (venueName !== '');
  3007. }
  3008. else
  3009. {
  3010. noKeywordMatch = (venueName.indexOf(uFP_NameKeyword) === -1);
  3011. }
  3012. if(!noKeywordMatch && uFP_filterNoKeyword) placeStyle = 'hidden';
  3013. if(noKeywordMatch && uFP_filterKeyword) placeStyle = 'hidden';
  3014. }
  3015. }
  3016. if(placeStyle == 'visible')
  3017. {
  3018. if(filterNameID != null)
  3019. {
  3020. if(uFP_filterCreatedBy === true)
  3021. {
  3022. if(filterNameID != lmObj.model.attributes.createdBy) placeStyle = 'hidden';
  3023. }
  3024. if(uFP_filterEditedBy === true)
  3025. {
  3026. if(filterNameID != lmObj.model.attributes.updatedBy) placeStyle = 'hidden';
  3027. }
  3028. }
  3029. }
  3030. }
  3031.  
  3032. if(uFP_invertFilters === true)
  3033. {
  3034. if(placeStyle == 'hidden') placeStyle = 'visible';
  3035. else placeStyle = 'hidden';
  3036. }
  3037. }
  3038.  
  3039. var geoID = W.map.landmarkLayer.features[v].geometry.id;
  3040. if(document.getElementById(geoID) !== null)
  3041. {
  3042. document.getElementById(geoID).style.visibility = placeStyle;
  3043. }
  3044. }
  3045. var uFP_filterUneditable = uroGetCBChecked('_cbFilterUneditablePlaceUpdates');
  3046. var uFP_filterLockRanked = uroGetCBChecked('_cbFilterLockRankedPlaceUpdates');
  3047. var uFP_filterFlagged = uroGetCBChecked("_cbFilterFlaggedPUR");
  3048. var uFP_filterNewPlace = uroGetCBChecked("_cbFilterNewPlacePUR");
  3049. var uFP_filterUpdatedDetails = uroGetCBChecked("_cbFilterUpdatedDetailsPUR");
  3050. var uFP_filterNewPhoto = uroGetCBChecked("_cbFilterNewPhotoPUR");
  3051. var uFP_filterMinPURAge = uroGetCBChecked('_cbEnablePURMinAgeFilter');
  3052. var uFP_filterMaxPURAge = uroGetCBChecked('_cbEnablePURMaxAgeFilter');
  3053. var uFP_invertPURFilters = uroGetCBChecked('_cbInvertPURFilters');
  3054. var uFP_filterHighSeverity = uroGetCBChecked('_cbPURFilterHighSeverity');
  3055. var uFP_filterMedSeverity = uroGetCBChecked('_cbPURFilterMediumSeverity');
  3056. var uFP_filterLowSeverity = uroGetCBChecked('_cbPURFilterLowSeverity');
  3057. var uFP_leavePURGeos = uroGetCBChecked('_cbLeavePURGeos');
  3058. var uFP_thresholdMinPURDays = uroGetElmValue('_inputPURFilterMinDays');
  3059. var uFP_thresholdMaxPURDays = uroGetElmValue('_inputPURFilterMaxDays');
  3060. var uFP_isLoggedIn = W.loginManager.isLoggedIn();
  3061. var uFP_userRank = W.loginManager.user.rank;
  3062. var purAge = null;
  3063. for(var pu in W.map.placeUpdatesLayer.markers)
  3064. {
  3065. if(W.map.placeUpdatesLayer.markers.hasOwnProperty(pu))
  3066. {
  3067. var puObj = W.map.placeUpdatesLayer.markers[pu];
  3068. if(W.map.placeUpdatesLayer.getVisibility() === true)
  3069. {
  3070. placeStyle = 'visible';
  3071. if(uFP_masterEnable === true)
  3072. {
  3073. if(uFP_masterEnable === true)
  3074. {
  3075. if(uFP_filterUneditable === true)
  3076. {
  3077. if(puObj.model.attributes.permissions === 0)
  3078. {
  3079. placeStyle = 'hidden';
  3080. }
  3081. if((placeStyle == 'visible') && (uFP_isLoggedIn))
  3082. {
  3083. if(uFP_userRank < puObj.model.attributes.lockRank)
  3084. {
  3085. placeStyle = 'hidden';
  3086. }
  3087. }
  3088. if((placeStyle == 'visible') && (puObj.model.attributes.adLocked))
  3089. {
  3090. placeStyle = 'hidden';
  3091. }
  3092. }
  3093.  
  3094. if((placeStyle == 'visible') && (uFP_filterLockRanked === true))
  3095. {
  3096. if(puObj.model.attributes.lockRank !== 0)
  3097. {
  3098. placeStyle = 'hidden';
  3099. }
  3100. }
  3101.  
  3102. if((placeStyle == 'visible') && (uFP_filterFlagged === true))
  3103. {
  3104. if(puObj.icon.imageDiv.className.indexOf('flag') != -1)
  3105. {
  3106. placeStyle = 'hidden';
  3107. }
  3108. }
  3109. if((placeStyle == 'visible') && (uFP_filterNewPlace === true))
  3110. {
  3111. if(puObj.icon.imageDiv.className.indexOf('add_venue') != -1)
  3112. {
  3113. placeStyle = 'hidden';
  3114. }
  3115. }
  3116. if((placeStyle == 'visible') && (uFP_filterUpdatedDetails === true))
  3117. {
  3118. if((puObj.icon.imageDiv.className.indexOf('update_venue') != -1) || (puObj.icon.imageDiv.className.indexOf('multiple') != -1))
  3119. {
  3120. placeStyle = 'hidden';
  3121. }
  3122. }
  3123. if((placeStyle == 'visible') && (uFP_filterNewPhoto === true))
  3124. {
  3125. if(puObj.icon.imageDiv.className.indexOf('add_image') != -1)
  3126. {
  3127. placeStyle = 'hidden';
  3128. }
  3129. }
  3130.  
  3131. if(uFP_invertPURFilters === true)
  3132. {
  3133. if(placeStyle == 'hidden') placeStyle = 'visible';
  3134. else placeStyle = 'hidden';
  3135. }
  3136.  
  3137. if(uFP_filterMinPURAge || uFP_filterMaxPURAge)
  3138. {
  3139. purAge = uroGetPURAge(puObj.model);
  3140. if(uFP_filterMinPURAge === true)
  3141. {
  3142. if(purAge < uFP_thresholdMinPURDays) placeStyle = 'hidden';
  3143. }
  3144. if(uFP_filterMaxPURAge === true)
  3145. {
  3146. if(purAge > uFP_thresholdMaxPURDays) placeStyle = 'hidden';
  3147. }
  3148. }
  3149.  
  3150. if(placeStyle == 'visible')
  3151. {
  3152. var purSeverity = puObj._getSeverity();
  3153. if((uFP_filterHighSeverity) && (purSeverity == "high")) placeStyle = 'hidden';
  3154. if((placeStyle == 'visible') && (uFP_filterMedSeverity) && (purSeverity == "medium")) placeStyle = 'hidden';
  3155. if((placeStyle == 'visible') && (uFP_filterLowSeverity) && (purSeverity == "low")) placeStyle = 'hidden';
  3156. }
  3157. }
  3158. }
  3159.  
  3160. puObj.icon.imageDiv.style.visibility = placeStyle;
  3161.  
  3162. if(uFP_leavePURGeos === false)
  3163. {
  3164. if(puObj.model != null)
  3165. {
  3166. if(puObj.model.geometry != null)
  3167. {
  3168. var puGeo = document.getElementById(puObj.model.geometry.id);
  3169. if(puGeo !== null)
  3170. {
  3171. puGeo.style.visibility = placeStyle;
  3172. }
  3173. }
  3174. }
  3175. }
  3176. }
  3177. }
  3178. }
  3179. }
  3180. function uroFilterCameras()
  3181. {
  3182. if(uroFilterPreamble() === false) return;
  3183. var camLayer = document.getElementById(uroRootContainer+'_svgRoot');
  3184. if(camLayer === null)
  3185. {
  3186. if(uroNullCamLayer === false)
  3187. {
  3188. uroAddLog('caught null camLayer');
  3189. uroNullCamLayer = true;
  3190. }
  3191. return;
  3192. }
  3193. uroNullCamLayer = false;
  3194. if(uroMouseIsDown === false) W.map.camerasLayer.redraw();
  3195. if(uroGetCBChecked('_cbMasterEnable') === true)
  3196. {
  3197. uroUpdateCamEditorList();
  3198. var filterNameID = null;
  3199. var tbUserName = uroGetElmValue('_textCameraEditor');
  3200. var selector = document.getElementById('_selectCameraUserID');
  3201. if(selector.selectedIndex > 0)
  3202. {
  3203. var selUserName = document.getElementById('_selectCameraUserID').selectedOptions[0].innerHTML;
  3204. if(selUserName == tbUserName)
  3205. {
  3206. filterNameID = document.getElementById('_selectCameraUserID').selectedOptions[0].value;
  3207. }
  3208. }
  3209. if(filterNameID === null)
  3210. {
  3211. var userObj = W.model.users.getByAttributes({userName:tbUserName})[0];
  3212. if(userObj !== undefined)
  3213. {
  3214. filterNameID = userObj.id;
  3215. }
  3216. }
  3217.  
  3218. for (var uroCamObj in W.model.cameras.objects)
  3219. {
  3220. if(W.model.cameras.objects.hasOwnProperty(uroCamObj))
  3221. {
  3222. var uroCamUpdater = '';
  3223. var uroCamUpdaterRank = -1;
  3224. var uroCamCreator = '';
  3225. var uroCamCreatorRank = -1;
  3226. var uroCam = W.model.cameras.objects[uroCamObj];
  3227. var uroCamStyle = 'visible';
  3228. if(uroCam.attributes.createdBy !== null)
  3229. {
  3230. if(W.model.users.objects[uroCam.attributes.createdBy] != null)
  3231. {
  3232. uroCamCreator = W.model.users.objects[uroCam.attributes.createdBy].userName;
  3233. uroCamCreatorRank = W.model.users.objects[uroCam.attributes.createdBy].rank;
  3234. }
  3235. }
  3236.  
  3237. if(uroCam.attributes.updatedBy !== null)
  3238. {
  3239. if(W.model.users.objects[uroCam.attributes.updatedBy] != null)
  3240. {
  3241. uroCamUpdater = W.model.users.objects[uroCam.attributes.updatedBy].userName;
  3242. uroCamUpdaterRank = W.model.users.objects[uroCam.attributes.updatedBy].rank;
  3243. }
  3244. }
  3245.  
  3246. var uroCamApproved = uroCam.attributes.validated;
  3247. var uroCamType = uroCam.attributes.type;
  3248.  
  3249. if(filterNameID != null)
  3250. {
  3251. if(uroGetCBChecked('_cbShowOnlyCamsCreatedBy') === true)
  3252. {
  3253. if(filterNameID != uroCam.attributes.createdBy) uroCamStyle = 'hidden';
  3254. }
  3255. if(uroGetCBChecked('_cbShowOnlyCamsEditedBy') === true)
  3256. {
  3257. if(filterNameID != uroCam.attributes.updatedBy) uroCamStyle = 'hidden';
  3258. }
  3259. }
  3260. if(uroGetCBChecked('_cbShowOnlyMyCams') === true)
  3261. {
  3262. if((uroUserID != uroCam.attributes.createdBy)&&(uroUserID != uroCam.attributes.updatedBy)) uroCamStyle = 'hidden';
  3263. }
  3264.  
  3265. if((uroGetCBChecked('_cbShowWorldCams') === false) || (uroGetCBChecked('_cbShowUSACams') === false) || (uroGetCBChecked('_cbShowNonWorldCams') === false))
  3266. {
  3267. var posWorld = uroCamCreator.indexOf('world_');
  3268. var posUSA = uroCamCreator.indexOf('usa_');
  3269.  
  3270. if((uroGetCBChecked('_cbShowWorldCams') === false) && (posWorld === 0)) uroCamStyle = 'hidden';
  3271. if((uroGetCBChecked('_cbShowUSACams') === false) && (posUSA === 0)) uroCamStyle = 'hidden';
  3272. if((uroGetCBChecked('_cbShowNonWorldCams') === false) && (posWorld !== 0) && (posUSA !== 0)) uroCamStyle = 'hidden';
  3273. }
  3274.  
  3275. if((uroGetCBChecked('_cbShowApprovedCams') === false) || (uroGetCBChecked('_cbShowNonApprovedCams') === false))
  3276. {
  3277. if((uroGetCBChecked('_cbShowApprovedCams') === false) && (uroCamApproved === true)) uroCamStyle = 'hidden';
  3278. if((uroGetCBChecked('_cbShowNonApprovedCams') === false) && (uroCamApproved === false)) uroCamStyle = 'hidden';
  3279. }
  3280.  
  3281. if((uroGetCBChecked('_cbShowNonApprovedCams') === true) && (uroCamApproved === false))
  3282. {
  3283. if(((uroGetCBChecked('_cbShowOlderCreatedNonApproved') === true)) && (uroGetCameraAge(uroCam,1) <= uroGetElmValue('_inputCameraMinCreatedDays'))) uroCamStyle = 'hidden';
  3284. if(((uroGetCBChecked('_cbShowOlderUpdatedNonApproved') === true)) && (uroGetCameraAge(uroCam,0) <= uroGetElmValue('_inputCameraMinUpdatedDays'))) uroCamStyle = 'hidden';
  3285. }
  3286.  
  3287. if((uroGetCBChecked('_cbShowSpeedCams') === false) || (uroGetCBChecked('_cbShowRedLightCams') === false) || (uroGetCBChecked('_cbShowDummyCams') === false))
  3288. {
  3289. if((uroGetCBChecked('_cbShowSpeedCams') === false) && (uroCamType == 2)) uroCamStyle = 'hidden';
  3290. if((uroGetCBChecked('_cbShowRedLightCams') === false) && (uroCamType == 4)) uroCamStyle = 'hidden';
  3291. if((uroGetCBChecked('_cbShowDummyCams') === false) && (uroCamType == 3)) uroCamStyle = 'hidden';
  3292. }
  3293.  
  3294. if(uroGetCBChecked('_cbShowSpeedCams') === true)
  3295. {
  3296. if((uroGetCBChecked('_cbShowIfNoSpeedSet') === false) && (uroCam.attributes.speed === null)) uroCamStyle = 'hidden';
  3297. if((uroGetCBChecked('_cbShowIfSpeedSet') === false) && (uroCam.attributes.speed !== null)) uroCamStyle = 'hidden';
  3298. }
  3299.  
  3300. if(uroGetCBChecked('_cbHideCreatedByMe') === true)
  3301. {
  3302. if(uroUserID == uroCam.attributes.createdBy) uroCamStyle = 'hidden';
  3303. }
  3304. if((uroGetCBChecked('_cbHideCreatedByRank0') === true) && (uroCamCreatorRank === 0)) uroCamStyle = 'hidden';
  3305. if((uroGetCBChecked('_cbHideCreatedByRank1') === true) && (uroCamCreatorRank == 1)) uroCamStyle = 'hidden';
  3306. if((uroGetCBChecked('_cbHideCreatedByRank2') === true) && (uroCamCreatorRank == 2)) uroCamStyle = 'hidden';
  3307. if((uroGetCBChecked('_cbHideCreatedByRank3') === true) && (uroCamCreatorRank == 3)) uroCamStyle = 'hidden';
  3308. if((uroGetCBChecked('_cbHideCreatedByRank4') === true) && (uroCamCreatorRank == 4)) uroCamStyle = 'hidden';
  3309. if((uroGetCBChecked('_cbHideCreatedByRank5') === true) && (uroCamCreatorRank == 5)) uroCamStyle = 'hidden';
  3310.  
  3311. if(uroGetCBChecked('_cbHideUpdatedByMe') === true)
  3312. {
  3313. if(uroUserID == uroCam.attributes.updatedBy) uroCamStyle = 'hidden';
  3314. }
  3315. if((uroGetCBChecked('_cbHideUpdatedByRank0') === true) && (uroCamUpdaterRank === 0)) uroCamStyle = 'hidden';
  3316. if((uroGetCBChecked('_cbHideUpdatedByRank1') === true) && (uroCamUpdaterRank == 1)) uroCamStyle = 'hidden';
  3317. if((uroGetCBChecked('_cbHideUpdatedByRank2') === true) && (uroCamUpdaterRank == 2)) uroCamStyle = 'hidden';
  3318. if((uroGetCBChecked('_cbHideUpdatedByRank3') === true) && (uroCamUpdaterRank == 3)) uroCamStyle = 'hidden';
  3319. if((uroGetCBChecked('_cbHideUpdatedByRank4') === true) && (uroCamUpdaterRank == 4)) uroCamStyle = 'hidden';
  3320. if((uroGetCBChecked('_cbHideUpdatedByRank5') === true) && (uroCamUpdaterRank == 5)) uroCamStyle = 'hidden';
  3321.  
  3322. if((uroGetCBChecked('_cbHideCWLCams') === true) && (uroIsCamOnWatchList(uroCam.attributes.id) != -1)) uroCamStyle = 'hidden';
  3323.  
  3324. var uroCamGeometryID = uroCam.geometry.id;
  3325. if(camLayer.getElementById(uroCamGeometryID) !== null)
  3326. {
  3327. if(uroCamStyle == "hidden")
  3328. {
  3329. camLayer.getElementById(uroCamGeometryID).remove();
  3330. }
  3331. }
  3332. }
  3333. }
  3334. }
  3335. }
  3336. function uroFilterURs_onObjectsChanged()
  3337. {
  3338. if(uroFilterPreamble())
  3339. {
  3340. if(uroBackfilling === false)
  3341. {
  3342. if(uroURDialogIsOpen === false)
  3343. {
  3344. uroURBackfill();
  3345. }
  3346. else
  3347. {
  3348. uroFilterURs();
  3349. }
  3350. }
  3351. }
  3352. }
  3353. function uroFilterURs_onObjectsAdded()
  3354. {
  3355. if(uroFilterPreamble())
  3356. {
  3357. if(uroBackfilling === false)
  3358. {
  3359. uroURBackfill();
  3360. }
  3361. }
  3362. }
  3363. function uroFilterURs_onObjectsRemoved()
  3364. {
  3365. if(uroFilterPreamble())
  3366. {
  3367. if(uroBackfilling === false)
  3368. {
  3369. uroURBackfill();
  3370. }
  3371. }
  3372. }
  3373. function uroBackfillQueueObj(lon, lat, blockSize)
  3374. {
  3375. this.lon = lon;
  3376. this.lat = lat;
  3377. this.blockSize = blockSize;
  3378. }
  3379. function uroURBackfill_GetData()
  3380. {
  3381. if(uroBackfillQueue.length === 0)
  3382. {
  3383. uroBackfilling = false;
  3384. uroFilterURs();
  3385. return;
  3386. }
  3387. var nextBFQueueObj = uroBackfillQueue.shift();
  3388. var lon = parseFloat(nextBFQueueObj.lon);
  3389. var lat = parseFloat(nextBFQueueObj.lat);
  3390. var blockSize = parseFloat(nextBFQueueObj.blockSize);
  3391. uroAddLog('Backfill square '+lon+','+lat);
  3392. var backfillReq = new XMLHttpRequest();
  3393. backfillReq.onreadystatechange = function ()
  3394. {
  3395. if (backfillReq.readyState == 4)
  3396. {
  3397. uroAddLog('backfill data request, response '+backfillReq.status+' received');
  3398. if (backfillReq.status == 200)
  3399. {
  3400. var tResp = JSON.parse(backfillReq.responseText);
  3401. var urCount = tResp.mapUpdateRequests.objects.length;
  3402.  
  3403. uroAddLog(urCount+' URs loaded for backfill processing');
  3404. if(urCount == 500)
  3405. {
  3406. uroAddLog('WARNING - backfill data may have been pre-filtered by server');
  3407. }
  3408.  
  3409. var backfilled = 0;
  3410. for(var i=0; i<urCount; i++)
  3411. {
  3412. var urID = tResp.mapUpdateRequests.objects[i].id;
  3413. if(W.model.mapUpdateRequests.objects[urID] === undefined)
  3414. {
  3415. var newUR = require('Waze/Feature/Vector/UpdateRequest');
  3416. var tUR = new newUR(tResp.mapUpdateRequests.objects[i]);
  3417. var tPoint = new OpenLayers.Geometry.Point();
  3418. tPoint.x = tResp.mapUpdateRequests.objects[i].geometry.coordinates[0];
  3419. tPoint.y = tResp.mapUpdateRequests.objects[i].geometry.coordinates[1];
  3420. tPoint.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
  3421. tUR.geometry = tPoint;
  3422. var tReqBounds = new OpenLayers.Geometry.Polygon();
  3423. var tBounds = new OpenLayers.Bounds();
  3424. tBounds.left = tPoint.x;
  3425. tBounds.right = tPoint.x;
  3426. tBounds.top = tPoint.y;
  3427. tBounds.bottom = tPoint.y;
  3428. tReqBounds.bounds = tBounds;
  3429. tUR.requestBounds = tReqBounds;
  3430. W.model.mapUpdateRequests.put(tUR);
  3431. backfilled++;
  3432. }
  3433. }
  3434. uroAddLog(backfilled+' URs backfilled');
  3435. }
  3436. uroURBackfill_GetData();
  3437. }
  3438. };
  3439. var tURL = 'https://' + document.location.host;
  3440. tURL += Waze.Config.api_base;
  3441. tURL += '/Features?language=en&mapUpdateRequestFilter=0';
  3442. tURL += '&bbox='+(lon)+','+(lat)+','+(lon + blockSize)+','+(lat + blockSize);
  3443. backfillReq.open('GET',tURL,true);
  3444. backfillReq.send();
  3445. }
  3446. function uroURBackfill()
  3447. {
  3448. if((uroGetCBChecked('_cbURBackfill') === false) || (uroGetCBChecked('_cbMasterEnable') === false))
  3449. {
  3450. uroFilterURs();
  3451. return;
  3452. }
  3453.  
  3454. var nativeURCount = Object.keys(W.model.mapUpdateRequests.objects).length;
  3455. if(nativeURCount < 500)
  3456. {
  3457. uroAddLog(nativeURCount+' URs loaded natively, no backfilling required');
  3458. uroFilterURs();
  3459. return;
  3460. }
  3461.  
  3462. uroAddLog('exactly 500 URs loaded, possible server-side filtering requiring backfill...');
  3463.  
  3464. var subSize = 0.1;
  3465. var vpWidth = W.map.getExtent().getWidth();
  3466. var vpHeight = W.map.getExtent().getHeight();
  3467. var vpCentre = W.map.getCenter();
  3468. var vpLL = new OpenLayers.LonLat();
  3469. var vpUR = new OpenLayers.LonLat();
  3470. vpLL.lon = vpCentre.lon - (vpWidth / 2);
  3471. vpLL.lat = vpCentre.lat - (vpHeight / 2);
  3472. vpUR.lon = vpCentre.lon + (vpWidth / 2);
  3473. vpUR.lat = vpCentre.lat + (vpHeight / 2);
  3474. vpLL = vpLL.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  3475. vpUR = vpUR.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  3476. vpLL.lon -= (subSize / 2);
  3477. vpLL.lat -= (subSize / 2);
  3478. vpUR.lon += (subSize / 2);
  3479. vpUR.lat += (subSize / 2);
  3480. vpLL.lon = +vpLL.lon.toFixed(1);
  3481. vpLL.lat = +vpLL.lat.toFixed(1);
  3482. vpUR.lon = +vpUR.lon.toFixed(1);
  3483. vpUR.lat = +vpUR.lat.toFixed(1);
  3484.  
  3485. uroBackfilling = true;
  3486. uroBackfillQueue = [];
  3487. for(var bfLat = vpLL.lat; bfLat <= vpUR.lat; bfLat += subSize)
  3488. {
  3489. for(var bfLon = vpLL.lon; bfLon <= vpUR.lon; bfLon += subSize)
  3490. {
  3491. uroBackfillQueue.push(new uroBackfillQueueObj(bfLon, bfLat, subSize));
  3492. }
  3493. }
  3494. uroURBackfill_GetData();
  3495. }
  3496. function uroFilterURs()
  3497. {
  3498. if(uroUserID === -1) return;
  3499. // compatibility fix for URComments - based on code supplied by RickZabel
  3500. var hasActiveURFilters = false;
  3501. if(uroGetCBChecked('_cbMasterEnable') === true)
  3502. {
  3503. var urTabInputs = document.getElementById('uroCtrlURs').getElementsByTagName('input');
  3504. for(var loop = 0; loop < urTabInputs.length; loop++)
  3505. {
  3506. if(urTabInputs[loop].type == 'checkbox')
  3507. {
  3508. var ignoreCB = false;
  3509. ignoreCB = ignoreCB || (urTabInputs[loop].id == '_cbCaseInsensitive');
  3510. ignoreCB = ignoreCB || (urTabInputs[loop].id == '_cbNoFilterForTaggedURs');
  3511. if((urTabInputs[loop].checked) && (ignoreCB === false))
  3512. {
  3513. hasActiveURFilters = true;
  3514. break;
  3515. }
  3516. }
  3517. }
  3518. }
  3519. sessionStorage.UROverview_hasActiveURFilters = hasActiveURFilters;
  3520. if(uroFilterPreamble() === false) return;
  3521. uroRefreshUpdateRequestSessions();
  3522. var selectorResolver = document.getElementById('_selectURResolverID');
  3523. var selectorCommentUser = document.getElementById('_selectURUserID');
  3524.  
  3525. if(uroGetCBChecked('_cbURResolverIDFilter') === false)
  3526. {
  3527. while(selectorResolver.options.length > 0)
  3528. {
  3529. selectorResolver.options.remove(0);
  3530. }
  3531. }
  3532. if(uroGetCBChecked('_cbURUserIDFilter') === false)
  3533. {
  3534. while(selectorCommentUser.options.length > 0)
  3535. {
  3536. selectorCommentUser.options.remove(0);
  3537. }
  3538. }
  3539. if(Object.keys(W.model.updateRequestSessions.objects).length === 0)
  3540. {
  3541. return;
  3542. }
  3543. var commenterUser = null;
  3544. if(uroGetCBChecked('_cbURUserIDFilter') === true)
  3545. {
  3546. if(selectorCommentUser.options.length === 0)
  3547. {
  3548. uroUpdateUserList();
  3549. }
  3550. if(selectorCommentUser.selectedOptions[0] != null)
  3551. {
  3552. commenterUser = parseInt(selectorCommentUser.selectedOptions[0].value);
  3553. }
  3554. }
  3555. var resolverUser = null;
  3556. if(uroGetCBChecked('_cbURResolverIDFilter') === true)
  3557. {
  3558. if(selectorResolver.options.length === 0)
  3559. {
  3560. uroUpdateResolverList();
  3561. }
  3562. if(selectorResolver.selectedOptions[0] != null)
  3563. {
  3564. resolverUser = parseInt(selectorResolver.selectedOptions[0].value);
  3565. }
  3566. }
  3567. uroCustomMarkerList = [];
  3568.  
  3569. var uFURs_masterEnable = uroGetCBChecked('_cbMasterEnable');
  3570. var filterOutsideEditableArea = uroGetCBChecked('_cbURFilterOutsideArea');
  3571. var filterSolved = uroGetCBChecked('_cbFilterSolved');
  3572. var filterUnidentified = uroGetCBChecked('_cbFilterUnidentified');
  3573. var filterClosed = uroGetCBChecked('_cbFilterClosedUR');
  3574. var filterOpen = uroGetCBChecked('_cbFilterOpenUR');
  3575. var filterDescMustBePresent = uroGetCBChecked('_cbURDescriptionMustBePresent');
  3576. var filterDescMustBeAbsent = uroGetCBChecked('_cbURDescriptionMustBeAbsent');
  3577. var filterKeywordMustBePresent = uroGetCBChecked('_cbEnableKeywordMustBePresent');
  3578. var filterKeywordMustBeAbsent = uroGetCBChecked('_cbEnableKeywordMustBeAbsent');
  3579. var filterMinURAge = uroGetCBChecked('_cbEnableMinAgeFilter');
  3580. var filterMaxURAge = uroGetCBChecked('_cbEnableMaxAgeFilter');
  3581. var filterMinComments = uroGetCBChecked('_cbEnableMinCommentsFilter');
  3582. var filterMaxComments = uroGetCBChecked('_cbEnableMaxCommentsFilter');
  3583. var filterReporterLastCommenter = uroGetCBChecked('_cbHideIfReporterLastCommenter');
  3584. var filterReporterNotLastCommenter = uroGetCBChecked('_cbHideIfReporterNotLastCommenter');
  3585. var filterHideAnyComments = uroGetCBChecked('_cbHideAnyComments');
  3586. var filterHideNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
  3587. var filterHideMyComments = uroGetCBChecked('_cbHideMyComments');
  3588. var filterIfLastCommenter = uroGetCBChecked('_cbHideIfLastCommenter');
  3589. var filterIfNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
  3590. var filterCommentMinAge = uroGetCBChecked('_cbEnableCommentAgeFilter2');
  3591. var filterCommentMaxAge = uroGetCBChecked('_cbEnableCommentAgeFilter');
  3592. var filterUserID = uroGetCBChecked('_cbURUserIDFilter');
  3593. var filterMyFollowed = uroGetCBChecked('_cbHideMyFollowed');
  3594. var filterMyUnfollowed = uroGetCBChecked('_cbHideMyUnfollowed');
  3595. var filterWazeAuto = uroGetCBChecked('_cbFilterWazeAuto');
  3596. var filterRoadworks = uroGetCBChecked('_cbFilterRoadworks');
  3597. var filterConstruction = uroGetCBChecked('_cbFilterConstruction');
  3598. var filterClosure = uroGetCBChecked('_cbFilterClosure');
  3599. var filterEvent = uroGetCBChecked('_cbFilterEvent');
  3600. var filterNote = uroGetCBChecked('_cbFilterNote');
  3601. var filterWSLM = uroGetCBChecked('_cbFilterWSLM');
  3602. var filterBOG = uroGetCBChecked('_cbFilterBOG');
  3603. var filterDifficult = uroGetCBChecked('_cbFilterDifficult');
  3604. var filterIncorrectTurn = uroGetCBChecked('_cbFilterIncorrectTurn');
  3605. var filterIncorrectAddress = uroGetCBChecked('_cbFilterIncorrectAddress');
  3606. var filterIncorrectRoute = uroGetCBChecked('_cbFilterIncorrectRoute');
  3607. var filterMissingRoundabout = uroGetCBChecked('_cbFilterMissingRoundabout');
  3608. var filterGeneralError = uroGetCBChecked('_cbFilterGeneralError');
  3609. var filterTurnNotAllowed = uroGetCBChecked('_cbFilterTurnNotAllowed');
  3610. var filterIncorrectJunction = uroGetCBChecked('_cbFilterIncorrectJunction');
  3611. var filterMissingBridgeOverpass = uroGetCBChecked('_cbFilterMissingBridgeOverpass');
  3612. var filterWrongDrivingDirection = uroGetCBChecked('_cbFilterWrongDrivingDirection');
  3613. var filterMissingExit = uroGetCBChecked('_cbFilterMissingExit');
  3614. var filterMissingRoad = uroGetCBChecked('_cbFilterMissingRoad');
  3615. var filterMissingLandmark = uroGetCBChecked('_cbFilterMissingLandmark');
  3616. var filterNativeSpeedLimit = uroGetCBChecked('_cbFilterSpeedLimits');
  3617. var filterBlockedRoad = uroGetCBChecked('_cbFilterBlockedRoad');
  3618. var filterUndefined = uroGetCBChecked('_cbFilterUndefined');
  3619. var invertURFilters = uroGetCBChecked('_cbInvertURFilter');
  3620. var invertURStateFilters = uroGetCBChecked('_cbInvertURStateFilter');
  3621. var noFilterTaggedURs = uroGetCBChecked('_cbNoFilterForTaggedURs');
  3622. var noFilterURInURL = uroGetCBChecked('_cbNoFilterForURInURL');
  3623. var keywordPresent = uroGetElmValue('_textKeywordPresent');
  3624. var keywordAbsent = uroGetElmValue('_textKeywordAbsent');
  3625. var thresholdMinAge = uroGetElmValue('_inputFilterMinDays');
  3626. var thresholdMaxAge = uroGetElmValue('_inputFilterMaxDays');
  3627. var thresholdMinComments = uroGetElmValue('_inputFilterMinComments');
  3628. var thresholdMaxComments = uroGetElmValue('_inputFilterMaxComments');
  3629. var thresholdMaxCommentAge = uroGetElmValue('_inputFilterCommentDays');
  3630. var thresholdMinCommentAge = uroGetElmValue('_inputFilterCommentDays2');
  3631. var ignoreOtherEditorComments = uroGetCBChecked('_cbIgnoreOtherEditorComments');
  3632. var urcFilteringIsActive = false;
  3633. var urcCB = document.getElementById('URCommentsFilterEnabled');
  3634. if(urcCB !== null)
  3635. {
  3636. if(urcCB.checked)
  3637. {
  3638. urcFilteringIsActive = true;
  3639. }
  3640. }
  3641. urcCB = document.getElementById('URCommentUROOnlyMyUR');
  3642. if(urcCB !== null)
  3643. {
  3644. if(urcCB.checked)
  3645. {
  3646. urcFilteringIsActive = true;
  3647. }
  3648. }
  3649. urcCB = document.getElementById('URCommentUROHideTagged');
  3650. if(urcCB !== null)
  3651. {
  3652. if(urcCB.checked)
  3653. {
  3654. urcFilteringIsActive = true;
  3655. }
  3656. }
  3657.  
  3658. for (var urobj in W.model.mapUpdateRequests.objects)
  3659. {
  3660. if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
  3661. {
  3662. var ureq = W.model.mapUpdateRequests.objects[urobj];
  3663. var ureqID = null;
  3664. if(ureq.fid === null) ureqID = ureq.attributes.id;
  3665. else ureqID = ureq.fid;
  3666. var urStyle = 'visible';
  3667. var inhibitFiltering = ((ureqID == uroURIDInURL) && (noFilterURInURL));
  3668. var hasMyComments = false;
  3669. var nComments = 0;
  3670. var customType = uroGetCustomType(ureqID, "ur");
  3671. if(W.model.updateRequestSessions.objects[ureqID] != null)
  3672. {
  3673. nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
  3674. if((uFURs_masterEnable === false) && (nComments === 0))
  3675. {
  3676. // when master enable is turned off, we want to make sure that all URs, including ones that were previously hidden, are correctly
  3677. // displayed in their native form - i.e. no comment count or custom conversation bubbles. The easiest way to achieve this is to
  3678. // force the uroRenderCustomMarkers code to test for the presence of these bubbles on each UR, which we do by setting a non-zero
  3679. // comment count for each UR... For URs which genuinely do have no comments we use -1 to indicate that we're not really setting
  3680. // a comment count, but that we still need to do something that wouldn't be achieved by using 0.
  3681. nComments = -1;
  3682. }
  3683. }
  3684. if((uFURs_masterEnable === true) && (inhibitFiltering === false))
  3685. {
  3686. var wazeauto_ur = false;
  3687. var ukroadworks_ur = false;
  3688. var construction_ur = false;
  3689. var closure_ur = false;
  3690. var event_ur = false;
  3691. var note_ur = false;
  3692. var wslm_ur = false;
  3693. var bog_ur = false;
  3694. var difficult_ur = false;
  3695.  
  3696. var filterByNotIncludedKeyword = false;
  3697. var filterByIncludedKeyword = true;
  3698.  
  3699. var desc = '';
  3700. if(ureq.attributes.description !== null) desc = ureq.attributes.description.replace(/<\/?[^>]+(>|$)/g, "");
  3701.  
  3702. if(customType === 0) ukroadworks_ur = true;
  3703. else if(customType === 1) construction_ur = true;
  3704. else if(customType === 2) closure_ur = true;
  3705. else if(customType === 3) event_ur = true;
  3706. else if(customType === 4) note_ur = true;
  3707. else if(customType === 5) wslm_ur = true;
  3708. else if(customType === 6) bog_ur = true;
  3709. else if(customType === 7) difficult_ur = true;
  3710.  
  3711. // check UR against editable area...
  3712.  
  3713. if(filterOutsideEditableArea === true)
  3714. {
  3715. if(ureq.canEdit() === false) urStyle = 'hidden';
  3716. }
  3717. // check UR against current session ignore list...
  3718. if(uroIsOnIgnoreList(ureqID)) urStyle = 'hidden';
  3719. // state-age filtering
  3720. if(urStyle == 'visible')
  3721. {
  3722. // check against closed/not identified filtering if enabled...
  3723. if(filterSolved === true)
  3724. {
  3725. if(ureq.attributes.resolution === 0) urStyle = 'hidden';
  3726. }
  3727. if(filterUnidentified === true)
  3728. {
  3729. if(ureq.attributes.resolution == 1) urStyle = 'hidden';
  3730. }
  3731.  
  3732. if((ureq.attributes.resolvedOn !== null) && (filterClosed === true))
  3733. {
  3734. urStyle = 'hidden';
  3735. }
  3736.  
  3737. if((ureq.attributes.resolvedOn === null) && (filterOpen === true))
  3738. {
  3739. urStyle = 'hidden';
  3740. }
  3741.  
  3742. if(urStyle == 'visible')
  3743. {
  3744. // check UR against keyword filtering if enabled...
  3745. if(filterDescMustBePresent === true)
  3746. {
  3747. if(desc === '') urStyle = 'hidden';
  3748. }
  3749. if(filterDescMustBeAbsent === true)
  3750. {
  3751. if(desc !== '') urStyle = 'hidden';
  3752. }
  3753.  
  3754. if(filterKeywordMustBePresent === true)
  3755. {
  3756. var keywordIsPresentInDesc = uroKeywordPresent(desc,keywordPresent);
  3757. filterByIncludedKeyword &= (!keywordIsPresentInDesc);
  3758. }
  3759. if(filterKeywordMustBeAbsent === true)
  3760. {
  3761. var keywordIsAbsentInDesc = uroKeywordPresent(desc,keywordAbsent);
  3762. filterByNotIncludedKeyword |= keywordIsAbsentInDesc;
  3763. }
  3764. }
  3765.  
  3766. if(urStyle == 'visible')
  3767. {
  3768. // do age-based filtering if enabled
  3769. if(filterMinURAge === true)
  3770. {
  3771. if(uroGetURAge(ureq,0,false) < thresholdMinAge) urStyle = 'hidden';
  3772. }
  3773. if(filterMaxURAge === true)
  3774. {
  3775. if(uroGetURAge(ureq,0,false) > thresholdMaxAge) urStyle = 'hidden';
  3776. }
  3777. }
  3778.  
  3779. if(urStyle == 'visible')
  3780. {
  3781. if(resolverUser !== null)
  3782. {
  3783. if(ureq.attributes.resolvedBy != resolverUser) urStyle = 'hidden';
  3784. }
  3785. }
  3786.  
  3787. if(urStyle == 'visible')
  3788. {
  3789. // do comments/following filtering
  3790. if(W.model.updateRequestSessions.objects[ureqID] != null)
  3791. {
  3792. nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
  3793. var commentDaysOld = -1;
  3794.  
  3795.  
  3796. if(filterMinComments === true)
  3797. {
  3798. if(nComments < thresholdMinComments) urStyle = 'hidden';
  3799. }
  3800. if(filterMaxComments === true)
  3801. {
  3802. if(nComments > thresholdMaxComments) urStyle = 'hidden';
  3803. }
  3804.  
  3805.  
  3806. if(nComments > 0)
  3807. {
  3808. var reporterIsLastCommenter = false;
  3809. if(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1].userID == -1) reporterIsLastCommenter = true;
  3810.  
  3811. if(filterReporterLastCommenter === true)
  3812. {
  3813. if(reporterIsLastCommenter === true) urStyle = 'hidden';
  3814. }
  3815. else if(filterReporterNotLastCommenter === true)
  3816. {
  3817. if(reporterIsLastCommenter === false) urStyle = 'hidden';
  3818. }
  3819.  
  3820. hasMyComments = uroURHasMyComments(ureqID);
  3821. if(hasMyComments === false)
  3822. {
  3823. if(filterHideAnyComments === true) urStyle = 'hidden';
  3824. if(filterHideNotLastCommenter === true) urStyle = 'hidden';
  3825. }
  3826. else
  3827. {
  3828. if(filterHideMyComments === true) urStyle = 'hidden';
  3829.  
  3830. var userIsLastCommenter = false;
  3831. if(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1].userID == uroUserID) userIsLastCommenter = true;
  3832.  
  3833. if(filterIfLastCommenter === true)
  3834. {
  3835. if(userIsLastCommenter === true) urStyle = 'hidden';
  3836. }
  3837. else if(filterIfNotLastCommenter === true)
  3838. {
  3839. if(userIsLastCommenter === false) urStyle = 'hidden';
  3840. }
  3841. }
  3842. var cidx;
  3843. if(ignoreOtherEditorComments === false)
  3844. {
  3845. commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1]);
  3846. }
  3847. else
  3848. {
  3849. for(cidx=0; cidx<nComments; cidx++)
  3850. {
  3851. var cObj = W.model.updateRequestSessions.objects[ureqID].comments[cidx];
  3852. if((cObj.userID == uroUserID) || (cObj.userID == -1))
  3853. {
  3854. commentDaysOld = uroGetCommentAge(cObj);
  3855. }
  3856. }
  3857. }
  3858. if((filterCommentMinAge === true) && (commentDaysOld != -1))
  3859. {
  3860. if(thresholdMinCommentAge > commentDaysOld) urStyle = 'hidden';
  3861. }
  3862. if((filterCommentMaxAge === true) && (commentDaysOld != -1))
  3863. {
  3864. if(thresholdMaxCommentAge < commentDaysOld) urStyle = 'hidden';
  3865. }
  3866. if((commenterUser !== null) && (urStyle != 'hidden'))
  3867. {
  3868. urStyle = 'hidden';
  3869. for(cidx=0; cidx<nComments; cidx++)
  3870. {
  3871. if(W.model.updateRequestSessions.objects[ureqID].comments[cidx].userID == commenterUser)
  3872. {
  3873. urStyle = 'visible';
  3874. break;
  3875. }
  3876. }
  3877. }
  3878.  
  3879. var commentText = '';
  3880. for(cidx=0; cidx<nComments; cidx++)
  3881. {
  3882. commentText += W.model.updateRequestSessions.objects[ureqID].comments[cidx].text;
  3883. }
  3884.  
  3885. if(filterKeywordMustBePresent === true)
  3886. {
  3887. var keywordIsPresentInComments = uroKeywordPresent(commentText,keywordPresent);
  3888. filterByIncludedKeyword &= (!keywordIsPresentInComments);
  3889. }
  3890. if(filterKeywordMustBeAbsent === true)
  3891. {
  3892. var keywordIsAbsentInComments = uroKeywordPresent(commentText,keywordAbsent);
  3893. filterByNotIncludedKeyword |= keywordIsAbsentInComments;
  3894. }
  3895. }
  3896. else
  3897. {
  3898. if(filterUserID === true)
  3899. {
  3900. urStyle = 'hidden';
  3901. }
  3902. }
  3903.  
  3904. filterByNotIncludedKeyword &= filterKeywordMustBeAbsent;
  3905. filterByIncludedKeyword &= filterKeywordMustBePresent;
  3906. if(filterByNotIncludedKeyword || filterByIncludedKeyword)
  3907. {
  3908. urStyle = 'hidden';
  3909. }
  3910.  
  3911. if(W.model.updateRequestSessions.objects[ureqID].isFollowing === true)
  3912. {
  3913. if(filterMyFollowed === true) urStyle = 'hidden';
  3914. }
  3915. else
  3916. {
  3917. if(filterMyUnfollowed === true) urStyle = 'hidden';
  3918. }
  3919. }
  3920. }
  3921.  
  3922. if(invertURStateFilters === true)
  3923. {
  3924. if(urStyle == 'hidden') urStyle = 'visible';
  3925. else urStyle = 'hidden';
  3926. }
  3927. }
  3928.  
  3929. // type filtering
  3930. if(urStyle == 'visible')
  3931. {
  3932. // Test for Waze automatic URs before any others - these always (?) get inserted as General Error URs,
  3933. // so we can't filter them by type...
  3934. if(desc.indexOf('Waze Automatic:') != -1)
  3935. {
  3936. wazeauto_ur = true;
  3937. }
  3938.  
  3939. if(wazeauto_ur === true)
  3940. {
  3941. if(filterWazeAuto === true) urStyle = 'hidden';
  3942. }
  3943.  
  3944. else if(ukroadworks_ur === true)
  3945. {
  3946. if(filterRoadworks === true) urStyle = 'hidden';
  3947. }
  3948. else if(construction_ur === true)
  3949. {
  3950. if(filterConstruction === true) urStyle = 'hidden';
  3951. }
  3952. else if(closure_ur === true)
  3953. {
  3954. if(filterClosure === true) urStyle = 'hidden';
  3955. }
  3956. else if(event_ur === true)
  3957. {
  3958. if(filterEvent === true) urStyle = 'hidden';
  3959. }
  3960. else if(note_ur === true)
  3961. {
  3962. if(filterNote === true) urStyle = 'hidden';
  3963. }
  3964. else if(wslm_ur === true)
  3965. {
  3966. if(filterWSLM === true) urStyle = 'hidden';
  3967. }
  3968. else if(bog_ur === true)
  3969. {
  3970. if(filterBOG === true) urStyle = 'hidden';
  3971. }
  3972. else if(difficult_ur === true)
  3973. {
  3974. if(filterDifficult === true) urStyle = 'hidden';
  3975. }
  3976.  
  3977. else if(ureq.attributes.type == 6)
  3978. {
  3979. if(filterIncorrectTurn === true) urStyle = 'hidden';
  3980. }
  3981. else if(ureq.attributes.type == 7)
  3982. {
  3983. if (filterIncorrectAddress === true) urStyle = 'hidden';
  3984. }
  3985. else if(ureq.attributes.type == 8)
  3986. {
  3987. if(filterIncorrectRoute === true) urStyle = 'hidden';
  3988. }
  3989. else if(ureq.attributes.type == 9)
  3990. {
  3991. if(filterMissingRoundabout === true) urStyle = 'hidden';
  3992. }
  3993. else if(ureq.attributes.type == 10)
  3994. {
  3995. if(filterGeneralError === true) urStyle = 'hidden';
  3996. }
  3997. else if(ureq.attributes.type == 11)
  3998. {
  3999. if(filterTurnNotAllowed === true) urStyle = 'hidden';
  4000. }
  4001. else if(ureq.attributes.type == 12)
  4002. {
  4003. if(filterIncorrectJunction === true) urStyle = 'hidden';
  4004. }
  4005. else if(ureq.attributes.type == 13)
  4006. {
  4007. if(filterMissingBridgeOverpass === true) urStyle = 'hidden';
  4008. }
  4009. else if(ureq.attributes.type == 14)
  4010. {
  4011. if(filterWrongDrivingDirection === true) urStyle = 'hidden';
  4012. }
  4013. else if(ureq.attributes.type == 15)
  4014. {
  4015. if(filterMissingExit === true) urStyle = 'hidden';
  4016. }
  4017. else if(ureq.attributes.type == 16)
  4018. {
  4019. if(filterMissingRoad === true) urStyle = 'hidden';
  4020. }
  4021. else if(ureq.attributes.type == 18)
  4022. {
  4023. if(filterMissingLandmark === true) urStyle = 'hidden';
  4024. }
  4025. else if(ureq.attributes.type == 19)
  4026. {
  4027. if(filterBlockedRoad === true) urStyle = 'hidden';
  4028. }
  4029. else if(ureq.attributes.type == 23)
  4030. {
  4031. if(filterNativeSpeedLimit === true) urStyle = 'hidden';
  4032. }
  4033. else if(filterUndefined === true) urStyle = 'hidden';
  4034.  
  4035. if(invertURFilters === true)
  4036. {
  4037. if(urStyle == 'hidden') urStyle = 'visible';
  4038. else urStyle = 'hidden';
  4039. }
  4040. }
  4041.  
  4042. // stage-age filtering override for tagged URs
  4043. if(noFilterTaggedURs === true)
  4044. {
  4045. if(ukroadworks_ur === true)
  4046. {
  4047. if(filterRoadworks === false) urStyle = 'visible';
  4048. }
  4049. else if(construction_ur === true)
  4050. {
  4051. if(filterConstruction === false) urStyle = 'visible';
  4052. }
  4053. else if(closure_ur === true)
  4054. {
  4055. if(filterClosure === false) urStyle = 'visible';
  4056. }
  4057. else if(event_ur === true)
  4058. {
  4059. if(filterEvent === false) urStyle = 'visible';
  4060. }
  4061. else if(note_ur === true)
  4062. {
  4063. if(filterNote === false) urStyle = 'visible';
  4064. }
  4065. else if(wslm_ur === true)
  4066. {
  4067. if(filterWSLM === false) urStyle = 'visible';
  4068. }
  4069. }
  4070. }
  4071. // only touch marker visibility if we've got active filter settings, or if URComments is not
  4072. // doing any filtering of its own
  4073. if((hasActiveURFilters === true) || (urcFilteringIsActive === false) || (uFURs_masterEnable === false))
  4074. {
  4075. W.map.updateRequestLayer.markers[urobj].icon.imageDiv.style.visibility = urStyle;
  4076. }
  4077. if(urStyle != 'hidden')
  4078. {
  4079. uroAddCustomMarkers(ureqID,'ur',customType, hasMyComments, nComments);
  4080. }
  4081. }
  4082. }
  4083. uroRenderCustomMarkers('ur');
  4084. }
  4085. function uroGetProblemTypes()
  4086. {
  4087. uroKnownProblemTypeIDs = [];
  4088. uroKnownProblemTypeNames = [];
  4089. var tProblemList = I18n.lookup("problems.types");
  4090. for(var tObj in tProblemList)
  4091. {
  4092. if(tObj !== undefined)
  4093. {
  4094. uroKnownProblemTypeIDs.push(parseInt(tObj));
  4095. uroKnownProblemTypeNames.push(tProblemList[tObj].title);
  4096. }
  4097. }
  4098. }
  4099. function uroFilterProblems()
  4100. {
  4101. if(uroFilterPreamble() === false) return;
  4102. var selector;
  4103.  
  4104. if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === false) && (uroGetCBChecked('_cbMPClosedUserIDFilter') === false))
  4105. {
  4106. selector = document.getElementById('_selectMPUserID');
  4107. while(selector.options.length > 0)
  4108. {
  4109. selector.options.remove(0);
  4110. }
  4111. }
  4112.  
  4113. var solverUser = null;
  4114. if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === true) || (uroGetCBChecked('_cbMPClosedUserIDFilter') === true))
  4115. {
  4116. selector = document.getElementById('_selectMPUserID');
  4117. if(selector.options.length === 0)
  4118. {
  4119. uroUpdateMPSolverList();
  4120. }
  4121. if(selector.selectedOptions[0] != null)
  4122. {
  4123. solverUser = parseInt(selector.selectedOptions[0].value);
  4124. }
  4125. }
  4126.  
  4127. var urobj;
  4128. var problem;
  4129. var problemStyle;
  4130. var problem_marker_img;
  4131.  
  4132. var uFP_masterEnable = uroGetCBChecked('_cbMasterEnable');
  4133. var filter_OutsideEditableArea = uroGetCBChecked('_cbMPFilterOutsideArea');
  4134. var filter_Solved = uroGetCBChecked('_cbMPFilterSolved');
  4135. var filter_Unidentified = uroGetCBChecked('_cbMPFilterUnidentified');
  4136. var filter_Closed = uroGetCBChecked('_cbMPFilterClosed');
  4137. var filter_NotClosedUserID = uroGetCBChecked('_cbMPNotClosedUserIDFilter');
  4138. var filter_ClosedUserID = uroGetCBChecked('_cbMPClosedUserIDFilter');
  4139. var filter_Reopened = uroGetCBChecked('_cbMPFilterReopenedProblem');
  4140. var filter_LowSeverity = uroGetCBChecked('_cbMPFilterLowSeverity');
  4141. var filter_MediumSeverity = uroGetCBChecked('_cbMPFilterMediumSeverity');
  4142. var filter_HighSeverity = uroGetCBChecked('_cbMPFilterHighSeverity');
  4143. var filter_TurnProblems = uroGetCBChecked('_cbMPFilterTurnProblem');
  4144. var filterTypes = [];
  4145. var i;
  4146. for(i=0; i<uroKnownProblemTypeIDs.length; i++)
  4147. {
  4148. if(uroGetCBChecked('_cbMPFilter_T'+uroKnownProblemTypeIDs[i])) filterTypes.push(uroKnownProblemTypeIDs[i]);
  4149. }
  4150. var filter_TypeUnknown = uroGetCBChecked('_cbMPFilterUnknownProblem');
  4151. var filter_TaggedElgin = uroGetCBChecked('_cbFilterElgin');
  4152. var filter_TaggedTrafficCast = uroGetCBChecked('_cbFilterTrafficCast');
  4153. var filter_TaggedTrafficMaster = uroGetCBChecked('_cbFilterTrafficMaster');
  4154. var filter_TaggedCaltrans = uroGetCBChecked('_cbFilterCaltrans');
  4155. var filter_TaggedTFL = uroGetCBChecked('_cbFilterTFL');
  4156. var filter_Invert = uroGetCBChecked('_cbInvertMPFilter');
  4157.  
  4158.  
  4159. for (urobj in W.model.problems.objects)
  4160. {
  4161. if(W.model.problems.objects.hasOwnProperty(urobj))
  4162. {
  4163. problem = W.model.problems.objects[urobj];
  4164. problemStyle = 'visible';
  4165. var ureqID = null;
  4166. var customType = null;
  4167.  
  4168. if(uFP_masterEnable === true)
  4169. {
  4170. var elgin_mp = false;
  4171. var trafficcast_mp = false;
  4172. var trafficmaster_mp = false;
  4173. var caltrans_mp = false;
  4174. var tfl_mp = false;
  4175.  
  4176. ureqID = problem.attributes.id;
  4177. customType = uroGetCustomType(ureqID, "mp");
  4178. if(customType === 100) elgin_mp = true;
  4179. else if(customType === 101) trafficcast_mp = true;
  4180. else if(customType === 102) trafficmaster_mp = true;
  4181. else if(customType === 103) caltrans_mp = true;
  4182. else if(customType === 104) tfl_mp = true;
  4183. // check problem against current session ignore list...
  4184. if(uroIsOnIgnoreList(ureqID)) problemStyle = 'hidden';
  4185.  
  4186. if(filter_OutsideEditableArea === true)
  4187. {
  4188. if(problem.canEdit() === false)
  4189. {
  4190. problemStyle = 'hidden';
  4191. }
  4192. }
  4193.  
  4194. // check against closed/not identified filtering if enabled...
  4195. problem_marker_img = '';
  4196. if(problem.geometry.id !== null)
  4197. {
  4198. if(document.getElementById(problem.geometry.id) !== null)
  4199. {
  4200. problem_marker_img = document.getElementById(problem.geometry.id).href.baseVal;
  4201. if(filter_Solved === true)
  4202. {
  4203. if(problem_marker_img.indexOf('_solved') != -1) problemStyle = 'hidden';
  4204. }
  4205. if(filter_Unidentified === true)
  4206. {
  4207. if(problem_marker_img.indexOf('_rejected') != -1) problemStyle = 'hidden';
  4208. }
  4209. }
  4210. }
  4211.  
  4212. if(filter_Closed === true)
  4213. {
  4214. if(problem.attributes.open === false)
  4215. {
  4216. problemStyle = 'hidden';
  4217. }
  4218. }
  4219.  
  4220. if(problemStyle == 'visible')
  4221. {
  4222. if(solverUser !== null)
  4223. {
  4224. if((filter_NotClosedUserID === true) && (problem.attributes.resolvedBy == solverUser)) problemStyle = 'hidden';
  4225. if((filter_ClosedUserID === true) && (problem.attributes.resolvedBy != solverUser)) problemStyle = 'hidden';
  4226. }
  4227. }
  4228.  
  4229. if(problemStyle == 'visible')
  4230. {
  4231. var problemType = null;
  4232. if(uroDOMHasTurnProblems)
  4233. {
  4234. problemType = problem.attributes.problemType;
  4235. }
  4236. else
  4237. {
  4238. problemType = problem.attributes.subType;
  4239. }
  4240.  
  4241. if(elgin_mp === true)
  4242. {
  4243. if(filter_TaggedElgin === true) problemStyle = 'hidden';
  4244. }
  4245. else if(trafficcast_mp === true)
  4246. {
  4247. if(filter_TaggedTrafficCast === true) problemStyle = 'hidden';
  4248. }
  4249. else if(trafficmaster_mp === true)
  4250. {
  4251. if(filter_TaggedTrafficMaster === true) problemStyle = 'hidden';
  4252. }
  4253. else if(caltrans_mp === true)
  4254. {
  4255. if(filter_TaggedCaltrans === true) problemStyle = 'hidden';
  4256. }
  4257. else if(tfl_mp === true)
  4258. {
  4259. if(filter_TaggedTFL === true) problemStyle = 'hidden';
  4260. }
  4261.  
  4262. else if(uroKnownProblemTypeIDs.indexOf(problemType) !== -1)
  4263. {
  4264. if(filterTypes.indexOf(problemType) !== -1)
  4265. {
  4266. problemStyle = 'hidden';
  4267. }
  4268. }
  4269. else if(filter_TypeUnknown === true) problemStyle = 'hidden';
  4270.  
  4271. if(filter_Reopened === true)
  4272. {
  4273. if((problem.attributes.open === true) && (problem.attributes.resolvedOn !== null))
  4274. {
  4275. problemStyle = 'hidden';
  4276. }
  4277. }
  4278.  
  4279.  
  4280. if(filter_Invert === true)
  4281. {
  4282. if(problemStyle == 'hidden') problemStyle = 'visible';
  4283. else problemStyle = 'hidden';
  4284. }
  4285.  
  4286.  
  4287. if(problem.attributes.weight <= 3)
  4288. {
  4289. if(filter_LowSeverity === true) problemStyle = 'hidden';
  4290. }
  4291. else if(problem.attributes.weight <= 7)
  4292. {
  4293. if(filter_MediumSeverity === true) problemStyle = 'hidden';
  4294. }
  4295. else if(filter_HighSeverity === true) problemStyle = 'hidden';
  4296. }
  4297. }
  4298.  
  4299. W.map.problemLayer.markers[urobj].icon.imageDiv.style.visibility = problemStyle;
  4300.  
  4301. if((problemStyle != 'hidden') && (ureqID !== null) && (customType !== null))
  4302. {
  4303. uroAddCustomMarkers(ureqID,'mp',customType, false, 0);
  4304. }
  4305. }
  4306. }
  4307.  
  4308. if(uroDOMHasTurnProblems)
  4309. {
  4310. for (urobj in W.model.turnProblems.objects)
  4311. {
  4312. if(W.model.turnProblems.objects.hasOwnProperty(urobj))
  4313. {
  4314. problem = W.model.turnProblems.objects[urobj];
  4315. problemStyle = 'visible';
  4316.  
  4317. if(uFP_masterEnable === true)
  4318. {
  4319. // check problem against current session ignore list...
  4320. if(uroIsOnIgnoreList(problem.attributes.id)) problemStyle = 'hidden';
  4321.  
  4322. // check against closed/not identified filtering if enabled...
  4323. problem_marker_img = '';
  4324. if(problem.geometry.id !== null)
  4325. {
  4326. if(document.getElementById(problem.geometry.id) !== null)
  4327. {
  4328. problem_marker_img = document.getElementById(problem.geometry.id).href.baseVal;
  4329. if(filter_Solved === true)
  4330. {
  4331. if(problem_marker_img.indexOf('_solved') != -1) problemStyle = 'hidden';
  4332. }
  4333. if(filter_Unidentified === true)
  4334. {
  4335. if(problem_marker_img.indexOf('_rejected') != -1) problemStyle = 'hidden';
  4336. }
  4337. }
  4338. }
  4339.  
  4340. if(filter_Closed === true)
  4341. {
  4342. if(problem.attributes.open === false)
  4343. {
  4344. problemStyle = 'hidden';
  4345. }
  4346. }
  4347.  
  4348. if(problemStyle == 'visible')
  4349. {
  4350. if(filter_TurnProblems === true) problemStyle = 'hidden';
  4351.  
  4352. if(filter_Reopened === true)
  4353. {
  4354. if((problem.attributes.open === true) && (problem.attributes.resolvedOn !== null))
  4355. {
  4356. problemStyle = 'hidden';
  4357. }
  4358. }
  4359.  
  4360. if(filter_Invert === true)
  4361. {
  4362. if(problemStyle == 'hidden') problemStyle = 'visible';
  4363. else problemStyle = 'hidden';
  4364. }
  4365. }
  4366. }
  4367. W.map.problemLayer.markers[urobj].icon.imageDiv.style.visibility = problemStyle;
  4368. }
  4369. }
  4370. }
  4371. uroRenderCustomMarkers('mp');
  4372. }
  4373.  
  4374. function uroToHex(decValue,digits)
  4375. {
  4376. var modifier = 1;
  4377. for(var i=0; i<digits; i++)
  4378. {
  4379. modifier *= 16;
  4380. }
  4381. decValue = parseInt(decValue);
  4382. decValue += modifier;
  4383. var retval = decValue.toString(16);
  4384. retval = retval.substr(-digits);
  4385. retval = retval.toUpperCase();
  4386. return retval;
  4387. }
  4388. function uroFilterPreamble()
  4389. {
  4390. var mapviewport = document.getElementsByClassName("olMapViewport")[0];
  4391. if(mapviewport === null)
  4392. {
  4393. if(uroNullMapViewport === false)
  4394. {
  4395. uroAddLog('caught null mapviewport');
  4396. uroNullMapViewport = true;
  4397. }
  4398. return false;
  4399. }
  4400. var uiElms = document.getElementById('uroCtrlMisc');
  4401. if(uiElms == null)
  4402. {
  4403. uroAddLog('caught missing UI');
  4404. return false;
  4405. }
  4406. if(uiElms.innerHTML.length === 0)
  4407. {
  4408. uroAddLog('caught empty UI');
  4409. return false;
  4410. }
  4411. uroNullMapViewport = false;
  4412.  
  4413. return true;
  4414. }
  4415. function uroFilterItems_URTabClick()
  4416. {
  4417. uroFilterURs();
  4418. }
  4419. function uroFilterItems_MPTabClick()
  4420. {
  4421. uroFilterProblems();
  4422. }
  4423. function uroFilterItems_PlacesTabClick()
  4424. {
  4425. uroFilterPlaces();
  4426. }
  4427. function uroFilterItems_CamerasTabClick()
  4428. {
  4429. uroFilterCameras();
  4430. }
  4431. function uroFilterItems_MiscTabClick()
  4432. {
  4433. uroFilterItems();
  4434. }
  4435. function uroFilterItems_MasterEnableClick()
  4436. {
  4437. if(uroGetCBChecked('_cbMasterEnable') === false)
  4438. {
  4439. uroHidePopup();
  4440. }
  4441. uroFilterItems();
  4442. }
  4443.  
  4444. function uroScaleTheScaleBar()
  4445. {
  4446. // adjust the scale bar to more accurately reflect true distances at all latitudes
  4447. var currLat = W.map.getCenter().transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326")).lat;
  4448.  
  4449. if((currLat < 85) && (currLat > -85))
  4450. {
  4451. var cosLat = Math.cos((currLat * Math.PI) / 180);
  4452. var scaleElm;
  4453. var elmWidth;
  4454. scaleElm = document.getElementsByClassName('olControlScaleLineTop')[0];
  4455. if(scaleElm.innerHTML.indexOf(' ') !== -1)
  4456. {
  4457. elmWidth = Math.round((scaleElm.clientWidth + 2) / cosLat);
  4458. scaleElm.innerHTML = scaleElm.innerHTML.replace(' ','');
  4459. scaleElm.style.width = elmWidth + 'px';
  4460. }
  4461. scaleElm = document.getElementsByClassName('olControlScaleLineBottom')[0];
  4462. if(scaleElm.innerHTML.indexOf(' ') !== -1)
  4463. {
  4464. elmWidth = Math.round((scaleElm.clientWidth + 2) / cosLat);
  4465. scaleElm.innerHTML = scaleElm.innerHTML.replace(' ','');
  4466. scaleElm.style.width = elmWidth + 'px';
  4467. }
  4468. }
  4469. }
  4470.  
  4471. function uroFilterItems()
  4472. {
  4473. uroScaleTheScaleBar();
  4474. uroFilterProblems();
  4475. uroFilterPlaces();
  4476. uroFilterCameras();
  4477. uroFilterURs();
  4478. }
  4479. function uroFilterItemsOnMove()
  4480. {
  4481. W.map.events.unregister('mousemove',null,uroFilterItemsOnMove);
  4482. uroFilterItems();
  4483. }
  4484.  
  4485. function uroDeleteObject()
  4486. {
  4487. uroAddLog('delete camera ID '+uroShownFID);
  4488. if(W.model.cameras.objects[uroShownFID] === null)
  4489. {
  4490. uroAddLog('camera object not found...');
  4491. return false;
  4492. }
  4493. uroRemoveCamFromWatchList();
  4494. var actionObj = require('Waze/Action/DeleteObject');
  4495. var deleteAction = new actionObj(W.model.cameras.objects[uroShownFID], null);
  4496. W.model.actionManager.add(deleteAction);
  4497. uroExitPopup();
  4498. uroHidePopup();
  4499. return false;
  4500. }
  4501.  
  4502. function uroGetUserNameAndRank(userID)
  4503. {
  4504. var userName;
  4505. var userLevel;
  4506. if(W.model.users.objects[userID] != null)
  4507. {
  4508. userName = W.model.users.objects[userID].userName;
  4509. if(userName === undefined)
  4510. {
  4511. userName = userID;
  4512. }
  4513. userLevel = W.model.users.objects[userID].rank + 1;
  4514. }
  4515. else
  4516. {
  4517. userName = userID;
  4518. userLevel = '?';
  4519. }
  4520. return userName + ' (' + userLevel + ')';
  4521. }
  4522. function uroCheckCommentsForTag(idSrc, customText)
  4523. {
  4524. var ursObj = W.model.updateRequestSessions.objects[idSrc];
  4525. if(typeof(ursObj) == 'undefined') return -1;
  4526. if(ursObj.comments.length === 0) return -1;
  4527.  
  4528. for(var idx=ursObj.comments.length-1; idx>=0; idx--)
  4529. {
  4530. for(var tag=0; tag<uroCustomURTags.length; tag++)
  4531. {
  4532. var keyword = uroCustomURTags[tag];
  4533. if(ursObj.comments[idx].text.indexOf(keyword) != -1)
  4534. {
  4535. return tag;
  4536. }
  4537. }
  4538. if(customText !== '')
  4539. {
  4540. if(ursObj.comments[idx].text.toLowerCase().indexOf(customText) != -1)
  4541. {
  4542. return 99;
  4543. }
  4544. }
  4545. }
  4546. return -1;
  4547. }
  4548.  
  4549. function uroGetCustomMarkerIdx(customType)
  4550. {
  4551. if(customType === 0) return 1; // ROADWORKS
  4552. if(customType === 1) return 1; // CONSTRUCTION
  4553. if(customType === 2) return 0; // CLOSURE
  4554. if(customType === 3) return 4; // EVENT
  4555. if(customType === 4) return 3; // NOTE
  4556. if(customType === 5) return 5; // WSLM
  4557. if(customType === 6) return 11; // BOG
  4558. if(customType === 7) return 12; // DIFFICULT
  4559. if(customType === 98) return 5; // Native speed limit URs
  4560. if(customType === 99) return 2; // custom text
  4561. if(customType === 100) return 6; // ELGIN
  4562. if(customType === 101) return 7; // TRAFFICCAST
  4563. if(customType === 102) return 8; // TRAFFICMASTER
  4564. if(customType === 103) return 9; // CALTRANS
  4565. if(customType === 104) return 10; // TFL
  4566. return -1;
  4567. }
  4568. function uroGetCustomType(idSrc, markerType)
  4569. {
  4570. var desc = '';
  4571. var provider = '';
  4572. var customText = '';
  4573. if(uroGetCBChecked('_cbCustomKeywordMarkers')) customText = document.getElementById('_textCustomKeyword').value.toLowerCase();
  4574. if(markerType == "ur")
  4575. {
  4576. var ureq = W.model.mapUpdateRequests.objects[idSrc];
  4577. // early test for native speed limit URs
  4578. if(ureq.attributes.type == 23) return 98;
  4579. // UR objects always have a .description attribute, which is set to null if empty...
  4580. if(ureq.attributes.description !== null)
  4581. {
  4582. desc = ureq.attributes.description;
  4583. }
  4584. }
  4585. else if(markerType == "mp")
  4586. {
  4587. var mp = W.model.problems.objects[idSrc];
  4588. // ...whereas MP objects with a blank description don't even get the attribute
  4589. if(mp.attributes.description != null)
  4590. {
  4591. desc = mp.attributes.description;
  4592. }
  4593. if(mp.attributes.provider != null)
  4594. {
  4595. provider = mp.attributes.provider;
  4596. }
  4597. }
  4598.  
  4599. if(desc !== '')
  4600. {
  4601. if(markerType == 'ur')
  4602. {
  4603. for(var tag=0; tag<uroCustomURTags.length; tag++)
  4604. {
  4605. var keyword = uroCustomURTags[tag];
  4606. if(desc.indexOf(keyword) != -1)
  4607. {
  4608. return tag;
  4609. }
  4610. }
  4611. }
  4612. if((uroGetCBChecked('_cbCustomKeywordMarkers')) && (customText !== ''))
  4613. {
  4614. if(desc.toLowerCase().indexOf(customText) != -1) return 99;
  4615. }
  4616.  
  4617. if(markerType == 'mp')
  4618. {
  4619. if(desc.indexOf('[Elgin]') != -1) return 100;
  4620. if(desc.indexOf('[TrafficCast]') != -1) return 101;
  4621. if(desc.indexOf('[TM]') != -1) return 102;
  4622. if(desc.indexOf('[Caltrans]') != -1) return 103;
  4623. if(desc.indexOf('[TfL Open Data]') != -1) return 104;
  4624. if(provider.indexOf('London TFL Closures') != -1) return 104;
  4625. }
  4626. }
  4627. if(markerType == "ur")
  4628. {
  4629. return uroCheckCommentsForTag(idSrc, customText);
  4630. }
  4631.  
  4632. return -1;
  4633. }
  4634.  
  4635. function uroFormatRestriction(restObj)
  4636. {
  4637. var retval = '<tr>';
  4638. retval += '<td style="text-align:center;">';
  4639. if((restObj._days & 1) == 1) retval += 'S';
  4640. else retval += '-';
  4641. retval += '</td><td style="text-align:center;">';
  4642. if((restObj._days & 2) == 2) retval += 'M';
  4643. else retval += '-';
  4644. retval += '</td><td style="text-align:center;">';
  4645. if((restObj._days & 4) == 4) retval += 'T';
  4646. else retval += '-';
  4647. retval += '</td><td style="text-align:center;">';
  4648. if((restObj._days & 8) == 8) retval += 'W';
  4649. else retval += '-';
  4650. retval += '</td><td style="text-align:center;">';
  4651. if((restObj._days & 16) == 16) retval += 'T';
  4652. else retval += '-';
  4653. retval += '</td><td style="text-align:center;">';
  4654. if((restObj._days & 32) == 32) retval += 'F';
  4655. else retval += '-';
  4656. retval += '</td><td style="text-align:center;">';
  4657. if((restObj._days & 64) == 64) retval += 'S';
  4658. else retval += '-';
  4659.  
  4660. retval += '</td><td>';
  4661.  
  4662. if(restObj._fromDate === null) retval += 'All dates';
  4663. else retval += restObj._fromDate+' to '+restObj._toDate;
  4664.  
  4665. retval += '</td><td>';
  4666.  
  4667. if(restObj._allDay === true) retval += 'All day';
  4668. else retval += restObj._fromTime+' to '+restObj._toTime;
  4669.  
  4670. retval += '</td><td>';
  4671.  
  4672. if(restObj.allVehicleTypes == restObj._vehicleTypes) retval += 'All vehicles';
  4673. else retval += 'Some vehicles';
  4674.  
  4675. retval += '</td><td>';
  4676.  
  4677. if(restObj._description !== null)
  4678. {
  4679. var desc = restObj._description.replace(/<\/?[^>]+(>|$)/g, "");
  4680. desc = uroClickify(desc);
  4681. retval += desc;
  4682. }
  4683.  
  4684. retval += '</td></tr>';
  4685.  
  4686. return retval;
  4687. }
  4688.  
  4689. function uroHidePopup()
  4690. {
  4691. if(uroPopupShown)
  4692. {
  4693. uroDiv.style.visibility = 'hidden';
  4694. uroPopupShown = false;
  4695. uroPopupTimer = -2;
  4696. uroShownFID = -1;
  4697. }
  4698. uroPopupSuppressed = false;
  4699. }
  4700. function uroSuppressPopup()
  4701. {
  4702. uroDiv.style.visibility = 'hidden';
  4703. window.getSelection().removeAllRanges();
  4704. uroPopupSuppressed = true;
  4705. }
  4706.  
  4707. function uroRecentreSessionOnUR()
  4708. {
  4709. W.map.updateRequestLayer.markers[uroShownFID].icon.imageDiv.click();
  4710. W.map.moveTo(W.map.updateRequestLayer.markers[uroShownFID].lonlat, 5);
  4711. uroHidePopup();
  4712. return false;
  4713. }
  4714. function uroRecentreSessionOnMP()
  4715. {
  4716. W.map.problemLayer.markers[uroShownFID].icon.imageDiv.click();
  4717. W.map.moveTo(W.map.problemLayer.markers[uroShownFID].lonlat, 5);
  4718. uroHidePopup();
  4719. return false;
  4720. }
  4721. function uroRecentreSessionOnPUR()
  4722. {
  4723. W.map.placeUpdatesLayer.markers[uroShownFID].icon.imageDiv.click();
  4724. W.map.moveTo(W.map.placeUpdatesLayer.markers[uroShownFID].lonlat, 5);
  4725. uroHidePopup();
  4726. return false;
  4727. }
  4728. function uroRecentreSessionOnVenueNavPoint()
  4729. {
  4730. W.map.moveTo(uroGetVenueNavPoint(uroShownFID), 5);
  4731. uroHidePopup();
  4732. return false;
  4733. }
  4734.  
  4735. function uroGetDateTimeString(ts)
  4736. {
  4737. var tDateObj = new Date(ts);
  4738. var dateLocale;
  4739. var timeLocale;
  4740. if(uroGetCBChecked('_cbDateFmtDDMMYY')) dateLocale = 'en-gb';
  4741. if(uroGetCBChecked('_cbDateFmtMMDDYY')) dateLocale = 'en-us';
  4742. if(uroGetCBChecked('_cbDateFmtYYMMDD')) dateLocale = 'ja';
  4743. if(uroGetCBChecked('_cbTimeFmt24H')) timeLocale = 'en-gb';
  4744. if(uroGetCBChecked('_cbTimeFmt12H')) timeLocale = 'en-us';
  4745. return tDateObj.toLocaleDateString(dateLocale) + ' ' + tDateObj.toLocaleTimeString(timeLocale);
  4746. }
  4747. function uroParsePxString(pxString)
  4748. {
  4749. return parseInt(pxString.split("px")[0]);
  4750. }
  4751.  
  4752. function uroStackListObj(fid,x,y)
  4753. {
  4754. this.fid = fid;
  4755. this.x = uroTypeCast(x);
  4756. this.y = uroTypeCast(y);
  4757. }
  4758. function uroRestackMarkers()
  4759. {
  4760. if(uroStackList.length === 0) return;
  4761. var markerCollection = null;
  4762. if(uroStackType == 1) markerCollection = W.map.updateRequestLayer.markers;
  4763. else if(uroStackType == 2) markerCollection = W.map.problemLayer.markers;
  4764. else if(uroStackType == 3) markerCollection = W.map.placeUpdatesLayer.markers;
  4765.  
  4766. if(markerCollection !== null)
  4767. {
  4768. uroAddLog('restacking markers...');
  4769. // strip off the .realX/realY attributes from any UR object we've previously added it to, to allow
  4770. // the native recentering to work again...
  4771. for(var marker in markerCollection)
  4772. {
  4773. if(markerCollection.hasOwnProperty(marker))
  4774. {
  4775. var testMarkerObj = markerCollection[marker];
  4776. if(testMarkerObj.model.attributes.geometry.realX != null)
  4777. {
  4778. testMarkerObj.model.attributes.geometry.x = testMarkerObj.model.attributes.geometry.realX;
  4779. testMarkerObj.model.attributes.geometry.y = testMarkerObj.model.attributes.geometry.realY;
  4780. delete(testMarkerObj.model.attributes.geometry.realX);
  4781. delete(testMarkerObj.model.attributes.geometry.realY);
  4782. }
  4783. }
  4784. }
  4785. // now restack any markers that were repositioned...
  4786. for(var idx=0; idx<uroStackList.length; idx++)
  4787. {
  4788. var orig_x = uroStackList[idx].x + 'px';
  4789. var orig_y = uroStackList[idx].y + 'px';
  4790. var fid = uroStackList[idx].fid;
  4791.  
  4792. if(markerCollection[fid] != null)
  4793. {
  4794. markerCollection[fid].icon.imageDiv.style.left = orig_x;
  4795. markerCollection[fid].icon.imageDiv.style.top = orig_y;
  4796. }
  4797. }
  4798. uroStackList = [];
  4799. uroUnstackedMasterID = null;
  4800. uroStackType = null;
  4801. }
  4802. }
  4803. function uroIsIDAlreadyUnstacked(idSrc)
  4804. {
  4805. if(uroStackList.length === 0) return false;
  4806. for(var idx=0; idx<uroStackList.length; idx++)
  4807. {
  4808. if(uroStackList[idx].fid == idSrc) return true;
  4809. }
  4810. return false;
  4811. }
  4812. function uroCheckStacking(stackType, masterID, unstackedX, unstackedY)
  4813. {
  4814. if(uroIsIDAlreadyUnstacked(masterID) === true) return;
  4815. if(uroStackType !== null) return;
  4816. if(uroPopupDwellTimer > 0) return;
  4817.  
  4818. uroAddLog('checking for marker stack, masterID: '+masterID+', stackType: '+stackType);
  4819. var stackList = [];
  4820. stackList.push(masterID);
  4821. var threshSquared = uroGetElmValue('_inputUnstackSensitivity');
  4822. threshSquared *= threshSquared;
  4823.  
  4824. var markerCollection = null;
  4825. var marker;
  4826. if(stackType == 1) markerCollection = W.map.updateRequestLayer.markers;
  4827. else if(stackType == 2) markerCollection = W.map.problemLayer.markers;
  4828. else if(stackType == 3) markerCollection = W.map.placeUpdatesLayer.markers;
  4829.  
  4830. var offset = 1000000000;
  4831. if(markerCollection !== null)
  4832. {
  4833. for(marker in markerCollection)
  4834. {
  4835. if(markerCollection.hasOwnProperty(marker))
  4836. {
  4837. var testMarkerObj = markerCollection[marker];
  4838. var includeInStack = (testMarkerObj.icon.imageDiv.style.visibility != 'hidden');
  4839. var suppressClosed = (testMarkerObj.icon.imageDiv.classList.contains("recently-closed") & (W.map.updateRequestLayer.showHidden === false));
  4840.  
  4841. // 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
  4842. // top of the stack in the absence of any URO+ filtering, regardless of which UR pin actually receives the click event. To prevent
  4843. // this, we give each pin in the stack a unique set of false coordinates, storing the original coordinates in newly created
  4844. // properties so they can be restored later on
  4845. //
  4846. // originally this fix changed the x coordinate for each UR in the stack to be a unique value that ought to have been well out of the range
  4847. // of any real coordinate and therefore unable to clash with any UR marker coordinates that weren't in the current stack. However it now
  4848. // appears this could then cause WME to think a completely different UR was being opened - possibly as a result of some change in coordinate
  4849. // handling allowing out of range values to be wrapped around back into the normal range? As a workaround for this new WME behaviour, both the
  4850. // x and y coordinates are now set to valid values somewhere in the North Atlantic Ocean - the likelihood of there being any real URs in this
  4851. // area is so vanishingly small as to be not worth worrying about...
  4852. if(testMarkerObj.model.attributes.geometry.realX === undefined)
  4853. {
  4854. testMarkerObj.model.attributes.geometry.realX = testMarkerObj.model.attributes.geometry.x;
  4855. testMarkerObj.model.attributes.geometry.x += offset;
  4856. testMarkerObj.model.attributes.geometry.realY = testMarkerObj.model.attributes.geometry.y;
  4857. testMarkerObj.model.attributes.geometry.y += offset;
  4858. offset += 1000;
  4859. }
  4860.  
  4861. if((includeInStack) && (!suppressClosed))
  4862. {
  4863. if(testMarkerObj.id != masterID)
  4864. {
  4865. var xdiff = unstackedX - uroParsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.left);
  4866. var ydiff = unstackedY - uroParsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.top);
  4867. var distSquared = ((xdiff * xdiff) + (ydiff * ydiff));
  4868. if(distSquared < threshSquared)
  4869. {
  4870. stackList.push(testMarkerObj.id);
  4871. }
  4872. }
  4873. }
  4874. }
  4875. }
  4876. }
  4877.  
  4878. // as it's the fiddling with .geometry.x which seems to inhibit the autocentering behaviour when a UR/MP marker is clicked, we need to
  4879. // allow this to occur even if unstacking isn't required at this zoom level. To then reenable recentering when clicking on the crosshairs
  4880. // or feed entry, we need to reinstate the correct .x value once the marker is no longer being highlighted, which means we pretty much need
  4881. // to do all of the unstacking *except* for actually unstacking the stack and hiding the other markers...
  4882. var inhibitUnstacking = (W.map.getZoom() < uroGetElmValue('_inputUnstackZoomLevel'));
  4883. // also inhibit unstacking if there's only a single marker in the list - this will be true if we're highlighting an isolated marker that
  4884. // doesn't need to be unstacked, and where we're only using the .geometry.x fiddle to prevent autocentering...
  4885. inhibitUnstacking |= (stackList.length == 1);
  4886. uroStackType = stackType;
  4887. if(stackList.length > 0)
  4888. {
  4889. if(inhibitUnstacking) uroAddLog('single marker highlighted, adjusting geometry properties to prevent recentering...');
  4890. else uroAddLog('markers are stacked!');
  4891. if(uroUnstackedMasterID != masterID)
  4892. {
  4893. uroAddLog('unstacked ID mismatch, relocating markers...');
  4894. uroRestackMarkers();
  4895. uroUnstackedMasterID = masterID;
  4896. uroStackList = [];
  4897.  
  4898. // push the highlighted marker onto the stacklist so uroIsIDAlreadyUnstacked() will return true
  4899. uroStackList.push(new uroStackListObj(masterID,unstackedX,unstackedY));
  4900.  
  4901. for(var shoveIdx=0; shoveIdx < stackList.length; shoveIdx++)
  4902. {
  4903. var fid = stackList[shoveIdx];
  4904. var x = uroParsePxString(markerCollection[fid].icon.imageDiv.style.left);
  4905. var y = uroParsePxString(markerCollection[fid].icon.imageDiv.style.top);
  4906. // store the unstacked marker positions so they can be reinstated later
  4907. uroStackList.push(new uroStackListObj(fid,x,y));
  4908. if(!inhibitUnstacking)
  4909. {
  4910. markerCollection[fid].icon.imageDiv.style.left = unstackedX + 'px';
  4911. markerCollection[fid].icon.imageDiv.style.top = unstackedY + 'px';
  4912. unstackedX += 10;
  4913. unstackedY -= 30;
  4914. }
  4915. }
  4916.  
  4917. if(!inhibitUnstacking)
  4918. {
  4919. // hide other markers to prevent confusion with the unstacked markers
  4920. for(marker in markerCollection)
  4921. {
  4922. if(markerCollection.hasOwnProperty(marker))
  4923. {
  4924. var toHideID = markerCollection[marker].id;
  4925. if(uroIsIDAlreadyUnstacked(toHideID) === false)
  4926. {
  4927. markerCollection[toHideID].icon.imageDiv.style.visibility = 'hidden';
  4928. }
  4929. }
  4930. }
  4931. }
  4932. }
  4933. }
  4934. else
  4935. {
  4936. uroRestackMarkers();
  4937. }
  4938. }
  4939.  
  4940. function uroGetVenueNavPoint(uroFID)
  4941. {
  4942. for(var vObj in W.model.venues.objects)
  4943. {
  4944. if(W.model.venues.objects.hasOwnProperty(vObj))
  4945. {
  4946. if(uroFID == vObj)
  4947. {
  4948. return W.model.venues.objects[vObj].getNavigationPoint().point.toLonLat();
  4949. }
  4950. }
  4951. }
  4952. // just in case... return a safe value if the requested venue object wasn't found
  4953. return W.map.getCenter();
  4954. }
  4955.  
  4956. function uroOpenNewTab()
  4957. {
  4958. // flush the current settings into localStorage before the new tab opens, so that when its instance of
  4959. // URO+ fires up it'll have the same settings as this one
  4960. uroSaveSettings();
  4961. return true;
  4962. }
  4963.  
  4964. function uroEditTBR()
  4965. {
  4966. if(uroTBRObj === null)
  4967. {
  4968. return;
  4969. }
  4970. uroTBRObj.getElementsByClassName('waze-icon-clock')[0].click();
  4971. return false;
  4972. }
  4973.  
  4974. function uroKillCentering()
  4975. {
  4976. return W.map.getExtent();
  4977. }
  4978. function uroRestoreCentering()
  4979. {
  4980. if(uroAutoCentreDisabledOn.length > 0)
  4981. {
  4982. if(uroAutoCentreDisabledOn[0] == 'PUR')
  4983. {
  4984. if(W.map.placeUpdatesLayer.markers[uroAutoCentreDisabledOn[1]] != null)
  4985. {
  4986. W.map.placeUpdatesLayer.markers[uroAutoCentreDisabledOn[1]].model.geometry.getBounds = W.map.placeUpdatesLayer.markers[uroAutoCentreDisabledOn[1]].model.geometry.origGetBounds;
  4987. }
  4988. }
  4989. else if(uroAutoCentreDisabledOn[0] == 'MP')
  4990. {
  4991. if(W.map.problemLayer.markers[uroAutoCentreDisabledOn[1]] != null)
  4992. {
  4993. W.map.problemLayer.markers[uroAutoCentreDisabledOn[1]].model.getDisconnectBounds = W.map.problemLayer.markers[uroAutoCentreDisabledOn[1]].model.origGetDisconnectBounds;
  4994. }
  4995. }
  4996. uroAutoCentreDisabledOn = [];
  4997. }
  4998. }
  4999.  
  5000. function uroAddClosureRowToTable(rcObj)
  5001. {
  5002. var result = '';
  5003.  
  5004. if(rcObj.active === true)
  5005. {
  5006. result += '<tr>';
  5007. }
  5008. else
  5009. {
  5010. result += '<tr bgcolor="#C0C0C0">';
  5011. }
  5012.  
  5013. var startDate = rcObj.startDate;
  5014. var endDate = "unknown";
  5015. if(rcObj.endDate !== null)
  5016. {
  5017. endDate = rcObj.endDate;
  5018. }
  5019. var provider = "---";
  5020. if(rcObj.provider !== null)
  5021. {
  5022. provider = rcObj.provider;
  5023. }
  5024. else if(rcObj.createdBy !== null)
  5025. {
  5026. provider = uroGetUserNameAndRank(rcObj.createdBy);
  5027. }
  5028. var reason = "---";
  5029. if(rcObj.reason !== null)
  5030. {
  5031. reason = rcObj.reason;
  5032. }
  5033. var mte = "---";
  5034. if(rcObj.eventId !== null)
  5035. {
  5036. try
  5037. {
  5038. mte = W.model.majorTrafficEvents.objects[rcObj.eventId].attributes.names[0].value;
  5039. }
  5040. catch(err)
  5041. {
  5042. }
  5043. }
  5044. result += '<td>' + startDate + ' to ' + endDate + '</td>';
  5045. result += '<td>' + provider + '</td>';
  5046. result += '<td>' + reason + '</td>';
  5047. result += '<td>' + mte + '</td>';
  5048. result += '</td></tr>';
  5049. return result;
  5050. }
  5051.  
  5052.  
  5053. function uroGetAddress(streetID, houseNumber, formatForSegmentPopup)
  5054. {
  5055. var result = '';
  5056. if((houseNumber !== undefined) && (houseNumber !== null))
  5057. {
  5058. result += houseNumber + ' ';
  5059. }
  5060.  
  5061. if(streetID != null)
  5062. {
  5063. var streetName = I18n.lookup('edit.address.no_street');
  5064. var doesStreetIDExist = true;
  5065. if(W.model.streets.objects[streetID] === undefined)
  5066. {
  5067. streetName = 'non-existent streetID';
  5068. doesStreetIDExist = false;
  5069. }
  5070. else
  5071. {
  5072. if((streetName !== null) && (W.model.streets.objects[streetID].isEmpty === false))
  5073. {
  5074. streetName = W.model.streets.objects[streetID].name;
  5075. }
  5076. }
  5077. if(formatForSegmentPopup === true)
  5078. {
  5079. result += '<b>'+streetName+'</b><br>';
  5080. }
  5081. else
  5082. {
  5083. result += streetName + ', ';
  5084. }
  5085. if(doesStreetIDExist === true)
  5086. {
  5087. var cityName = I18n.lookup('edit.address.no_city');
  5088. var doesCityIDExist = true;
  5089. var cityID = W.model.streets.objects[streetID].cityID;
  5090. if(W.model.cities.objects[cityID] === undefined)
  5091. {
  5092. cityName = 'non-existent cityID';
  5093. doesCityIDExist = false;
  5094. }
  5095. else
  5096. {
  5097. if(W.model.cities.objects[cityID].attributes.name !== "")
  5098. {
  5099. cityName = W.model.cities.objects[cityID].attributes.name;
  5100. }
  5101. }
  5102. result += cityName + ', ';
  5103. if(doesCityIDExist === true)
  5104. {
  5105. var stateID = W.model.cities.objects[cityID].attributes.stateID;
  5106. if(W.model.states.objects[stateID] === undefined)
  5107. {
  5108. result += 'non-existent stateID';
  5109. }
  5110. else
  5111. {
  5112. result += W.model.states.objects[stateID].name;
  5113. }
  5114. }
  5115. }
  5116. }
  5117. result += '<br>';
  5118. return result;
  5119. }
  5120.  
  5121.  
  5122. function uroNewLookHighlightedItemsCheck(e)
  5123. {
  5124. var result = '';
  5125. var rw;
  5126. var rh;
  5127. var objHasIgnoreLink = false;
  5128. var objHasDeleteLink = false;
  5129. var objHasAddWatchLink = false;
  5130. var objHasRemoveWatchLink = false;
  5131. var objHasUpdateWatchLink = false;
  5132. var objHasRecentreSessionLink = false;
  5133. var objHasOpenInNewTabLink = false;
  5134. var objHasCloneLink = false;
  5135. var isVenue = false;
  5136. var isMapComment = false;
  5137. var newPopupType = null;
  5138. var markerObj;
  5139. var markerPos;
  5140. var markerImg;
  5141. var ureq = null;
  5142. var idx;
  5143. var hovered = false;
  5144. var targetTab = '';
  5145. var unstackedX;
  5146. var unstackedY;
  5147. var ureqID = null;
  5148. var isUR = false;
  5149. var isProblem = false;
  5150. var isTurnProb = false;
  5151. var isPlaceUpdate = false;
  5152. var idSrc = null;
  5153. var mouseX;
  5154. var mouseY;
  5155. var uroDaysResolved;
  5156. var renderIntent;
  5157. // function preamble...
  5158. {
  5159. if(uroMTEMode) return;
  5160. if(!uroInitialised) return;
  5161. if(e == 'dwellTimeout')
  5162. {
  5163. }
  5164. else
  5165. {
  5166. if((uroMouseIsDown) && (e.buttons === 0))
  5167. {
  5168. uroAddLog('trapped erroneous mousedown state');
  5169. uroMouseIsDown = false;
  5170. }
  5171. }
  5172. if(uroMouseIsDown)
  5173. {
  5174. return;
  5175. }
  5176.  
  5177. if(OpenLayers === null)
  5178. {
  5179. if(uroNullOpenLayers === false)
  5180. {
  5181. uroAddLog('caught null OpenLayers');
  5182. uroNullOpenLayers = true;
  5183. }
  5184. return;
  5185. }
  5186. uroNullOpenLayers = false;
  5187.  
  5188. if(W.map.updateRequestLayer === null)
  5189. {
  5190. if(uroNullURLayer === false)
  5191. {
  5192. uroAddLog('caught null UR layer');
  5193. uroNullURLayer = true;
  5194. }
  5195. return;
  5196. }
  5197. uroNullURLayer = false;
  5198.  
  5199. if(W.map.problemLayer === null)
  5200. {
  5201. if(uroNullProblemLayer === false)
  5202. {
  5203. uroAddLog('caught null problem layer');
  5204. uroNullProblemLayer = true;
  5205. }
  5206. return;
  5207. }
  5208. uroNullProblemLayer = false;
  5209.  
  5210. if(uroGetCBChecked('_cbMasterEnable') === false)
  5211. {
  5212. return;
  5213. }
  5214.  
  5215. if(e == 'dwellTimeout')
  5216. {
  5217. mouseX = uroPrevMouseX;
  5218. mouseY = uroPrevMouseY;
  5219. }
  5220. else
  5221. {
  5222. mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
  5223. mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top;
  5224.  
  5225. var maxJitter = uroGetElmValue('_inputMaxJitter');
  5226. if((Math.abs(uroPrevMouseX - mouseX) > maxJitter) || (Math.abs(uroPrevMouseY - mouseY) > maxJitter))
  5227. {
  5228. uroPopupDwellTimer = uroGetElmValue('_inputPopupDwellTimeout');
  5229. }
  5230. uroPrevMouseX = mouseX;
  5231. uroPrevMouseY = mouseY;
  5232. }
  5233. }
  5234.  
  5235. var popupXOffset = uroParsePxString(window.getComputedStyle(document.getElementById('sidebar')).getPropertyValue("width"));
  5236. var popupYOffset = $(document.getElementById("WazeMap")).offset().top - 80;
  5237. var uroPopupX = mouseX + popupXOffset + 10;
  5238. var uroPopupY = mouseY + popupYOffset - 10;
  5239. // popup for segment restrictions
  5240. if((uroMousedOverMarkerType === null) && (uroGetCBChecked('_cbInhibitSegPopup') === false))
  5241. {
  5242. for(var slIdx=0; slIdx < W.map.segmentLayer.features.length; slIdx++)
  5243. {
  5244. if(W.map.segmentLayer.features[slIdx].renderIntent == 'highlight')
  5245. {
  5246. if(W.map.getExtent().intersectsBounds(W.map.segmentLayer.features[slIdx].geometry.getBounds()))
  5247. {
  5248. var doPopUp = false;
  5249. var segObj;
  5250. var restObj;
  5251. if(W.map.segmentLayer.features[slIdx].fid === null) segObj = W.map.segmentLayer.features[slIdx].model;
  5252. else segObj = W.map.segmentLayer.features[slIdx];
  5253.  
  5254. // generic segment data
  5255. if(uroGetCBChecked('_cbInhibitSegGenericPopup') === false)
  5256. {
  5257. doPopUp = true;
  5258. var streetID = segObj.attributes.primaryStreetID;
  5259. uroAddLog('building popup for segment '+streetID);
  5260. result += uroGetAddress(streetID, null, true);
  5261. result += '<b>ID: </b>'+segObj.attributes.id+'<br>';
  5262. var fwdSpeed = segObj.attributes.fwdMaxSpeed;
  5263. var revSpeed = segObj.attributes.revMaxSpeed;
  5264. var fwdUnverified = segObj.attributes.fwdMaxSpeedUnverified;
  5265. var revUnverified = segObj.attributes.revMaxSpeedUnverified;
  5266. var roadType = segObj.attributes.roadType;
  5267. var verifyLimits = true;
  5268. if((roadType === 17) || (roadType === 20))
  5269. {
  5270. verifyLimits = false;
  5271. }
  5272. if(segObj.attributes.fwdDirection)
  5273. {
  5274. result += '<b>A-B speed: </b>'+uroGetLocalisedSpeedString(fwdSpeed, verifyLimits);
  5275. if(fwdUnverified) result += ' (unverified)';
  5276. result += '<br>';
  5277. }
  5278. if(segObj.attributes.revDirection)
  5279. {
  5280. result += '<b>B-A speed: </b>'+uroGetLocalisedSpeedString(revSpeed, verifyLimits);
  5281. if(revUnverified) result += ' (unverified)';
  5282. result += '<br>';
  5283. }
  5284. if((segObj.attributes.fwdDirection) && (segObj.attributes.revDirection) && (fwdSpeed != revSpeed) && (!fwdUnverified) && (!revUnverified))
  5285. {
  5286. result += 'Two-way segment has different verified speed limits...<br>';
  5287. }
  5288. }
  5289.  
  5290. // segment restrictions
  5291. result += '<table cellpadding=4 border=1">';
  5292. if(segObj.attributes.fwdRestrictions.length > 0)
  5293. {
  5294. doPopUp = true;
  5295. result += '<tr><td colspan=11><b>A-B restrictions:</b></td></tr>';
  5296. for(idx = 0; idx < segObj.attributes.fwdRestrictions.length; idx++)
  5297. {
  5298. restObj = segObj.attributes.fwdRestrictions[idx];
  5299. result += uroFormatRestriction(restObj);
  5300. }
  5301. }
  5302.  
  5303. if (segObj.attributes.revRestrictions.length > 0)
  5304. {
  5305. doPopUp = true;
  5306. result += '<tr><td colspan=11><b>B-A restrictions:</b></td></tr>';
  5307. for(idx = 0; idx < segObj.attributes.revRestrictions.length; idx++)
  5308. {
  5309. restObj = segObj.attributes.revRestrictions[idx];
  5310. result += uroFormatRestriction(restObj);
  5311. }
  5312. }
  5313. result += '</table>';
  5314. if(W.map.closuresMarkerLayer.getVisibility() === true)
  5315. {
  5316. result += '<table cellpadding=4 border=1" width="100%">';
  5317. if(segObj.attributes.hasClosures === true)
  5318. {
  5319. var hasFwd = false;
  5320. var hasRev = false;
  5321. var rcObj;
  5322. var roadClosure;
  5323. for(roadClosure in W.model.roadClosures.objects)
  5324. {
  5325. if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
  5326. {
  5327. rcObj = W.model.roadClosures.objects[roadClosure];
  5328. if(rcObj.segID == segObj.attributes.id)
  5329. {
  5330. if(rcObj.forward === true)
  5331. {
  5332. if(hasFwd === false)
  5333. {
  5334. result += '<tr><td colspan=4><b>A-B closures:</b></td></tr>';
  5335. hasFwd = true;
  5336. }
  5337. result += uroAddClosureRowToTable(rcObj);
  5338. }
  5339. else
  5340. {
  5341. hasRev = true;
  5342. }
  5343. }
  5344. }
  5345. }
  5346. if(hasRev === true)
  5347. {
  5348. result += '<tr><td colspan=4><b>B-A closures:</b></td></tr>';
  5349. for(roadClosure in W.model.roadClosures.objects)
  5350. {
  5351. if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
  5352. {
  5353. rcObj = W.model.roadClosures.objects[roadClosure];
  5354. if(rcObj.segID == segObj.attributes.id)
  5355. {
  5356. if(rcObj.forward === false)
  5357. {
  5358. result += uroAddClosureRowToTable(rcObj);
  5359. }
  5360. }
  5361. }
  5362. }
  5363. }
  5364. if((hasFwd === true) || (hasRev === true))
  5365. {
  5366. doPopUp = true;
  5367. }
  5368. }
  5369. result += '</table>';
  5370. }
  5371.  
  5372. if(doPopUp === true)
  5373. {
  5374. if(segObj.attributes.id === null) uroFID = segObj.id;
  5375. else uroFID = segObj.attributes.id;
  5376. newPopupType = 'segment_restriction';
  5377. }
  5378.  
  5379. break;
  5380. }
  5381. else
  5382. {
  5383. uroAddLog('segment '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup');
  5384. }
  5385. }
  5386. }
  5387. }
  5388. // popup for restricted turns
  5389. if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitTurnsPopup') === false))
  5390. {
  5391. var turnMarkerCount = W.map.layers[uroTurnsLayerIdx].markers.length;
  5392. if(turnMarkerCount > 0)
  5393. {
  5394. for(idx=0; idx<turnMarkerCount; idx++)
  5395. {
  5396. markerObj = W.map.layers[uroTurnsLayerIdx].markers[idx];
  5397. var arrowElm = markerObj.icon.imageDiv.childNodes[0];
  5398. markerImg = window.getComputedStyle(arrowElm).getPropertyValue("background-image");
  5399. markerPos = window.getComputedStyle(arrowElm).getPropertyValue("background-position");
  5400.  
  5401. hovered = false;
  5402.  
  5403. if(markerImg.indexOf('turns96765f551688fd5082b619129499bbb3.png') != -1)
  5404. {
  5405. if(markerPos == '-72px 0px')
  5406. {
  5407. hovered = true;
  5408. }
  5409. }
  5410. if(hovered === true)
  5411. {
  5412. uroAddLog('hover over restricted turn marker');
  5413. uroTBRObj = arrowElm.childNodes[0];
  5414. var trObj = ($(arrowElm).data('model'));
  5415. var Vertex = new require("Waze/Model/Graph/Vertex");
  5416. var turnGraph = W.model.getTurnGraph().getTurn(Vertex.forwardOf(trObj.fromSeg.attributes.id),Vertex.reverseOf(trObj.toSeg.attributes.id));
  5417. var resObj = turnGraph._turnData._restrictions;
  5418. uroAddLog('building popup for turn restriction');
  5419.  
  5420. result += '<label id="_editTBR">Click to edit</label><br>';
  5421. result += '<table cellpadding=4 border=1">';
  5422. for(var resIdx=0; resIdx < resObj.length; resIdx++)
  5423. {
  5424. result += uroFormatRestriction(resObj[resIdx]);
  5425. }
  5426. result += '</table>';
  5427. uroFID = markerObj.icon.imageDiv._eventCacheID;
  5428. newPopupType = 'turn_restriction';
  5429. break;
  5430. }
  5431. }
  5432. }
  5433. }
  5434.  
  5435. // popup for landmarks
  5436. if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitLandmarkPopup') === false))
  5437. {
  5438. uroPlaceSelected = false;
  5439. var venueObj = null;
  5440. renderIntent = null;
  5441. var navpointPos=new OpenLayers.LonLat();
  5442. for(var llFeatureIdx=0; llFeatureIdx < W.map.landmarkLayer.features.length; llFeatureIdx++)
  5443. {
  5444. renderIntent = W.map.landmarkLayer.features[llFeatureIdx].renderIntent;
  5445. if(renderIntent == 'highlight')
  5446. {
  5447. if(W.map.getExtent().intersectsBounds(W.map.landmarkLayer.features[llFeatureIdx].geometry.getBounds()))
  5448. {
  5449. if(W.map.landmarkLayer.features[llFeatureIdx].fid === null) venueObj = W.map.landmarkLayer.features[llFeatureIdx].model;
  5450. else venueObj = W.map.landmarkLayer.features[llFeatureIdx];
  5451. if(newPopupType === null)
  5452. {
  5453. if(venueObj.attributes.id === null) uroFID = venueObj.id;
  5454. else uroFID = venueObj.attributes.id;
  5455. uroAddLog('building popup for place '+uroFID);
  5456. navpointPos = uroGetVenueNavPoint(uroFID);
  5457. navpointPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  5458. result += '<b>';
  5459. if(venueObj.attributes.name === '')
  5460. {
  5461. if(venueObj.attributes.residential === true) result += '<i>Residential</i>';
  5462. else result += '<i>Unnamed</i>';
  5463. }
  5464. else result += venueObj.attributes.name;
  5465. if(venueObj.attributes.externalProviderIDs.length > 0)
  5466. {
  5467. result += ' <i>(linked)</i>';
  5468. }
  5469. if(venueObj.attributes.adLocked)
  5470. {
  5471. result += ' <i>(AdLocked)</i>';
  5472. }
  5473. result += '</b><br>';
  5474. result += uroGetAddress(venueObj.attributes.streetID, venueObj.attributes.houseNumber, false);
  5475. result += '<ul>';
  5476. for(idx = 0; idx < venueObj.attributes.categories.length; idx++)
  5477. {
  5478. result += '<li>' + I18n.lookup("venues.categories." + venueObj.attributes.categories[idx]);
  5479. }
  5480. result += '</ul>';
  5481. if(venueObj.attributes.residential === true)
  5482. {
  5483. if(venueObj.geometry.CLASS_NAME == 'OpenLayers.Geometry.Point')
  5484. {
  5485. result += '<a href="#" id="_cloneRP">Clone place</a>';
  5486. objHasCloneLink = true;
  5487. }
  5488. }
  5489. var npLink = document.location.href;
  5490. ////var npLayers = '&layers='+...
  5491. var npLayers = '';
  5492. npLink = npLink.substr(0,npLink.indexOf('?zoom'));
  5493. npLink += '?zoom=5&lat='+navpointPos.lat+'&lon='+navpointPos.lon+npLayers;
  5494.  
  5495. targetTab = "_uroTab_" + Math.round(Math.random()*1000000);
  5496. result += '<hr>Jump to nav point: <a href="'+npLink+'" id="_openInNewTab" target="'+targetTab+'">in new tab</a> - ';
  5497. objHasOpenInNewTabLink = true;
  5498. result += '<a href="#" id="_recentreSession">in this tab</a>';
  5499. objHasRecentreSessionLink = true;
  5500.  
  5501. newPopupType = 'venue';
  5502. isVenue = true;
  5503. break;
  5504. }
  5505. else
  5506. {
  5507. var otherID;
  5508. if(venueObj.attributes.id === null) otherID = venueObj.id;
  5509. else otherID = venueObj.attributes.id;
  5510. uroAddLog('venue '+otherID+' is also highlighted');
  5511. }
  5512. }
  5513. else
  5514. {
  5515. uroAddLog('landmark '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup');
  5516. }
  5517. }
  5518. else if((renderIntent == 'select') || (renderIntent == 'highlightselected'))
  5519. {
  5520. uroPlaceSelected = true;
  5521. }
  5522. }
  5523. }
  5524. // popup for map comments
  5525. if((uroMousedOverMarkerType === null) && (newPopupType === null) && (uroGetCBChecked('_cbInhibitMapCommentPopup') === false))
  5526. {
  5527. uroMCSelected = false;
  5528. var mcObj = null;
  5529. renderIntent = null;
  5530. for(var mcFeatureIdx=0; mcFeatureIdx < W.map.layers[uroMCLayerIdx].features.length; mcFeatureIdx++)
  5531. {
  5532. renderIntent = W.map.layers[uroMCLayerIdx].features[mcFeatureIdx].renderIntent;
  5533. if(renderIntent == 'highlight')
  5534. {
  5535. if(W.map.getExtent().intersectsBounds(W.map.layers[uroMCLayerIdx].features[mcFeatureIdx].geometry.getBounds()))
  5536. {
  5537. mcObj = W.map.layers[uroMCLayerIdx].features[mcFeatureIdx].model;
  5538. if(newPopupType === null)
  5539. {
  5540. uroFID = mcObj.attributes.id;
  5541. uroAddLog('building popup for map comment '+uroFID);
  5542. result += '<b>';
  5543. if(mcObj.attributes.subject === '')
  5544. {
  5545. result += '<i>No subject</i>';
  5546. }
  5547. else result += mcObj.attributes.subject;
  5548. result += '</b><br>';
  5549. if(mcObj.attributes.body !== '')
  5550. {
  5551. var mcDesc = mcObj.attributes.body.replace(/<\/?[^>]+(>|$)/g, "");
  5552. if(mcDesc != "null")
  5553. {
  5554. mcDesc = uroClickify(mcDesc);
  5555. result += mcDesc + '<br>';
  5556. }
  5557. }
  5558. var mcDaysOld = uroGetMCAge(mcObj,0,false);
  5559. var mcSubmittedTS = uroGetMCAge(mcObj,0,true);
  5560. if(mcSubmittedTS != -1)
  5561. {
  5562. mcSubmittedTS = uroGetDateTimeString(mcSubmittedTS);
  5563. }
  5564. if(mcDaysOld != -1)
  5565. {
  5566. result += '<i>Submitted ' + uroParseDaysAgo(mcDaysOld) + ' ';
  5567. if(mcSubmittedTS != -1) result += '(' + mcSubmittedTS + ') ';
  5568. if(mcObj.attributes.createdBy != null)
  5569. {
  5570. result += ' by '+uroGetUserNameAndRank(mcObj.attributes.createdBy);
  5571. }
  5572. result += '</i><br>';
  5573. }
  5574. mcDaysOld = uroGetMCAge(mcObj,1,false);
  5575. mcSubmittedTS = uroGetMCAge(mcObj,1,true);
  5576. if(mcSubmittedTS != -1)
  5577. {
  5578. mcSubmittedTS = uroGetDateTimeString(mcSubmittedTS);
  5579. }
  5580. if(mcDaysOld != -1)
  5581. {
  5582. result += '<i>Updated ' + uroParseDaysAgo(mcDaysOld) + ' ';
  5583. if(mcSubmittedTS != -1) result += '(' + mcSubmittedTS + ') ';
  5584. if(mcObj.attributes.createdBy != null)
  5585. {
  5586. result += ' by '+uroGetUserNameAndRank(mcObj.attributes.updatedBy);
  5587. }
  5588. result += '</i><br>';
  5589. }
  5590. var mcHasMyComments = false;
  5591. var mcNComments = mcObj.attributes.conversation.length;
  5592. if(mcNComments > 0)
  5593. {
  5594. for(var i=0; i<mcNComments; i++)
  5595. {
  5596. if(mcObj.attributes.conversation[i].userID == uroUserID)
  5597. {
  5598. mcHasMyComments = true;
  5599. break;
  5600. }
  5601. }
  5602. }
  5603. result += '<br>' + mcNComments +' comment';
  5604. if(mcNComments != 1) result += 's';
  5605. if((mcHasMyComments === false) && (mcNComments > 0)) result += ' (none by me)';
  5606.  
  5607. newPopupType = 'map_comment';
  5608. isMapComment = true;
  5609. break;
  5610. }
  5611. else
  5612. {
  5613. var mcOtherID = mcObj.attributes.id;
  5614. uroAddLog('map comment '+mcOtherID+' is also highlighted');
  5615. }
  5616. }
  5617. else
  5618. {
  5619. uroAddLog('map comment '+uroFID+' has renderIntent==highlight but is offscreen... blocking popup');
  5620. }
  5621. }
  5622. else if((renderIntent == 'select') || (renderIntent == 'highlightselected'))
  5623. {
  5624. uroMCSelected = true;
  5625. }
  5626. }
  5627. }
  5628.  
  5629. // look for URs, place updates and problems
  5630. if(newPopupType === null)
  5631. {
  5632. if((uroMousedOverMarkerType == 'ur') && (newPopupType === null) && (uroGetCBChecked('_cbInhibitURPopup') === false))
  5633. {
  5634. hovered = true;
  5635. isUR = true;
  5636. newPopupType = 'ur';
  5637. ureq = W.model.mapUpdateRequests.objects[uroMousedOverMarkerID];
  5638.  
  5639. unstackedX = uroParsePxString(W.map.updateRequestLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.left);
  5640. unstackedY = uroParsePxString(W.map.updateRequestLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.top);
  5641.  
  5642. // override popup base position
  5643. uroPopupX = unstackedX + popupXOffset + 6;
  5644. uroPopupY = unstackedY + popupYOffset + 66;
  5645. uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
  5646. uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
  5647.  
  5648. // check for stacking...
  5649. if(uroShownFID != uroMousedOverMarkerID)
  5650. {
  5651. uroCheckStacking(1,uroMousedOverMarkerID, unstackedX, unstackedY);
  5652. }
  5653. uroFID = uroMousedOverMarkerID;
  5654. uroAddLog('building popup for UR '+uroMousedOverMarkerID);
  5655. result = '<b>Update Request ('+uroMousedOverMarkerID+'): ' + I18n.lookup("update_requests.types." + ureq.attributes.type) + '</b><br>';
  5656. if(ureq.attributes.description !== null)
  5657. {
  5658. var desc = ureq.attributes.description.replace(/<\/?[^>]+(>|$)/g, "");
  5659. if(desc != "null")
  5660. {
  5661. desc = uroClickify(desc);
  5662. result += desc + '<br>';
  5663. }
  5664. }
  5665. var uroDaysOld = uroGetURAge(ureq,0,false);
  5666. var uroSubmittedTS = uroGetURAge(ureq,0,true);
  5667. if(uroSubmittedTS != -1)
  5668. {
  5669. uroSubmittedTS = uroGetDateTimeString(uroSubmittedTS);
  5670. }
  5671. if(uroDaysOld != -1)
  5672. {
  5673. result += '<i>Submitted ' + uroParseDaysAgo(uroDaysOld) + ' ';
  5674. if(uroSubmittedTS != -1) result += '(' + uroSubmittedTS + ') ';
  5675. if(ureq.attributes.guestUserName != null)
  5676. {
  5677. result += 'via Livemap';
  5678. if(ureq.attributes.guestUserName !== '')
  5679. {
  5680. result += ' by '+ureq.attributes.guestUserName.replace(/<\/?[^>]+(>|$)/g, "");
  5681. }
  5682. }
  5683. result += '</i>';
  5684. }
  5685. if(ureq.attributes.resolvedOn !== null)
  5686. {
  5687. uroDaysResolved = uroGetURAge(ureq,1,false);
  5688. var uroResolvedTS = uroGetURAge(ureq,1,true);
  5689. if(uroResolvedTS != -1)
  5690. {
  5691. uroResolvedTS = uroGetDateTimeString(uroResolvedTS);
  5692. }
  5693.  
  5694. if(uroDaysResolved != -1)
  5695. {
  5696. result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' ';
  5697. if(uroResolvedTS != -1) result += '(' + uroResolvedTS + ')</i>';
  5698.  
  5699. result += '<br><i>Marked as ';
  5700. if(ureq.attributes.resolution === 0) result += 'solved';
  5701. else if(ureq.attributes.resolution == 1) result += 'not identified';
  5702. else result += 'unknown';
  5703. if(ureq.attributes.resolvedBy !== null)
  5704. {
  5705. result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy);
  5706. }
  5707. result += '</i>';
  5708. }
  5709. }
  5710. if(W.model.updateRequestSessions.objects[uroMousedOverMarkerID] != null)
  5711. {
  5712. var hasMyComments = uroURHasMyComments(uroMousedOverMarkerID);
  5713. var nComments = W.model.updateRequestSessions.objects[uroMousedOverMarkerID].comments.length;
  5714. result += '<br>' + nComments + ' comment';
  5715. if(nComments != 1) result += 's';
  5716. if((hasMyComments === false) && (nComments > 0)) result += ' (none by me)';
  5717. if(nComments > 0)
  5718. {
  5719. var commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[uroMousedOverMarkerID].comments[nComments-1]);
  5720. if(commentDaysOld != -1)
  5721. {
  5722. result += ', last update '+uroParseDaysAgo(commentDaysOld);
  5723. }
  5724. }
  5725. }
  5726. }
  5727.  
  5728. if((uroMousedOverMarkerType == 'pur') && (newPopupType === null) && (uroGetCBChecked('_cbInhibitPUPopup') === false))
  5729. {
  5730. hovered = true;
  5731. isPlaceUpdate = true;
  5732. newPopupType = 'pur';
  5733. ureq = W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].model;
  5734. unstackedX = uroParsePxString(W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.left);
  5735. unstackedY = uroParsePxString(W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.top);
  5736.  
  5737. // override popup base position
  5738. uroPopupX = unstackedX + popupXOffset + 6;
  5739. uroPopupY = unstackedY + popupYOffset + 66;
  5740. uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
  5741. uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
  5742. if(uroShownFID != uroMousedOverMarkerID)
  5743. {
  5744. // check for stacking...
  5745. uroCheckStacking(3,uroMousedOverMarkerID, unstackedX, unstackedY);
  5746. }
  5747. uroFID = uroMousedOverMarkerID;
  5748. // to inhibit auto-centering only when the PUR marker is clicked, we wait for the marker to get highlighted, then
  5749. // make a copy of the original getBounds() function before replacing it with a call to W.map.getExtent(). Clicking
  5750. // the marker causes a call to getBounds() which will then return the current map extent, and thus no change in the
  5751. // map view will occur...
  5752. uroRestoreCentering();
  5753. W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.origGetBounds = W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.getBounds;
  5754. W.map.placeUpdatesLayer.markers[uroMousedOverMarkerID].model.geometry.getBounds = uroKillCentering;
  5755. uroAutoCentreDisabledOn.push('PUR', uroMousedOverMarkerID);
  5756. uroAddLog('building popup for placeUpdate '+uroMousedOverMarkerID);
  5757. result = '<b>';
  5758. if(ureq.attributes.name === '') result += 'Unnamed landmark';
  5759. else result += ureq.attributes.name;
  5760. result += '</b><br>';
  5761.  
  5762. result += '<ul>';
  5763. for(idx = 0; idx < ureq.attributes.categories.length; idx++)
  5764. {
  5765. result += '<li>' + I18n.lookup("venues.categories." + ureq.attributes.categories[idx]);
  5766. }
  5767. result += '</ul>';
  5768.  
  5769. if(ureq.attributes.residential === true)
  5770. {
  5771. result += '<i>Residential</i>';
  5772. }
  5773.  
  5774. var daysOld = uroGetPURAge(ureq);
  5775. if(daysOld != -1)
  5776. {
  5777. result += '<br><i>Submitted '+uroParseDaysAgo(daysOld)+'</i>';
  5778. }
  5779. }
  5780.  
  5781. if((uroMousedOverMarkerType == 'mp') && (newPopupType === null) && (uroGetCBChecked('_cbInhibitMPPopup') === false))
  5782. {
  5783. hovered = true;
  5784. isProblem = true;
  5785. newPopupType = 'map_problem';
  5786. ureq = W.model.problems.objects[uroMousedOverMarkerID];
  5787. if(ureq === undefined)
  5788. {
  5789. if(uroDOMHasTurnProblems)
  5790. {
  5791. ureq = W.model.turnProblems.objects[uroMousedOverMarkerID];
  5792. if(ureq != null) isTurnProb = true;
  5793. }
  5794. }
  5795. unstackedX = uroParsePxString(W.map.problemLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.left);
  5796. unstackedY = uroParsePxString(W.map.problemLayer.markers[uroMousedOverMarkerID].icon.imageDiv.style.top);
  5797.  
  5798. // override popup base position
  5799. uroPopupX = unstackedX + popupXOffset + 6;
  5800. uroPopupY = unstackedY + popupYOffset + 66;
  5801. uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
  5802. uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
  5803.  
  5804. // check for stacking...
  5805. if(uroShownFID != uroMousedOverMarkerID)
  5806. {
  5807. uroCheckStacking(2,uroMousedOverMarkerID, unstackedX, unstackedY);
  5808. }
  5809. uroFID = uroMousedOverMarkerID;
  5810. // same method of disabling the on-click auto-centre behaviour as for PURs above...
  5811. uroRestoreCentering();
  5812. W.map.problemLayer.markers[uroMousedOverMarkerID].model.origGetDisconnectBounds = W.map.problemLayer.markers[uroMousedOverMarkerID].model.getDisconnectBounds;
  5813. W.map.problemLayer.markers[uroMousedOverMarkerID].model.getDisconnectBounds = uroKillCentering;
  5814. uroAutoCentreDisabledOn.push('MP', uroMousedOverMarkerID);
  5815.  
  5816. uroAddLog('building popup for problem '+uroMousedOverMarkerID);
  5817. if(isTurnProb) result = '<b>Turn Problem ('+uroMousedOverMarkerID+'): ' + I18n.lookup("problems.types.turn.title");
  5818. else
  5819. {
  5820. result = '<b>Map Problem ('+uroMousedOverMarkerID+'): ';
  5821.  
  5822. var problemType = null;
  5823. if(uroDOMHasTurnProblems)
  5824. {
  5825. problemType = ureq.attributes.problemType;
  5826. }
  5827. else
  5828. {
  5829. problemType = ureq.attributes.subType;
  5830. }
  5831.  
  5832. if(problemType == 300)
  5833. {
  5834. result += I18n.lookup("problems.panel.closure.title");
  5835. }
  5836. else
  5837. {
  5838. if(I18n.lookup("problems.types." + problemType) === undefined) result += 'Unknown problem type ('+problemType+')';
  5839. else result += I18n.lookup("problems.types." + problemType + ".title");
  5840. }
  5841. }
  5842. result += '</b><br>';
  5843. if(ureq.attributes.description != null)
  5844. {
  5845. result += 'Description: ' + ureq.attributes.description + '<br>';
  5846. }
  5847. if(ureq.attributes.extraInfo != null)
  5848. {
  5849. result += 'ExtraInfo: ' + ureq.attributes.extraInfo + '<br>';
  5850. }
  5851. if(ureq.attributes.provider != null)
  5852. {
  5853. result += 'Provider: ' + ureq.attributes.provider + '<br>';
  5854. }
  5855. if(ureq.attributes.resolvedOn != null)
  5856. {
  5857. uroDaysResolved = uroGetURAge(ureq,1,false);
  5858. if(uroDaysResolved != -1)
  5859. {
  5860. result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' ';
  5861. if(ureq.attributes.resolvedBy != null)
  5862. {
  5863. result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy);
  5864. }
  5865.  
  5866. if((ureq.attributes.open === true) && (ureq.attributes.resolvedOn != null))
  5867. {
  5868. result += '<br>Reopened by Waze';
  5869. }
  5870. result += '</i>';
  5871. }
  5872. }
  5873. }
  5874.  
  5875. if(hovered === false)
  5876. {
  5877. uroFID = -1;
  5878. if(uroStackType !== null)
  5879. {
  5880. var tStackType = uroStackType;
  5881. uroRestackMarkers();
  5882. if(tStackType == 1) uroFilterURs();
  5883. else if(tStackType == 2) uroFilterProblems();
  5884. else if(tStackType == 3) uroFilterPlaces();
  5885. }
  5886. }
  5887. else
  5888. {
  5889. // add "open new WME tab" link
  5890. var urPos=new OpenLayers.LonLat();
  5891. if(isPlaceUpdate)
  5892. {
  5893. urPos=ureq.geometry.bounds.centerLonLat.clone();
  5894. }
  5895. else
  5896. {
  5897. if(ureq.geometry.realX === undefined)
  5898. {
  5899. urPos.lon = ureq.geometry.x;
  5900. urPos.lat = ureq.geometry.y;
  5901. }
  5902. else
  5903. {
  5904. urPos.lon = ureq.geometry.realX;
  5905. urPos.lat = ureq.geometry.realY;
  5906. }
  5907. }
  5908. urPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  5909. var urLink = document.location.href;
  5910. ////var urLayers = '&layers='+...
  5911. var urLayers = '';
  5912. urLink = urLink.substr(0,urLink.indexOf('?zoom'));
  5913. urLink += '?zoom=5&lat='+urPos.lat+'&lon='+urPos.lon+urLayers;
  5914.  
  5915. if(isUR) urLink += '&mapUpdateRequest='+idSrc;
  5916. else if(isTurnProb) urLink += '&showturn='+idSrc+'&endshow';
  5917. else if(isProblem) urLink += '&mapProblem='+idSrc;
  5918. else if(isPlaceUpdate) urLink += '&showpur='+idSrc+'&endshow';
  5919.  
  5920. targetTab = "_uroTab_" + Math.round(Math.random()*1000000);
  5921. result += '<hr><ul><li><a href="'+urLink+'" id="_openInNewTab" target="'+targetTab+'">Open in new tab</a> - ';
  5922. objHasOpenInNewTabLink = true;
  5923. result += '<a href="#" id="_recentreSession">centre in current tab</a>';
  5924. objHasRecentreSessionLink = true;
  5925.  
  5926. // add "open new livemap tab" link
  5927. var lmLink = null;
  5928. if(document.getElementById("livemap-link") != null)
  5929. {
  5930. uroAddLog('Livemap link in livemap-link id element');
  5931. lmLink = document.getElementById("livemap-link").href;
  5932. }
  5933. else if(document.getElementsByClassName("livemap-link") != null)
  5934. {
  5935. uroAddLog('Livemap link in livemap-link class element');
  5936. lmLink = document.getElementsByClassName("livemap-link")[0].href;
  5937. }
  5938. else
  5939. {
  5940. uroAddLog('Livemap link not found...');
  5941. }
  5942. if(lmLink !== null)
  5943. {
  5944. var zpos = lmLink.indexOf('?');
  5945. if(zpos > -1) lmLink = lmLink.substr(0,zpos);
  5946. lmLink += '?zoom=17&lat='+urPos.lat+'&lon='+urPos.lon+'&layers=BTTTT';
  5947. result += '<li><a href="'+lmLink+'" target="_lmTab">Open in new livemap tab</a>';
  5948. }
  5949. if(!isPlaceUpdate)
  5950. {
  5951. // add "ignore for this session" link
  5952. result += '<li><a href="#" id="_addtoignore">Hide for this session</a></ul>';
  5953. objHasIgnoreLink = true;
  5954. }
  5955. }
  5956. }
  5957. if((newPopupType != 'map_problem') && (newPopupType != 'pur'))
  5958. {
  5959. uroRestoreCentering();
  5960. }
  5961.  
  5962. // look for cameras
  5963. if((newPopupType === null) && (uroGetCBChecked('_cbInhibitCamPopup') === false))
  5964. {
  5965. for(var clFeature in W.map.camerasLayer._featureMap)
  5966. {
  5967. if(W.map.camerasLayer._featureMap[clFeature].renderIntent == 'highlight')
  5968. {
  5969. ureq = W.map.camerasLayer._featureMap[clFeature].model;
  5970. ureqID = ureq.attributes.id;
  5971.  
  5972. // test isSelected() so that we only do overview data on cameras that are being hovered over
  5973. if(ureq.isSelected() === false)
  5974. {
  5975. uroPopupY -= 20;
  5976. newPopupType = 'camera';
  5977. uroFID = ureqID;
  5978. uroAddLog('building popup for camera '+uroFID);
  5979. if(I18n.lookup("edit.camera.fields.type") === undefined)
  5980. {
  5981. result += '<b>Camera: ' + ureq.TYPES[ureq.attributes.type] + '</b><br>';
  5982. }
  5983. else
  5984. {
  5985. result += '<b>Camera: ' + I18n.lookup("edit.camera.fields.type." + ureq.attributes.type) + '</b><br>';
  5986. }
  5987. result += 'ID: '+uroFID+'<br>';
  5988. result += 'Created by ';
  5989. var userID;
  5990. if(W.model.users.get(ureq.attributes.createdBy) != null)
  5991. {
  5992. userID = ureq.attributes.createdBy;
  5993. result += uroGetUserNameAndRank(userID);
  5994. }
  5995. else result += 'unknown';
  5996. result += ', ';
  5997. var camAge = uroGetCameraAge(ureq,1);
  5998. if(camAge != -1)
  5999. {
  6000. result += uroParseDaysAgo(camAge);
  6001. }
  6002. else result += 'unknown days ago';
  6003. result += '<br>Updated by ';
  6004. if(W.model.users.get(ureq.attributes.updatedBy) != null)
  6005. {
  6006. userID = ureq.attributes.updatedBy;
  6007. var userName = W.model.users.objects[userID].userName;
  6008. var userLevel = W.model.users.objects[userID].rank + 1;
  6009. result += userName + ' (' + userLevel + ')';
  6010. }
  6011. else result += 'unknown';
  6012. result += ', ';
  6013. camAge = uroGetCameraAge(ureq,0);
  6014. if(camAge != -1)
  6015. {
  6016. result += uroParseDaysAgo(camAge);
  6017. }
  6018. else result += 'unknown days ago';
  6019. result += '<br>Speed data: ';
  6020. result += uroGetLocalisedSpeedString(ureq.attributes.speed, true);
  6021. result += '<hr><ul>';
  6022. if(uroIsCamOnWatchList(uroFID) != -1)
  6023. {
  6024. result += '<li><a href="#" id="_updatewatchlist">Update watchlist entry</a>';
  6025. result += '<li><a href="#" id="_removefromwatchlist">Remove from watchlist</a>';
  6026. objHasUpdateWatchLink = true;
  6027. objHasRemoveWatchLink = true;
  6028. }
  6029. else
  6030. {
  6031. result += '<li><a href="#" id="_addtowatchlist">Add to watchlist</a>';
  6032. objHasAddWatchLink = true;
  6033. }
  6034. if(ureq.attributes.permissions !== 0)
  6035. {
  6036. result += '<li><a href="#" id="_deleteobject">Delete Camera</a>';
  6037. objHasDeleteLink = true;
  6038. }
  6039. result += '</ul>';
  6040. }
  6041. break;
  6042. }
  6043. }
  6044. }
  6045.  
  6046. if((newPopupType !== null) && (uroPopupDwellTimer === 0) && (uroPopupSuppressed === false))
  6047. {
  6048. if((uroFID != uroShownFID) || (newPopupType != uroShownPopupType))
  6049. {
  6050. if(uroFID != uroShownFID) uroAddLog('FID mismatch, show popup: '+uroFID+'/'+uroShownFID);
  6051. else uroAddLog('Popup type mismatch: '+newPopupType+'/'+uroShownPopupType);
  6052. uroShownFID = uroFID;
  6053. uroShownPopupType = newPopupType;
  6054. uroPopupShown = false;
  6055. }
  6056. if(uroPopupShown === false)
  6057. {
  6058. uroAddLog('display popup at '+uroPopupX+','+uroPopupY);
  6059. uroPopupShown = true;
  6060. uroDiv.style.height = "auto";
  6061. uroDiv.style.width = "auto";
  6062. uroDiv.innerHTML = result;
  6063. if((uroFID != -1) && (objHasIgnoreLink === true))
  6064. {
  6065. uroAddEventListener('_addtoignore','click', uroAddToIgnoreList, true);
  6066. }
  6067. if(objHasDeleteLink === true)
  6068. {
  6069. uroAddEventListener('_deleteobject','click', uroDeleteObject, true);
  6070. }
  6071. if(objHasRemoveWatchLink === true)
  6072. {
  6073. uroAddEventListener('_removefromwatchlist','click', uroRemoveCamFromWatchList, true);
  6074. }
  6075. if(objHasAddWatchLink === true)
  6076. {
  6077. uroAddEventListener('_addtowatchlist','click', uroAddCamToWatchList, true);
  6078. }
  6079. if(objHasUpdateWatchLink === true)
  6080. {
  6081. uroAddEventListener('_updatewatchlist','click', uroUpdateCamWatchList, true);
  6082. }
  6083. if(objHasOpenInNewTabLink === true)
  6084. {
  6085. uroAddEventListener('_openInNewTab','mouseup', uroOpenNewTab, true);
  6086. }
  6087. if(objHasRecentreSessionLink === true)
  6088. {
  6089. if(isUR) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnUR, true);
  6090. else if((isProblem)||(isTurnProb)) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnMP, true);
  6091. else if(isPlaceUpdate) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnPUR, true);
  6092. else if(isVenue) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnVenueNavPoint, true);
  6093. }
  6094. if(objHasCloneLink === true)
  6095. {
  6096. uroAddEventListener('_cloneRP', 'click', uroCloneResidentialPlace, true);
  6097. }
  6098. if(newPopupType == 'turn_restriction')
  6099. {
  6100. uroAddEventListener('_editTBR','click', uroEditTBR, true);
  6101. }
  6102.  
  6103. // restrict the popup width to be no wider than just under half the window width to avoid it
  6104. // completely overlapping the marker it's associated with - by keeping it to just below half
  6105. // the window width we guarantee that it'll fit either to the left or the right of the marker
  6106. // no matter how far across the screen the marker is located...
  6107. rw = parseInt(uroDiv.clientWidth);
  6108. if(rw > (window.innerWidth * 0.45))
  6109. {
  6110. rw = (window.innerWidth * 0.45);
  6111. uroDiv.style.width = rw+'px';
  6112. }
  6113. // get the div height after any adjustment of the width above, to account for whatever content
  6114. // reflow may have occurred as a result of reducing the width...
  6115. rh = parseInt(uroDiv.clientHeight);
  6116.  
  6117. if((uroPopupX + rw) > window.innerWidth)
  6118. {
  6119. // where the popup would be off the right hand side of the screen, move it completely over to the
  6120. // other side of the mouse pointer
  6121. uroPopupX -= (rw + 20);
  6122. if(uroPopupX < 0) uroPopupX = 0;
  6123. }
  6124. if((uroPopupY + rh) > window.innerHeight)
  6125. {
  6126. // where the popup would be off the bottom of the screen, shift it up just far enough to be
  6127. // fully visible
  6128. uroPopupY -= (((uroPopupY + rh) - window.innerHeight) + 30);
  6129. if(uroPopupY < 0) uroPopupY = 0;
  6130. }
  6131. uroDiv.style.top = uroPopupY+'px';
  6132. uroDiv.style.left = uroPopupX+'px';
  6133. uroDiv.style.visibility = 'visible';
  6134. }
  6135. uroPopupTimer = -1;
  6136. }
  6137. else if((newPopupType === null) && (uroPopupDwellTimer !== 0) && (uroPopupShown === true))
  6138. {
  6139. uroHidePopup();
  6140. }
  6141. else
  6142. {
  6143. if((uroPopupTimer == -1) && (uroFID != uroShownFID))
  6144. {
  6145. uroPopupTimer = uroGetElmValue('_inputPopupEntryTimeout');
  6146. }
  6147. }
  6148. }
  6149.  
  6150. function uroExclusiveCB()
  6151. {
  6152. var cbChecked = uroGetCBChecked(this.id);
  6153.  
  6154. if(cbChecked === true)
  6155. {
  6156. var pairedList = this.attributes.pairedWith.value.split(',');
  6157. for(var i=0; i<pairedList.length; i++)
  6158. {
  6159. uroSetCBChecked(pairedList[i], false);
  6160. }
  6161. }
  6162. }
  6163.  
  6164. function uroGetAMs(e)
  6165. {
  6166. if(uroMTEMode) return;
  6167. if(!uroFilterPreamble) return;
  6168. if(!uroInitialised) return;
  6169. var amList = '';
  6170. if(W.map.managedAreasLayer.getVisibility() === true)
  6171. {
  6172. var mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
  6173. var mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top - document.getElementById('toolbar').clientHeight;
  6174. var mousePixel = new OL.Pixel(mouseX, mouseY);
  6175. var mousePoint = W.map.getLonLatFromPixel(mousePixel).toPoint();
  6176.  
  6177. for(var amObj in W.model.managedAreas.objects)
  6178. {
  6179. if(W.model.managedAreas.objects[amObj].geometry.containsPoint(mousePoint))
  6180. {
  6181. if(amList !== '') amList += ', ';
  6182. amList += uroGetUserNameAndRank(W.model.managedAreas.objects[amObj].userID);
  6183. }
  6184. }
  6185. if(amList === '')
  6186. {
  6187. amList = 'none';
  6188. }
  6189. amList = "<b>Area Managers:</b> "+amList;
  6190. }
  6191. document.getElementById("uroAMList").innerHTML = amList;
  6192. }
  6193.  
  6194. function uroMouseDown()
  6195. {
  6196. uroMouseIsDown = true;
  6197. }
  6198. function uroMouseUp()
  6199. {
  6200. uroMouseIsDown = false;
  6201. }
  6202. function uroMouseOut()
  6203. {
  6204. //console.debug('Elvis has left the building...');
  6205. }
  6206.  
  6207. function uroUREvent_onObjectsChanged()
  6208. {
  6209. }
  6210. function uroUREvent_onObjectsAdded()
  6211. {
  6212. if(uroGetCBChecked('_cbURResolverIDFilter') === true)
  6213. {
  6214. uroUpdateResolverList();
  6215. }
  6216. uroFilterURs();
  6217. }
  6218. function uroUREvent_onObjectsRemoved()
  6219. {
  6220. }
  6221.  
  6222. function uroGetSelectedURCommentCount()
  6223. {
  6224. if(W.model.updateRequestSessions.objects[uroSelectedURID] != null)
  6225. {
  6226. var cachedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
  6227. uroAddLog(uroSelectedURID+':'+cachedCommentCount+' '+uroExpectedCommentCount);
  6228.  
  6229. // if there aren't the same number of cached comments as there are comments in the UR dialog list, initiate
  6230. // a refresh of the comment data...
  6231. if(cachedCommentCount != uroExpectedCommentCount)
  6232. {
  6233. if(uroPendingCommentDataRefresh === true)
  6234. {
  6235. if(cachedCommentCount > 0)
  6236. {
  6237. uroCachedLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].comments[cachedCommentCount-1].id;
  6238. }
  6239. else
  6240. {
  6241. uroCachedLastCommentID = null;
  6242. }
  6243. uroAddLog('updateRequestSessions refresh required for UR '+uroSelectedURID);
  6244. if(uroCachedLastCommentID !== null)
  6245. {
  6246. uroAddLog('last comment ID for this UR is '+uroCachedLastCommentID);
  6247. }
  6248. else
  6249. {
  6250. uroAddLog('first comment for this UR, no previous comment to ID');
  6251. }
  6252. var idList = [];
  6253. idList.push(uroSelectedURID);
  6254. // need to delete the existing cache object first, as .get() is only capable of creating new objects,
  6255. // it doesn't seem able to update an existing object with new data
  6256. W.model.updateRequestSessions.remove(W.model.updateRequestSessions.objects[uroSelectedURID]);
  6257. W.model.updateRequestSessions.get(idList);
  6258. // the call to .get() initiates a XMLHttpRequest for the data, so we now need to switch modes - the
  6259. // refresh process has started so we're no longer pending, but we are now waiting for the XMLHttpRequest
  6260. // to return something...
  6261. uroPendingCommentDataRefresh = false;
  6262. uroWaitingCommentDataRefresh = true;
  6263. }
  6264. else
  6265. {
  6266. if(cachedCommentCount > 0)
  6267. {
  6268. var currentLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].comments[cachedCommentCount-1].id;
  6269. if(currentLastCommentID == uroCachedLastCommentID)
  6270. {
  6271. // most recent comment loaded for this UR is the same one that was present at the start of this
  6272. // refresh process, so kick back into pending mode so we can retry the .get()...
  6273. uroAddLog('latest comment ID still the same, reverting to pending mode...');
  6274. uroPendingCommentDataRefresh = true;
  6275. }
  6276. else
  6277. {
  6278. // something may have gone awry here - the most recent comment loaded for this UR doesn't have the
  6279. // same ID as the one present at the start of the refresh process, yet the comment counts still don't
  6280. // match up, which suggests either a comment got lost along the way or someone else has commented on
  6281. // the same UR at almost the same time. To get out of the loop this would create, assume that a
  6282. // mismatch in the IDs means the .get() has completed successfully no matter what the new comment
  6283. // count is, and take this new count to be the count we were expecting all along...
  6284. uroAddLog('latest comment ID different, but expected count not correct...');
  6285. uroExpectedCommentCount = cachedCommentCount;
  6286. }
  6287. }
  6288. else
  6289. {
  6290. uroAddLog('first comment on this UR not received yet, reverting to pending mode...');
  6291. uroPendingCommentDataRefresh = true;
  6292. }
  6293. }
  6294.  
  6295. }
  6296. else
  6297. {
  6298. // if the WME session is loaded with a UR already selected, such that WME has opened the UR dialog as part
  6299. // of the session startup process, adding new comments to the UR cause the cached data to be updated immediately.
  6300. // This prevents URO+ from switching into waiting mode in the above block of code, so we have to instead do
  6301. // it here by comparing the cached count against the expected count following the Send click event.
  6302. if(cachedCommentCount >= uroExpectedCommentCount)
  6303. {
  6304. uroPendingCommentDataRefresh = false;
  6305. uroWaitingCommentDataRefresh = true;
  6306. uroExpectedCommentCount = null;
  6307. }
  6308.  
  6309. // once the cached data has been updated, refilter the URs so that the new comment count is taken into account
  6310. // immediately for filtering and display purposes
  6311. if(uroWaitingCommentDataRefresh === true)
  6312. {
  6313. uroWaitingCommentDataRefresh = false;
  6314. uroFilterURs();
  6315. uroAddLog('refresh complete');
  6316. }
  6317. }
  6318. }
  6319. }
  6320.  
  6321. function uroAddedComment()
  6322. {
  6323. // when the user clicks the Send button to submit a new UR comment, this event handler fires before the new comment is
  6324. // posted to the server and thus also before the comment list gets updated in the UR dialog. So we take the current
  6325. // comment count and, if the new comment edit box isn't empty, increment it by 1 to get the expected count. Then we
  6326. // set the pending flag true to initiate a session refresh on the next 100ms tick
  6327. uroExpectedCommentCount = W.map.panelRegion.currentView.conversationView.viewModel.attributes.commentCount;
  6328. if(document.getElementsByClassName('new-comment-text')[0].value !== '')
  6329. {
  6330. uroExpectedCommentCount++;
  6331. uroAddLog('new comment added to UR '+uroSelectedURID+', cache refresh required...');
  6332. uroPendingCommentDataRefresh = true;
  6333. }
  6334. else
  6335. {
  6336. uroPendingCommentDataRefresh = false;
  6337. }
  6338. }
  6339.  
  6340. function uroInhibitNextUpdateRequestButton(e)
  6341. {
  6342. var doClick = true;
  6343. e.stopPropagation();
  6344. if(document.getElementsByClassName('form-control new-comment-text').length > 0)
  6345. {
  6346. if(document.getElementsByClassName('form-control new-comment-text')[0].textLength > 0)
  6347. {
  6348. doClick = (confirm('Comment not sent, close report panel anyway?'));
  6349. }
  6350. }
  6351. if(doClick)
  6352. {
  6353. document.getElementsByClassName('close-panel')[0].click();
  6354. }
  6355. }
  6356.  
  6357. function uroAddLZ(valueToPad, newLength)
  6358. {
  6359. var padString = '';
  6360. for(var i=0; i<newLength; i++)
  6361. {
  6362. padString += '0';
  6363. }
  6364. padString += valueToPad.toString();
  6365. return padString.slice(-newLength);
  6366. }
  6367.  
  6368. function uroIncrementClosureDate(oldDate, incByDays)
  6369. {
  6370. var dateBits = oldDate.split('-');
  6371. var year = parseInt(dateBits[0]);
  6372. var month = parseInt(dateBits[1])-1;
  6373. var date = parseInt(dateBits[2])+1;
  6374. var incrementedDate = new Date(year, month, date);
  6375. return (uroAddLZ(incrementedDate.getFullYear(),4) + '-' + uroAddLZ(incrementedDate.getMonth()+1,2) + '-' + uroAddLZ(incrementedDate.getDate(),2));
  6376. }
  6377.  
  6378. function uroGetElementProperty(elmName, elmOffset, elmProperty)
  6379. {
  6380. var retval = null;
  6381. if(document.getElementsByName(elmName).length !== 0)
  6382. {
  6383. retval = document.getElementsByName(elmName)[elmOffset][elmProperty];
  6384. }
  6385. return retval;
  6386. }
  6387.  
  6388. // Residential Place Cloning
  6389. //{
  6390. var uroCRPStreetID;
  6391. var uroCRPHouseNumber;
  6392. function uroCompleteRPClone()
  6393. {
  6394. // as with closure cloning, the place details edit form requires us to push the new value into the relevant
  6395. // edit field and then generate a change event on that field, otherwise WME doesn't bother reading the value...
  6396. // street name
  6397. var streetObj = W.model.streets.get(uroCRPStreetID);
  6398. if(streetObj !== undefined)
  6399. {
  6400. document.getElementsByClassName('street-name')[0].value = streetObj.name;
  6401. document.getElementsByClassName('street-name')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
  6402. // city name
  6403. var cityObj = W.model.cities.get(streetObj.cityID);
  6404. if(cityObj !== undefined)
  6405. {
  6406. if(cityObj.attributes.isEmpty === true)
  6407. {
  6408. // The donor point place we create to take the cloned RPP properties may have been automatically given
  6409. // a city name by WME, and thus the city name field will already be filled in and activated... If our
  6410. // RPP doesn't however have a city name, we need to deactivate the city name field again, so that WME
  6411. // doesn't complain about the user trying to save the new RPP with an empty city name
  6412. if(document.getElementsByClassName("empty-city")[0].checked === false)
  6413. {
  6414. document.getElementsByClassName("empty-city")[0].click();
  6415. }
  6416. }
  6417. document.getElementsByClassName('city-name')[0].value = cityObj.attributes.name;
  6418. document.getElementsByClassName('city-name')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
  6419. // county
  6420. document.getElementsByClassName('state-id')[0].value = cityObj.attributes.stateID;
  6421. document.getElementsByClassName('state-id')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
  6422. // country
  6423. document.getElementsByClassName('country-id')[0].value = cityObj.attributes.countryID;
  6424. document.getElementsByClassName('country-id')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
  6425. }
  6426. }
  6427. // house number
  6428. document.getElementsByClassName('house-number')[0].value = uroCRPHouseNumber;
  6429. document.getElementsByClassName('house-number')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
  6430.  
  6431. // now wait for the user to confirm everything and click Apply...
  6432. }
  6433. function uroConvertToRP()
  6434. {
  6435. // panel isn't open yet, which means the user either hasn't clicked yet or WME is still processing the
  6436. // placement of the venue, so wait a while and then check again...
  6437. if(document.getElementById('edit-panel').getElementsByClassName('landmark').length === 0)
  6438. {
  6439. setTimeout(uroConvertToRP, 100);
  6440. return;
  6441. }
  6442. // panel is open, so move to the next step of the cloning procedure by converting the newly created
  6443. // place to residential by generating a click event on the "convert to residential" link...
  6444. document.getElementsByClassName("toggle-residential")[0].click();
  6445. // and then click on the address edit icon...
  6446. document.getElementsByClassName('waze-icon-edit')[0].click();
  6447. // now click on the "none" checkbox for the street name edit field so we can enter the street name
  6448. document.getElementById('empty-street').click();
  6449. // WME automatically clears the checkbox associated with the city name edit field if we set the street
  6450. // name to be one that has a city associated with it, which is nice :-)
  6451. // the click event seems to take a while to execute, and if we call dispatchEvent on the edit field whilst
  6452. // it's still tagged as disabled then it gets ignored, causing the value in that field to be dropped when
  6453. // we apply the changes to the place. Trying to programatically detect when the field has been activated
  6454. // doesn't seem to be reliable, however a fixed delay of 1s seems to work nicely
  6455. setTimeout(uroCompleteRPClone, 1000);
  6456. }
  6457. function uroCloneResidentialPlace()
  6458. {
  6459. // trying to clone a RPP when one is already selected causes the selected one to be changed back to
  6460. // a non-residential, as uroConvertToRP() thinks the user has already clicked to place the new RPP...
  6461. if(document.getElementById('edit-panel').getElementsByClassName('landmark').length === 0)
  6462. {
  6463. var venueObj = W.model.venues.objects[uroFID];
  6464. if(venueObj !== undefined)
  6465. {
  6466. // copy address from highlighted residential place
  6467. uroCRPHouseNumber = venueObj.attributes.houseNumber;
  6468. uroCRPStreetID = venueObj.attributes.streetID;
  6469.  
  6470. // generate a click event on the first new point venue entry in the venues menu in order to generate a
  6471. // new point venue object that we can manipulate...
  6472. document.getElementsByClassName('toolbar-group-venues')[0].getElementsByClassName("drawing-control main-control point")[0].click();
  6473.  
  6474. // now wait for the user to click on the map to place the new point venue
  6475. uroConvertToRP();
  6476. }
  6477. }
  6478. }
  6479. //}
  6480.  
  6481. // Closure Cloning
  6482. //{
  6483. var uroConfirmClosureDelete = true;
  6484. var uroClosuresToDelete = 0;
  6485. var uroCLocation;
  6486. var uroCReason;
  6487. var uroCEvent;
  6488. var uroCDirection;
  6489. var uroCHasStartDate;
  6490. var uroCStartDate;
  6491. var uroCStartTime;
  6492. var uroCEndDate;
  6493. var uroCEndTime;
  6494. var uroCIgnoreTraffic;
  6495.  
  6496. function uroCompleteClosureCloning()
  6497. {
  6498. if(document.getElementsByClassName('edit-closure').length === 0)
  6499. {
  6500. window.setTimeout(uroCompleteClosureCloning,100);
  6501. return;
  6502. }
  6503.  
  6504. // need to generate a change event on each of the form fields, because WME appears to be silently populating some hidden
  6505. // closure object with the details as they're entered manually, and if we just set the form values without then forcing
  6506. // the change event as well then WME will end up using its default values instead of the ones we've so lovingly copied...
  6507.  
  6508. if(uroCLocation !== null)
  6509. {
  6510. document.getElementsByName('closure_location')[0].value = uroCLocation;
  6511. document.getElementsByName('closure_location')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6512. }
  6513. if(uroCReason !== null)
  6514. {
  6515. document.getElementsByName('closure_reason')[0].value = uroCReason;
  6516. document.getElementsByName('closure_reason')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6517. }
  6518. if(uroCDirection !== null)
  6519. {
  6520. document.getElementsByName('closure_direction')[0].selectedIndex = uroCDirection;
  6521. document.getElementsByName('closure_direction')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6522. }
  6523. if(uroCHasStartDate !== null)
  6524. {
  6525. document.getElementsByName('closure_hasStartDate')[0].checked = uroCHasStartDate;
  6526. document.getElementsByName('closure_hasStartDate')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6527. }
  6528. if(uroCStartDate !== null)
  6529. {
  6530. document.getElementsByName('closure_startDate')[0].value = uroCStartDate;
  6531. document.getElementsByName('closure_startDate')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6532. }
  6533. if(uroCStartTime !== null)
  6534. {
  6535. document.getElementsByName('closure_startTime')[0].value = uroCStartTime;
  6536. document.getElementsByName('closure_startTime')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6537. }
  6538. if(uroCIgnoreTraffic !== null)
  6539. {
  6540. document.getElementsByName('closure_permanent')[0].checked = uroCIgnoreTraffic;
  6541. document.getElementsByName('closure_permanent')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6542. }
  6543. if(uroCEndTime !== null)
  6544. {
  6545. document.getElementsByName('closure_endTime')[0].value = uroCEndTime;
  6546. // the cloning process doesn't alter the end time, which seems to confuse WME when it then receives the
  6547. // change event - the MTE dropdown ends up being reset to just the "Choose event" and "None" entries
  6548. // regardless of how many MTE entries there ought to be. The fix for this appears to simply be to then
  6549. // submit a second change event...
  6550. document.getElementsByName('closure_endTime')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6551. document.getElementsByName('closure_endTime')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6552. }
  6553.  
  6554. // the current version of WME wipes any existing end date as soon as the end time is altered, so we now need
  6555. // to set the date after the time instead of before as in earlier versions of this function...
  6556. if(uroCEndDate !== null)
  6557. {
  6558. document.getElementsByName('closure_endDate')[0].value = uroCEndDate;
  6559. document.getElementsByName('closure_endDate')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6560. }
  6561. uroTempFixMTEDropDown();
  6562. if(uroCEvent !== null)
  6563. {
  6564. if(document.getElementsByName('closure_eventId')[0].options.length > 1)
  6565. {
  6566. for(var loop=0; loop<document.getElementsByName('closure_eventId')[0].options.length; loop++)
  6567. {
  6568. if(document.getElementsByName('closure_eventId')[0].options[loop].value == uroCEvent)
  6569. {
  6570. document.getElementsByName('closure_eventId')[0].selectedIndex = loop;
  6571. break;
  6572. }
  6573. }
  6574. }
  6575. else
  6576. {
  6577. document.getElementsByName('closure_eventId')[0].selectedIndex = 0;
  6578. }
  6579. document.getElementsByName('closure_eventId')[0].dispatchEvent(new Event('change', {'bubbles':true}));
  6580. }
  6581. }
  6582. function uroCloneClosure()
  6583. {
  6584. var closureOffset = parseInt(this.id.split('-')[1]);
  6585. // grab the current closure details from the UI...
  6586. document.getElementsByClassName('closure-item')[closureOffset].children[0].children[0].children[1].click();
  6587. uroCLocation = uroGetElementProperty('closure_location', 0, 'value');
  6588. uroCReason = uroGetElementProperty('closure_reason', 0, 'value');
  6589. uroCEvent = uroGetElementProperty('closure_eventId', 0, 'value');
  6590. uroCDirection = uroGetElementProperty('closure_direction', 0, 'selectedIndex');
  6591. uroCHasStartDate = uroGetElementProperty('closure_hasStartDate', 0, 'checked');
  6592. uroCStartDate = uroGetElementProperty('closure_startDate', 0, 'value');
  6593. uroCStartTime = uroGetElementProperty('closure_startTime', 0, 'value');
  6594. uroCEndDate = uroGetElementProperty('closure_endDate', 0, 'value');
  6595. uroCEndTime = uroGetElementProperty('closure_endTime', 0, 'value');
  6596. uroCIgnoreTraffic = uroGetElementProperty('closure_permanent', 0, 'checked');
  6597. document.getElementsByClassName('closures')[0].getElementsByClassName('cancel-button')[0].click();
  6598. // auto-increment the start and end dates...
  6599. uroCStartDate = uroIncrementClosureDate(uroCStartDate,1);
  6600. uroCEndDate = uroIncrementClosureDate(uroCEndDate,1);
  6601. // generate a click event on the Add a closure button to open up the closure editing UI, then
  6602. // wait for the UI to finish opening...
  6603. document.getElementsByClassName('add-closure-button')[0].click();
  6604. window.setTimeout(uroCompleteClosureCloning,100);
  6605. }
  6606.  
  6607. function uroDeleteNextClosureOnList()
  6608. {
  6609. var nClosures = document.getElementsByClassName('closure-item').length;
  6610. if(nClosures > 0)
  6611. {
  6612. if (nClosures != uroClosuresToDelete)
  6613. {
  6614. uroClosuresToDelete = nClosures;
  6615. document.getElementsByClassName('closure-item')[0].getElementsByClassName('delete')[0].click();
  6616. }
  6617. setTimeout(uroDeleteNextClosureOnList,100);
  6618. }
  6619. else
  6620. {
  6621. uroConfirmClosureDelete = true;
  6622. }
  6623. }
  6624.  
  6625. function uroDeleteAllClosures()
  6626. {
  6627. uroConfirmClosureDelete = true;
  6628. if(window.confirm(I18n.lookup("closures.delete_confirm_no_reason")+' ('+I18n.lookup("closures.apply_to_all")+')'))
  6629. {
  6630. uroConfirmClosureDelete = false;
  6631. var nClosures = document.getElementsByClassName('closure-item').length;
  6632. if(nClosures > 0)
  6633. {
  6634. uroClosuresToDelete = -1;
  6635. uroDeleteNextClosureOnList();
  6636. }
  6637. else
  6638. {
  6639. uroConfirmClosureDelete = true;
  6640. }
  6641. }
  6642. }
  6643. //}
  6644.  
  6645. // Feed Filtering
  6646. //{
  6647. function uroToggleFFCtrls()
  6648. {
  6649. if(uroShowFeedFilter === false)
  6650. {
  6651. uroShowFeedFilter = true;
  6652. document.getElementById('_uroFFCtrlVisibility').className = "fa fa-minus-square-o";
  6653. document.getElementById('uroFFCtrls').style.display = "block";
  6654. }
  6655. else
  6656. {
  6657. uroShowFeedFilter = false;
  6658. document.getElementById('_uroFFCtrlVisibility').className = "fa fa-plus-square-o";
  6659. document.getElementById('uroFFCtrls').style.display = "none";
  6660. }
  6661. }
  6662.  
  6663. function uroForceFeedRefresh()
  6664. {
  6665. uroFeedFilterReloads = 0;
  6666. }
  6667. function uroAddFeedFilterControls()
  6668. {
  6669. if(document.getElementById('sidepanel-feed') != null)
  6670. {
  6671. if(document.getElementById('sidepanel-feed').childNodes[0] != null)
  6672. {
  6673. var nDiv = document.createElement('div');
  6674. var iHTML = '';
  6675. nDiv.id = "uroFeedFilter";
  6676. iHTML += '<i class="fa fa-plus-square-o" style="cursor:pointer;font-size:14px;" id="_uroFFCtrlVisibility"> </i><b>Feed Filter Controls</b><br>';
  6677. iHTML += '<div id="uroFFCtrls">';
  6678. iHTML += '<b>Filter feed by listing type:</b><br>';
  6679. iHTML += '<input type="checkbox" id="_cbFeedFilter_TypeUR" />UR<br>';
  6680. iHTML += '<input type="checkbox" id="_cbFeedFilter_TypeMP" />MP<br>';
  6681. iHTML += '<input type="checkbox" id="_cbFeedFilter_TypePUR" />PUR<br>';
  6682. iHTML += '<input type="checkbox" id="_cbFeedFilter_TypePM" />PM notifications<br>';
  6683. iHTML += '<br><b>Filter feed by listing reason:</b><br>';
  6684. for(var loop=0; loop < uroFeedFilterFilters.length; loop++)
  6685. {
  6686. iHTML += '<input type="checkbox" id="_cbFeedFilter_'+loop+'" />'+I18n.lookup(uroFeedFilterFilters[loop][0])+'<br>';
  6687. }
  6688. iHTML += '<input type="checkbox" id="_cbFeedFilter_MotNone" />None of the above...<br>';
  6689. iHTML += '<br><input type="checkbox" id="_cbFeedFilter_Invert" />Invert behaviour of above filters<br>';
  6690. iHTML += '<br><b>Filter feed by keyword/phrase:</b><br>';
  6691. iHTML += '<input type="checkbox" id="_cbFeedFilter_HideKeyword" pairedWith="_cbFeedFilter_ShowKeyword" />Hide or ';
  6692. iHTML += '<input type="checkbox" id="_cbFeedFilter_ShowKeyword" pairedWith="_cbFeedFilter_HideKeyword" />show if keyword is present<br>';
  6693. iHTML += '<input type="text" id="_cbFeedFilter_Keyword" />';
  6694.  
  6695. iHTML += '</div>';
  6696. nDiv.innerHTML = iHTML;
  6697. document.getElementById('sidepanel-feed').insertBefore(nDiv,document.getElementById('sidepanel-feed').childNodes[0]);
  6698. uroAddEventListener('_uroFFCtrlVisibility','click',uroToggleFFCtrls, true);
  6699. uroShowFeedFilter = true;
  6700. uroToggleFFCtrls();
  6701. var nDiv2 = document.createElement('div');
  6702. nDiv2.id = "uroFeedRefresher";
  6703. nDiv2.style.display = 'none';
  6704. nDiv2.innerHTML = '<br><div id="_btnFeedRefresh" class="btn btn-block btn-default" style="display: block;"><i class="fa fa-refresh"></div>';
  6705. document.getElementById('sidepanel-feed').appendChild(nDiv2);
  6706. uroAddEventListener('_btnFeedRefresh','click',uroForceFeedRefresh, true);
  6707. }
  6708. }
  6709. }
  6710. function uroTSTFeedFilter()
  6711. {
  6712. var feedEntries = document.getElementsByClassName('feed-item');
  6713. var feedLength = feedEntries.length;
  6714. if(feedLength === 0) return;
  6715. if(document.getElementById('uroFeedFilter') === null) return;
  6716. var hideFI;
  6717. var iHTML;
  6718. var fClass;
  6719. var ffKeyword = uroGetElmValue('_cbFeedFilter_Keyword');
  6720. var ffShowKW = uroGetCBChecked('_cbFeedFilter_ShowKeyword');
  6721. var ffHideKW = uroGetCBChecked('_cbFeedFilter_HideKeyword');
  6722. var kwPresent;
  6723. var isChecked_cbFeedFilter_TypeUR = uroGetCBChecked('_cbFeedFilter_TypeUR');
  6724. var isChecked_cbFeedFilter_TypeMP = uroGetCBChecked('_cbFeedFilter_TypeMP');
  6725. var isChecked_cbFeedFilter_TypePUR = uroGetCBChecked('_cbFeedFilter_TypePUR');
  6726. var isChecked_cbFeedFilter_TypePM = uroGetCBChecked('_cbFeedFilter_TypePM');
  6727. var isChecked_cbFeedFilter_MotNone = uroGetCBChecked('_cbFeedFilter_MotNone');
  6728. var isChecked_cbFeedFilter_Invert = uroGetCBChecked('_cbFeedFilter_Invert');
  6729.  
  6730. var trans = [];
  6731. var isChecked = [];
  6732. var nFilters = uroFeedFilterFilters.length;
  6733. for(var loop=0; loop < nFilters; loop++)
  6734. {
  6735. trans.push(I18n.lookup(uroFeedFilterFilters[loop][0]));
  6736. isChecked.push(uroGetCBChecked('_cbFeedFilter_'+loop));
  6737. }
  6738.  
  6739. for(var i=0; i<feedLength; i++)
  6740. {
  6741. hideFI = false;
  6742. iHTML = feedEntries[i].innerHTML;
  6743. fClass = feedEntries[i].className;
  6744. if(isChecked_cbFeedFilter_TypeUR)
  6745. {
  6746. hideFI |= (fClass.indexOf('feed-issue-ur') != -1);
  6747. }
  6748. if(isChecked_cbFeedFilter_TypeMP)
  6749. {
  6750. hideFI |= (fClass.indexOf('feed-issue-mp') != -1);
  6751. }
  6752. if(isChecked_cbFeedFilter_TypePUR)
  6753. {
  6754. hideFI |= (fClass.indexOf('feed-issue-pu') != -1);
  6755. }
  6756. if(isChecked_cbFeedFilter_TypePM)
  6757. {
  6758. hideFI |= (fClass.indexOf('feed-notification-pm') != -1);
  6759. }
  6760. for(var filters=0; filters < nFilters; filters++)
  6761. {
  6762. if(isChecked[filters])
  6763. {
  6764. hideFI |= (iHTML.indexOf(trans[filters]) != -1);
  6765. }
  6766. }
  6767. if(isChecked_cbFeedFilter_MotNone)
  6768. {
  6769. hideFI |= (iHTML.indexOf('motivation') == -1);
  6770. }
  6771. if(isChecked_cbFeedFilter_Invert)
  6772. {
  6773. hideFI = !hideFI;
  6774. }
  6775. if((ffShowKW) || (ffHideKW))
  6776. {
  6777. kwPresent = (iHTML.indexOf(ffKeyword) != -1);
  6778. hideFI |= (kwPresent & ffHideKW);
  6779. hideFI |= ((!kwPresent) & ffShowKW);
  6780. }
  6781. if(hideFI) feedEntries[i].style.display = 'none';
  6782. else feedEntries[i].style.display = 'block';
  6783. }
  6784. var nFeedItems = document.getElementById('sidepanel-feed').getElementsByClassName('feed-item').length;
  6785. var nVisibleItems = 0;
  6786. for(var j=0; j < nFeedItems; j++)
  6787. {
  6788. if(document.getElementById('sidepanel-feed').getElementsByClassName('feed-item')[j].style.display == 'block')
  6789. {
  6790. nVisibleItems++;
  6791. }
  6792. }
  6793. if(nVisibleItems < 5)
  6794. {
  6795. if(document.getElementById('sidepanel-feed').getElementsByClassName('feed-loading-more')[0].style.display == 'none')
  6796. {
  6797. if(uroFeedFilterReloads < 5)
  6798. {
  6799. document.getElementById('sidepanel-feed').getElementsByClassName('feed-load-more')[0].click();
  6800. uroFeedFilterReloads++;
  6801. document.getElementById('uroFeedRefresher').style.display = 'none';
  6802. }
  6803. else
  6804. {
  6805. document.getElementById('uroFeedRefresher').style.display = 'block';
  6806. }
  6807. }
  6808. }
  6809. else
  6810. {
  6811. uroFeedFilterReloads = 0;
  6812. document.getElementById('uroFeedRefresher').style.display = 'none';
  6813. }
  6814. }
  6815. //}
  6816.  
  6817. function uroGetMarkerType(className)
  6818. {
  6819. var markerType = null;
  6820. if(className.indexOf('user-generated') !== -1) markerType = 'ur';
  6821. else if(className.indexOf('map-problem') !== -1) markerType = 'mp';
  6822. else if(className.indexOf('place-update') !== -1) markerType = 'pur';
  6823. return markerType;
  6824. }
  6825. function uroMarkerMouseOver(e)
  6826. {
  6827. var markerType;
  6828. markerType = uroGetMarkerType(this.className);
  6829. if(markerType !== null)
  6830. {
  6831. var markerID = null;
  6832. markerID = this.attributes["data-id"].value;
  6833. uroAddLog('hover over '+markerType+' ID '+markerID);
  6834. uroMousedOverMarkerID = markerID;
  6835. uroMousedOverMarkerType = markerType;
  6836. if(markerType == 'ur') uroHoveredURID = markerID;
  6837. }
  6838. else
  6839. {
  6840. uroAddLog('hover over unknown object...');
  6841. }
  6842. }
  6843. function uroMarkerMouseOut(e)
  6844. {
  6845. var markerType;
  6846. markerType = uroGetMarkerType(this.className);
  6847. if(markerType !== null)
  6848. {
  6849. var markerID = null;
  6850. markerID = this.attributes["data-id"].value;
  6851. uroAddLog('hover off '+markerType+' ID '+markerID);
  6852. uroMousedOverMarkerID = null;
  6853. uroMousedOverMarkerType = null;
  6854. uroHoveredURID = null;
  6855. }
  6856. else
  6857. {
  6858. uroAddLog('hover off unknown object...');
  6859. }
  6860. }
  6861. function uroMarkerClick()
  6862. {
  6863. var markerType = uroGetMarkerType(this.className);
  6864. if(markerType !== null)
  6865. {
  6866. var markerID = this.attributes["data-id"].value;
  6867. uroAddLog('clicked on '+markerType+' marker '+markerID);
  6868. uroClickedOnMarkerID = markerID;
  6869. uroClickedOnMarkerType = markerType;
  6870. }
  6871. }
  6872.  
  6873. function uroBlobMouseOver(e)
  6874. {
  6875. var blobType = this.attributes.uroBlobType;
  6876. if(blobType !== undefined)
  6877. {
  6878. var blobID = this.attributes.uroBlobID;
  6879. uroAddLog('hover over '+blobType+' ID '+blobID);
  6880. //uroMousedOverMarkerID = markerID;
  6881. //uroMousedOverMarkerType = markerType;
  6882. }
  6883. else
  6884. {
  6885. uroAddLog('hover over unknown blob...');
  6886. }
  6887. }
  6888. function uroBlobMouseOut(e)
  6889. {
  6890. var blobType = this.attributes.uroBlobType;
  6891. if(blobType !== undefined)
  6892. {
  6893. var blobID = this.attributes.uroBlobID;
  6894. uroAddLog('hover off '+blobType+' ID '+blobID);
  6895. //uroMousedOverMarkerID = markerID;
  6896. //uroMousedOverMarkerType = markerType;
  6897. }
  6898. else
  6899. {
  6900. uroAddLog('hover off unknown blob...');
  6901. }
  6902. }
  6903. function uroBlobClick()
  6904. {
  6905. var blobType = this.attributes.uroBlobType;
  6906. if(blobType !== undefined)
  6907. {
  6908. var blobID = this.attributes.uroBlobID;
  6909. uroAddLog('clicked on '+blobType+' blob '+blobID);
  6910. }
  6911. }
  6912.  
  6913. function uroMCLayerChanged()
  6914. {
  6915. if(uroMCLayerIdx != null)
  6916. {
  6917. uroAddLog('adding MC blob event handlers');
  6918. for(var mObj=0; mObj<W.map.layers[uroMCLayerIdx].features.length; mObj++)
  6919. {
  6920. var mcBlobID = W.map.layers[uroMCLayerIdx].features[mObj].model.attributes.geometry.id;
  6921. var mcID = W.map.layers[uroMCLayerIdx].features[mObj].model.attributes.id;
  6922. var mcBlob = document.getElementById(mcBlobID);
  6923. if(mcBlob !== null)
  6924. {
  6925. mcBlob.addEventListener("mouseover", uroBlobMouseOver, false);
  6926. mcBlob.addEventListener("mouseout", uroBlobMouseOut, false);
  6927. mcBlob.addEventListener("click", uroBlobClick, false);
  6928. mcBlob.attributes.uroBlobID = mcID;
  6929. mcBlob.attributes.uroBlobType = "map_comment";
  6930. }
  6931. }
  6932. }
  6933. }
  6934. function uroPlaceLayerChanged()
  6935. {
  6936. uroAddLog('adding place blob event handlers');
  6937. for(var mObj=0; mObj<W.map.landmarkLayer.features.length; mObj++)
  6938. {
  6939. var mcBlobID = W.map.landmarkLayer.features[mObj].model.attributes.geometry.id;
  6940. var mcID = W.map.landmarkLayer.features[mObj].model.attributes.id;
  6941. var mcBlob = document.getElementById(mcBlobID);
  6942. if(mcBlob !== null)
  6943. {
  6944. mcBlob.addEventListener("mouseover", uroBlobMouseOver, false);
  6945. mcBlob.addEventListener("mouseout", uroBlobMouseOut, false);
  6946. mcBlob.addEventListener("click", uroBlobClick, false);
  6947. mcBlob.attributes.uroBlobID = mcID;
  6948. mcBlob.attributes.uroBlobType = "place";
  6949. }
  6950. }
  6951. }
  6952. function uroURLayerChanged()
  6953. {
  6954. uroAddLog('adding UR marker event handlers');
  6955. for(var mObj in W.map.updateRequestLayer.markers)
  6956. {
  6957. var mIcon = W.map.updateRequestLayer.markers[mObj].icon.div;
  6958. mIcon.addEventListener("mouseover",uroMarkerMouseOver, false);
  6959. mIcon.addEventListener("mouseout",uroMarkerMouseOut, false);
  6960. mIcon.addEventListener("click",uroMarkerClick, false);
  6961. }
  6962. }
  6963. function uroMPLayerChanged()
  6964. {
  6965. uroAddLog('adding MP marker event handlers');
  6966. for(var mObj in W.map.problemLayer.markers)
  6967. {
  6968. var mIcon = W.map.problemLayer.markers[mObj].icon.div;
  6969. mIcon.addEventListener("mouseover", uroMarkerMouseOver, false);
  6970. mIcon.addEventListener("mouseout", uroMarkerMouseOut, false);
  6971. mIcon.addEventListener("click", uroMarkerClick, false);
  6972. }
  6973. }
  6974. function uroPURLayerChanged()
  6975. {
  6976. uroAddLog('adding PUR marker event handlers');
  6977. for(var mObj in W.map.placeUpdatesLayer.markers)
  6978. {
  6979. var mIcon = W.map.placeUpdatesLayer.markers[mObj].icon.div;
  6980. mIcon.addEventListener("mouseover", uroMarkerMouseOver, false);
  6981. mIcon.addEventListener("mouseout", uroMarkerMouseOut, false);
  6982. mIcon.addEventListener("click", uroMarkerClick, false);
  6983. }
  6984. }
  6985. function uroFinalizeListenerSetup()
  6986. {
  6987. uroFinalisingListenerSetup = true;
  6988. // filter markers when the marker objects are modified (this happens whenever WME needs to load fresh marker data
  6989. // due to having panned/zoomed the map beyond the extents of the previously loaded data)
  6990. W.model.mapUpdateRequests.on("objectschanged", uroFilterURs_onObjectsChanged);
  6991. W.model.mapUpdateRequests.on("objectsadded", uroFilterURs_onObjectsAdded);
  6992. W.model.mapUpdateRequests.on("objectsremoved", uroFilterURs_onObjectsRemoved);
  6993.  
  6994. W.model.updateRequestSessions.on("objectschanged", uroUREvent_onObjectsChanged);
  6995. W.model.updateRequestSessions.on("objectsadded", uroUREvent_onObjectsAdded);
  6996. W.model.updateRequestSessions.on("objectsremoved", uroUREvent_onObjectsRemoved);
  6997.  
  6998. W.model.cameras.on("objectschanged", uroFilterCameras);
  6999. W.model.cameras.on("objectsadded", uroFilterCameras);
  7000. W.model.cameras.on("objectsremoved", uroFilterCameras);
  7001.  
  7002. W.model.problems.on("objectschanged", uroFilterProblems);
  7003. W.model.problems.on("objectsadded", uroFilterProblems);
  7004. W.model.problems.on("objectsremoved", uroFilterProblems);
  7005. W.model.venues.on("objectschanged", uroFilterPlaces);
  7006. W.model.venues.on("objectsadded", uroFilterPlaces);
  7007. W.model.venues.on("objectsremoved", uroFilterPlaces);
  7008. ////var uroMO_MCLayer = new MutationObserver(uroMCLayerChanged);
  7009. ////uroMO_MCLayer.observe(W.map.layers[uroMCLayerIdx].div,{childList: true, attributes : true, characterData: true, subtree: true});
  7010. ////var uroMO_PlaceLayer = new MutationObserver(uroPlaceLayerChanged);
  7011. ////uroMO_PlaceLayer.observe(W.map.landmarkLayer.div,{childList: true, attributes : true, characterData : true, subtree: true});
  7012. var uroMO_URLayer = new MutationObserver(uroURLayerChanged);
  7013. uroMO_URLayer.observe(W.map.updateRequestLayer.div,{childList : true});
  7014. var uroMO_MPLayer = new MutationObserver(uroMPLayerChanged);
  7015. uroMO_MPLayer.observe(W.map.problemLayer.div,{childList : true});
  7016. var uroMO_PURLayer = new MutationObserver(uroPURLayerChanged);
  7017. uroMO_PURLayer.observe(W.map.placeUpdatesLayer.div,{childList : true});
  7018. var userTabs = document.getElementById(uroUserTabId);
  7019. var tabContent = null;
  7020.  
  7021. var navTabs = userTabs.getElementsByClassName('nav-tabs')[0];
  7022. tabContent = document.getElementById('user-info').getElementsByClassName('tab-content')[0];
  7023. var newtabUR = document.createElement('li');
  7024. newtabUR.innerHTML = '<a href="#sidepanel-uroverview" data-toggle="tab">URO+</a>';
  7025. navTabs.appendChild(newtabUR);
  7026. uroControls.id = "sidepanel-uroverview";
  7027. uroControls.className = "tab-pane";
  7028. tabContent.appendChild(uroControls);
  7029.  
  7030. uroAddEventListener('_btnUndoLastHide',"click", uroRemoveLastAddedIgnore, true);
  7031. uroAddEventListener('_btnClearSessionHides',"click", uroRemoveAllIgnores, true);
  7032. uroEnableIgnoreListControls();
  7033.  
  7034. uroAddEventListener('_btnClearCamWatchList',"click", uroClearCamWatchList, true);
  7035. uroAddEventListener('_btnSettingsToText',"click", uroSettingsToText, true);
  7036. uroAddEventListener('_btnTextToSettings',"click", uroTextToSettings, true);
  7037. uroAddEventListener('_btnResetSettings',"click", uroDefaultSettings, true);
  7038. uroAddEventListener('_btnClearSettingsText',"click", uroClearSettingsText, true);
  7039. uroAddEventListener('_cbMasterEnable',"click", uroFilterItems_MasterEnableClick, true);
  7040. /*
  7041. uroAddEventListener('_btnDebugToScreen',"click", uroDumpDebug, true);
  7042. */
  7043. uroAddEventListener('uroDiv',"dblclick",uroSuppressPopup,true);
  7044. uroAddEventListener('_selectCameraUserID',"change", uroCamEditorSelected, true);
  7045. uroAddEventListener('_selectPlacesUserID',"change", uroPlacesEditorSelected, true);
  7046. uroAddEventListener('uroAlertTickBtn','click',uroCloseAlertBoxWithTick,true);
  7047. uroAddEventListener('uroAlertCrossBtn','click',uroCloseAlertBoxWithCross,true);
  7048. uroSetOnClick("_linkSelectUserRequests",uroShowURTab);
  7049. uroSetOnClick("_linkSelectMapProblems",uroShowMPTab);
  7050. uroSetOnClick("_linkSelectPlaces",uroShowPlacesTab);
  7051. uroSetOnClick("_linkSelectCameras",uroShowCameraTab);
  7052. uroSetOnClick("_linkSelectMisc",uroShowMiscTab);
  7053. uroSetOnClick("_linkSelectOWL",uroShowOWLTab);
  7054.  
  7055. for(var idx=0;idx<W.Config.venues.categories.length;idx++)
  7056. {
  7057. uroSetOnClick('_uroPlacesGroupState-'+idx,uroPlacesGroupCollapseExpand);
  7058. }
  7059.  
  7060. uroAddLog('finalise onload');
  7061. uroLoadSettings();
  7062. uroNewLookCheckDetailsRequest();
  7063. if(uroGetCBChecked('_cbEnableDTE'))
  7064. {
  7065. if(dteControlsIdx != -1)
  7066. {
  7067. dteSetNewTabLength();
  7068. }
  7069. else
  7070. {
  7071. uroAddLog('ERROR - archive panel not found!');
  7072. uroSetStyleDisplay(uroUserTabId,'');
  7073. }
  7074. }
  7075. // filter markers as and when the map is moved
  7076. W.map.events.register("moveend", null, uroFilterItems);
  7077. W.map.events.register("mousemove", null, uroGetAMs);
  7078. W.map.events.register("mousemove", null, uroNewLookHighlightedItemsCheck);
  7079. W.map.events.registerPriority("mousedown", null, uroMouseDown);
  7080.  
  7081. // trap mousedown on Streetview marker drag
  7082. document.getElementsByClassName('street-view-control')[0].onmousedown = uroMouseDown;
  7083.  
  7084. W.map.events.register("mouseup", null, uroMouseUp);
  7085. W.map.events.register("mouseout", null, uroMouseOut);
  7086.  
  7087. uroSetStyles(uroCtrlURs);
  7088. uroSetStyles(uroCtrlMPs);
  7089. uroSetStyles(uroCtrlPlaces);
  7090. uroSetStyles(uroCtrlCameras);
  7091. uroSetStyles(uroCtrlMisc);
  7092. uroSetStyles(uroOWL);
  7093.  
  7094. uroAddFeedFilterControls();
  7095.  
  7096. uroShowURTab();
  7097.  
  7098. uroUserID = W.loginManager.getLoggedInUser().id;
  7099.  
  7100. uroFilterItems();
  7101.  
  7102. uroShowDebugOutput = uroPersistentDebugOutput;
  7103. var dbgMode = "none";
  7104. if(uroShowDebugOutput)
  7105. {
  7106. dbgMode = "inline";
  7107. }
  7108. document.getElementById('_uroDebugMode').style.display = dbgMode;
  7109. uroAddEventListener('_uroVersion',"click", uroToggleDebug, true);
  7110.  
  7111. // add exclusiveCB click handlers to all checkboxes with a pairedWith attribute
  7112. var cbList = document.getElementsByTagName('input');
  7113. for (var optIdx=0;optIdx<cbList.length;optIdx++)
  7114. {
  7115. if((cbList[optIdx].id.indexOf('_cb') === 0) && (cbList[optIdx].attributes.pairedWith != null))
  7116. {
  7117. uroSetOnClick(cbList[optIdx].id,uroExclusiveCB);
  7118. }
  7119. }
  7120. // manually call the layer-change handlers on startup, since there's a good chance WME will already have
  7121. // completed its own startup layer changes before our handlers get registered, preventing the marker handlers
  7122. // from being set up as expected on any markers which are visible in the startup map view before the user forces
  7123. // a layer update by panning/zooming/etc...
  7124. ////uroMCLayerChanged();
  7125. ////uroPlaceLayerChanged();
  7126. uroURLayerChanged();
  7127. uroMPLayerChanged();
  7128. uroPURLayerChanged();
  7129. uroSetupListeners = false;
  7130. uroInitialised = true;
  7131. }
  7132.  
  7133. function uroTSTPopupHandler()
  7134. {
  7135. if(document.getElementsByClassName('panel')[0] === undefined)
  7136. {
  7137. uroHidePopupOnPanelOpen = true;
  7138. }
  7139.  
  7140. if(uroPopupShown === true)
  7141. {
  7142. var hidePopup = false;
  7143. if(document.getElementsByClassName('dropdown action open').length > 0.5)
  7144. {
  7145. hidePopup = true;
  7146. }
  7147. if(hidePopup === false)
  7148. {
  7149. if(document.getElementsByClassName('layer-switcher-container').length === 0)
  7150. {
  7151. if(document.getElementById('layer-switcher-list') !== null)
  7152. {
  7153. hidePopup = (window.getComputedStyle(document.getElementById('layer-switcher-list').parentNode).getPropertyValue('opacity') > 0.5);
  7154. }
  7155. else if(document.getElementsByClassName('layer-switcher').length !== 0)
  7156. {
  7157. hidePopup = (window.getComputedStyle(document.getElementsByClassName('layer-switcher')[0].getElementsByClassName('content')[0]).getPropertyValue('visibility') == 'visible');
  7158. }
  7159. }
  7160. }
  7161. if(hidePopup === false)
  7162. {
  7163. hidePopup = (window.getComputedStyle(document.getElementsByClassName('toolbar-group-drawing')[0].childNodes[0]).getPropertyValue('opacity') > 0.5);
  7164. }
  7165. if(hidePopup === false)
  7166. {
  7167. hidePopup = (window.getComputedStyle(document.getElementsByClassName('toolbar-group-venues')[0].childNodes[0]).getPropertyValue('opacity') > 0.5);
  7168. }
  7169. if(hidePopup === false)
  7170. {
  7171. hidePopup = (window.getComputedStyle(document.getElementsByClassName('toolbar-group-map-comments')[0].childNodes[0]).getPropertyValue('opacity') > 0.5);
  7172. }
  7173.  
  7174. if(document.getElementsByClassName('panel')[0] != null)
  7175. {
  7176. if(uroHidePopupOnPanelOpen === true)
  7177. {
  7178. hidePopup = true;
  7179. uroHidePopupOnPanelOpen = false;
  7180. }
  7181. }
  7182.  
  7183. if(hidePopup === true)
  7184. {
  7185. uroHidePopup();
  7186. }
  7187. }
  7188.  
  7189. if((uroAreaNameHoverObj !== null) && (uroAreaNameHoverTime != -1) && (uroAreaNameOverlayShown === false))
  7190. {
  7191. if(++uroAreaNameHoverTime > 5)
  7192. {
  7193. uroAreaNameOverlaySetup();
  7194. }
  7195. }
  7196. uroReplaceAreaNames(false);
  7197.  
  7198. if(uroPopupTimer > 0)
  7199. {
  7200. if(uroMouseInPopup === false)
  7201. {
  7202. uroPopupTimer--;
  7203. }
  7204. }
  7205. if(uroPopupTimer === 0)
  7206. {
  7207. uroHidePopup();
  7208. }
  7209.  
  7210. if(uroPopupDwellTimer > 0)
  7211. {
  7212. uroPopupDwellTimer--;
  7213. if(uroPopupDwellTimer === 0)
  7214. {
  7215. uroNewLookHighlightedItemsCheck('dwellTimeout');
  7216. }
  7217. }
  7218. }
  7219.  
  7220. function uroTSTDTEHandler()
  7221. {
  7222. if(document.getElementsByClassName("archive-panel")[0] === undefined)
  7223. {
  7224. if(dteClearHighlightsOnPanelClose)
  7225. {
  7226. dteClearListHighlight();
  7227. dteClearHighlightsOnPanelClose = false;
  7228. }
  7229. }
  7230. else
  7231. {
  7232. if(dteArmClearHighlightsOnPanelClose)
  7233. {
  7234. dteArmClearHighlightsOnPanelClose = false;
  7235. dteClearHighlightsOnPanelClose = true;
  7236. }
  7237. }
  7238. }
  7239.  
  7240. function uroTSTNextBtnHandler()
  7241. {
  7242. // replace the "next xxx" button on UR, MP and PUR editing UIs
  7243. if(W.map.panelRegion.hasView() === true)
  7244. {
  7245. var nurButton = W.map.panelRegion.$el[0].getElementsByClassName('btn btn-block')[0];
  7246. if(nurButton != null)
  7247. {
  7248. var doneString = I18n.lookup('problems.panel.done');
  7249. var nextURString = (nurButton.innerHTML.indexOf(I18n.lookup('update_requests.panel.next')) !== -1);
  7250. var nextMPString = (nurButton.innerHTML.indexOf(I18n.lookup('problems.panel.next')) !== -1);
  7251. var nextPURString = (nurButton.innerHTML.indexOf(I18n.lookup('venues.update_requests.panel.next_venue')) !== -1);
  7252. var nextIssueString = (nurButton.innerHTML.indexOf(I18n.lookup('feed.issues.next')) !== -1);
  7253.  
  7254. var updateButton = false;
  7255. var panelClass = W.map.panelRegion.$el[0].childNodes[0].childNodes[0].className;
  7256. var isURorMPPanel = (panelClass.indexOf('problem-edit') !== -1);
  7257. var isPURPanel = (panelClass.indexOf('place-update') !== -1);
  7258. // "next" button class used for UR and MP edit panel
  7259. if(isURorMPPanel === true)
  7260. {
  7261. // user has enabled UR button mod?
  7262. if(uroGetCBChecked('_cbInhibitNURButton') === true)
  7263. {
  7264. // mod the button if this is a UR/MP panel rather than a PUR panel, and if the "Next update request" or "Next issue"
  7265. // text is currently being used on the button - WME uses the former text when a UR is selected directly from the map
  7266. // pin, and the latter text when a UR is selected from the feed. However, it *also* uses the latter text for PURs
  7267. // selected from the feed, hence the need to test the panel type here as well - having to work around odd
  7268. // inconsistencies in WME behaviour, whatever next...
  7269. updateButton = updateButton || ((nextURString === true) || (nextIssueString === true));
  7270. }
  7271. // user has enabled MP button mod?
  7272. if(uroGetCBChecked('_cbInhibitNMPButton') === true)
  7273. {
  7274. // although there's no easy way to determine whether the panel is opened for a UR or MP, since the feed doesn't yet seem
  7275. // to provide a means of accessing MPs then we don't yet have to worry about finding the "Next issue" text in a MP panel,
  7276. // which makes life slightly easier here...
  7277. updateButton = updateButton || (nextMPString === true);
  7278. }
  7279. }
  7280. else if(isPURPanel === true)
  7281. {
  7282. // "next-venue" class only used for PUR edit panel
  7283. if(uroGetCBChecked('_cbInhibitNPURButton') === true)
  7284. {
  7285. // as noted above, PURs opened from the feed may also use the "Next issue" text, so we need to test for that as well,
  7286. // knowing that in order to get this far we've already determined that the user really does want to mod the PUR
  7287. // next button *and* that the currently open panel is a PUR panel...
  7288. updateButton = updateButton || ((nextPURString === true) || (nextIssueString === true));
  7289. }
  7290. }
  7291.  
  7292. if(updateButton === true)
  7293. {
  7294. // if we need to change the button label back to "Done", do it here...
  7295. nurButton.innerHTML = doneString;
  7296. uroAddLog('inhibit Next UR/MP/PUR button');
  7297. }
  7298. // if updateButton isn't already set here, it suggests the panel is using the native "Done" button...
  7299. if(updateButton === false)
  7300. {
  7301. nurButton = W.map.panelRegion.$el[0].getElementsByClassName('btn btn-block done')[0];
  7302. if(nurButton != null)
  7303. {
  7304. updateButton = true;
  7305. }
  7306. }
  7307.  
  7308. if(updateButton === true)
  7309. {
  7310. // 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
  7311. // moving to the next UR/MP/PUR, and also allows us to warn about closing the UR panel if there's an unsent comment...
  7312. nurButton.addEventListener("click", uroInhibitNextUpdateRequestButton, false);
  7313. }
  7314. }
  7315. }
  7316. }
  7317.  
  7318. function uroTSTCommentAddedHandler()
  7319. {
  7320. // test for the opening or closing of the UR editing dialog so we can detect when a new comment is added
  7321. var URDialogIsOpen = false;
  7322. var panelOpen = (document.getElementById('panel-container').firstChild !== null);
  7323. if(panelOpen)
  7324. {
  7325. if(document.getElementById('panel-container').firstChild.firstChild !== null)
  7326. {
  7327. var panelClass = document.getElementById('panel-container').firstChild.firstChild.className;
  7328. if(panelClass === 'problem-edit severity-low')
  7329. {
  7330. URDialogIsOpen = true;
  7331. }
  7332. }
  7333. }
  7334. if(URDialogIsOpen)
  7335. {
  7336. var thisSelectedURID = document.getElementsByClassName('permalink')[0].href.split('&mapUpdateRequest=');
  7337. if(thisSelectedURID.length > 1)
  7338. {
  7339. thisSelectedURID = thisSelectedURID[1].split('&')[0];
  7340. }
  7341. else
  7342. {
  7343. thisSelectedURID = null;
  7344. }
  7345.  
  7346. if(thisSelectedURID != uroSelectedURID)
  7347. {
  7348. // if the user selects a new UR whilst the editing dialog is still open, treat it in the
  7349. // same way as if the user had selected that UR with the dialog closed
  7350. uroURDialogIsOpen = false;
  7351. }
  7352. if(uroURDialogIsOpen === false)
  7353. {
  7354. // user is editing a new UR
  7355. uroSelectedURID = thisSelectedURID;
  7356. // add our own click event handler to the Send button, so we can do stuff whenever a new comment is added
  7357. document.getElementsByClassName('new-comment-form')[0].getElementsByClassName('btn')[0].addEventListener("click", uroAddedComment, false);
  7358. uroAddLog('user is editing UR '+uroSelectedURID);
  7359. uroExpectedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
  7360. ////if(uroShowDebugOutput === true)
  7361. {
  7362. if((uroHoveredURID !== null) && (uroSelectedURID !== null) && (parseInt(uroHoveredURID) !== parseInt(uroSelectedURID)))
  7363. {
  7364. if(uroURReclickAttempts === 0)
  7365. {
  7366. uroAddLog('DANGER, WILL ROBINSON! You clicked on UR ID '+uroHoveredURID+' but WME has loaded the details for UR ID '+uroSelectedURID+' instead, attempting to fix...');
  7367. }
  7368. if(++uroURReclickAttempts < 3)
  7369. {
  7370. //uroRestackMarkers();
  7371. W.map.updateRequestLayer.markers[uroHoveredURID].model.attributes.geometry.x = W.map.updateRequestLayer.markers[uroHoveredURID].model.attributes.geometry.realX;
  7372. W.map.updateRequestLayer.markers[uroHoveredURID].model.attributes.geometry.y = W.map.updateRequestLayer.markers[uroHoveredURID].model.attributes.geometry.realY;
  7373. W.map.updateRequestLayer.markers[uroHoveredURID].icon.$div.click();
  7374. return;
  7375. }
  7376. else
  7377. {
  7378. uroAddLog('Woe is me, attempting to open UR ID '+uroHoveredURID+' has failed...');
  7379. 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);
  7380. }
  7381. }
  7382. uroURReclickAttempts = 0;
  7383. }
  7384. }
  7385. }
  7386. else if(uroURDialogIsOpen === true)
  7387. {
  7388. // dialog was open and has now been closed
  7389. uroSelectedURID = null;
  7390. }
  7391. uroURDialogIsOpen = URDialogIsOpen;
  7392.  
  7393. if(((uroPendingCommentDataRefresh === true) || (uroWaitingCommentDataRefresh === true)) && (uroSelectedURID !== null))
  7394. {
  7395. uroAddLog('check completion of comment data refresh for UR '+uroSelectedURID+' ('+uroPendingCommentDataRefresh+','+uroWaitingCommentDataRefresh+')');
  7396. uroGetSelectedURCommentCount();
  7397. }
  7398.  
  7399. }
  7400.  
  7401. function uroTSTOWLHandler()
  7402. {
  7403. /*
  7404. var selectedTotal = W.selectionManager.selectedItems.length;
  7405. if((selectedTotal > 0) && (document.getElementById('_uroDivOWLBtns') === null))
  7406. {
  7407. var selectedClass = W.selectionManager.selectedItems[0].model.CLASS_NAME;
  7408. var displayAddToOWLBtn = false;
  7409. var displayUpdateOWLBtn = false;
  7410. var displayRemoveFromOWLBtn = false;
  7411. var selectedSegments = false;
  7412. var selectedLandmarks = false;
  7413. var fid;
  7414. var loop;
  7415.  
  7416. // WME only seems to allow multi-object selections for segments, so testing the class of the first object in the
  7417. // selection list tells us the class of any other objects in the list too...
  7418. if(selectedClass == "Waze.Feature.Vector.Segment")
  7419. {
  7420. selectedSegments = true;
  7421. for(loop=0; loop<selectedTotal; loop++)
  7422. {
  7423. fid = W.selectionManager.selectedItems[loop].model.attributes.id;
  7424. var segIdx = uroIsSegOnWatchList(fid);
  7425. if(segIdx == -1)
  7426. {
  7427. displayAddToOWLBtn = true;
  7428. }
  7429. else
  7430. {
  7431. if(uroSegDataChanged(segIdx))
  7432. {
  7433. displayUpdateOWLBtn = true;
  7434. }
  7435. displayRemoveFromOWLBtn = true;
  7436. }
  7437. }
  7438. }
  7439.  
  7440. else if(selectedClass == "Waze.Feature.Vector.Landmark")
  7441. {
  7442. selectedLandmarks = true;
  7443. for(loop=0; loop<selectedTotal; loop++)
  7444. {
  7445. fid = W.selectionManager.selectedItems[loop].model.attributes.id;
  7446. var placeIdx = uroIsPlaceOnWatchList(fid);
  7447. if(placeIdx == -1)
  7448. {
  7449. displayAddToOWLBtn = true;
  7450. }
  7451. else
  7452. {
  7453. if(uroPlaceDataChanged(placeIdx))
  7454. {
  7455. displayUpdateOWLBtn = true;
  7456. }
  7457. displayRemoveFromOWLBtn = true;
  7458. }
  7459. }
  7460. }
  7461.  
  7462. var btnHTML = '<div id="_uroDivOWLBtns">';
  7463. if((displayAddToOWLBtn === true) && (displayUpdateOWLBtn === false))
  7464. {
  7465. btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Add to OWL</button>';
  7466. }
  7467. else if((displayUpdateOWLBtn === true) && (displayAddToOWLBtn === false))
  7468. {
  7469. btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Update OWL</button>';
  7470. }
  7471. else if((displayAddToOWLBtn === true) && (displayUpdateOWLBtn === true))
  7472. {
  7473. btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Add to & Update OWL</button>';
  7474. }
  7475.  
  7476. if(displayRemoveFromOWLBtn === true)
  7477. {
  7478. btnHTML += '<button class="btn btn-default" id="_btnRemoveOWL">Remove from OWL</button>';
  7479. }
  7480. btnHTML += '</div>';
  7481.  
  7482. // note to self... altering the inner HTML of the segment-edit-general panel when the selected
  7483. // segment is part of a roundabout always used to disable the onclick handler for the select
  7484. // roundabout button. will need to see how this behaves in the current WME given the changes in
  7485. // panel arrangement and the introduction of the native select roundabout button
  7486. if(selectedSegments === true)
  7487. {
  7488. document.getElementById("segment-edit-general").innerHTML += btnHTML;
  7489. }
  7490. else if(selectedLandmarks === true)
  7491. {
  7492. document.getElementById("landmark-edit-general").innerHTML += btnHTML;
  7493. }
  7494.  
  7495. if((displayAddToOWLBtn === true)||(displayUpdateOWLBtn === true))
  7496. {
  7497. if(selectedSegments === true)
  7498. {
  7499. uroAddEventListener('_btnAddUpdateOWL','click', uroAddUpdateSegWatchList, true);
  7500. }
  7501. else
  7502. {
  7503. uroAddEventListener('_btnAddUpdateOWL','click', uroAddUpdatePlaceWatchList, true);
  7504. }
  7505. }
  7506.  
  7507. if(displayRemoveFromOWLBtn === true)
  7508. {
  7509. if(selectedSegments === true)
  7510. {
  7511. uroAddEventListener('_btnRemoveOWL','click', uroRemoveSegFromWatchList, true);
  7512. }
  7513. else
  7514. {
  7515. uroAddEventListener('_btnRemoveOWL','click', uroRemovePlaceFromWatchList, true);
  7516. }
  7517. }
  7518. }
  7519. */
  7520. }
  7521.  
  7522. function uroTSTClosureCloningHandler()
  7523. {
  7524. // closure cloning support...
  7525. //
  7526. // has the closures tab been generated?
  7527. if(document.getElementById('segment-edit-closures') !== null)
  7528. {
  7529. // and is it active?
  7530. if(document.getElementById('segment-edit-closures').className === 'tab-pane active')
  7531. {
  7532. // and are there any closures defined for all of the selected segment(s)...
  7533. if(document.getElementsByClassName('full-closures').length > 0)
  7534. {
  7535. var nClosures = document.getElementsByClassName('full-closures')[0].childNodes.length;
  7536. if(nClosures > 0)
  7537. {
  7538. // and last but by no means least, have we already added the clone icon to this closure?
  7539. for(var cLoop = 0; cLoop < nClosures; cLoop++)
  7540. {
  7541. var btnElm = document.getElementsByClassName('full-closures')[0].childNodes[cLoop].children[0].children[0];
  7542. if(btnElm.innerHTML.indexOf('_uroCloneClosure-') == -1)
  7543. {
  7544. var newAnchor = document.createElement('a');
  7545. var anchorID = '_uroCloneClosure-'+cLoop;
  7546. newAnchor.href="#";
  7547. newAnchor.innerHTML = "<i class='fa fa-copy'></i>";
  7548. newAnchor.id = anchorID;
  7549. btnElm.appendChild(newAnchor);
  7550. uroAddEventListener(anchorID,"click",uroCloneClosure,false);
  7551. }
  7552. }
  7553. }
  7554. }
  7555. // if there's more than one closure (full or partial) listed, also add the delete all button if not already present...
  7556. if(document.getElementsByClassName('closure-item').length > 1)
  7557. {
  7558. if(document.getElementById('_btnDeleteAllClosures') === null)
  7559. {
  7560. var daDiv = document.createElement('div');
  7561. daDiv.className = 'delete-all-button btn btn-primary';
  7562. daDiv.id = '_btnDeleteAllClosures';
  7563. daDiv.innerHTML = '<i class="fa fa-trash"></i> '+I18n.lookup("closures.delete_confirm_no_reason")+' ('+I18n.lookup("closures.apply_to_all")+')';
  7564. daDiv.style.width = '100%';
  7565. daDiv.style.marginBottom = '10px';
  7566. var acBtn = document.getElementsByClassName('add-closure-button')[0];
  7567. acBtn.parentNode.insertBefore(daDiv, acBtn.nextSibling);
  7568. uroAddEventListener('_btnDeleteAllClosures',"click", uroDeleteAllClosures, false);
  7569. }
  7570. }
  7571. }
  7572. }
  7573. }
  7574.  
  7575. function uroMiscUITweaksHandler()
  7576. {
  7577. if(uroFilterPreamble())
  7578. {
  7579. // give user the option of setting their own background colour...
  7580. {
  7581. var mapviewport = document.getElementsByClassName("olMapViewport")[0];
  7582. if((uroGetCBChecked('_cbWhiteBackground') === true) && (uroGetCBChecked('_cbMasterEnable') === true))
  7583. {
  7584. var customColour = '#' + uroToHex(uroGetElmValue('_inputCustomBackgroundRed'),2);
  7585. customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundGreen'),2);
  7586. customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundBlue'),2);
  7587. mapviewport.style.backgroundColor = customColour;
  7588. }
  7589. else
  7590. {
  7591. mapviewport.style.backgroundColor = "#C2C2C2";
  7592. }
  7593. }
  7594.  
  7595. // allows user to hide the area managers layer without switching off the layer completely...
  7596. {
  7597. // ...if this sounds like a weird option - why not just switch off the layer from the layers menu? - then
  7598. // remember that in order for URO+ to be able to display in its own tab the list of AMs under the current
  7599. // mouse pointer location, which is somewhat more useful than the list given in the topbar, it needs the
  7600. // AM layer to be activated so that the AM areas data is loaded into WME. It doesn't however need the layer
  7601. // to then be visible, and since having a bunch of purple polygons covering the map can make for a rather
  7602. // difficult editing experience, being able to hide the polys whilst retaining the area information is
  7603. // of real benefit...
  7604. if((uroGetCBChecked('_cbHideAMLayer')) && (uroGetCBChecked('_cbMasterEnable')))
  7605. {
  7606. W.map.managedAreasLayer.setOpacity(0);
  7607. }
  7608. else
  7609. {
  7610. W.map.managedAreasLayer.setOpacity(1);
  7611. }
  7612. }
  7613.  
  7614. // gives user the option of minimising the size of the sidebar tabs to save space
  7615. {
  7616. if(!uroGetCBChecked('_cbDisableTabStyling'))
  7617. {
  7618. // The nav-tabs class is now also used for the General/Closures tabs on the segment edit panel, so we have
  7619. // to restrict the scope of this code to just those nav-tab classed elements within the user-tabs element.
  7620. var navTabs = document.getElementById('user-tabs').getElementsByClassName("nav-tabs")[0].children;
  7621. for(var loop = 0; loop<navTabs.length; loop++)
  7622. {
  7623. navTabs[loop].children[0].style.padding = "4px";
  7624. }
  7625. }
  7626. }
  7627.  
  7628. // gives user the option of hiding the somewhat unnecessary editor info panel at the top of the sidebar
  7629. {
  7630. var panelDisplay = '';
  7631. if(uroGetCBChecked('_cbHideEditorInfo'))
  7632. {
  7633. panelDisplay = "none";
  7634. }
  7635. document.getElementById("user-details").style.display = panelDisplay;
  7636. }
  7637. }
  7638. }
  7639.  
  7640. function uroTenthSecondTick()
  7641. {
  7642. if(uroMTEMode) return;
  7643. if(uroSetupListeners)
  7644. {
  7645. if(uroFinalisingListenerSetup === false)
  7646. {
  7647. if(W.loginManager.isLoggedIn())
  7648. {
  7649. uroFinalizeListenerSetup();
  7650. }
  7651. }
  7652. }
  7653. else
  7654. {
  7655. uroTSTPopupHandler();
  7656. uroTSTDTEHandler();
  7657. uroTSTNextBtnHandler();
  7658.  
  7659. uroTSTCommentAddedHandler();
  7660. uroTSTOWLHandler();
  7661. uroTSTClosureCloningHandler();
  7662. uroTSTFeedFilter();
  7663. uroMiscUITweaksHandler();
  7664.  
  7665. uroTempFixMTEDropDown();
  7666. }
  7667. }
  7668.  
  7669. function uroActiveTab(_id)
  7670. {
  7671. var e = document.getElementById(_id);
  7672. e.style.backgroundColor = "aliceblue";
  7673. e.style.borderTop = "1px solid";
  7674. e.style.borderLeft = "1px solid";
  7675. e.style.borderRight = "1px solid";
  7676. e.style.borderBottom = "0px solid";
  7677. }
  7678.  
  7679. function uroInactiveTab(_id)
  7680. {
  7681. var e = document.getElementById(_id);
  7682. e.style.backgroundColor = "white";
  7683. e.style.borderTop = "0px solid";
  7684. e.style.borderLeft = "0px solid";
  7685. e.style.borderRight = "0px solid";
  7686. e.style.borderBottom = "1px solid";
  7687. }
  7688.  
  7689. function uroInactiveAllTabs()
  7690. {
  7691. uroInactiveTab("_tabSelectCameras");
  7692. uroInactiveTab("_tabSelectMapProblems");
  7693. uroInactiveTab("_tabSelectMisc");
  7694. uroInactiveTab("_tabSelectUserRequests");
  7695. uroInactiveTab("_tabSelectCWL");
  7696. uroInactiveTab("_tabSelectPlaces");
  7697.  
  7698. if(!uroCtrlsHidden)
  7699. {
  7700. uroSetStyleDisplay('uroCtrlURs','none');
  7701. uroSetStyleDisplay('uroCtrlMPs','none');
  7702. uroSetStyleDisplay('uroCtrlCameras','none');
  7703. uroSetStyleDisplay('uroCtrlMisc','none');
  7704. uroSetStyleDisplay('uroOWL','none');
  7705. uroSetStyleDisplay('uroCtrlPlaces','none');
  7706. }
  7707. }
  7708.  
  7709. function uroShowURTab()
  7710. {
  7711. uroInactiveAllTabs();
  7712. uroActiveTab("_tabSelectUserRequests");
  7713. uroCurrentTab = 1;
  7714. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlURs','block');
  7715. return false;
  7716. }
  7717.  
  7718. function uroShowMPTab()
  7719. {
  7720. uroInactiveAllTabs();
  7721. uroActiveTab("_tabSelectMapProblems");
  7722. uroCurrentTab = 2;
  7723. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMPs','block');
  7724. return false;
  7725. }
  7726.  
  7727. function uroShowPlacesTab()
  7728. {
  7729. uroInactiveAllTabs();
  7730. uroActiveTab("_tabSelectPlaces");
  7731. uroCurrentTab = 3;
  7732. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlPlaces','block');
  7733. for(var idx=0;idx<uroPlacesGroupsCollapsed.length;idx++)
  7734. {
  7735. uroPlacesGroupCEHandler(idx);
  7736. }
  7737. return false;
  7738. }
  7739.  
  7740. function uroShowCameraTab()
  7741. {
  7742. uroInactiveAllTabs();
  7743. uroActiveTab("_tabSelectCameras");
  7744. uroCurrentTab = 4;
  7745. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlCameras','block');
  7746. return false;
  7747. }
  7748.  
  7749. function uroShowOWLTab()
  7750. {
  7751. uroInactiveAllTabs();
  7752. uroActiveTab("_tabSelectCWL");
  7753. uroCurrentTab = 5;
  7754. if(!uroCtrlsHidden) uroSetStyleDisplay('uroOWL','block');
  7755. uroOWLUpdateHTML();
  7756. return false;
  7757. }
  7758.  
  7759. function uroShowMiscTab()
  7760. {
  7761. uroInactiveAllTabs();
  7762. uroActiveTab("_tabSelectMisc");
  7763. uroCurrentTab = 6;
  7764. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMisc','block');
  7765. return false;
  7766. }
  7767.  
  7768. function uroNewLookCheckDetailsRequest()
  7769. {
  7770. var thisurl = document.location.href;
  7771. var doRetry = true;
  7772. var urID;
  7773. var endmarkerpos = thisurl.indexOf('&endshow');
  7774. var showmarkerpos = thisurl.indexOf('&showturn=');
  7775. if((endmarkerpos != -1) && (showmarkerpos != -1))
  7776. {
  7777. showmarkerpos += 10;
  7778. uroAddLog('showturn tab opened');
  7779. urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
  7780. uroAddLog(' turn problem ID = '+urID);
  7781.  
  7782. try
  7783. {
  7784. W.map.problemLayer.markers[urID].icon.imageDiv.click();
  7785. doRetry = false;
  7786. }
  7787. catch(err)
  7788. {
  7789. uroAddLog('problems not fully loaded, retrying...');
  7790. }
  7791.  
  7792. if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
  7793. }
  7794. else
  7795. {
  7796. showmarkerpos = thisurl.indexOf('&showpur=');
  7797. if((endmarkerpos != -1) && (showmarkerpos != -1))
  7798. {
  7799. showmarkerpos += 9;
  7800. uroAddLog('showPUR tab opened');
  7801. urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
  7802. uroAddLog(' PUR ID = '+urID);
  7803.  
  7804. try
  7805. {
  7806. W.map.placeUpdatesLayer.markers[urID].icon.imageDiv.click();
  7807. doRetry = false;
  7808. }
  7809. catch(err)
  7810. {
  7811. uroAddLog('PURs not fully loaded, retrying...');
  7812. }
  7813.  
  7814. if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
  7815. }
  7816. }
  7817.  
  7818. }
  7819.  
  7820. function uroUpdateVenueEditorList()
  7821. {
  7822. if(Object.keys(W.model.venues.objects).length === 0) return;
  7823.  
  7824. var selector = document.getElementById('_selectPlacesUserID');
  7825. var selectedUser = null;
  7826. if(selector.selectedOptions[0] != null)
  7827. {
  7828. selectedUser = parseInt(selector.selectedOptions[0].value);
  7829. }
  7830. while(selector.options.length > 0)
  7831. {
  7832. selector.options.remove(0);
  7833. }
  7834.  
  7835. var selectedIdx = null;
  7836. var listedIDs = [];
  7837. var idx;
  7838. for(idx in W.model.venues.objects)
  7839. {
  7840. if(W.model.venues.objects.hasOwnProperty(idx))
  7841. {
  7842. var obj = W.model.venues.objects[idx].attributes;
  7843. var cbID = obj.createdBy;
  7844. var ubID = obj.updatedBy;
  7845. if((cbID !== null) && (listedIDs.indexOf(cbID) == -1))
  7846. {
  7847. listedIDs.push(cbID);
  7848. }
  7849. if((ubID !== null) && (ubID !== cbID) && (listedIDs.indexOf(ubID) == -1))
  7850. {
  7851. listedIDs.push(ubID);
  7852. }
  7853. }
  7854. }
  7855.  
  7856. selector.options.add(new Option('<select a user>', null));
  7857. if(listedIDs.length > 0)
  7858. {
  7859. var users = W.model.users.getByIds(listedIDs);
  7860. var selectorEntry = '';
  7861. for(idx=0; idx<users.length; idx++)
  7862. {
  7863. if(users[idx].userName === undefined)
  7864. {
  7865. selectorEntry = users[idx].id;
  7866. }
  7867. else
  7868. {
  7869. selectorEntry = users[idx].userName;
  7870. }
  7871. selector.options.add(new Option(selectorEntry, users[idx].id));
  7872. if(users[idx].id == selectedUser)
  7873. {
  7874. selectedIdx = idx+1;
  7875. }
  7876. }
  7877. }
  7878.  
  7879. if(selectedIdx !== null)
  7880. {
  7881. selector.selectedIndex = selectedIdx;
  7882. }
  7883. }
  7884.  
  7885. function uroPlacesEditorSelected()
  7886. {
  7887. var selector = document.getElementById('_selectPlacesUserID');
  7888. if(selector.selectedIndex > 0)
  7889. {
  7890. document.getElementById('_textPlacesEditor').value = document.getElementById('_selectPlacesUserID').selectedOptions[0].innerHTML;
  7891. }
  7892. }
  7893.  
  7894. function uroUpdateMPSolverList()
  7895. {
  7896. if(Object.keys(W.model.problems.objects).length === 0)
  7897. {
  7898. return;
  7899. }
  7900.  
  7901. var resolverList = [];
  7902. var selector = document.getElementById('_selectMPUserID');
  7903. var selectedUser = null;
  7904. if(selector.selectedOptions[0] != null)
  7905. {
  7906. selectedUser = parseInt(selector.selectedOptions[0].value);
  7907. }
  7908. while(selector.options.length > 0)
  7909. {
  7910. selector.options.remove(0);
  7911. }
  7912. var selectedIdx = 0;
  7913. var idx = 0;
  7914.  
  7915. for (var mpobj in W.model.problems.objects)
  7916. {
  7917. if(W.model.problems.objects.hasOwnProperty(mpobj))
  7918. {
  7919. var prob = W.model.problems.objects[mpobj];
  7920. if(prob.attributes.resolvedBy !== null)
  7921. {
  7922. var userID = prob.attributes.resolvedBy;
  7923. var userName = W.model.users.objects[userID].userName;
  7924. if(resolverList.indexOf(userName) == -1)
  7925. {
  7926. resolverList.push(userName);
  7927. selector.options.add(new Option(userName, userID));
  7928. if(userID == selectedUser)
  7929. {
  7930. selectedIdx = idx;
  7931. }
  7932. idx++;
  7933. }
  7934. }
  7935. }
  7936. }
  7937.  
  7938. if(selectedIdx !== null)
  7939. {
  7940. selector.selectedIndex = selectedIdx;
  7941. }
  7942. }
  7943.  
  7944. function uroUpdateResolverList()
  7945. {
  7946. if(Object.keys(W.model.mapUpdateRequests.objects).length === 0)
  7947. {
  7948. return;
  7949. }
  7950.  
  7951. var resolverList = [];
  7952. var selector = document.getElementById('_selectURResolverID');
  7953. var selectedUser = null;
  7954. if(selector.selectedOptions[0] != null)
  7955. {
  7956. selectedUser = parseInt(selector.selectedOptions[0].value);
  7957. }
  7958. while(selector.options.length > 0)
  7959. {
  7960. selector.options.remove(0);
  7961. }
  7962. var selectedIdx = 0;
  7963. var idx = 0;
  7964.  
  7965. for (var urobj in W.model.mapUpdateRequests.objects)
  7966. {
  7967. if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
  7968. {
  7969. var ureq = W.model.mapUpdateRequests.objects[urobj];
  7970. if(ureq.attributes.resolvedBy !== null)
  7971. {
  7972. var userID = ureq.attributes.resolvedBy;
  7973. var userName = W.model.users.objects[userID].userName;
  7974. if(resolverList.indexOf(userName) == -1)
  7975. {
  7976. resolverList.push(userName);
  7977. selector.options.add(new Option(userName, userID));
  7978. if(userID == selectedUser)
  7979. {
  7980. selectedIdx = idx;
  7981. }
  7982. idx++;
  7983. }
  7984. }
  7985. }
  7986. }
  7987. if(selectedIdx !== null)
  7988. {
  7989. selector.selectedIndex = selectedIdx;
  7990. }
  7991. }
  7992.  
  7993. function uroUpdateUserList()
  7994. {
  7995. if(Object.keys(W.model.updateRequestSessions.objects).length === 0) return;
  7996.  
  7997. var selector = document.getElementById('_selectURUserID');
  7998.  
  7999. var selectedUser = null;
  8000. if(selector.selectedOptions[0] != null)
  8001. {
  8002. selectedUser = parseInt(selector.selectedOptions[0].value);
  8003. }
  8004.  
  8005. while(selector.options.length > 0)
  8006. {
  8007. selector.options.remove(0);
  8008. }
  8009.  
  8010. var selectedIdx = null;
  8011.  
  8012. var listedIDs = [];
  8013. for(var ursIdx in W.model.updateRequestSessions.objects)
  8014. {
  8015. if(W.model.updateRequestSessions.objects.hasOwnProperty(ursIdx))
  8016. {
  8017. var ursObj = W.model.updateRequestSessions.objects[ursIdx];
  8018. if(ursObj.comments.length > 0)
  8019. {
  8020. for(var cidx=0; cidx < ursObj.comments.length; cidx++)
  8021. {
  8022. var userID = ursObj.comments[cidx].userID;
  8023. if((listedIDs.indexOf(userID) == -1) && (userID != -1))
  8024. {
  8025. listedIDs.push(userID);
  8026. }
  8027. }
  8028. }
  8029. }
  8030. }
  8031.  
  8032. if(listedIDs.length > 0)
  8033. {
  8034. var users = W.model.users.getByIds(listedIDs);
  8035. for(var idx=0; idx<listedIDs.length; idx++)
  8036. {
  8037. selector.options.add(new Option(users[idx].userName, listedIDs[idx]));
  8038. if(listedIDs[idx] == selectedUser)
  8039. {
  8040. selectedIdx = idx;
  8041. }
  8042. }
  8043. }
  8044.  
  8045.  
  8046. if(selectedIdx !== null)
  8047. {
  8048. selector.selectedIndex = selectedIdx;
  8049. }
  8050. }
  8051.  
  8052. function uroUpdateCamEditorList()
  8053. {
  8054. if(Object.keys(W.model.cameras.objects).length === 0) return;
  8055.  
  8056. var selector = document.getElementById('_selectCameraUserID');
  8057.  
  8058. var selectedUser = null;
  8059. if(selector.selectedOptions[0] != null)
  8060. {
  8061. selectedUser = parseInt(selector.selectedOptions[0].value);
  8062. }
  8063.  
  8064. while(selector.options.length > 0)
  8065. {
  8066. selector.options.remove(0);
  8067. }
  8068.  
  8069. var selectedIdx = null;
  8070. var listedIDs = [];
  8071. for(var camIdx in W.model.cameras.objects)
  8072. {
  8073. if(W.model.cameras.objects.hasOwnProperty(camIdx))
  8074. {
  8075. var camObj = W.model.cameras.objects[camIdx].attributes;
  8076. var cbID = camObj.createdBy;
  8077. var ubID = camObj.updatedBy;
  8078. if((cbID !== null) && (listedIDs.indexOf(cbID) == -1))
  8079. {
  8080. listedIDs.push(cbID);
  8081. }
  8082. if((ubID !== null) && (ubID !== cbID) && (listedIDs.indexOf(ubID) == -1))
  8083. {
  8084. listedIDs.push(ubID);
  8085. }
  8086. }
  8087. }
  8088.  
  8089. selector.options.add(new Option('<select a user>', null));
  8090. if(listedIDs.length > 0)
  8091. {
  8092. var users = W.model.users.getByIds(listedIDs);
  8093. var selectorEntry = '';
  8094. for(var idx=0; idx<users.length; idx++)
  8095. {
  8096. if(users[idx].userName === undefined)
  8097. {
  8098. selectorEntry = users[idx].id;
  8099. }
  8100. else
  8101. {
  8102. selectorEntry = users[idx].userName;
  8103. }
  8104. selector.options.add(new Option(selectorEntry, users[idx].id));
  8105. if(users[idx].id == selectedUser)
  8106. {
  8107. selectedIdx = idx+1;
  8108. }
  8109. }
  8110. }
  8111.  
  8112. if(selectedIdx !== null)
  8113. {
  8114. selector.selectedIndex = selectedIdx;
  8115. }
  8116. }
  8117.  
  8118. function uroCamEditorSelected()
  8119. {
  8120. var selector = document.getElementById('_selectCameraUserID');
  8121. if(selector.selectedIndex > 0)
  8122. {
  8123. document.getElementById('_textCameraEditor').value = document.getElementById('_selectCameraUserID').selectedOptions[0].innerHTML;
  8124. }
  8125. }
  8126.  
  8127. function uroSetStyles(obj)
  8128. {
  8129. obj.style.fontSize = '12px';
  8130. obj.style.lineHeight = '100%';
  8131. obj.style.overflow = 'auto';
  8132. obj.style.height = (window.innerHeight * 0.55) + 'px';
  8133. }
  8134.  
  8135. function uroPlacesGroupCEHandler(groupidx)
  8136. {
  8137. if(uroPlacesGroupsCollapsed[groupidx] === false)
  8138. {
  8139. document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "block";
  8140. document.getElementById('_uroPlacesGroupState-'+groupidx).className = "fa fa-minus-square-o";
  8141. }
  8142. else
  8143. {
  8144. document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "none";
  8145. document.getElementById('_uroPlacesGroupState-'+groupidx).className = "fa fa-plus-square-o";
  8146. }
  8147. }
  8148. function uroPlacesGroupCollapseExpand()
  8149. {
  8150. var groupidx = this.id.substr(21);
  8151. if(uroPlacesGroupsCollapsed[groupidx] === true) uroPlacesGroupsCollapsed[groupidx] = false;
  8152. else uroPlacesGroupsCollapsed[groupidx] = true;
  8153. uroPlacesGroupCEHandler(groupidx);
  8154. return false;
  8155. }
  8156. function uroPopulateProblemsTab()
  8157. {
  8158. var tHTML = '';
  8159. tHTML += '<input type="checkbox" id="_cbMPFilterOutsideArea">Hide MPs outside my editable area</input><br><br>';
  8160. tHTML += '<b>Filter MPs by type:</b><br>';
  8161. var i;
  8162. for(i=0; i<uroKnownProblemTypeNames.length; i++)
  8163. {
  8164. tHTML += '<input type="checkbox" id="_cbMPFilter_T'+uroKnownProblemTypeIDs[i]+'">'+uroKnownProblemTypeNames[i]+'</input><br>';
  8165. }
  8166. tHTML += '<br><input type="checkbox" id="_cbMPFilterUnknownProblem">Unknown problem type</input><br><br>';
  8167.  
  8168. tHTML += '&nbsp;&nbsp;<i>Specially tagged types</i><br>';
  8169. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterElgin">[Elgin]</input><br>';
  8170. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterTrafficCast">[TrafficCast]</input><br>';
  8171. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterTrafficMaster">[TM]</input><br>';
  8172. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterCaltrans">[Caltrans]</input><br>';
  8173. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterTFL">TfL</input><br>';
  8174. tHTML += '<input type="checkbox" id="_cbMPFilterReopenedProblem">Reopened Problems</input><br><br>';
  8175.  
  8176. tHTML += '<input type="checkbox" id="_cbInvertMPFilter">Invert operation of type filters?</input><br>';
  8177.  
  8178.  
  8179. tHTML += '<br><b>Hide closed/solved/unidentified Problems:</b><br>';
  8180. tHTML += '<input type="checkbox" id="_cbMPFilterClosed">Closed</input><br>';
  8181. tHTML += '<input type="checkbox" id="_cbMPFilterSolved">Solved</input><br>';
  8182. tHTML += '<input type="checkbox" id="_cbMPFilterUnidentified">Not identified</input><br><br>';
  8183.  
  8184. tHTML += '<input type="checkbox" id="_cbMPClosedUserIDFilter" pairedWith="_cbMPNotClosedUserIDFilter">Closed</input> or ';
  8185. tHTML += '<input type="checkbox" id="_cbMPNotClosedUserIDFilter" pairedWith="_cbMPClosedUserIDFilter">Not Closed</input> by user';
  8186. tHTML += '<select id="_selectMPUserID" style="width:80%; height:22px;"></select><br>';
  8187.  
  8188. tHTML += '<br><b>Hide problems (not turn) by severity:</b><br>';
  8189. tHTML += '<input type="checkbox" id="_cbMPFilterLowSeverity">Low</input>&nbsp;&nbsp;';
  8190. tHTML += '<input type="checkbox" id="_cbMPFilterMediumSeverity">Medium</input>&nbsp;&nbsp;';
  8191. tHTML += '<input type="checkbox" id="_cbMPFilterHighSeverity">High</input><br>';
  8192. uroCtrlMPs.innerHTML = tHTML;
  8193. }
  8194. function uroPopulatePlacesTab()
  8195. {
  8196. var tHTML = '';
  8197. tHTML += '<b>Filter PURs by category/status:</b><br>';
  8198. tHTML += '<input type="checkbox" id="_cbFilterUneditablePlaceUpdates">Ones I can\'t edit</input><br>';
  8199. tHTML += '<input type="checkbox" id="_cbFilterLockRankedPlaceUpdates">Ones with non-zero lockRanks</input><br>';
  8200. tHTML += '<input type="checkbox" id="_cbFilterNewPlacePUR">Ones for new places</input><br>';
  8201. tHTML += '<input type="checkbox" id="_cbFilterUpdatedDetailsPUR">Ones for updated place details</input><br>';
  8202. tHTML += '<input type="checkbox" id="_cbFilterNewPhotoPUR">Ones for new photos</input><br>';
  8203. tHTML += '<input type="checkbox" id="_cbFilterFlaggedPUR">Ones flagged for attention</input><br>';
  8204. tHTML += '<br><input type="checkbox" id="_cbLeavePURGeos">Don\'t hide place polygons/points</input><br>';
  8205. tHTML += '<br><input type="checkbox" id="_cbInvertPURFilters">Invert PUR filters</input><br>';
  8206.  
  8207. tHTML += '<br><b>Filter PURs by severity:</b><br>';
  8208. tHTML += '<input type="checkbox" id="_cbPURFilterLowSeverity">Low</input>&nbsp;&nbsp;';
  8209. tHTML += '<input type="checkbox" id="_cbPURFilterMediumSeverity">Medium</input>&nbsp;&nbsp;';
  8210. tHTML += '<input type="checkbox" id="_cbPURFilterHighSeverity">High</input>';
  8211.  
  8212. tHTML += '<br><b>Filter PURs by age of submission:</b><br>';
  8213. tHTML += '<input type="checkbox" id="_cbEnablePURMinAgeFilter">Hide PURs less than </input>';
  8214. tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPURFilterMinDays"> days old<br>';
  8215. tHTML += '<input type="checkbox" id="_cbEnablePURMaxAgeFilter">Hide PURs more than </input>';
  8216. tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPURFilterMaxDays"> days old<br>';
  8217.  
  8218. tHTML += '<hr>';
  8219.  
  8220. tHTML += '<br><b>Filter Places by state:</b><br>';
  8221. tHTML += 'Hide if last edited<br>';
  8222. tHTML += '<input type="checkbox" id="_cbPlaceFilterEditedLessThan"> less than </input>';
  8223. tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterPlaceEditMinDays"> days ago<br>';
  8224. tHTML += '<input type="checkbox" id="_cbPlaceFilterEditedMoreThan"> more than </input>';
  8225. tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterPlaceEditMaxDays"> days ago<br>';
  8226.  
  8227. tHTML += '<br>Hide if locked at level:<br>';
  8228. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL0">1</input>';
  8229. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL1">2</input>';
  8230. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL2">3</input>';
  8231. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL3">4</input>';
  8232. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL4">5</input>';
  8233. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL5">6</input>';
  8234. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesStaff">Staff</input>';
  8235. tHTML += '<br>&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesAdLocked">AdLocked</input><br>';
  8236. tHTML += '<br>Hide by geometry:<br>';
  8237. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideAreaPlaces">Areas</input>';
  8238. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePointPlaces">Points</input>';
  8239.  
  8240. tHTML += '<br><br><input type="checkbox" id="_cbHidePhotoPlaces" pairedWith="_cbHideNoPhotoPlaces">Hide or </input>';
  8241. tHTML += '<input type="checkbox" id="_cbHideNoPhotoPlaces" pairedWith="_cbHidePhotoPlaces">show ones with photos</input><br>';
  8242.  
  8243. tHTML += '<input type="checkbox" id="_cbHideLinkedPlaces" pairedWith="_cbHideNoLinkedPlaces">Hide or </input>';
  8244. tHTML += '<input type="checkbox" id="_cbHideNoLinkedPlaces" pairedWith="_cbHideLinkedPlaces">show ones with external links</input><br>';
  8245. tHTML += '<input type="checkbox" id="_cbHideKeywordPlaces" pairedWith="_cbHideNoKeywordPlaces">Hide or </input>';
  8246. tHTML += '<input type="checkbox" id="_cbHideNoKeywordPlaces" pairedWith="_cbHideKeywordPlaces">show ones with a name including</input><br>';
  8247. tHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPlace"><br>';
  8248. tHTML += '<br><b>Show Places touched by a specific editor:</b><br>';
  8249. tHTML += '<input type="checkbox" id="_cbShowOnlyPlacesCreatedBy">Created by</input>&nbsp;/&nbsp;';
  8250. tHTML += '<input type="checkbox" id="_cbShowOnlyPlacesEditedBy">edited by</input><br>';
  8251. tHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textPlacesEditor"><br>';
  8252. tHTML += '<select id="_selectPlacesUserID" style="width:80%; height:22px;"></select><br>';
  8253.  
  8254. tHTML += '<br><br><b>Filter Places by category:</b><br>';
  8255.  
  8256. var nCategories = W.Config.venues.categories.length;
  8257. var i;
  8258. if(uroPlacesGroupsCollapsed.length != nCategories)
  8259. {
  8260. for(i=0; i<nCategories; i++)
  8261. {
  8262. uroPlacesGroupsCollapsed.push(false);
  8263. }
  8264. }
  8265.  
  8266. for(i=0; i<nCategories; i++)
  8267. {
  8268. var parentCategory = W.Config.venues.categories[i];
  8269. var localisedName = I18n.lookup("venues.categories." + parentCategory);
  8270.  
  8271. if(uroPlacesGroupsCollapsed[i] === true)
  8272. {
  8273. tHTML += '<i class="fa fa-plus-square-o" style="cursor:pointer;font-size:14px;" id="_uroPlacesGroupState-'+i+'"></i>';
  8274. }
  8275. else
  8276. {
  8277. tHTML += '<i class="fa fa-minus-square-o" style="cursor:pointer;font-size:14px;" id="_uroPlacesGroupState-'+i+'"></i>';
  8278. }
  8279.  
  8280. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPlacesFilter-'+parentCategory+'"><b>'+localisedName+'</b></input><br>';
  8281. tHTML += '<div id="_uroPlacesGroup-'+i+'" style="padding:3px;border-width:2px;border-style:solid;border-color:#FFFFFF">';
  8282.  
  8283. for(var ii=0; ii<W.Config.venues.subcategories[parentCategory].length; ii++)
  8284. {
  8285. var subCategory = W.Config.venues.subcategories[parentCategory][ii];
  8286. localisedName = I18n.lookup("venues.categories." + subCategory);
  8287. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPlacesFilter-'+subCategory+'">'+localisedName+'</input><br>';
  8288. }
  8289. tHTML += '</div>';
  8290. }
  8291. tHTML += '<input type="checkbox" id="_cbFilterPrivatePlaces"><b>Residential Places</b></input><br>';
  8292. tHTML += '<br><input type="checkbox" id="_cbInvertPlacesFilter">Invert Place filters?</input>';
  8293.  
  8294. uroCtrlPlaces.innerHTML = tHTML;
  8295. }
  8296.  
  8297. function uroWazeBits()
  8298. {
  8299. // "fake" uroWazeBits() function which only performs layer scan, to stop the uroWazeBits() call in WMETB from
  8300. // messing around with other stuff in the actual uroWazeBits() function (now renamed uroRealWazeBits...) that
  8301. // really only ought to be called once.
  8302. var i;
  8303. for(i=0;i<W.map.layers.length;i++)
  8304. {
  8305. if(W.map.layers[i].CLASS_NAME == 'OpenLayers.Layer.Vector.RootContainer') uroRootContainer = W.map.layers[i].div.id;
  8306. if(W.map.layers[i].name == 'Node Connections') uroTurnsLayerIdx = i;
  8307. if(W.map.layers[i].name == 'Map comments') uroMCLayerIdx = i;
  8308. }
  8309. uroPlacesRoot = W.map.landmarkLayer.id + '_vroot';
  8310.  
  8311. for(i=0;i<W.map.controls.length;i++)
  8312. {
  8313. if(W.map.controls[i].CLASS_NAME == 'Waze.View.ArchivePanel') dteControlsIdx = i;
  8314. else if(W.map.controls[i].CLASS_NAME == 'Waze.Control.Archive') dteControlsIdx = i;
  8315.  
  8316. if(W.map.controls[i].id !== null)
  8317. {
  8318. if(W.map.controls[i].id.indexOf('UpdateRequests') != -1) uroURControlsIdx = i;
  8319. if(W.map.controls[i].id.indexOf('MapProblems') != -1) uroProblemControlsIdx = i;
  8320. }
  8321. }
  8322. uroAddLog('Turns layer at idx '+uroTurnsLayerIdx);
  8323. uroAddLog('MC layer at idx '+uroMCLayerIdx);
  8324. uroAddLog('uroRootContainer = '+uroRootContainer);
  8325. uroAddLog('Places root layer = '+uroPlacesRoot);
  8326. }
  8327. function uroRealWazeBits()
  8328. {
  8329. if(document.getElementsByClassName("sandbox").length > 0)
  8330. {
  8331. uroAddLog('WME practice mode detected, script is disabled...');
  8332. return;
  8333. }
  8334.  
  8335. if(document.location.href.indexOf('user') !== -1)
  8336. {
  8337. uroAddLog('User profile page detected, script is disabled...');
  8338. return;
  8339. }
  8340. uroAddLog('adding WazeBits...'+uroToHex(uroWazeBitsPresent,4));
  8341. if((uroWazeBitsPresent & 0x0001) === 0)
  8342. {
  8343. if(typeof W != "undefined")
  8344. {
  8345. if(typeof W.map != "undefined")
  8346. {
  8347. uroAddLog(' W.map OK');
  8348. uroWazeBitsPresent |= 0x0001;
  8349. }
  8350. }
  8351. }
  8352. if((uroWazeBitsPresent & 0x0002) === 0)
  8353. {
  8354. if(typeof W != "undefined")
  8355. {
  8356. if(typeof W.model != "undefined")
  8357. {
  8358. uroAddLog(' W.model OK');
  8359. uroWazeBitsPresent |= 0x0002;
  8360. }
  8361. }
  8362. }
  8363. if((uroWazeBitsPresent & 0x0004) === 0)
  8364. {
  8365. if(typeof W != "undefined")
  8366. {
  8367. if(typeof W.loginManager != "undefined")
  8368. {
  8369. uroAddLog(' loginManager OK');
  8370. uroWazeBitsPresent |= 0x0004;
  8371. }
  8372. }
  8373. }
  8374. if((uroWazeBitsPresent & 0x0008) === 0)
  8375. {
  8376. if(typeof W != "undefined")
  8377. {
  8378. if(typeof W.selectionManager != "undefined")
  8379. {
  8380. uroAddLog(' selectionManager OK');
  8381. uroWazeBitsPresent |= 0x0008;
  8382. }
  8383. }
  8384. }
  8385. if((uroWazeBitsPresent & 0x0010) === 0)
  8386. {
  8387. if(typeof OpenLayers != "undefined")
  8388. {
  8389. uroAddLog(' OpenLayers OK');
  8390. uroWazeBitsPresent |= 0x0010;
  8391. }
  8392. }
  8393. if((uroWazeBitsPresent & 0x0020) === 0)
  8394. {
  8395. if(typeof Waze != "undefined")
  8396. {
  8397. uroAddLog(' Waze OK');
  8398. uroWazeBitsPresent |= 0x0020;
  8399. }
  8400. }
  8401. if((uroWazeBitsPresent & 0x0040) === 0)
  8402. {
  8403. if(document.getElementById('user-tabs') !== null)
  8404. {
  8405. uroUserTabId = 'user-tabs';
  8406. uroAddLog(' user-tabs OK');
  8407. uroWazeBitsPresent |= 0x0040;
  8408. }
  8409. }
  8410. if((uroWazeBitsPresent & 0x0080) === 0)
  8411. {
  8412. if(document.getElementById('sidepanel-drives') !== null)
  8413. {
  8414. uroAddLog(' sidepanel-drives OK');
  8415. uroWazeBitsPresent |= 0x0080;
  8416. }
  8417. }
  8418. if((uroWazeBitsPresent & 0x0100) === 0)
  8419. {
  8420. if(typeof I18n != "undefined")
  8421. {
  8422. uroAddLog(' I18n OK');
  8423. uroWazeBitsPresent |= 0x0100;
  8424. }
  8425. }
  8426.  
  8427. if(uroWazeBitsPresent !== 0x01FF)
  8428. {
  8429. setTimeout(uroRealWazeBits,250);
  8430. }
  8431. else if(W.loginManager.isLoggedIn() === false)
  8432. {
  8433. uroAddLog('Waiting for user log-in...');
  8434. setTimeout(uroRealWazeBits,1000);
  8435. }
  8436. else
  8437. {
  8438. uroAddLog('All WazeBits present and correct...');
  8439. W.app.modeController.model.bind("change:mode",uroInitialise);
  8440. if(W.app.modeController.mode.mteModeState !== undefined)
  8441. {
  8442. uroMTEMode = true;
  8443. uroSetupListeners = true;
  8444. uroFinalisingListenerSetup = false;
  8445. uroHidePopup();
  8446. uroAddLog('MTE mode, sleeping until normal service is resumed...');
  8447. return;
  8448. }
  8449. uroMTEMode = false;
  8450. uroSetupUI();
  8451. uroDOMHasTurnProblems = (W.model.turnProblems != null);
  8452. uroGetProblemTypes();
  8453. uroPopulateProblemsTab();
  8454. uroPopulatePlacesTab();
  8455. uroControls.appendChild(uroCtrlURs);
  8456. uroControls.appendChild(uroCtrlMPs);
  8457. uroControls.appendChild(uroCtrlPlaces);
  8458. uroControls.appendChild(uroCtrlCameras);
  8459. uroControls.appendChild(uroOWL);
  8460. uroControls.appendChild(uroCtrlMisc);
  8461. uroControls.appendChild(uroCtrlHides);
  8462. uroControls.appendChild(uroAMList);
  8463.  
  8464. uroCtrlURs.onclick = uroFilterItems_URTabClick;
  8465. uroCtrlMPs.onclick = uroFilterItems_MPTabClick;
  8466. uroCtrlPlaces.onclick = uroFilterItems_PlacesTabClick;
  8467. uroCtrlCameras.onclick = uroFilterItems_CamerasTabClick;
  8468. uroCtrlMisc.onclick = uroFilterItems_MiscTabClick;
  8469.  
  8470. uroWazeBits();
  8471.  
  8472. uroDiv.addEventListener("mouseover", uroEnterPopup, false);
  8473. uroDiv.addEventListener("mouseout", uroExitPopup, false);
  8474.  
  8475. if(sessionStorage.UROverview_FID_IgnoreList === undefined) sessionStorage.UROverview_FID_IgnoreList = '';
  8476. if(sessionStorage.UROverview_FID_WatchList === undefined) sessionStorage.UROverview_FID_WatchList = '';
  8477. if(uroConfirmIntercepted === false) uroAddInterceptor();
  8478. setInterval(uroTenthSecondTick,100);
  8479. }
  8480. }
  8481.  
  8482. function uroAddInterceptor()
  8483. {
  8484. uroAddLog('Adding interceptor function...');
  8485. // add interceptor function for confirm(), so that we can auto-select the "OK" option when solving URs
  8486. // which have pending question...
  8487.  
  8488. var _confirm = window.confirm;
  8489. window.confirm = function(msg)
  8490. {
  8491. var cm_delete_confirm = I18n.lookup("closures.delete_confirm").split('"')[0].trimRight(1);
  8492. if((I18n.lookup("update_requests.panel.confirm") == msg) && (uroGetCBChecked('_cbDisablePendingQuestions') === true))
  8493. {
  8494. uroAddLog('Intercepted pending comments confirmation...');
  8495. return true;
  8496. }
  8497. else if(msg.indexOf(cm_delete_confirm) != -1)
  8498. {
  8499. uroAddLog('intercepted closure delete confirmation...');
  8500. if(uroConfirmClosureDelete)
  8501. {
  8502. return _confirm(msg);
  8503. }
  8504. else
  8505. {
  8506. return true;
  8507. }
  8508. }
  8509. else if(typeof(msg) == 'undefined')
  8510. {
  8511. uroAddLog('Intercepted blank confirmation...');
  8512. return true;
  8513. }
  8514. else
  8515. {
  8516. return _confirm(msg);
  8517. }
  8518. };
  8519. uroConfirmIntercepted = true;
  8520. }
  8521.  
  8522. function uroEnterPopup()
  8523. {
  8524. uroMouseInPopup = true;
  8525. }
  8526.  
  8527. function uroExitPopup()
  8528. {
  8529. uroMouseInPopup = false;
  8530. }
  8531.  
  8532. function uroToggleDebug()
  8533. {
  8534. uroShowDebugOutput = !uroShowDebugOutput;
  8535. var dbgMode = "none";
  8536. if(uroShowDebugOutput)
  8537. {
  8538. dbgMode = "inline";
  8539. }
  8540. document.getElementById('_uroDebugMode').style.display = dbgMode;
  8541. }
  8542.  
  8543. function uroInitialise()
  8544. {
  8545. uroInitialised = false;
  8546. if(document.URL.indexOf('beta') != -1) uroBetaEditor = true;
  8547. var urlBits = document.URL.split("&mapUpdateRequest=");
  8548. if(urlBits.length == 2)
  8549. {
  8550. uroURIDInURL = parseInt(urlBits[1].split('&')[0]);
  8551. uroAddLog('found UR ID '+uroURIDInURL+' in URL');
  8552. }
  8553. uroRealWazeBits();
  8554. }
  8555.  
  8556. function uroSetupUI()
  8557. {
  8558. // create a new div to display the UR details floaty-box
  8559. uroDiv = document.createElement('div');
  8560. uroDiv.id = "uroDiv";
  8561. uroDiv.style.position = 'absolute';
  8562. uroDiv.style.visibility = 'hidden';
  8563. uroDiv.style.top = '0';
  8564. uroDiv.style.left = '0';
  8565. uroDiv.style.zIndex = 100;
  8566. uroDiv.style.backgroundColor = 'aliceblue';
  8567. uroDiv.style.borderWidth = '3px';
  8568. uroDiv.style.borderStyle = 'solid';
  8569. uroDiv.style.borderRadius = '10px';
  8570. uroDiv.style.boxShadow = '5px 5px 10px Silver';
  8571. uroDiv.style.padding = '4px';
  8572. document.body.appendChild(uroDiv);
  8573.  
  8574. // create a new div to display script alerts
  8575. uroAlerts = document.createElement('div');
  8576. uroAlerts.id = "uroAlerts";
  8577. uroAlerts.style.position = 'fixed';
  8578. uroAlerts.style.visibility = 'hidden';
  8579. uroAlerts.style.top = '50%';
  8580. uroAlerts.style.left = '50%';
  8581. uroAlerts.style.zIndex = 100;
  8582. uroAlerts.style.backgroundColor = 'aliceblue';
  8583. uroAlerts.style.borderWidth = '3px';
  8584. uroAlerts.style.borderStyle = 'solid';
  8585. uroAlerts.style.borderRadius = '10px';
  8586. uroAlerts.style.boxShadow = '5px 5px 10px Silver';
  8587. uroAlerts.style.padding = '4px';
  8588. uroAlerts.style.webkitTransform = "translate(-50%, -50%)";
  8589. uroAlerts.style.transform = "translate(-50%, -50%)";
  8590. var alertsHTML = '<div id="header" style="padding: 4px; background-color:LightGreen; font-weight: bold;">Alert title goes here...</div>';
  8591. alertsHTML += '<div id="content" style="padding: 4px; background-color:White">Alert content goes here...</div>';
  8592. alertsHTML += '<div id="controls" style="padding: 4px;">';
  8593. alertsHTML += '<span id="uroAlertTickBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px;">';
  8594. alertsHTML += '<i class="fa fa-check"> </i>';
  8595. alertsHTML += '<span id="uroAlertTickBtnCaption" style="font-weight: bold;"></span>';
  8596. alertsHTML += '</span>';
  8597. alertsHTML += '&nbsp;&nbsp;';
  8598. alertsHTML += '<span id="uroAlertCrossBtn" style="cursor:pointer;font-size:14px;border:thin outset black;padding:2px;">';
  8599. alertsHTML += '<i class="fa fa-times"> </i>';
  8600. alertsHTML += '<span id="uroAlertCrossBtnCaption" style="font-weight: bold;"></span>';
  8601. alertsHTML += '</span>';
  8602. alertsHTML += '</div>';
  8603. uroAlerts.innerHTML = alertsHTML;
  8604. document.body.appendChild(uroAlerts);
  8605.  
  8606.  
  8607. uroControls = document.createElement('section');
  8608. uroControls.style.fontSize = '12px';
  8609. uroControls.id = 'uroControls';
  8610. var updateURL;
  8611. if(navigator.userAgent.indexOf('Chrome') == -1)
  8612. {
  8613. updateURL = 'https://greasyfork.org/scripts/1952-uroverview-plus-uro';
  8614. }
  8615. else
  8616. {
  8617. updateURL = 'https://chrome.google.com/webstore/detail/uroverview/amdamgkgchnbaopmphhjapmjcdghdphi';
  8618. }
  8619. var tabbyHTML = '<b><a href="'+updateURL+'" target="_blank">UROverview Plus</a></b> <label id="_uroVersion">'+uroVersion+'</label>';
  8620. tabbyHTML += '<label id="_uroDebugMode">(dbg)</label>';
  8621. tabbyHTML += '&nbsp;<input type="checkbox" id="_cbMasterEnable" checked>Enabled</input>';
  8622. tabbyHTML += '<p><table border=0 width="100%"><tr>';
  8623. tabbyHTML += '<td valign="center" align="center" id="_tabSelectUserRequests"><a href="#" id="_linkSelectUserRequests" style="text-decoration:none;font-size:12px">URs</a></td>';
  8624. tabbyHTML += '<td valign="center" align="center" id="_tabSelectMapProblems"><a href="#" id="_linkSelectMapProblems" style="text-decoration:none;font-size:12px">MPs</a></td>';
  8625. tabbyHTML += '<td valign="center" align="center" id="_tabSelectPlaces"><a href="#" id="_linkSelectPlaces" style="text-decoration:none;font-size:12px">Places</a></td>';
  8626. tabbyHTML += '<td valign="center" align="center" id="_tabSelectCameras"><a href="#" id="_linkSelectCameras" style="text-decoration:none;font-size:12px">Cams</a></td>';
  8627. tabbyHTML += '<td valign="center" align="center" id="_tabSelectCWL"><a href="#" id="_linkSelectOWL" style="text-decoration:none;font-size:12px">OWL</a></td>';
  8628. tabbyHTML += '<td valign="center" align="center" id="_tabSelectMisc"><a href="#" id="_linkSelectMisc" style="text-decoration:none;font-size:12px">Misc</a></td>';
  8629. tabbyHTML += '</tr></table>';
  8630. uroControls.innerHTML = tabbyHTML;
  8631.  
  8632.  
  8633. uroCtrlURs = document.createElement('p');
  8634. uroCtrlMPs = document.createElement('p');
  8635. uroCtrlCameras = document.createElement('p');
  8636. uroOWL = document.createElement('p');
  8637. uroCtrlMisc = document.createElement('p');
  8638. uroAMList = document.createElement('div');
  8639. uroCtrlHides = document.createElement('div');
  8640. uroCtrlPlaces = document.createElement('p');
  8641.  
  8642. // UR controls tab
  8643. uroCtrlURs.id = "uroCtrlURs";
  8644. uroCtrlURs.innerHTML = '<br>';
  8645.  
  8646. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURFilterOutsideArea">Hide URs outside my editable area</input><br>';
  8647. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbNoFilterForURInURL">Don\'t filter UR in URL</input><br><br>';
  8648. uroCtrlURs.innerHTML += '<b>Filter by type:</b><br>';
  8649. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterWazeAuto">Waze Automatic</input><br>';
  8650. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectTurn">Incorrect turn</input><br>';
  8651. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectAddress">Incorrect address</input><br>';
  8652. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectRoute">Incorrect route</input><br>';
  8653. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingRoundabout">Missing roundabout</input><br>';
  8654. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterGeneralError">General error</input><br>';
  8655. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterTurnNotAllowed">Turn not allowed</input><br>';
  8656. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectJunction">Incorrect junction</input><br>';
  8657. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingBridgeOverpass">Missing bridge overpass</input><br>';
  8658. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterWrongDrivingDirection">Wrong driving direction</input><br>';
  8659. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingExit">Missing exit</input><br>';
  8660. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingRoad">Missing road</input><br>';
  8661. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterBlockedRoad">Blocked road</input><br>';
  8662. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingLandmark">Missing Landmark</input><br>';
  8663. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterSpeedLimits">Missing or Invalid Speed limit</input><br>';
  8664. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterUndefined">Undefined</input><br>';
  8665.  
  8666. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<i>Specially tagged types</i><br>';
  8667. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterRoadworks">[ROADWORKS]</input><br>';
  8668. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterConstruction">[CONSTRUCTION]</input><br>';
  8669. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterClosure">[CLOSURE]</input><br>';
  8670. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterEvent">[EVENT]</input><br>';
  8671. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterNote">[NOTE]</input><br>';
  8672. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterBOG">[BOG]</input><br>';
  8673. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterDifficult">[DIFFICULT]</input><br>';
  8674. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterWSLM">[WSLM]</input><br><br>';
  8675. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbInvertURFilter">Invert operation of type filters?</input><br>';
  8676.  
  8677. uroCtrlURs.innerHTML += '<hr>';
  8678.  
  8679. uroCtrlURs.innerHTML += '<br><b>Hide by state:</b><br>';
  8680. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterOpenUR">Open</input><br>';
  8681. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterClosedUR">Closed</input><br>';
  8682. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterSolved">Solved</input><br>';
  8683. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterUnidentified">Not identified</input><br><br>';
  8684.  
  8685.  
  8686. uroCtrlURs.innerHTML += '<br><b>Filter by age of submission:</b><br>';
  8687. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMinAgeFilter">Hide URs less than </input>';
  8688. uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinDays"> days old<br>';
  8689. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMaxAgeFilter">Hide URs more than </input>';
  8690. uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxDays"> days old<br>';
  8691.  
  8692. uroCtrlURs.innerHTML += '<br><b>Filter by description/comments/following:</b><br>';
  8693. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyFollowed" pairedWith="_cbHideMyUnfollowed">Ones I am or </input>';
  8694. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyUnfollowed" pairedWith="_cbHideMyFollowed">am not following</input><br><br>';
  8695.  
  8696. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURDescriptionMustBePresent" pairedWith="_cbURDescriptionMustBeAbsent">Hide</input> or ';
  8697. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURDescriptionMustBeAbsent" pairedWith="_cbURDescriptionMustBePresent">show</input> URs with no description<br>';
  8698. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableKeywordMustBePresent">Hide URs not including </input>';
  8699. uroCtrlURs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPresent"><br>';
  8700. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableKeywordMustBeAbsent">Hide URs including </input>';
  8701. uroCtrlURs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordAbsent"><br>';
  8702. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbCaseInsensitive"><i>Case-insensitive matches?</i></input><br><br>';
  8703.  
  8704. uroCtrlURs.innerHTML += 'With comments from me?<br>';
  8705. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyComments" pairedWith="_cbHideAnyComments">Yes </input>';
  8706. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideAnyComments" pairedWith="_cbHideMyComments">No</input><br>';
  8707. uroCtrlURs.innerHTML += 'If last comment made by me?<br>';
  8708. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfLastCommenter" pairedWith="_cbHideIfNotLastCommenter">Yes </input>';
  8709. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfNotLastCommenter" pairedWith="_cbHideIfLastCommenter">No </input><br>';
  8710. uroCtrlURs.innerHTML += 'If last comment made by UR reporter?<br>';
  8711. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfReporterLastCommenter" pairedWith="_cbHideIfReporterNotLastCommenter">Yes </input>';
  8712. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfReporterNotLastCommenter" pairedWith="_cbHideIfReporterLastCommenter">No</input><br>';
  8713.  
  8714. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMinCommentsFilter">With less than </input>';
  8715. uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinComments"> comments<br>';
  8716. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMaxCommentsFilter">With more than </input>';
  8717. uroCtrlURs.innerHTML += '<input type="number" min="0" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxComments"> comments<br><br>';
  8718.  
  8719. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter2">Last comment less than </input>';
  8720. uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays2"> days ago<br>';
  8721. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter">Last comment more than </input>';
  8722. uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterCommentDays"> days ago<br>';
  8723. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbIgnoreOtherEditorComments"><i>Ignore other editor comments?</i></input><br><br>';
  8724.  
  8725. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURUserIDFilter">Without comments from user</input>';
  8726. uroCtrlURs.innerHTML += '<select id="_selectURUserID" style="width:80%; height:22px;"></select><br>';
  8727. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURResolverIDFilter">Not resolved by user</input>';
  8728. uroCtrlURs.innerHTML += '<select id="_selectURResolverID" style="width:80%; height:22px;"></select>';
  8729.  
  8730. uroCtrlURs.innerHTML += '<br><br><input type="checkbox" id="_cbInvertURStateFilter">Invert operation of state/age filters?</input><br>';
  8731. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbNoFilterForTaggedURs">Don\'t apply state/age filters to tagged URs</input><br>';
  8732.  
  8733.  
  8734. // Map problems controls tab
  8735. uroCtrlMPs.id = "uroCtrlMPs";
  8736. uroCtrlMPs.innerHTML = 'MP filter list being populated, please wait...';
  8737.  
  8738.  
  8739. // Places filtering tab
  8740. uroCtrlPlaces.id = "uroCtrlPlaces";
  8741. uroCtrlPlaces.innerHTML = 'Places filter list being populated, please wait...';
  8742.  
  8743.  
  8744. // Camera controls tab
  8745. uroCtrlCameras.id = "uroCtrlCameras";
  8746. uroCtrlCameras.innerHTML = '<br><b>Show Cameras created by:</b><br>';
  8747. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowWorldCams" checked>world_* users</input><br>';
  8748. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowUSACams" checked>usa_* users</input><br>';
  8749. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowNonWorldCams" checked>other users</input><br>';
  8750.  
  8751. uroCtrlCameras.innerHTML += '<br><b>Show Cameras touched by a specific editor:</b><br>';
  8752. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowOnlyCamsCreatedBy">Created by</input>&nbsp;/&nbsp;';
  8753. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowOnlyCamsEditedBy">edited by</input><br>';
  8754. uroCtrlCameras.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textCameraEditor"><br>';
  8755. uroCtrlCameras.innerHTML += '<select id="_selectCameraUserID" style="width:80%; height:22px;"></select><br>';
  8756. uroCtrlCameras.innerHTML += '<br><input type="checkbox" id="_cbShowOnlyMyCams">Show ONLY cameras created/edited by me</input><br>';
  8757.  
  8758.  
  8759. uroCtrlCameras.innerHTML += '<br><b>Show Cameras by approval status:</b><br>';
  8760. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowApprovedCams" checked>approved</input><br>';
  8761. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowNonApprovedCams" checked>non-approved</input><br>';
  8762. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowOlderCreatedNonApproved"> if created more than </input>';
  8763. uroCtrlCameras.innerHTML += '<input type="number" min="1" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCameraMinCreatedDays"> days ago<br>';
  8764. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowOlderUpdatedNonApproved"> if updated more than </input>';
  8765. uroCtrlCameras.innerHTML += '<input type="number" min="1" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCameraMinUpdatedDays"> days ago<br>';
  8766.  
  8767. uroCtrlCameras.innerHTML += '<br><b>Show Cameras by type:</b><br>';
  8768. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowSpeedCams" checked>Speed</input><br>';
  8769. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowIfSpeedSet" checked> with speed data</input><br>';
  8770. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowIfNoSpeedSet" checked> with no speed data</input><br>';
  8771. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowRedLightCams" checked>Red Light</input><br>';
  8772. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowDummyCams" checked>Dummy</input><br>';
  8773.  
  8774. uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by creator:</b><br>';
  8775. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByMe">me</input>';
  8776. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank0">L1</input>';
  8777. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank1">L2</input>';
  8778. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank2">L3</input>';
  8779. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank3">L4</input>';
  8780. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank4">L5</input>';
  8781. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank5">L6</input>';
  8782.  
  8783. uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by updater:</b><br>';
  8784. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByMe">me</input>';
  8785. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank0">L1</input>';
  8786. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank1">L2</input>';
  8787. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank2">L3</input>';
  8788. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank3">L4</input>';
  8789. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank4">L5</input>';
  8790. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank5">L6</input>';
  8791.  
  8792. uroCtrlCameras.innerHTML += '<br><br><b><input type="checkbox" id="_cbHideCWLCams">Hide cameras on watchlist</input></b><br>';
  8793.  
  8794.  
  8795. // Object watchlist tab
  8796. uroOWL.id = "uroOWL";
  8797. uroCWLGroups = [];
  8798. uroOWLUpdateHTML();
  8799.  
  8800.  
  8801. // Misc controls tab
  8802. uroCtrlMisc.id = "uroCtrlMisc";
  8803. uroCtrlMisc.innerHTML = '<br><b>Use default conversation markers:</b><br>';
  8804. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbNativeConvoMarkers" checked />in public WME<br>';
  8805. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbNativeBetaConvoMarkers" checked />in beta WME<br>';
  8806.  
  8807. uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbCommentCount" />Show comment count on UR markers</b><br>';
  8808.  
  8809. uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbURBackfill" />Backfill UR data</b><br>';
  8810.  
  8811. uroCtrlMisc.innerHTML += '<br><br><b>Marker Unstacking:</b><br>';
  8812. uroCtrlMisc.innerHTML += 'Distance threshold: <input type="number" min="1" max="30" value="15" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackSensitivity" /><br>';
  8813. uroCtrlMisc.innerHTML += 'Disable below zoom: <input type="number" min="0" max="10" value="3" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputUnstackZoomLevel" /><br>';
  8814.  
  8815. uroCtrlMisc.innerHTML += '<br><br><b>Use custom marker for URs tagged as:</b><br>';
  8816. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomRoadworksMarkers" />[ROADWORKS]<br>';
  8817. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomConstructionMarkers" />[CONSTRUCTION]<br>';
  8818. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomClosuresMarkers" />[CLOSURE]<br>';
  8819. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomEventsMarkers" />[EVENT]<br>';
  8820. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomNotesMarkers" />[NOTE]<br>';
  8821. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomBOGMarkers" />[BOG]<br>';
  8822. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomDifficultMarkers" />[DIFFICULT]<br>';
  8823. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomWSLMMarkers" />[WSLM]<br>';
  8824. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomNativeSLMarkers" />Native speed limit reports<br>';
  8825. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomKeywordMarkers" />';
  8826. uroCtrlMisc.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textCustomKeyword" /><br>';
  8827.  
  8828. uroCtrlMisc.innerHTML += '<br><br><b>Use custom marker for MPs tagged as:</b><br>';
  8829. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomElginMarkers" />[Elgin]<br>';
  8830. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTrafficMasterMarkers" />[TM]<br>';
  8831. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTrafficCastMarkers" />[TrafficCast]<br>';
  8832. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomCaltransMarkers" />[Caltrans]<br>';
  8833. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTFLMarkers" />[TfL Open Data]<br>';
  8834.  
  8835. uroCtrlMisc.innerHTML += '<br><br><b>Popup mouse behaviour:</b><br>';
  8836. uroCtrlMisc.innerHTML += 'Mouse idle <input type="number" min="1" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupDwellTimeout" /> *100ms<br>';
  8837. uroCtrlMisc.innerHTML += 'Mouse over <input type="number" min="1" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPopupEntryTimeout" /> *100ms<br>';
  8838. uroCtrlMisc.innerHTML += 'Distance <input type="number" min="0" max="10" value="2" size="2" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputMaxJitter" /> pixels<br>';
  8839.  
  8840. uroCtrlMisc.innerHTML += '<br><br><b>Disable popup for:</b><br>';
  8841. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitURPopup" />URs<br>';
  8842. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitMPPopup" />MPs<br>';
  8843. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitCamPopup" />Cameras<br>';
  8844. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitSegPopup" />Segments<br>';
  8845. uroCtrlMisc.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbInhibitSegGenericPopup" />Speed limit info<br>';
  8846. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitTurnsPopup" />Restricted Turns<br>';
  8847. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitLandmarkPopup" />Places<br>';
  8848. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitPUPopup" />Place Updates<br>';
  8849. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitMapCommentPopup" />Map Comments<br>';
  8850.  
  8851. uroCtrlMisc.innerHTML += '<br><br><b>Date/Time formatting for popups:</b><br>';
  8852. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtDDMMYY" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtYYMMDD" checked />day/month/year<br>';
  8853. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtMMDDYY" pairedWith="_cbDateFmtDDMMYY,_cbDateFmtYYMMDD" />month/day/year<br>';
  8854. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtYYMMDD" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtDDMMYY" />year/month/day<br><br>';
  8855. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbTimeFmt24H" pairedWith="_cbTimeFmt12H" checked />24 hour<br>';
  8856. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbTimeFmt12H" pairedWith="_cbTimeFmt24H" />12 hour<br><br>';
  8857. uroCtrlMisc.innerHTML += '<i>Unticked uses browser default setting</i>';
  8858.  
  8859. uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbWhiteBackground" />Use custom background colour</b><br>';
  8860. uroCtrlMisc.innerHTML += 'R:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundRed" />';
  8861. uroCtrlMisc.innerHTML += 'G:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundGreen" />';
  8862. uroCtrlMisc.innerHTML += 'B:<input type="number" min="0" max="255" value="255" size="3" style="width:50px;;line-height:14px;height:22px;margin-bottom:4px;" id="_inputCustomBackgroundBlue" /><br>';
  8863.  
  8864. uroCtrlMisc.innerHTML += '<br><br><b>Replace "Next ..." button with "Done" for:</b><br>';
  8865. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNURButton" />URs<br>';
  8866. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNMPButton" />MPs<br>';
  8867. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNPURButton" />PURs<br>';
  8868. uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbHideAMLayer" />Hide Area Manager polygons</b><br>';
  8869. uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisablePlacesFiltering" />Disable Places filtering</b><br>';
  8870. ////uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisablePendingQuestions">Disable UR Pending Questions confirmation</input></b><br>';
  8871. uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisableTabStyling" />Use default tab styling</b><br>';
  8872. uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbHideEditorInfo" />Hide sidebar editor info</b><br>';
  8873. uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbEnableDTE" />Drive Tab Enhancement (DTE)</b><br>';
  8874.  
  8875. uroCtrlMisc.innerHTML += '<br><br><b>Settings backup/restore/reset:</b><br>';
  8876. uroCtrlMisc.innerHTML += '<input type="button" id="_btnSettingsToText" value="Backup" />&nbsp;&nbsp;&nbsp;';
  8877. uroCtrlMisc.innerHTML += '<input type="button" id="_btnTextToSettings" value="Restore" />&nbsp;&nbsp;|&nbsp;&nbsp;';
  8878. uroCtrlMisc.innerHTML += '<input type="button" id="_btnResetSettings" value="Reset" /><br><br>';
  8879. uroCtrlMisc.innerHTML += '<textarea id="_txtSettings" value="" /><br>';
  8880. uroCtrlMisc.innerHTML += '<input type="button" id="_btnClearSettingsText" value="Clear" /><br>';
  8881.  
  8882. /*
  8883. uroCtrlMisc.innerHTML += '<br><br><b>Debug:</b><br>';
  8884. uroCtrlMisc.innerHTML += '<input type="button" id="_btnDebugToScreen" value="Show debug data" />';
  8885. */
  8886. // footer for tabs container
  8887. uroCtrlHides.id = 'uroCtrlHides';
  8888. uroCtrlHides.innerHTML = '<input type="button" id="_btnUndoLastHide" value="Undo last hide" />&nbsp;&nbsp;&nbsp;';
  8889. uroCtrlHides.innerHTML += '<input type="button" id="_btnClearSessionHides" value="Undo all hides" /><p>';
  8890.  
  8891. // footer for AM list
  8892. uroAMList.id = 'uroAMList';
  8893. window.addEventListener("beforeunload", uroSaveSettings, false);
  8894.  
  8895. //uroRealWazeBits();
  8896. }
  8897.  
  8898. function dteAddHeader()
  8899. {
  8900. if(uroMTEMode) return;
  8901. if(!uroInitialised) return;
  8902. var rlcObj = document.getElementsByClassName("result-list-container");
  8903. if(typeof rlcObj == "undefined") return;
  8904. if(typeof rlcObj[0].children[0] == "undefined") return;
  8905. if(typeof rlcObj[0].children[0].innerHTML == "undefined") return;
  8906.  
  8907. var thtml = rlcObj[0].children[0].innerHTML;
  8908. if(thtml.indexOf('Full drive history') == -1)
  8909. {
  8910. thtml += '<br><br><i><small>Full drive history goes back to '+dteOldestFullDrive.toDateString()+'</small></i>';
  8911. rlcObj[0].children[0].innerHTML = thtml;
  8912. }
  8913. }
  8914.  
  8915. function dteGetData()
  8916. {
  8917. var loc = 'https://'+window.location.hostname+Waze.Config.api_base+'/Archive/MyList?minDistance=1000';
  8918. loc += '&offset='+dteOffset+'&count=5';
  8919. var dteReq = new XMLHttpRequest();
  8920. dteReq.onreadystatechange = function()
  8921. {
  8922. var foundMissingDrive = false;
  8923. if(dteReq.readyState == 4)
  8924. {
  8925. uroAddLog('drive data request, response '+dteReq.status+' received');
  8926. if(dteReq.status == 200)
  8927. {
  8928. if(dteReq.responseText !== "")
  8929. {
  8930. var drives = JSON.parse(dteReq.responseText);
  8931. var loadedDrives = drives.archives.objects.length;
  8932. uroAddLog('received '+loadedDrives+' drives');
  8933. if(loadedDrives != 5) foundMissingDrive = true;
  8934.  
  8935. for(var loop=0; loop < loadedDrives; loop++)
  8936. {
  8937. if(drives.archives.objects[loop].hasFullSession === false)
  8938. {
  8939. foundMissingDrive = true;
  8940. }
  8941. else
  8942. {
  8943. dteOffset++;
  8944. dteOldestFullDrive = new Date(drives.archives.objects[loop].startTime);
  8945. }
  8946. }
  8947. }
  8948. else
  8949. {
  8950. foundMissingDrive = true;
  8951. }
  8952. }
  8953. if(foundMissingDrive === false)
  8954. {
  8955. dteGetData();
  8956. }
  8957. else
  8958. {
  8959. uroAddLog(dteOffset+' full drives in history');
  8960. uroAddLog('oldest drives are on '+dteOldestFullDrive.toDateString());
  8961. if(dteOffset < 5)
  8962. {
  8963. dteOffset = 5;
  8964. uroAddLog('insufficient full drives, using standard drives tab');
  8965. }
  8966. else if(dteOffset > 50)
  8967. {
  8968. var nPages = Math.ceil(dteOffset / 50);
  8969. uroAddLog('too many full drives for a single tab page, splitting over '+nPages+' pages...');
  8970. dteOffset = Math.ceil(dteOffset/nPages);
  8971. }
  8972.  
  8973. if((dteOldestFullDrive - dteEpoch) > 0)
  8974. {
  8975. var totalDrives = 0;
  8976. if(W.model.archives.additionalInfo !== null)
  8977. {
  8978. totalDrives = W.model.archives.additionalInfo.totalSessions;
  8979. }
  8980. if(totalDrives !== null)
  8981. {
  8982. uroAddLog('updating drives tab...');
  8983. W.map.controls[dteControlsIdx].sidePanelView.ResultsPerPage = dteOffset;
  8984. uroAddLog(totalDrives+' drives in history');
  8985. W.map.controls[dteControlsIdx].sidePanelView.setSessions(totalDrives);
  8986. W.map.controls[dteControlsIdx].loadSessions(0);
  8987. }
  8988. setInterval(dteAddHeader,250);
  8989. setInterval(dteCheckDriveListChanges,250);
  8990. }
  8991. }
  8992. }
  8993. };
  8994. dteReq.open('GET',loc,true);
  8995. dteReq.send();
  8996.  
  8997. }
  8998.  
  8999. function dteSetNewTabLength()
  9000. {
  9001. uroAddLog('altering ResultsPerPage parameter...');
  9002.  
  9003. var t = document.getElementById('sidepanel-drives');
  9004. t.style.overflow = 'auto';
  9005. t.style.height = (window.innerHeight * 0.6) + 'px';
  9006. dteOffset = 0;
  9007. dteGetData();
  9008. }
  9009.  
  9010. function dteListClick()
  9011. {
  9012. dteClearListHighlight();
  9013. this.style.backgroundColor = "lightgreen";
  9014. dteArmClearHighlightsOnPanelClose = true;
  9015. }
  9016.  
  9017. function dteClearListHighlight()
  9018. {
  9019. var drivesShown = document.getElementById('sidepanel-drives').getElementsByClassName('result session').length;
  9020. if(drivesShown > 0)
  9021. {
  9022. for(var loop = 0;loop < drivesShown; loop++)
  9023. {
  9024. var listEntry = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[loop];
  9025. listEntry.style.backgroundColor = "";
  9026. }
  9027. }
  9028. }
  9029.  
  9030. function dteCheckDriveListChanges()
  9031. {
  9032. if(uroMTEMode) return;
  9033. if(!uroInitialised) return;
  9034. var drivesShown = document.getElementById('sidepanel-drives').getElementsByClassName('result session').length;
  9035. if(drivesShown > 0)
  9036. {
  9037. var topID = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[0].getAttribute('data-id');
  9038. if(topID != dteTopID)
  9039. {
  9040. dteTopID = topID;
  9041. for(var loop = 0;loop < drivesShown; loop++)
  9042. {
  9043. var listEntry = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[loop];
  9044. var driveID = listEntry.getAttribute('data-id');
  9045. var driveObj = W.model.archives.objects[driveID];
  9046. var driveSecs = Math.floor((driveObj.endTime - driveObj.startTime) / 1000);
  9047. var driveHours = Math.floor(driveSecs / 3600);
  9048. driveSecs -= (driveHours * 3600);
  9049. var driveMins = Math.floor(driveSecs / 60);
  9050. driveSecs -= (driveMins * 60);
  9051. var trueTime = (driveHours+':'+("0"+driveMins).slice(-2)+'.'+("0"+driveSecs).slice(-2));
  9052. listEntry.getElementsByTagName('span')[1].innerHTML = trueTime;
  9053. listEntry.addEventListener("click", dteListClick, false);
  9054. }
  9055. }
  9056. }
  9057. }
  9058.  
  9059. uroInitialise();