UROverview Plus (URO+)

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

当前为 2016-01-19 提交的版本,查看 最新版本

  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://*.waze.com/*editor/*
  6. // @include https://editor-beta.waze.com/*
  7. // @exclude https://*.waze.com/*/user/editor/*
  8. // @grant none
  9. // @version 3.64
  10. // ==/UserScript==
  11.  
  12. /*
  13.  
  14. TO-DO ITEMS
  15. =======================================================================================================================
  16. Bug fixes - MUST BE CLEARED BEFORE RELEASE
  17. =======================================================================================================================
  18.  
  19.  
  20.  
  21. =======================================================================================================================
  22. Things to be checked
  23. =======================================================================================================================
  24.  
  25.  
  26.  
  27. =======================================================================================================================
  28. Proposed functionality
  29. =======================================================================================================================
  30.  
  31. ***HIGH PRIORITY***
  32. Stop backfilling code from firing off on each UR data change - when performing a save which includes any URs being
  33. closed, it seems like each time one of the closures is saved it triggers an onchange event in the UR data, which then
  34. fires off backfilling...
  35.  
  36. Convert camera XHR code to async operation
  37.  
  38. ***OTHER STUFF***
  39. Use a completely new custom marker for any UR which matches a description keyword - i.e. combine the existing custom
  40. marker idea with the existing keyword filter
  41.  
  42. Implement some sort of UR "hotspot" marking to highlight areas of the map with significant clustering of URs
  43.  
  44. Flush settings to localStorage whenever a change is made, or at least before opening a new tab via a popup
  45.  
  46. User-defined setting presets
  47.  
  48. Extend unstacking to cameras
  49.  
  50. Place filtering
  51. - by last user to edit
  52.  
  53. More localisation
  54.  
  55. First-run information
  56. - show quickstart guide to URO features if no existing settings are present (i.e. new installation)
  57.  
  58. =======================================================================================================================
  59. New functionality in progress
  60. =======================================================================================================================
  61.  
  62. Addition of segment and place watchlist functionality
  63. */
  64.  
  65. /* JSHint Directives */
  66. /* globals $: */
  67. /* globals W: true */
  68. /* globals I18n: */
  69. /* globals OL: true */
  70. /* globals OpenLayers: true */
  71. /* globals Waze: true */
  72. /* globals require: */
  73. /* jshint bitwise: false */
  74.  
  75. var uroVersion = "3.64";
  76. var uroReleaseDate = "20160119";
  77.  
  78. // list of changes affecting all users
  79. var uroChanges =
  80. [
  81. 'Correctly handles URs containing tags in multiple comments',
  82. 'Added road closure cloning functionality'
  83. ];
  84. // list of changes affecting only WME Beta users (at least until the next production release including these parts of the beta code...)
  85. var uroBetaChanges =
  86. [
  87.  
  88. ];
  89.  
  90. // true enables debug output during script startup
  91. var uroShowDebugOutput = true;
  92. // true keeps debug output enabled after script startup
  93. var uroPersistentDebugOutput = false;
  94. var uroRecentDebug = [];
  95.  
  96. var uroCtrlsHidden = false;
  97. var uroCurrentTab = 1;
  98. var uroFID = -1;
  99. var uroShownFID = -1;
  100. var uroShownPopupType = null;
  101. var uroInhibitSave = true;
  102. var uroPopupTimer = -2;
  103. var uroPopupDwellTimer = -1;
  104. var uroPopupShown = false;
  105. var uroSetupListeners = true;
  106. var uroRootContainer = null;
  107. var uroPlacesRoot = null;
  108. var uroMaskLayer = null;
  109. var uroCustomMarkerFID = null;
  110. var uroCustomMarkerType = null;
  111. var uroConfirmIntercepted = false;
  112. var uroCustomMarkerList = [];
  113. var uroPendingURSessionIDs = [];
  114. var uroRequestedURSessionIDs = [];
  115. var uroPlacesGroupsCollapsed = [];
  116.  
  117. var uroMouseInPopup = false;
  118. var uroURControlsIdx = null;
  119. var uroProblemControlsIdx = null;
  120. var uroTurnsLayerIdx = null;
  121.  
  122. var uroNullCamLayer = false;
  123. var uroNullOpenLayers = false;
  124. var uroNullRootContainer = false;
  125. var uroNullURLayer = false;
  126. var uroNullProblemLayer = false;
  127. var uroNullMapViewport = false;
  128.  
  129. var uroURDialogIsOpen = false;
  130. var uroSelectedURID = null;
  131. var uroPendingCommentDataRefresh = false;
  132. var uroWaitingCommentDataRefresh = false;
  133. var uroExpectedCommentCount = null;
  134. var uroCachedLastCommentID = null;
  135.  
  136. var uroPlaceSelected = false;
  137. var uroMouseIsDown = false;
  138. var uroBackfilling = false;
  139. var uroHidePopupOnPanelOpen = false;
  140.  
  141. var uroUserID = -1;
  142. var uroURIDInURL = null;
  143.  
  144. var uroDOMHasTurnProblems = false;
  145. var uroBetaEditor = false;
  146. var uroWazeBitsPresent = 0;
  147.  
  148. var uroOWL = null;
  149. var uroDiv = null;
  150. var uroControls = null;
  151. var uroCtrlURs = null;
  152. var uroCtrlMPs = null;
  153. var uroCtrlPlaces = null;
  154. var uroCtrlCameras = null;
  155. var uroCtrlMisc = null;
  156. var uroCtrlHides = null;
  157. var uroAMList = [];
  158.  
  159. var uroCWLGroups = [];
  160. var uroCamWatchObjects = [];
  161. var uroSegWatchObjects = [];
  162. var uroPlaceWatchObjects = [];
  163.  
  164. var uroFriendlyAreaNames = [];
  165. var uroAreaNameHoverTime = -1;
  166. var uroAreaNameHoverObj = null;
  167. var uroAreaNameOverlayShown = false;
  168. var uroANEditHovered = false;
  169. var uroANEditBox = null;
  170.  
  171. var uroPrevMouseX = -1;
  172. var uroPrevMouseY = -1;
  173.  
  174. var dteControlsIdx = -1;
  175. var dteOldestFullDrive = new Date(0);
  176. var dteEpoch = new Date(0);
  177. var dteTopID = '';
  178. var dteClearHighlightsOnPanelClose = false;
  179. var dteArmClearHighlightsOnPanelClose = false;
  180. var dteOffset = 0;
  181.  
  182. var uroUserTabId = '';
  183.  
  184. var uroTBRObj = null;
  185.  
  186. var uroBackfillQueue = [];
  187.  
  188. var uroUnstackedMasterID = null;
  189. var uroStackList = [];
  190. var uroStackType = null;
  191.  
  192. var uroHasNewFontAwesome = null;
  193.  
  194. var uroIcons =
  195. [
  196. // stuff used within the URO tabbed UI
  197. // 0 = group collapse/group expand
  198. [
  199. "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAIAAACQkWg2AAAAAXNSR0IArs4c6QAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB94CAhYRIqo78SIAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAADtJREFUKM9j/P//PwNJgCQN////ZyFeGyMjIwMDAxMDiWAQamDB9Bb+kKTMBmICdxgFK64AxZKiSE3eAAOXFRnJRfN6AAAAAElFTkSuQmCC",
  200. "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAIAAACQkWg2AAAAAXNSR0IArs4c6QAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB94CAhYRDHbt/O0AAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAE5JREFUKM+9UkEKACAMcmP//7IdgghqMQvytoOKTiMJCRKBZNRpZgbAIcIzsa5XJcgOB8Qaaz3nSt4chlLX3nb9OXRlL7cO2V83I1Dn3QDKfhshMqWScAAAAABJRU5ErkJggg=="
  201. ],
  202. // 1 = addtogroup active/addtogroup idle
  203. [
  204. "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB94BGhYVKhKBubQAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAEZJREFUOMtjYBjygBHB/P8fjzJGIszCZQA+gxkYWIjTABPDdAkL8U7G7hIcYcDISEGYwDQSFyZM5AQckYA4F7CQZ8goQAYAMCsd/TxbqrwAAAAASUVORK5CYII=",
  205. "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAAZiS0dEAAAAAAAA+UO7fwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB94DAw0tDee2t8AAAABPSURBVDjL7ZAxCgAgDAMT6cP6dH+mUwfBaugmeFMpbTgCPA9j6H2M7MidvCZlAadgADDlIXY7E1OVM5NtB+5kuZN4VDtpleIkVAOrhHxWJoJkNv2QAd51AAAAAElFTkSuQmCC"
  206. ],
  207. // 2 = goto active/goto idle
  208. [
  209. "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB94BGhYfLAEN9AsAAAAdaVRYdENvbW1lbnQAAAAAAENyZWF0ZWQgd2l0aCBHSU1QZC5lBwAAAGdJREFUOMutk+EKACEIg7fD93/l3Y+uP5kmeYMgkH0NTUpCRw+acgBynHaCKsgBlpaIhDJQtQchaAtIBuNAlAQSV7OUwPYYGX2kL6Zi37hYYk6NU1ZPui/Y4XWeVsWSDpdkt8bftvEFOy0jJkeygWMAAAAASUVORK5CYII=",
  210. "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAAXNSR0IArs4c6QAAAAZiS0dEAAAAAAAA+UO7fwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB94DAw0tFm3TfiwAAABySURBVDjLrZNRDsAgCEPp4r306HKy7sMsMRMckZH4ZfqoYEFSMnVJshYAME7aQRS0AOaR9C5UFe5AoRnsQCbAW4wFAklRlaNd1ipIrxHeRwKGZeuuNcEjK554J5x7lohNS+g+4d39Kyqug2jGyqnwtzTengM2Gxwa0GsAAAAASUVORK5CYII="
  211. ],
  212.  
  213. // replacements for native UR/problem markers
  214. // 3 = normal/selected open roadworks UR
  215. [
  216. "data:image/png;base64,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",
  217. "data:image/png;base64,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"
  218. ],
  219. // 4 = normal/selected closed roadworks UR
  220. [
  221. "data:image/png;base64,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",
  222. "data:image/png;base64,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"
  223. ],
  224.  
  225. // 5 = normal/selected open closure UR
  226. [
  227. "data:image/png;base64,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",
  228. "data:image/png;base64,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"
  229. ],
  230. // 6 = normal/selected closed closure UR
  231. [
  232. "data:image/png;base64,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",
  233. "data:image/png;base64,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"
  234. ],
  235.  
  236. // 7 = normal/selected open event UR
  237. [
  238. "data:image/png;base64,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",
  239. "data:image/png;base64,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"
  240. ],
  241. // 8 = normal/selected closed event UR
  242. [
  243. "data:image/png;base64,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",
  244. "data:image/png;base64,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"
  245. ],
  246.  
  247. // 9 = normal/selected open note UR
  248. [
  249. "data:image/png;base64,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",
  250. "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACEAAAAoCAYAAABw65OnAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAAAAd0SU1FB94IGgkbBqSwfuUAAAqJSURBVFjDrVhtbFRXen7Ox/0c3/EMYfBagBnAkSEQNcYE7+BEHVksXch+KBsZ7YpUtKuETRVloyBVlbLSNqJCldqIVG1RRfMjirT8Iaqi/RGUDUmYVQjeQCBfJtjEMWPHZLAH2+O5Ht+599xzTn+s7QK1N8nG7587Ojpz3/e8z3Pe+z4vwTc3CoDk83ni+z6ZX/Q8TxcKBQ1AA1Df5IXka+4h+XyeFotFns1meRzHRhAEzLIsWq1WSTKZ1GEYKsdxJOdcFIvFOJvNxoVCQc0Fpf+UA/YVzmlHRwdXStnJZDLheV4yCIKUUioNYAXnPG2aZppSmpRSelJKNwxDO5VKmVJKJoQgmzZt0qVS6c/KBMnn8ywMQwOAwxhrUEold+zYsXb//v3fa2lp2d7Q0LDRtu2VlFJTKRXV6/WbMzMzn4+MjLx/4sSJ0+fPn/+CUlqVUs4ACCzLEoVCQS6WlcWCIB0dHdy2bbterzcYhpF66KGH2g4cOPCL1atX76KU8qVOFEQ1OGYCSqn4+vXrb7788svHX3vttQEhRMW27Zl6vV6/ePFifGcgZLEAOOeOZVmNURRljh49+uiOHTv+jlBiU0K/NtmUVtBK18+fP/9fhw4d+o1pmuUwDKfjWi242Nd3WyDkTgiEEA4hJO26bvPx48d/nc1m9+JbWrFYPPWLgwcPz9ZqJV2pTBlCBIXPPlsUGtbU1JTI5XKru7q6tl+7du2UXka7NjR0Krdly/bOlpbVTQ0NiVsvxfwPsmfPHsN1XS8Mw1UvvPDC39577737sVymNVLp9N2bh4fD377/fn+T60ZbmprE4OSkmi88yOfzbHx83NZaJ/fs2bOls7PzIJbTCAH+8Ad0fvDBwR+sX79Fa538cmbG7sxm2UL1KxQKrF6vO5ZlpR5//PHHGGMWltuOHwdmZqzHg+Axg9LUbBw77xWLDAChAGhXV5dp23Zi27Zt69etW9e9nDAAAN58ExgbA/v8c6zt6/vLzen0eouxxM5czgRAaT6fJ0II03XdxCOPPLJr2WGQEnj1VehqFRgfR6AUyyUSu7xMJhFLaebzeUILhQKTUpphGCZaWlralz0Lp08D4+MgfX3QAIoAUlK2x6aZkFKahUKB0VwuRy3LMh3HsVOpVHZZs+D7wLlzQLEITE+jDmAUwMYf/zibaGiwLcsyc7kcpZ7n0TAMeRRFpuu6qW916jvt8mXgk0+gr16FnguAt7Rgzfe/n4qEMMMw5J7nURpFEeGcU601JYQYf/apb4tJA7VZ4K23oIeGQGo1xAAGAXxn716kOjoMrTXlnNMoiv74MXAcRwOAUkosRxYIIZCzZahPzoNcGwWkxGcA2MqVWPPzn0NKKW71S03T1HEcK9M0Za1Wm/y2WZg34c5g8m+y8H/YjBqAYQCrurvReP/9qNVqk6Zpyjm/mvu+rxhjsVIqmpiYKFqO0WQZzrcnJmMItjZg+h86QO5pRPpsCXcfPoJQBJiYmLimlIoAxL7vK9rb26uq1bKI4zgYGBj40OT2N0BCL7lWqdQxWiwDQxNgrW1o/83vkWhrhWU4GBgY+CiO4yAIAtHb26tY5/ZtZHLK54xxZ3h4OOru7n5oZGSEnzlzBslkEpZlgfP/62OUUiBzEBBCUK/XUSqV8MYbb4AxBs45Zmdn8d57V3Dol/+NbbmfYvvP/hncbYTWGlLK8LnnnvvP6enpUcbYVGtra50AoLlczp2enl4lhFi/a9euvx8cHPyrrVu3oq+vDzt37sSDDz6IFStWYPPmzbBtG/V6HVeuXMHk5CTeeecdnDt3DvP7m5ubIYRAS0sLevbtQ8e2bbdl6qOPPvrdU0899a9a62uc8/FCoTDLAeje3l7R1tZWS6fTU1NTUycfffTRnfv27fMopTh58iQuXLiATZs2obW1FbZtQwiB4eFh9Pf3o7W1Fc8++yyUUjh58iR2794N13XR39+Pbe3tCxARQhCGoX/06NGTQoipmZmZWl9fnwCg56nNurq6XEJIhhCSPXjw4I/279//S0L+P/WVUqCULrwYAOJ4FkRJEGqCMHNhfX7PHE/0iRMn/v3YsX/7LZHhsIpV+b0PPp0FIBe6my+++AKZTAZSSly6dGmyubm54fXXX9/U2dkJAAvYnz59+jbs33rrDH76k59gY8bA3X/RBUIIgiDAjRs3Fnhy4cIFDA0NvXr48D/9z/hY+ToIm/zgk/4AQHxnj0k7Ojpsz/OStVrtO4ZhrHvmmWf+euPGjY98Hey33HPPkjwplUrj5XL5V57nXcxkMqUoiqoXL16szyu128RPW1ubBqC01hJA/O677w7dvHlz7KWXXmrfu3cvP3LkCLq7u8EYwxNPPIGHH34YmUwG92/fjiAIcOnSJXz88cfIZrM4cuQI3n77bdXQ0HBKa/2PlmUNGIYxnkgkfMuywmKxqJZs+fP5PANg+77vmaa5kjHWlEwm123evPmx559//ruLtvdzPLnVnnzyyb5qtfofQRAMffnll2NRFN30PM8HUL9TBJGl1Jfv+6brugnf91O2ba9gjDVxzte6rnvo1KlTG5YqYAcOHJiuVqvPMcaKo6OjN8IwnPA8rzI7O1vzPC9aTIWxJXSCLpVKasOGDXG9Xo8YYyGAOoB6EAQf9/f379i9e7d35/8qlUrl3Llzh69evTpQqVRGtdZjSqkp13Vne3t7ozkI9GIyf0lp6Pu+dBwnjOPYp5ROKKVuKKVGr1+//i++70/PVUAopaCUUi+++OJLH3744aCUskQpvSmE8B3HCcvlsrpV4X+VKic9PT30008/NZqamswVK1aYhBCLc25KKbkQgluWZVQqFVqr1aZ37tzZUQ2miGMlcObMmdPHjh17E8BNxphPKRVBEIBzzlauXMmam5v5mjVraGNjIymXy0vCQfL5PBsbG7PWrl2bSKfTHqW0UWud5Jw3UkqTlmV5hJCElNK9cuVKmM1mnXs2bc2OjIwUn3766VeklDOU0ogQQgFYjuM4cRy7URQ5ACzf97lt27jvvvv04ODgAjS3BjE/BHENw0jFcZw2TTPNGEsLIdKEkJQQopFSmtRar5FSrjp79qzf3t7edPjw4bPlchkArD9+xRnXWie01g0AEowxh1JqmqbJTNNUYRjKVatWqVKppO+8HSyXy5mEEI8Qko7jOG0YRgMARwhhmqZpEEJYFEWG1roFQDKOY4dzzqWUSmsdEkJmOOfjlNIprXXMGIsJIUIIEZqmWQuCoGoYxpRpmtO+79fmxwTkDpIaDzzwgAvAC8PQMwzDJYTYWmtDSskJIYxSSgkhTErJOeeMEMKEEEprLRljsdZaaq0lIURxzmMAcRiGkdY6EELU4jiuaq1rly9frgOQAPStAw/d09MTl8vlwPd9FYZh6DiOWalUuG3bBqWUSSmplBKcc8o5p2EYEq01IYRoy7I0IURLKVUcx9q2bR2GYSyllJZlxYQQQQgJhRAhgGiuZOtFK+bcoIRNTEywu+66i3qeR33fZ57nkTiOiZSSAMD88/aOji2wPggCxRjTtm2rwcFBtWHDBlWtVmUmk1Ffp2Leuk7mri3K5fJtI8OvsvmRYk9PD1555ZV5h4tO8v4XQ4S+ZuwgEBYAAAAASUVORK5CYII="
  251. ],
  252. // 10 = normal/selected closed note UR
  253. [
  254. "data:image/png;base64,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",
  255. "data:image/png;base64,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"
  256. ],
  257.  
  258. // 11 = normal/selected open Elgin MP
  259. [
  260. "data:image/png;base64,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",
  261. "data:image/png;base64,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"
  262. ],
  263. // 12 - normal/selected closed Elgin MP
  264. [
  265. "data:image/png;base64,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",
  266. "data:image/png;base64,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"
  267. ],
  268.  
  269. // 13 - normal/selected open TrafficCast MP
  270. [
  271. "data:image/png;base64,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",
  272. "data:image/png;base64,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"
  273. ],
  274. // 14 - normal/selected closed TrafficCast MP
  275. [
  276. "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACEAAAAoCAYAAABw65OnAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH3wcGFgESSv4YXQAAAB1pVFh0Q29tbWVudAAAAAAAQ3JlYXRlZCB3aXRoIEdJTVBkLmUHAAAABmJLR0QAAAAAAAD5Q7t/AAAKR0lEQVRYw61YB3AU1xm+vb2iq2rAcZW72726t6drqiBUEEhCFBVLtIgcHQzYoY3lEBsGcCbBEGMTwGDIUCR6s5l48BgP2BQhQEJUZzKZccbjTByDKTZgihH5/9M7vAgEgvHNfLO7b/f9//fX996JRM//EwNoa2+rRJ+ml8aBzzhO3v/qPwoFoxKA0lvhTfRWenVsMWtgB7BGa57V5CxzGh2lDoNrkKs7V8Ul2vvalYQU9asQcBQ5pMZ0o5av4fXhyWEu/Xfp+aGZoSr/dP9o/mV+DDeRG8NP5aNp09NGhmeGy9NnpudlTMvgQtGQPsWWogQv0S9MBiyhmXxG6R/u14cmhgLhWeGK8OLwfLae/UKzUXOJXkXfFb0nahMtEz0QrRC1SdZJ7mi3ab9jt7FH0pek/yEwOVCSNjrN4xroSjVFTLJQbej5wgTsJWx/NtE3zMcGpgSKIysifzY1mM5qt2ovyVplt0UXQPFFwHnAIcAewE7ANsBW0QPZIelPzD6mKXtF9muhqaF8CJUJQqcGueKuEqCdJc6k4LigLzA9MMK5wXlAtUF1VXlCeUPcIv5ZdA4UIY4SpQ2AzQRbCL4QPVC3qq9bmi3/jOyO1IMXy33DfU73IHdSeHSYflYIxEw/RuMf7feF6kLjjfXG89I10p8kzZK7D5W3Aj4G1BPFYLnoI8ABQCOgGXAWAN5Snlfe0Dfr/+3a7PosODs42l/r90PiakFPpzlCqbqrFJDpNn4yX2PZYjktWye7JWqBuJ9vFxojsIdYix74ENBEwnKRAJ9bfhmTnZXdTm1K/dbwtqGZG8/V+mp8DmPQKH8iAbYfK7UX2vXuanehZ4Pn7wkbEm5QLdT9hwrwupcoR+v3E2L47jNANUANEAN+LyAFkLRI7mqOaK4a5hgOuYe5S91D3N1An/ixMPT099Q6yhx8cFFwnmqT6ip9kr4Xc39c2KckBA3kHseOAaIAkQBJgAWPkkCyshOy2wkLEq755/rf9FR63LY+tse8Ie3m6qZzveQqsm62nlRtU12D2LY9FHKCKG8QeAAJhAXKSwDvEC8d6UACQDVT99WN6uvWPdbT3Cgu35xp1gr7B2XOMCuMGUY2bU7a9JRtKd/IjkIZtgpIfESScAeJN44VCAj8TRCazgBelTRJ7mp3aS8F64KToGR7Qh8Sx0NBGYIGrX2APexr8O1N2p30P+ok5EJcaAtRjiQ+IWOLifIEYvnFLpCA99Qp6j6QuBzaEqp3D3Uzjv4O6cMFCfIhxV5qL+yxo8dXykblDeoskLgoiHu8/k+SMR0hMbmLBOIhOUPd1zRprjr2O467K90RS5ZFkWhKpETQUmlYG3TOGmelcoPyGtUkqAjEQUFF4PNGgASQSsr14nPgnKhNeUp5w7TP9KV3lLfY1teWAvrFIogLrQ/qjWw1O1qxVnE9RkJo2SeERFzhXAANYJ6TAPGY9IT0To+9Pb5yjXSN6tWnlxn0yzAnJKYMk5mtYcepPlBdeYxEvDvuI8/TSC8IvhgJzLfkrcn/dY5yTrAV2Fy4PYitltCkTI6RjsmadZrL1PEO4djfgcQMQsL9giSgVFU7VFdB30S2hPVYcixqDIcYSBg8YzwTUjemfi1tlN55hMQBEo4d5HkZyQnTk/tBV0iod6qveqKeKDOAcYJ+FSam2FHi6I59Xb9J/w/lYeUPVKugOg4LEvMcWaSUpDz/8pwkoPcoTihu6vfp/+Wp9VRDOGzZ2dmKWLNyljqT+XH8EGY1c1CzQ/P9I3mBZbmdlOghMlZJStT/6GL1LC+ArDYMhWej52NnhbOfOctsBDnt7RuWFCUf5Xu769zLE9cnfiv9HEIS75ioYDdpVh+S59MAFSHSv4teALL0Efqeerv6CjeDm8uWsWFjxJicn58viZHwVfikgWjAyU3hphnXGs8iW7oJFrCOvWKLIA82klVTRJJ0LQnduScQQOKnRG0J9Qk/4trhGuaqYooYGxSForq6ur11q7qpKO9QbzI/lu/vft29MnV96teKg4r2pfwCceUuUiXbycYFhX8A6E6I0IRMHuD9DiTAc7JG2W3tHu1lz1RPnW2ALdKT75lMjgi//EK/CcnTatMc3nHesey77H7Nes1ldJ/oDCHRQkLSQLZ1zYK1ZQwhEV/Q6gBnBGFopO8p65XXvCu925lBTFmv3F7m3r17y564sYE9YCI3gstxjnHONK8wn5Qtld2EnVJbzMUXyL5yq2A/KXT/WeKB2eR9Szuow9R9+Rr5Tfsm+1Gmkqm15Fp8sFZpOj0kFRcXy2D7ZXRVuQocv3XMMi03nZL/Vf4jfRQ80kqIHCFE4htcXOY/FyzxF3/xHM5LWJ/wg+qPqu9MJabXbUW2LOhJunA4LO10owuZKi4pKVFVVFRYcZvHz+Rn2JfaP1UsUFyhFlM/P1Rwmqwl9YKd9jaSL3vaPYY7c9nbspvm98zH/a/4JzgGO3Ks+VYL6FCinqeeuOADury8XFtVVWXNnJOZG3wtOIlbxK3SLtR+E+uWCwUWHybnje2CfGlorxTVn1SX+AX8krSX02pwr2KIGAzYorEk582b98wTGRKR5OXlqcLjYe6USBCOd4O8k7yvOmY5Vsnny6+L1pCyu0CuTaSMV8Pu6S36tneJd1OgLvAqbBcHWPpYPKZMU3c4zyhJT3g6gQcPHlCCQzANkxLs+fZUWGhsfC2fw43jRrBRdq5ituJ7cSschJpJlZxuJyTfJb/Fz+cXs5XscHYgmwVVYNVxuiSUg/LAA+LnOYXHSGACYW+HckriqjmLq9wVYgYz5UyUWQQN7Zp4hfgeWo9nUciJNucbzvWWIkuNMdMY0of0JiSA8zmOkwn+Nnh2KKCD4cdynU6nyszM1IKQlJycnB5wb9JH9F5LgaXQVGQaz7zJ7JStl92iVlP36Y30Petaa6MhzzBZn6nvZ0g3+GBOL5/Pp8vIyEgtKipKBEM0AkKdkonlAlivxImoFCaxWVlZntzcXH+fPn3Cffv2zSkoKCgpLCyMVlZWLtQt132JByLNZs2lIUOGLARFYwEDAblAIh3mpKWnp3MgzwlyrECgJ8hPLC0tlXfWJ3BQjpbDZDtMDoDy3oB+IKAUBA8BYlXwbiSM1QHhd8rKyjYcO3bsP9FodAe8fxfevwHzx8H7l4DEUEAZjA2AsTwkhQbBdwb0CukVj3mDRneh6+EjF0zOAoVFgMGoHAQPB0+MAkFReJ4H3y4Foivhugau74PQZZFIZBG8fwVk1ML4CJhbDfPK0QiY2xcNwzCBAUmdkYh5AiYmAyzoRmRPXFuIFqEwtA69AveVRMlwuK9Bomg9KB+E38G1GL0IyINwZCMB+DkwTyAsatD1xHKNJSWWE8YNPwbyZr/fbyMx9SAxBCjgSbwDYH0Qr/iMuQNEfPgN3HtxHshi0HoMA+YaEhAkZ+d/kqGrrFYrksEGo0ZS6EL0EuYMAgV2RPwdAqxGl8eqAistXhldalgdekXs70LiIawcaVdBlNGk5MVPK8v/AwujhYTKrIZnAAAAAElFTkSuQmCC",
  277. "data:image/png;base64,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"
  278. ],
  279.  
  280. // 15 - normal/selected open TrafficMaster MP
  281. [
  282. "data:image/png;base64,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",
  283. "data:image/png;base64,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"
  284. ],
  285. // 16 - normal/selected closed TrafficMaster MP
  286. [
  287. "data:image/png;base64,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",
  288. "data:image/png;base64,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"
  289. ]
  290. ];
  291.  
  292. var uroMarkers =
  293. [
  294. // 0 = comment count circle
  295. ["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=="],
  296. // 1 = green comment marker
  297. ["data:image/png;base64,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"],
  298. // 2 = yellow (own) comment marker
  299. ["data:image/png;base64,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"]
  300. ];
  301.  
  302. function uroAddDebug(debugtext)
  303. {
  304. var ts = Math.round(performance.now());
  305. if(uroRecentDebug.length == 100)
  306. {
  307. uroRecentDebug.shift();
  308. }
  309. uroRecentDebug.push(ts+': '+debugtext);
  310. console.debug('URO+DBG '+ts+':'+debugtext);
  311. }
  312.  
  313. function uroDumpDebug()
  314. {
  315. if(uroRecentDebug.length > 0)
  316. {
  317. document.getElementById('WazeMap').innerHTML = '<textarea id="uroDbgOutput" style="width:100%;height:100%">';
  318. var dbgOutput = '';
  319. for(var i=0; i<uroRecentDebug.length; i++)
  320. {
  321. dbgOutput += uroRecentDebug[i]+'\n';
  322. }
  323. document.getElementById('uroDbgOutput').textContent = dbgOutput;
  324. }
  325. }
  326.  
  327. function uroAddLog(logtext)
  328. {
  329. if(uroShowDebugOutput) console.log('URO+: '+logtext);
  330. }
  331.  
  332. function uroGetCBChecked(cbID)
  333. {
  334. return(document.getElementById(cbID).checked);
  335. }
  336.  
  337. function uroSetCBChecked(cbID, state)
  338. {
  339. document.getElementById(cbID).checked = state;
  340. }
  341.  
  342. function uroGetElmValue(elmID)
  343. {
  344. return(document.getElementById(elmID).value);
  345. }
  346.  
  347. function uroSetStyleDisplay(elm,style)
  348. {
  349. document.getElementById(elm).style.display = style;
  350. }
  351.  
  352. function uroSetOnClick(elm,fn)
  353. {
  354. document.getElementById(elm).onclick = fn;
  355. }
  356.  
  357. function uroAddEventListener(elm,eventType,eventFn,eventBool)
  358. {
  359. document.getElementById(elm).addEventListener(eventType, eventFn, eventBool);
  360. }
  361.  
  362.  
  363. function uroFirstTimerWelcomePack()
  364. {
  365. uroAddLog('welcome new users to Club URO...');
  366.  
  367. // to be completed, perhaps...
  368. }
  369.  
  370. function uroShowUpdateNotes()
  371. {
  372. uroAddLog('let existing users know what\'s new in this release');
  373.  
  374. var alertMsg = 'URO+ Update Notes...\n\n';
  375. alertMsg += 'Thanks for upgrading to URO+ '+uroVersion+' ('+uroReleaseDate+'). What\'s changed?\n\n';
  376.  
  377. var loop;
  378. if(uroChanges.length > 0)
  379. {
  380. for(loop=0; loop < uroChanges.length; loop++)
  381. {
  382. alertMsg += '* '+uroChanges[loop]+'\n';
  383. }
  384. }
  385. if((uroBetaEditor) && (uroBetaChanges.length > 0))
  386. {
  387. alertMsg += '\nFor WME Beta:\n';
  388. for(loop=0; loop < uroBetaChanges.length; loop++)
  389. {
  390. alertMsg += '* '+uroBetaChanges[loop]+'\n';
  391. }
  392. }
  393.  
  394. alert(alertMsg);
  395. }
  396.  
  397. function uroAdvertiseCustomIcons()
  398. {
  399. uroAddLog('advertise the benefits of custom UR icons...');
  400.  
  401. var confirmMsg = 'URO+ Installation/Upgrade Processing...\n\n';
  402. confirmMsg += 'Hi there. One of the features of URO+ that a lot of users find useful is the ability to use a custom marker for URs and MPs which have been tagged with a specific keyword in their description text.\n\n';
  403. confirmMsg += 'Markers are defined for [ROADWORKS], [CONSTRUCTION], [CLOSURE], [EVENT] and [NOTE] tags in URs, and [Elgin], [TM] and [TrafficCast] in MPs.\n\n';
  404. confirmMsg += 'Would you like me to automatically enable these custom markers?\n\n';
  405. confirmMsg += 'If you change your mind later on, they can be enabled/disabled via the Misc tab within the URO+ settings';
  406.  
  407. if(confirm(confirmMsg) === true)
  408. {
  409. uroSetCBChecked('_cbCustomRoadworksMarkers', true);
  410. uroSetCBChecked('_cbCustomConstructionMarkers', true);
  411. uroSetCBChecked('_cbCustomClosuresMarkers', true);
  412. uroSetCBChecked('_cbCustomEventsMarkers', true);
  413. uroSetCBChecked('_cbCustomNotesMarkers', true);
  414. uroSetCBChecked('_cbCustomElginMarkers', true);
  415. uroSetCBChecked('_cbCustomTrafficMasterMarkers', true);
  416. uroSetCBChecked('_cbCustomTrafficCastMarkers', true);
  417. }
  418. }
  419.  
  420.  
  421. function uroGatherSettings(container)
  422. {
  423. var options = '';
  424. var urOptions = document.getElementById(container).getElementsByTagName('input');
  425. for (var optIdx=0;optIdx<urOptions.length;optIdx++)
  426. {
  427. var id = urOptions[optIdx].id;
  428. if((id.indexOf('_cb') === 0)||(id.indexOf('_text') === 0)||(id.indexOf('_input') === 0))
  429. {
  430. options += ':' + id;
  431. if(urOptions[optIdx].type == 'checkbox') options += ',' + urOptions[optIdx].checked.toString();
  432. else if((urOptions[optIdx].type == 'text')||(urOptions[optIdx].type == 'number')) options += ',' + urOptions[optIdx].value.toString();
  433. }
  434. }
  435. return options;
  436. }
  437.  
  438.  
  439. function uroGatherCamWatchList()
  440. {
  441. var liststr = '';
  442. for(var loop=0;loop<uroCamWatchObjects.length;loop++)
  443. {
  444. var camObj = uroCamWatchObjects[loop];
  445. if((camObj.fid !== undefined) && (camObj.persistent === true))
  446. {
  447. if(loop > 0) liststr += ':';
  448.  
  449. liststr += camObj.fid+',';
  450. liststr += camObj.watch.lon+',';
  451. liststr += camObj.watch.lat+',';
  452. liststr += camObj.watch.type+',';
  453. liststr += camObj.watch.azymuth+',';
  454. liststr += camObj.watch.speed+',';
  455. liststr += camObj.watch.validated+',';
  456. liststr += camObj.groupID+',';
  457. liststr += camObj.server;
  458. }
  459. }
  460. return liststr;
  461. }
  462. function uroGatherSegWatchList()
  463. {
  464. var liststr = '';
  465. for(var loop=0;loop<uroSegWatchObjects.length;loop++)
  466. {
  467. var segObj = uroSegWatchObjects[loop];
  468. if((segObj.fid !== undefined) && (segObj.persistent === true))
  469. {
  470. if(loop > 0) liststr += ':';
  471.  
  472. liststr += segObj.fid+',';
  473. liststr += segObj.watch.left+',';
  474. liststr += segObj.watch.right+',';
  475. liststr += segObj.watch.bottom+',';
  476. liststr += segObj.watch.top+',';
  477. liststr += segObj.watch.fromNode+',';
  478. liststr += segObj.watch.toNode+',';
  479. liststr += segObj.watch.fwdDir+',';
  480. liststr += segObj.watch.revDir+',';
  481. liststr += segObj.watch.length+',';
  482. liststr += segObj.watch.level+',';
  483. liststr += segObj.watch.rank+',';
  484. liststr += segObj.watch.roadType+',';
  485. liststr += segObj.watch.updatedOn+',';
  486. liststr += segObj.groupID+',';
  487. liststr += segObj.server;
  488. }
  489. }
  490. return liststr;
  491. }
  492. function uroGatherPlaceWatchList()
  493. {
  494. var liststr = '';
  495. for(var loop=0;loop<uroPlaceWatchObjects.length;loop++)
  496. {
  497. var placeObj = uroPlaceWatchObjects[loop];
  498. if((placeObj.fid !== undefined) && (placeObj.persistent === true))
  499. {
  500. if(loop > 0) liststr += ':';
  501.  
  502. liststr += placeObj.fid+',';
  503. liststr += placeObj.watch.left+',';
  504. liststr += placeObj.watch.right+',';
  505. liststr += placeObj.watch.bottom+',';
  506. liststr += placeObj.watch.top+',';
  507. liststr += placeObj.watch.name+',';
  508. liststr += placeObj.watch.imageCount+',';
  509. liststr += placeObj.watch.residential+',';
  510. liststr += placeObj.watch.updatedOn+',';
  511. liststr += placeObj.groupID+',';
  512. liststr += placeObj.server;
  513. }
  514. }
  515. return liststr;
  516. }
  517. function uroGatherCWLGroups()
  518. {
  519. var liststr = '';
  520. for(var loop=0;loop<uroCWLGroups.length;loop++)
  521. {
  522. var groupObj = uroCWLGroups[loop];
  523. if(groupObj.groupID != -1)
  524. {
  525. if(loop > 0) liststr += ':';
  526.  
  527. liststr += groupObj.groupID+',';
  528. liststr += groupObj.groupName+',';
  529. liststr += groupObj.groupCollapsed;
  530. }
  531. }
  532. return liststr;
  533. }
  534. function uroGatherPlacesGroups()
  535. {
  536. var liststr = '';
  537. for(var loop=0;loop<uroPlacesGroupsCollapsed.length;loop++)
  538. {
  539. if(loop > 0) liststr += ':';
  540. liststr += uroPlacesGroupsCollapsed[loop];
  541. }
  542. return liststr;
  543. }
  544. function uroGatherFriendlyAreaNames()
  545. {
  546. var liststr = '';
  547. for(var loop=0;loop<uroFriendlyAreaNames.length;loop++)
  548. {
  549. var fnObj = uroFriendlyAreaNames[loop];
  550. if(loop > 0) liststr += ':';
  551.  
  552. liststr += fnObj.fName+',';
  553. liststr += fnObj.area+',';
  554. liststr += fnObj.server;
  555. }
  556. return liststr;
  557. }
  558.  
  559. function uroSaveSettings()
  560. {
  561. if(uroInhibitSave)
  562. {
  563. uroAddLog('save inhibited');
  564. return;
  565. }
  566.  
  567. if (localStorage)
  568. {
  569. localStorage.UROverviewUROptions = uroGatherSettings('uroCtrlURs');
  570. localStorage.UROverviewMPOptions = uroGatherSettings('uroCtrlMPs');
  571. localStorage.UROverviewCameraOptions = uroGatherSettings('uroCtrlCameras');
  572. localStorage.UROverviewMiscOptions = uroGatherSettings('uroCtrlMisc');
  573. localStorage.UROverviewPlacesOptions = uroGatherSettings('uroCtrlPlaces');
  574. localStorage.UROverviewCamWatchList = uroGatherCamWatchList();
  575. localStorage.UROverviewSegWatchList = uroGatherSegWatchList();
  576. localStorage.UROverviewPlaceWatchList = uroGatherPlaceWatchList();
  577. localStorage.UROverviewCWLGroups = uroGatherCWLGroups();
  578. localStorage.UROverviewFriendlyAreaNames = uroGatherFriendlyAreaNames();
  579. localStorage.UROverviewPlacesGroups = uroGatherPlacesGroups();
  580.  
  581. localStorage.UROverviewMasterEnable = uroGetCBChecked('_cbMasterEnable');
  582. localStorage.UROverviewCurrentVersion = uroVersion;
  583.  
  584. uroAddLog('save complete');
  585. }
  586. else
  587. {
  588. uroAddLog('no localStorage, save blocked');
  589. }
  590. }
  591.  
  592. function uroApplySettings(settings)
  593. {
  594. var options = settings.split(':');
  595. for(var optIdx=0;optIdx<options.length;optIdx++)
  596. {
  597. var fields = options[optIdx].split(',');
  598. if(fields[0].indexOf('_cb') === 0)
  599. {
  600. if(document.getElementById(fields[0]) !== null)
  601. {
  602. uroSetCBChecked(fields[0], (fields[1] == 'true'));
  603. }
  604. }
  605. else if((fields[0].indexOf('_input') === 0)||(fields[0].indexOf('_text') === 0))
  606. {
  607. if(document.getElementById(fields[0]) !== null) document.getElementById(fields[0]).value = fields[1];
  608. }
  609. }
  610. }
  611.  
  612.  
  613. function uroApplyCamWatchList()
  614. {
  615. var objects = localStorage.UROverviewCamWatchList.split(':');
  616. uroCamWatchObjects = [];
  617. if(objects.length > 0)
  618. {
  619. for(var objIdx=0;objIdx<objects.length;objIdx++)
  620. {
  621. var fields = objects[objIdx].split(',');
  622. if(fields.length >= 7)
  623. {
  624. // following two bits of code add in blank fields if the user has updated their copy of URO+ from an
  625. // older version which didn't include support for either of these field types
  626.  
  627. // add default groupID field
  628. if(fields.length == 7)
  629. {
  630. fields.push(0);
  631. }
  632. // set default groupID value to 0 (no group)
  633. if(fields[7] == -1)
  634. {
  635. fields[7] = 0;
  636. }
  637.  
  638. // add default server field
  639. if(fields.length == 8)
  640. {
  641. fields.push('??');
  642. }
  643. // set default server value to unknown
  644. if(fields[8] === 0)
  645. {
  646. fields[8] = '??';
  647. }
  648.  
  649. uroCamWatchObjects.push(new uroCamWatchObj(true,fields[0],fields[1],fields[2],fields[3],fields[4],fields[5],fields[6],fields[7],fields[8]));
  650. }
  651. }
  652. }
  653. }
  654. function uroApplySegWatchList()
  655. {
  656. var objects = localStorage.UROverviewSegWatchList.split(':');
  657. uroSegWatchObjects = [];
  658.  
  659. for(var objIdx=0;objIdx<objects.length;objIdx++)
  660. {
  661. var fields = objects[objIdx].split(',');
  662. 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]));
  663. }
  664. }
  665. function uroApplyPlaceWatchList()
  666. {
  667. var objects = localStorage.UROverviewPlaceWatchList.split(':');
  668. uroPlaceWatchObjects = [];
  669.  
  670. for(var objIdx=0;objIdx<objects.length;objIdx++)
  671. {
  672. var fields = objects[objIdx].split(',');
  673. 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]));
  674. }
  675. }
  676.  
  677. function uroApplyCWLGroups()
  678. {
  679. var objects = localStorage.UROverviewCWLGroups.split(':');
  680. uroCWLGroups = [];
  681.  
  682. for(var objIdx=0;objIdx<objects.length;objIdx++)
  683. {
  684. var fields = objects[objIdx].split(',');
  685. if(fields.length < 2)
  686. {
  687. fields.push(false);
  688. }
  689. uroCWLGroups.push(new uroOWLGroupObj(fields[0],fields[1],(fields[2] == 'true')));
  690. }
  691. }
  692.  
  693. function uroApplyPlacesGroups()
  694. {
  695. var t = localStorage.UROverviewPlacesGroups.split(':');
  696. for(var i=0;i<t.length;i++)
  697. {
  698. uroPlacesGroupsCollapsed[i] = (t[i] == "true");
  699. }
  700. }
  701.  
  702.  
  703. function uroApplyFriendlyAreaNames()
  704. {
  705. var objects = localStorage.UROverviewFriendlyAreaNames.split(':');
  706. uroFriendlyAreaNames = [];
  707.  
  708. for(var objIdx=0;objIdx<objects.length;objIdx++)
  709. {
  710. var fields = objects[objIdx].split(',');
  711. uroFriendlyAreaNames.push(new uroAFNObj(fields[0],parseFloat(fields[1]),fields[2]));
  712. }
  713.  
  714. uroReplaceAreaNames(true);
  715. }
  716.  
  717. function uroLoadSettings()
  718. {
  719. var isNewInstall = true;
  720. var isUpgradeInstall = true;
  721. var notifyAboutCustomIcons = true;
  722.  
  723. uroAddLog('loadSettings()');
  724. if (localStorage.UROverviewUROptions !== undefined)
  725. {
  726. uroAddLog('recover UR tab settings');
  727. uroApplySettings(localStorage.UROverviewUROptions);
  728. isNewInstall = false;
  729. }
  730.  
  731. if (localStorage.UROverviewCameraOptions !== undefined)
  732. {
  733. uroAddLog('recover camera tab settings');
  734. uroApplySettings(localStorage.UROverviewCameraOptions);
  735. isNewInstall = false;
  736. }
  737.  
  738. if (localStorage.UROverviewMPOptions !== undefined)
  739. {
  740. uroAddLog('recover MP tab settings');
  741. uroApplySettings(localStorage.UROverviewMPOptions);
  742. isNewInstall = false;
  743. }
  744.  
  745. if (localStorage.UROverviewPlacesOptions !== undefined)
  746. {
  747. uroAddLog('recover Places tab settings');
  748. uroApplySettings(localStorage.UROverviewPlacesOptions);
  749. isNewInstall = false;
  750. }
  751.  
  752. if (localStorage.UROverviewMiscOptions !== undefined)
  753. {
  754. uroAddLog('recover misc tab settings');
  755. uroApplySettings(localStorage.UROverviewMiscOptions);
  756. isNewInstall = false;
  757.  
  758. if(localStorage.UROverviewCurrentVersion !== undefined)
  759. {
  760. notifyAboutCustomIcons = false;
  761. }
  762. else
  763. {
  764. if(uroGetCBChecked('_cbCustomRoadworksMarkers') === true) notifyAboutCustomIcons = false;
  765. if(uroGetCBChecked('_cbCustomConstructionMarkers')=== true) notifyAboutCustomIcons = false;
  766. if(uroGetCBChecked('_cbCustomClosuresMarkers') === true) notifyAboutCustomIcons = false;
  767. if(uroGetCBChecked('_cbCustomEventsMarkers') === true) notifyAboutCustomIcons = false;
  768. if(uroGetCBChecked('_cbCustomNotesMarkers') === true) notifyAboutCustomIcons = false;
  769. }
  770. }
  771.  
  772. if(localStorage.UROverviewCWLGroups !== undefined)
  773. {
  774. uroAddLog('recover CWL groups');
  775. uroApplyCWLGroups();
  776. isNewInstall = false;
  777. }
  778. else
  779. {
  780. uroAddLog('set default CWL group');
  781. uroCWLGroups.push(new uroOWLGroupObj(0,'No group',false));
  782. }
  783.  
  784. if(localStorage.UROverviewCamWatchList !== undefined)
  785. {
  786. uroAddLog('recover camera watchlist');
  787. uroApplyCamWatchList();
  788. uroGetCurrentCamWatchListObjects();
  789. isNewInstall = false;
  790. }
  791.  
  792. if(localStorage.UROverviewSegWatchList !== undefined)
  793. {
  794. uroAddLog('recover segment watchlist');
  795. uroApplySegWatchList();
  796. uroGetCurrentSegWatchListObjects();
  797. isNewInstall = false;
  798. }
  799.  
  800. if(localStorage.UROverviewPlaceWatchList !== undefined)
  801. {
  802. uroAddLog('recover places watchlist');
  803. uroApplyPlaceWatchList();
  804. //uroGetCurrentPlaceWatchListObjects();
  805. isNewInstall = false;
  806. }
  807.  
  808. if(localStorage.UROverviewPlacesGroups !== undefined)
  809. {
  810. uroAddLog('recover places groups');
  811. uroApplyPlacesGroups();
  812. isNewInstall = false;
  813. }
  814.  
  815. if(localStorage.UROverviewCurrentVersion !== undefined)
  816. {
  817. uroAddLog('comparing install versions');
  818. if(localStorage.UROverviewCurrentVersion == uroVersion)
  819. {
  820. isUpgradeInstall = false;
  821. }
  822. }
  823.  
  824. if(localStorage.UROverviewFriendlyAreaNames !== undefined)
  825. {
  826. uroAddLog('recover friendly area names');
  827. uroApplyFriendlyAreaNames();
  828. isNewInstall = false;
  829. }
  830.  
  831. if(localStorage.UROverviewMasterEnable !== undefined)
  832. {
  833. uroAddLog('recover master enable state');
  834. document.getElementById('_cbMasterEnable').checked = (localStorage.UROverviewMasterEnable == "true");
  835. }
  836.  
  837. if(isNewInstall)
  838. {
  839. uroFirstTimerWelcomePack();
  840. }
  841. else if(isUpgradeInstall)
  842. {
  843. uroShowUpdateNotes();
  844. }
  845.  
  846. if(notifyAboutCustomIcons)
  847. {
  848. uroAdvertiseCustomIcons();
  849. }
  850.  
  851. uroInhibitSave = false;
  852. }
  853.  
  854. function uroDefaultSettings()
  855. {
  856. if(confirm('Resetting URO+ settings cannot be undone\nAre you sure you want to do this?') === true)
  857. {
  858. var defaultSettings = '';
  859. defaultSettings += '[UROverviewMPOptions][len=628]:_cbMPFilterMissingJunction,false:_cbMPFilterMissingRoad,false:_cbMPFilterCrossroadsJunctionMissing,false:_cbMPFilterDrivingDirectionMismatch,false:_cbMPFilterRoadTypeMismatch,false:_cbMPFilterRestrictedTurn,false:_cbMPFilterRoadClosureProblem,false:_cbMPFilterUnknownProblem,false:_cbMPFilterTurnProblem,false:_cbMPFilterReopenedProblem,false:_cbInvertMPFilter,false:_cbMPFilterOutsideArea,false:_cbMPFilterClosed,false:_cbMPFilterSolved,false:_cbMPFilterUnidentified,false:_cbMPClosedUserIDFilter,false:_cbMPNotClosedUserIDFilter,false:_cbMPFilterLowSeverity,false:_cbMPFilterMediumSeverity,false:_cbMPFilterHighSeverity,false[END]';
  860. defaultSettings += '[UROverviewPlacesGroups][len=59]false:false:false:false:false:false:false:false:false:false[END]';
  861. defaultSettings += '[UROverviewCamWatchList][len=0][END]';
  862. defaultSettings += '[UROverviewFriendlyAreaNames][len=0][END]';
  863. defaultSettings += '[UROverviewMiscOptions][len=1141]:_cbNativeConvoMarkers,true:_cbNativeBetaConvoMarkers,true:_cbCommentCount,false:_cbURBackfill,false:_inputUnstackSensitivity,15:_inputUnstackZoomLevel,3:_cbCustomRoadworksMarkers,false:_cbCustomConstructionMarkers,false:_cbCustomClosuresMarkers,false:_cbCustomEventsMarkers,false:_cbCustomNotesMarkers,false:_cbCustomElginMarkers,false:_cbCustomTrafficMasterMarkers,false:_cbCustomTrafficCastMarkers,false:_inputPopupDwellTimeout,2:_inputPopupEntryTimeout,2:_inputMaxJitter,2:_cbInhibitURPopup,false:_cbInhibitMPPopup,false:_cbInhibitCamPopup,false:_cbInhibitSegPopup,false:_cbInhibitTurnsPopup,false:_cbInhibitLandmarkPopup,false:_cbInhibitPUPopup,false:_cbDateFmtDDMMYY,true:_cbDateFmtMMDDYY,false:_cbDateFmtYYMMDD,false:_cbTimeFmt24H,false:_cbTimeFmt12H,false:_cbWhiteBackground,false:_inputCustomBackgroundRed,255:_inputCustomBackgroundGreen,255:_inputCustomBackgroundBlue,255:_cbHideAMLayer,false:_cbDisablePlacesFiltering,false:_cbInhibitNURButton,false:_cbInhibitNMPButton,false:_cbInhibitNPURButton,false:_cbInhibitURCentering,false:_cbInhibitMPCentering,false:_cbInhibitPURCentering,false:_cbHideEditorInfo,false:_cbEnableDTE,false[END]';
  864. defaultSettings += '[UROverviewSegWatchList][len=0][END]';
  865. defaultSettings += '[UROverviewPlacesOptions][len=5106]:_cbFilterUneditablePlaceUpdates,false:_cbFilterLockRankedPlaceUpdates,false:_cbFilterNewPlacePUR,false:_cbFilterUpdatedDetailsPUR,false:_cbFilterNewPhotoPUR,false:_cbFilterFlaggedPUR,false:_cbLeavePURGeos,false:_cbInvertPURFilters,false:_cbPURFilterLowSeverity,false:_cbPURFilterMediumSeverity,false:_cbPURFilterHighSeverity,false:_cbEnablePURMinAgeFilter,false:_inputPURFilterMinDays,:_cbEnablePURMaxAgeFilter,false:_inputPURFilterMaxDays,:_cbPlaceFilterEditedLessThan,false:_inputFilterPlaceEditMinDays,:_cbPlaceFilterEditedMoreThan,false:_inputFilterPlaceEditMaxDays,:_cbHidePlacesL0,false:_cbHidePlacesL1,false:_cbHidePlacesL2,false:_cbHidePlacesL3,false:_cbHidePlacesL4,false:_cbHidePlacesL5,false:_cbHidePhotoPlaces,false:_cbHideNoPhotoPlaces,false:_cbPlacesFilter-CAR_SERVICES,false:_cbPlacesFilter-GAS_STATION,false:_cbPlacesFilter-PARKING_LOT,false:_cbPlacesFilter-GARAGE_AUTOMOTIVE_SHOP,false:_cbPlacesFilter-CAR_WASH,false:_cbPlacesFilter-CHARGING_STATION,false:_cbPlacesFilter-TRANSPORTATION,false:_cbPlacesFilter-AIRPORT,false:_cbPlacesFilter-BUS_STATION,false:_cbPlacesFilter-FERRY_PIER,false:_cbPlacesFilter-SEAPORT_MARINA_HARBOR,false:_cbPlacesFilter-SUBWAY_STATION,false:_cbPlacesFilter-TRAIN_STATION,false:_cbPlacesFilter-BRIDGE,false:_cbPlacesFilter-TUNNEL,false:_cbPlacesFilter-TAXI_STATION,false:_cbPlacesFilter-JUNCTION_INTERCHANGE,false:_cbPlacesFilter-PROFESSIONAL_AND_PUBLIC,false:_cbPlacesFilter-COLLEGE_UNIVERSITY,false:_cbPlacesFilter-SCHOOL,false:_cbPlacesFilter-CONVENTIONS_EVENT_CENTER,false:_cbPlacesFilter-GOVERNMENT,false:_cbPlacesFilter-LIBRARY,false:_cbPlacesFilter-CITY_HALL,false:_cbPlacesFilter-ORGANIZATION_OR_ASSOCIATION,false:_cbPlacesFilter-PRISON_CORRECTIONAL_FACILITY,false:_cbPlacesFilter-COURTHOUSE,false:_cbPlacesFilter-CEMETERY,false:_cbPlacesFilter-FIRE_DEPARTMENT,false:_cbPlacesFilter-POLICE_STATION,false:_cbPlacesFilter-MILITARY,false:_cbPlacesFilter-HOSPITAL_MEDICAL_CARE,false:_cbPlacesFilter-OFFICES,false:_cbPlacesFilter-POST_OFFICE,false:_cbPlacesFilter-RELIGIOUS_CENTER,false:_cbPlacesFilter-KINDERGARDEN,false:_cbPlacesFilter-FACTORY_INDUSTRIAL,false:_cbPlacesFilter-EMBASSY_CONSULATE,false:_cbPlacesFilter-INFORMATION_POINT,false:_cbPlacesFilter-SHOPPING_AND_SERVICES,false:_cbPlacesFilter-ARTS_AND_CRAFTS,false:_cbPlacesFilter-BANK_FINANCIAL,false:_cbPlacesFilter-SPORTING_GOODS,false:_cbPlacesFilter-BOOKSTORE,false:_cbPlacesFilter-PHOTOGRAPHY,false:_cbPlacesFilter-CAR_DEALERSHIP,false:_cbPlacesFilter-FASHION_AND_CLOTHING,false:_cbPlacesFilter-CONVENIENCE_STORE,false:_cbPlacesFilter-PERSONAL_CARE,false:_cbPlacesFilter-DEPARTMENT_STORE,false:_cbPlacesFilter-PHARMACY,false:_cbPlacesFilter-ELECTRONICS,false:_cbPlacesFilter-FLOWERS,false:_cbPlacesFilter-FURNITURE_HOME_STORE,false:_cbPlacesFilter-GIFTS,false:_cbPlacesFilter-GYM_FITNESS,false:_cbPlacesFilter-SWIMMING_POOL,false:_cbPlacesFilter-HARDWARE_STORE,false:_cbPlacesFilter-MARKET,false:_cbPlacesFilter-SUPERMARKET_GROCERY,false:_cbPlacesFilter-JEWELRY,false:_cbPlacesFilter-LAUNDRY_DRY_CLEAN,false:_cbPlacesFilter-SHOPPING_CENTER,false:_cbPlacesFilter-MUSIC_STORE,false:_cbPlacesFilter-PET_STORE_VETERINARIAN_SERVICES,false:_cbPlacesFilter-TOY_STORE,false:_cbPlacesFilter-TRAVEL_AGENCY,false:_cbPlacesFilter-ATM,false:_cbPlacesFilter-CURRENCY_EXCHANGE,false:_cbPlacesFilter-CAR_RENTAL,false:_cbPlacesFilter-FOOD_AND_DRINK,false:_cbPlacesFilter-RESTAURANT,false:_cbPlacesFilter-BAKERY,false:_cbPlacesFilter-DESSERT,false:_cbPlacesFilter-CAFE,false:_cbPlacesFilter-FAST_FOOD,false:_cbPlacesFilter-FOOD_COURT,false:_cbPlacesFilter-BAR,false:_cbPlacesFilter-ICE_CREAM,false:_cbPlacesFilter-CULTURE_AND_ENTERTAINEMENT,false:_cbPlacesFilter-ART_GALLERY,false:_cbPlacesFilter-CASINO,false:_cbPlacesFilter-CLUB,false:_cbPlacesFilter-TOURIST_ATTRACTION_HISTORIC_SITE,false:_cbPlacesFilter-MOVIE_THEATER,false:_cbPlacesFilter-MUSEUM,false:_cbPlacesFilter-MUSIC_VENUE,false:_cbPlacesFilter-PERFORMING_ARTS_VENUE,false:_cbPlacesFilter-GAME_CLUB,false:_cbPlacesFilter-STADIUM_ARENA,false:_cbPlacesFilter-THEME_PARK,false:_cbPlacesFilter-ZOO_AQUARIUM,false:_cbPlacesFilter-RACING_TRACK,false:_cbPlacesFilter-THEATER,false:_cbPlacesFilter-OTHER,false:_cbPlacesFilter-CONSTRUCTION_SITE,false:_cbPlacesFilter-LODGING,false:_cbPlacesFilter-HOTEL,false:_cbPlacesFilter-HOSTEL,false:_cbPlacesFilter-CAMPING_TRAILER_PARK,false:_cbPlacesFilter-COTTAGE_CABIN,false:_cbPlacesFilter-BED_AND_BREAKFAST,false:_cbPlacesFilter-OUTDOORS,false:_cbPlacesFilter-PARK,false:_cbPlacesFilter-PLAYGROUND,false:_cbPlacesFilter-BEACH,false:_cbPlacesFilter-SPORTS_COURT,false:_cbPlacesFilter-GOLF_COURSE,false:_cbPlacesFilter-PLAZA,false:_cbPlacesFilter-PROMENADE,false:_cbPlacesFilter-POOL,false:_cbPlacesFilter-SCENIC_LOOKOUT_VIEWPOINT,false:_cbPlacesFilter-SKI_AREA,false:_cbPlacesFilter-NATURAL_FEATURES,false:_cbPlacesFilter-ISLAND,false:_cbPlacesFilter-SEA_LAKE_POOL,false:_cbPlacesFilter-RIVER_STREAM,false:_cbPlacesFilter-FOREST_GROVE,false:_cbPlacesFilter-FARM,false:_cbPlacesFilter-CANAL,false:_cbPlacesFilter-SWAMP_MARSH,false:_cbPlacesFilter-DAM,false:_cbFilterPrivatePlaces,false:_cbInvertPlacesFilter,false[END]';
  866. defaultSettings += '[UROverviewUROptions][len=1601]:_cbURFilterOutsideArea,false:_cbNoFilterForURInURL,false:_cbFilterWazeAuto,false:_cbFilterIncorrectTurn,false:_cbFilterIncorrectAddress,false:_cbFilterIncorrectRoute,false:_cbFilterMissingRoundabout,false:_cbFilterGeneralError,false:_cbFilterTurnNotAllowed,false:_cbFilterIncorrectJunction,false:_cbFilterMissingBridgeOverpass,false:_cbFilterWrongDrivingDirection,false:_cbFilterMissingExit,false:_cbFilterMissingRoad,false:_cbFilterBlockedRoad,false:_cbFilterMissingLandmark,false:_cbFilterUndefined,false:_cbFilterRoadworks,false:_cbFilterConstruction,false:_cbFilterClosure,false:_cbFilterEvent,false:_cbFilterNote,false:_cbInvertURFilter,false:_cbFilterOpenUR,false:_cbFilterClosedUR,false:_cbFilterSolved,false:_cbFilterUnidentified,false:_cbEnableMinAgeFilter,false:_inputFilterMinDays,:_cbEnableMaxAgeFilter,false:_inputFilterMaxDays,:_cbURDescriptionMustBePresent,false:_cbURDescriptionMustBeAbsent,false:_cbEnableKeywordMustBePresent,false:_textKeywordPresent,:_cbEnableKeywordMustBeAbsent,false:_textKeywordAbsent,:_cbCaseInsensitive,false:_cbHideMyComments,false:_cbHideAnyComments,false:_cbHideIfLastCommenter,false:_cbHideIfNotLastCommenter,false:_cbHideIfReporterLastCommenter,false:_cbHideIfReporterNotLastCommenter,false:_cbEnableMinCommentsFilter,false:_inputFilterMinComments,:_cbEnableMaxCommentsFilter,false:_inputFilterMaxComments,:_cbHideMyFollowed,false:_cbHideMyUnfollowed,false:_cbEnableCommentAgeFilter2,false:_inputFilterCommentDays2,:_cbEnableCommentAgeFilter,false:_inputFilterCommentDays,:_cbURUserIDFilter,false:_cbURResolverIDFilter,false:_cbNoFilterForTaggedURs,false[END]';
  867. defaultSettings += '[UROverviewCameraOptions][len=798]:_cbShowWorldCams,true:_cbShowUSACams,true:_cbShowNonWorldCams,true:_cbShowOnlyMyCams,false:_cbShowApprovedCams,true:_cbShowNonApprovedCams,true:_cbShowOlderCreatedNonApproved,false:_inputCameraMinCreatedDays,:_cbShowOlderUpdatedNonApproved,false:_inputCameraMinUpdatedDays,:_cbShowSpeedCams,true:_cbShowIfSpeedSet,true:_cbShowIfNoSpeedSet,true:_cbShowRedLightCams,true:_cbShowDummyCams,true:_cbHideCreatedByMe,false:_cbHideCreatedByRank0,false:_cbHideCreatedByRank1,false:_cbHideCreatedByRank2,false:_cbHideCreatedByRank3,false:_cbHideCreatedByRank4,false:_cbHideCreatedByRank5,false:_cbHideUpdatedByMe,false:_cbHideUpdatedByRank0,false:_cbHideUpdatedByRank1,false:_cbHideUpdatedByRank2,false:_cbHideUpdatedByRank3,false:_cbHideUpdatedByRank4,false:_cbHideUpdatedByRank5,false:_cbHideCWLCams,false[END]';
  868. defaultSettings += '[UROverviewMasterEnable][len=4]true[END]';
  869. defaultSettings += '[UROverviewPlaceWatchList][len=0][END]';
  870. defaultSettings += '[UROverviewCurrentVersion][len=0][END]';
  871. defaultSettings += '[UROverviewCWLGroups][len=16]0,No group,false[END]';
  872. document.getElementById('_txtSettings').value = defaultSettings;
  873. uroTextToSettings();
  874. document.getElementById('_txtSettings').value = '';
  875. }
  876. }
  877.  
  878.  
  879. function uroSettingsToText()
  880. {
  881. var txtSettings = '';
  882.  
  883. uroSaveSettings();
  884.  
  885. for(var lsEntry in localStorage)
  886. {
  887. if(lsEntry.indexOf('UROverview') === 0)
  888. {
  889. txtSettings += '['+lsEntry+'][len=' + localStorage[lsEntry].length + ']' + localStorage[lsEntry] + '[END]\n';
  890. }
  891. }
  892.  
  893. document.getElementById('_txtSettings').value = txtSettings;
  894. document.getElementById('_txtSettings').focus();
  895. document.getElementById('_txtSettings').select();
  896. }
  897.  
  898. function uroTextToSettings()
  899. {
  900. var txtSettings = '';
  901. txtSettings = uroGetElmValue('_txtSettings');
  902. if(txtSettings.indexOf('[END]') == -1) return;
  903.  
  904. var subText = txtSettings.split('[END]');
  905. for(var i=0;i<subText.length;i++)
  906. {
  907. var bPos = subText[i].indexOf(']');
  908. if(bPos != -1)
  909. {
  910. var settingID = subText[i].substr(1,bPos-1);
  911. subText[i] = subText[i].substr(bPos+1);
  912. bPos = subText[i].indexOf(']');
  913. if(bPos != -1)
  914. {
  915. var settingLength = subText[i].substr(5,bPos-5);
  916. subText[i] = subText[i].substr(bPos+1);
  917. if(subText[i].length == settingLength)
  918. {
  919. localStorage[settingID] = subText[i];
  920. }
  921. }
  922. }
  923. }
  924. uroLoadSettings();
  925. }
  926.  
  927. function uroClearSettingsText()
  928. {
  929. document.getElementById('_txtSettings').value = '';
  930. }
  931.  
  932.  
  933. function uroDateToDays(dateToConvert)
  934. {
  935. var dateNow = new Date();
  936.  
  937. var elapsedSinceEpoch = dateNow.getTime();
  938. var elapsedSinceEvent = elapsedSinceEpoch - dateToConvert;
  939.  
  940. dateNow.setHours(0);
  941. dateNow.setMinutes(0);
  942. dateNow.setSeconds(0);
  943. dateNow.setMilliseconds(0);
  944.  
  945. var elapsedSinceMidnight = elapsedSinceEpoch - dateNow.getTime();
  946.  
  947. if(elapsedSinceEvent < elapsedSinceMidnight)
  948. {
  949. // event occurred today...
  950. return 0;
  951. }
  952. else
  953. {
  954. // event occurred at some point prior to midnight this morning, so return a minimum value of 1...
  955. return 1 + Math.floor((elapsedSinceEvent - elapsedSinceMidnight) / 86400000);
  956. }
  957. }
  958.  
  959. function uroGetURAge(urObj,ageType,getRaw)
  960. {
  961. if(ageType === 0)
  962. {
  963. if((urObj.attributes.driveDate === null)||(urObj.attributes.driveDate === 0)) return -1;
  964. if(getRaw) return urObj.attributes.driveDate;
  965. else return uroDateToDays(urObj.attributes.driveDate);
  966. }
  967. else if(ageType === 1)
  968. {
  969. if((urObj.attributes.resolvedOn === null)||(urObj.attributes.resolvedOn === 0)) return -1;
  970. if(getRaw) return urObj.attributes.resolvedOn;
  971. else return uroDateToDays(urObj.attributes.resolvedOn);
  972. }
  973. else
  974. {
  975. return -1;
  976. }
  977. }
  978.  
  979. function uroGetPURAge(purObj)
  980. {
  981. if(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded !== null)
  982. {
  983. return uroDateToDays(purObj.attributes.venueUpdateRequests[0].attributes.dateAdded);
  984. }
  985. else
  986. {
  987. return -1;
  988. }
  989. }
  990.  
  991. function uroGetCameraAge(camObj, mode)
  992. {
  993. if(mode === 0)
  994. {
  995. if(camObj.attributes.updatedOn === null) return -1;
  996. return uroDateToDays(camObj.attributes.updatedOn);
  997. }
  998. if(mode === 1)
  999. {
  1000. if(camObj.attributes.createdOn === null) return -1;
  1001. return uroDateToDays(camObj.attributes.createdOn);
  1002. }
  1003. }
  1004.  
  1005. function uroGetCommentAge(commentObj)
  1006. {
  1007. if(commentObj.createdOn === null) return -1;
  1008. return uroDateToDays(commentObj.createdOn);
  1009. }
  1010.  
  1011. function uroParseDaysAgo(days)
  1012. {
  1013. if(days === 0) return 'today';
  1014. else if(days === 1) return '1 day ago';
  1015. else return days+' days ago';
  1016. }
  1017.  
  1018. function uroGetCameraSpeedString(camSpeed)
  1019. {
  1020. if(camSpeed !== null)
  1021. {
  1022. var conversionFactor = 1; // default to metric
  1023. var multipleFactor = 10; // default to limits being set in multiples of 10
  1024.  
  1025. var country;
  1026. if(W.model.countries.top === undefined)
  1027. {
  1028. country = W.model.countries.additionalInfo[0].name;
  1029. }
  1030. else
  1031. {
  1032. country = W.model.countries.top.name;
  1033. }
  1034. if(country !== null)
  1035. {
  1036. // country-specific deviations from the above...
  1037. if
  1038. (
  1039. (country == "United Kingdom") ||
  1040. (country == "Jersey") ||
  1041. (country == "Guernsey") ||
  1042. (country == "United States")
  1043. )
  1044. {
  1045. // countries using MPH
  1046. conversionFactor = 1.609;
  1047. }
  1048. if
  1049. (
  1050. (country == "United States") ||
  1051. (country == "Guernsey")
  1052. )
  1053. {
  1054. // countries with speed limits set in multiples of 5
  1055. multipleFactor = 5;
  1056. }
  1057. }
  1058.  
  1059. var speed = Math.round(camSpeed / conversionFactor);
  1060. var retval = speed;
  1061. if(conversionFactor == 1) retval += "KM/H";
  1062. else retval += "MPH";
  1063. if(speed % multipleFactor !== 0) retval += " (not valid?)";
  1064. return retval;
  1065. }
  1066. else return "not set";
  1067. }
  1068.  
  1069.  
  1070. // --------------------------------------------------------------------------------------------------------------------
  1071. // AREA FRIENDLYNAME STUFF
  1072. // --------------------------------------------------------------------------------------------------------------------
  1073. function uroAFNObj(fName, area, server)
  1074. {
  1075. this.fName = fName;
  1076. this.area = area;
  1077. this.server = server;
  1078. }
  1079.  
  1080. function uroUpdateAreaName(name, server, area)
  1081. {
  1082. var foundExisting = false;
  1083. for(var i=0; i<uroFriendlyAreaNames.length; i++)
  1084. {
  1085. if((uroFriendlyAreaNames[i].server == server) && (uroFriendlyAreaNames[i].area == area))
  1086. {
  1087. if(name === "")
  1088. {
  1089. uroFriendlyAreaNames.splice(i,1);
  1090. foundExisting = true;
  1091. }
  1092. else
  1093. {
  1094. uroFriendlyAreaNames[i].fName = name;
  1095. foundExisting = true;
  1096. }
  1097. }
  1098. }
  1099.  
  1100. if((foundExisting === false) && (name !== ""))
  1101. {
  1102. uroFriendlyAreaNames.push(new uroAFNObj(name,area,server));
  1103. }
  1104. uroReplaceAreaNames(true);
  1105. }
  1106.  
  1107. function uroAreaNameHover()
  1108. {
  1109. if((uroAreaNameHoverObj === null) || (uroAreaNameHoverObj != this))
  1110. {
  1111. uroAreaNameHoverTime = 0;
  1112. }
  1113. uroAreaNameHoverObj = this;
  1114. }
  1115.  
  1116. function uroAreaNameUnHover()
  1117. {
  1118. if(uroANEditHovered === true)
  1119. {
  1120. return false;
  1121. }
  1122. if(uroAreaNameOverlayShown)
  1123. {
  1124. uroAreaNameHoverObj.removeChild(uroANEditBox);
  1125. }
  1126. uroAreaNameHoverObj = null;
  1127. uroAreaNameHoverTime = -1;
  1128. uroAreaNameOverlayShown = false;
  1129. }
  1130.  
  1131. function uroANEditHover()
  1132. {
  1133. uroANEditHovered = true;
  1134. uroAddEventListener('uroANEditBox','mouseout',uroANEditUnHover,false);
  1135. uroAddEventListener('uroANEditBox','click',uroANEditClick,false);
  1136. }
  1137.  
  1138. function uroANEditUnHover()
  1139. {
  1140. var newName = document.getElementById('_textAreaName').value;
  1141. // sanitise name to avoid conflicts with config storage delimiters...
  1142. newName = newName.replace(',','');
  1143. newName = newName.replace(':','');
  1144. var server = W.location.code;
  1145. var area = uroGetAreaArea(uroAreaNameHoverObj.parentNode.children[1]);
  1146. uroAreaNameHoverObj.removeChild(uroANEditBox);
  1147. uroAreaNameOverlayShown = false;
  1148. uroANEditHovered = false;
  1149. uroUpdateAreaName(newName, server, area);
  1150. }
  1151.  
  1152. function uroANEditClick(e)
  1153. {
  1154. // this traps the click to prevent it falling through to the underlying area name element and
  1155. // potentially causing the map view to be relocated to that area...
  1156. e.stopPropagation();
  1157. }
  1158.  
  1159. function uroGetAreaArea(listObj)
  1160. {
  1161. var area = listObj.getElementsByTagName('span')[0].innerHTML;
  1162. area = parseFloat(area.split(' ')[0]);
  1163. return area;
  1164. }
  1165.  
  1166. function uroAreaNameOverlaySetup()
  1167. {
  1168. uroAreaNameOverlayShown = true;
  1169.  
  1170. uroANEditBox = document.createElement('div');
  1171. uroANEditBox.id = "uroANEditBox";
  1172. uroANEditBox.style.position = "absolute";
  1173. uroANEditBox.style.top = '7px';
  1174. uroANEditBox.style.left = '2px';
  1175. uroANEditBox.style.width = "99%";
  1176. uroAreaNameHoverObj.appendChild(uroANEditBox);
  1177. uroANEditBox.onmouseover = uroANEditHover();
  1178. var existingName = uroAreaNameHoverObj.innerHTML;
  1179. var italicTagPos = existingName.indexOf(' <i>');
  1180. if(italicTagPos == -1)
  1181. {
  1182. existingName = "";
  1183. }
  1184. else
  1185. {
  1186. existingName = existingName.substr(0,italicTagPos);
  1187. }
  1188. uroANEditBox.innerHTML = '<input type="text" style="font-size:14px; line-height:16px; height:22px; width:100%" id="_textAreaName" value="'+existingName+'">';
  1189. }
  1190.  
  1191. function uroReplaceAreaNames(replaceAfterNameChange)
  1192. {
  1193. if(document.getElementById('sidepanel-areas') === undefined)
  1194. {
  1195. return;
  1196. }
  1197.  
  1198. if(replaceAfterNameChange === false)
  1199. {
  1200. if(document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id == "friendlyNamed")
  1201. {
  1202. return;
  1203. }
  1204. }
  1205.  
  1206. var panelRootObj = document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0];
  1207. var areaCount = panelRootObj.children.length;
  1208. if(areaCount === 0)
  1209. {
  1210. return;
  1211. }
  1212.  
  1213. var localisedManagedArea = I18n.lookup("user.areas.managed_area");
  1214. for(var loop=0; loop < areaCount; loop++)
  1215. {
  1216. var childObjPElems = panelRootObj.children[loop].getElementsByTagName('p');
  1217. var title = childObjPElems[0].innerHTML;
  1218. if(title.indexOf(localisedManagedArea) > -1)
  1219. {
  1220. var area = uroGetAreaArea(childObjPElems[1]);
  1221. childObjPElems[0].innerHTML = localisedManagedArea;
  1222.  
  1223. for(var fnIdx=0; fnIdx < uroFriendlyAreaNames.length; fnIdx++)
  1224. {
  1225. var fnObj = uroFriendlyAreaNames[fnIdx];
  1226. if((fnObj.area == area) && (fnObj.server == W.location.code))
  1227. {
  1228. childObjPElems[0].innerHTML = fnObj.fName +' <i>('+localisedManagedArea+')</i>';
  1229. break;
  1230. }
  1231. }
  1232. var titleObj = panelRootObj.getElementsByClassName('title')[loop];
  1233. titleObj.addEventListener("mouseover", uroAreaNameHover, false);
  1234. titleObj.addEventListener("mouseout", uroAreaNameUnHover, false);
  1235. titleObj.style.cursor = "text";
  1236. }
  1237. }
  1238. document.getElementById('sidepanel-areas').getElementsByClassName('result-list')[0].id = "friendlyNamed";
  1239. }
  1240.  
  1241. // --------------------------------------------------------------------------------------------------------------------
  1242. // WATCHLIST STUFF
  1243. // --------------------------------------------------------------------------------------------------------------------
  1244.  
  1245. // Generic Functions
  1246. function uroTypeCast(varin)
  1247. {
  1248. if(varin == "null") return null;
  1249. if(typeof varin == "string") return parseInt(varin);
  1250. return varin;
  1251. }
  1252. function uroTruncate(val)
  1253. {
  1254. if(val === null) return val;
  1255. if(val < 0) return Math.ceil(val);
  1256. return Math.floor(val);
  1257. }
  1258. function uroOWLGroupObj(groupID, groupName, groupCollapsed)
  1259. {
  1260. groupID = uroTypeCast(groupID);
  1261. this.groupID = groupID;
  1262. this.groupName = groupName;
  1263. this.groupCount = 0;
  1264. this.groupCollapsed = groupCollapsed;
  1265. }
  1266.  
  1267. // Camera Functions
  1268. function uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon)
  1269. {
  1270. if(type !== null) type = uroTypeCast(type);
  1271. if(azymuth !== null) azymuth = uroTruncate(uroTypeCast(azymuth)%360);
  1272. if(speed !== null) speed = uroTruncate(uroTypeCast(speed));
  1273. if(typeof validated == "string") validated = (validated == "true");
  1274. if(lat !== null) lat = uroTruncate(uroTypeCast(lat));
  1275. if(lon !== null) lon = uroTruncate(uroTypeCast(lon));
  1276.  
  1277. this.type = type;
  1278. this.azymuth = azymuth;
  1279. this.speed = speed;
  1280. this.validated = validated;
  1281. this.lat = lat;
  1282. this.lon = lon;
  1283. }
  1284. function uroCamWatchObj(persistent, fid, lon, lat, type, azymuth, speed, validated, groupID, server)
  1285. {
  1286. fid = uroTypeCast(fid);
  1287. groupID = uroTypeCast(groupID);
  1288. if(typeof persistent == "string") persistent = (persistent == "true");
  1289.  
  1290. this.fid = fid;
  1291. this.persistent = persistent;
  1292. this.loaded = false;
  1293. this.server = server;
  1294. this.groupID = groupID;
  1295. this.watch = new uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon);
  1296. this.current = new uroCamWatchObjCheckProps(null, null, null, null, null, null);
  1297. }
  1298. function uroCamDataChanged(idx)
  1299. {
  1300. var camObj = uroCamWatchObjects[idx];
  1301. if(camObj.loaded === false) return false;
  1302. if(camObj.current.type != camObj.watch.type) return true;
  1303. if(camObj.current.azymuth != camObj.watch.azymuth) return true;
  1304. if(camObj.current.speed != camObj.watch.speed) return true;
  1305. if(camObj.current.validated != camObj.watch.validated) return true;
  1306. if(camObj.current.lat != camObj.watch.lat) return true;
  1307. if(camObj.current.lon != camObj.watch.lon) return true;
  1308. return false;
  1309. }
  1310. function uroFindCWLGroupByIdx(groupIdx)
  1311. {
  1312. var groupName = '';
  1313. for(var loop=0;loop<uroCWLGroups.length;loop++)
  1314. {
  1315. if(uroCWLGroups[loop].groupID == groupIdx)
  1316. {
  1317. groupName = uroCWLGroups[loop].groupName;
  1318. break;
  1319. }
  1320. }
  1321. return groupName;
  1322. }
  1323. function uroIsCamOnWatchList(fid)
  1324. {
  1325. for(var loop=0;loop<uroCamWatchObjects.length;loop++)
  1326. {
  1327. if(uroCamWatchObjects[loop].fid == fid) return loop;
  1328. }
  1329. return -1;
  1330. }
  1331. function uroAddCurrentCamWatchData(idx, lat, lon, type, azymuth, speed, validated, server)
  1332. {
  1333. var camObj = uroCamWatchObjects[idx];
  1334. camObj.loaded = true;
  1335. camObj.server = server;
  1336. camObj.current = new uroCamWatchObjCheckProps(type, azymuth, speed, validated, lat, lon);
  1337. return(uroCamDataChanged(idx));
  1338. }
  1339. function uroAddCamToWatchList()
  1340. {
  1341. if(uroIsCamOnWatchList(uroShownFID) == -1)
  1342. {
  1343. var camObj = W.model.cameras.objects[uroShownFID];
  1344. 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));
  1345. 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);
  1346. uroAddLog('added camera '+uroShownFID+' to watchlist');
  1347. uroOWLUpdateHTML();
  1348. }
  1349. }
  1350. function uroRemoveCamFromWatchList()
  1351. {
  1352. var camidx = uroIsCamOnWatchList(uroShownFID);
  1353. if(camidx != -1)
  1354. {
  1355. uroCamWatchObjects.splice(camidx,1);
  1356. uroAddLog('removed camera '+uroShownFID+' from watchlist');
  1357. uroOWLUpdateHTML();
  1358. }
  1359. }
  1360. function uroUpdateCamWatchList()
  1361. {
  1362. var camIdx = uroIsCamOnWatchList(uroShownFID);
  1363. if(camIdx != -1)
  1364. {
  1365. var camObj = W.model.cameras.objects[uroShownFID];
  1366. uroCamWatchObjects[camIdx].watch = new uroCamWatchObjCheckProps(camObj.attributes.type, camObj.attributes.azymuth, camObj.attributes.speed, camObj.attributes.validated, camObj.geometry.y, camObj.geometry.x);
  1367. }
  1368. }
  1369. function uroClearCamWatchList()
  1370. {
  1371. if(confirm('Removing all cameras from the OWL cannot be undone\nAre you sure you want to do this?') === true)
  1372. {
  1373. uroCamWatchObjects = [];
  1374. uroOWLUpdateHTML();
  1375. }
  1376. }
  1377. function uroRetrieveCameras(lat, lon)
  1378. {
  1379. var camPos = new OpenLayers.LonLat();
  1380. var camChanged = false;
  1381.  
  1382. camPos.lon = lon;
  1383. camPos.lat = lat;
  1384. camPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  1385.  
  1386. var camURL = 'https://' + document.location.host;
  1387. camURL += Waze.Config.api_base;
  1388. camURL += '/Features?language=en&cameras=true&bbox=';
  1389. var latl = camPos.lat - 0.25;
  1390. var latu = camPos.lat + 0.25;
  1391. var lonl = camPos.lon - 0.25;
  1392. var lonr = camPos.lon + 0.25;
  1393. camURL += lonl+','+latl+','+lonr+','+latu;
  1394. uroAddLog('retrieving camera data around '+camPos.lon+','+camPos.lat);
  1395.  
  1396. var camReq = new XMLHttpRequest();
  1397. camReq.open('GET',camURL,false);
  1398. try
  1399. {
  1400. camReq.send();
  1401. uroAddLog('response '+camReq.status+' received');
  1402. if (camReq.status === 200)
  1403. {
  1404. var camData = JSON.parse(camReq.responseText);
  1405. for(var camIdx = 0; camIdx < camData.cameras.objects.length; camIdx++)
  1406. {
  1407. var camObj = camData.cameras.objects[camIdx];
  1408. var listIdx = uroIsCamOnWatchList(camObj.id);
  1409. if(listIdx != -1)
  1410. {
  1411. camPos.lon = camObj.geometry.coordinates[0];
  1412. camPos.lat = camObj.geometry.coordinates[1];
  1413. camPos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
  1414. camPos.lon = uroTruncate(camPos.lon);
  1415. camPos.lat = uroTruncate(camPos.lat);
  1416. camChanged |= uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, camObj.validated, W.location.code);
  1417. }
  1418. else if(camObj.validated === false)
  1419. {
  1420.  
  1421. }
  1422. }
  1423. }
  1424. else
  1425. {
  1426. uroAddLog('request failed (status != 200)');
  1427. }
  1428. }
  1429. catch(err)
  1430. {
  1431. uroAddLog('camera load request failed (exception '+err+' caught)');
  1432. }
  1433. return camChanged;
  1434. }
  1435. function uroGetCurrentCamWatchListObjects()
  1436. {
  1437. var camChanged = false;
  1438. var camsChanged = [];
  1439. var camsDeleted = [];
  1440. var camidx;
  1441. var camObj;
  1442. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  1443. {
  1444. camObj = uroCamWatchObjects[camidx];
  1445. if((camObj.loaded === false) && ((camObj.server == W.location.code) || (camObj.server == '??')))
  1446. {
  1447. if(typeof W.model.cameras.objects[camObj.fid] == 'object')
  1448. {
  1449. if(W.model.cameras.objects[camObj.fid].state != "Delete")
  1450. {
  1451. var wazeObj = W.model.cameras.objects[camObj.fid];
  1452. camChanged |= uroAddCurrentCamWatchData(camidx, wazeObj.geometry.y, wazeObj.geometry.x, wazeObj.attributes.type, wazeObj.attributes.azymuth, wazeObj.attributes.speed, wazeObj.attributes.validated);
  1453. }
  1454. else
  1455. {
  1456. camChanged |= uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon);
  1457. }
  1458. }
  1459. else
  1460. {
  1461. camChanged |= uroRetrieveCameras(camObj.watch.lat, camObj.watch.lon);
  1462. }
  1463. }
  1464. }
  1465.  
  1466. if(camChanged)
  1467. {
  1468. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  1469. {
  1470. if(uroCamDataChanged(camidx))
  1471. {
  1472. camsChanged.push(uroCamWatchObjects[camidx]);
  1473. }
  1474. }
  1475. }
  1476.  
  1477. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  1478. {
  1479. camObj = uroCamWatchObjects[camidx];
  1480. if((camObj.loaded === false) && (camObj.server == W.location.code))
  1481. {
  1482. camsDeleted.push(camObj);
  1483. }
  1484. }
  1485.  
  1486. if((camsChanged.length > 0) || (camsDeleted.length > 0))
  1487. {
  1488. var alertStr = 'Camera WatchList Alert!!!\r\n';
  1489. for(camidx=0;camidx<camsChanged.length;camidx++)
  1490. {
  1491. alertStr += 'Camera ID '+camsChanged[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsChanged[camidx].groupID)+'" has been changed\r\n';
  1492. }
  1493. for(camidx=0;camidx<camsDeleted.length;camidx++)
  1494. {
  1495. alertStr += 'Camera ID '+camsDeleted[camidx].fid+' in group "'+uroFindCWLGroupByIdx(camsDeleted[camidx].groupID)+'" has been deleted\r\n';
  1496. }
  1497. alert(alertStr);
  1498. }
  1499. }
  1500. function uroClearDeletedCameras()
  1501. {
  1502. for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
  1503. {
  1504. if(uroCamWatchObjects[camidx].loaded === false)
  1505. {
  1506. uroShownFID = uroCamWatchObjects[camidx].fid;
  1507. uroRemoveCamFromWatchList();
  1508. }
  1509. }
  1510. }
  1511. function uroClearUnknownServerCameras()
  1512. {
  1513. var confirmMsg = 'Cameras with an unknown server cannot be automatically verified by URO+.\n';
  1514. confirmMsg += 'It is recommended that you manually load WME from each server (World, USA/Canada and Israel) to give URO+ a chance of locating these cameras.\n';
  1515. confirmMsg += 'If the cameras then continue to show up as an unknown server, it is safe to delete them...\n\n';
  1516. confirmMsg += 'Do you still wish to proceed with deleting all unknown server cameras?';
  1517.  
  1518. if(confirm(confirmMsg) === true)
  1519. {
  1520. for(var camidx=uroCamWatchObjects.length-1;camidx>=0;camidx--)
  1521. {
  1522. if(uroCamWatchObjects[camidx].server == '??')
  1523. {
  1524. uroShownFID = uroCamWatchObjects[camidx].fid;
  1525. uroRemoveCamFromWatchList();
  1526. }
  1527. }
  1528. }
  1529. }
  1530. function uroRescanCamWatchList()
  1531. {
  1532. for(var camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  1533. {
  1534. uroCamWatchObjects[camidx].loaded = false;
  1535. }
  1536. uroGetCurrentCamWatchListObjects();
  1537. uroOWLUpdateHTML();
  1538. }
  1539. function uroGotoCam()
  1540. {
  1541. var camidx = this.id.substr(13);
  1542. var camPos = new OpenLayers.LonLat();
  1543. camPos.lon = uroCamWatchObjects[camidx].watch.lon;
  1544. camPos.lat = uroCamWatchObjects[camidx].watch.lat;
  1545. W.map.setCenter(camPos,4);
  1546. W.map.camerasLayer.setVisibility(true);
  1547. return false;
  1548. }
  1549.  
  1550. // Segment Functions
  1551. function uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn)
  1552. {
  1553. if(left !== null) left = uroTruncate(uroTypeCast(left));
  1554. if(right !== null) right = uroTruncate(uroTypeCast(right));
  1555. if(bottom !== null) bottom = uroTruncate(uroTypeCast(bottom));
  1556. if(top !== null) top = uroTruncate(uroTypeCast(top));
  1557. if(fromNode !== null) fromNode = uroTypeCast(fromNode);
  1558. if(toNode !== null) toNode = uroTypeCast(toNode);
  1559. if(fwdDir !== null) fwdDir = uroTypeCast(fwdDir);
  1560. if(revDir !== null) revDir = uroTypeCast(revDir);
  1561. if(length !== null) length = uroTypeCast(length);
  1562. if(level !== null) level = uroTypeCast(level);
  1563. if(rank !== null) rank = uroTypeCast(rank);
  1564. if(roadType !== null) roadType = uroTypeCast(roadType);
  1565. if(updatedOn !== null) updatedOn = uroTypeCast(updatedOn);
  1566.  
  1567. this.left = left;
  1568. this.right = right;
  1569. this.bottom = bottom;
  1570. this.top = top;
  1571. this.fromNode = fromNode;
  1572. this.toNode = toNode;
  1573. this.fwdDir = fwdDir;
  1574. this.revDir = revDir;
  1575. this.length = length;
  1576. this.level = level;
  1577. this.rank = rank;
  1578. this.roadType = roadType;
  1579. this.updatedOn = updatedOn;
  1580. }
  1581. function uroSegWatchObj(persistent, fid, left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn, groupID, server)
  1582. {
  1583. fid = uroTypeCast(fid);
  1584. groupID = uroTypeCast(groupID);
  1585. if(typeof persistent == "string") persistent = (persistent == "true");
  1586.  
  1587. this.fid = fid;
  1588. this.persistent = persistent;
  1589. this.loaded = false;
  1590. this.server = server;
  1591. this.groupID = groupID;
  1592.  
  1593. this.watch = new uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn);
  1594. this.current = new uroSegWatchObjCheckProps(null, null, null, null, null, null, null, null, null, null, null, null, null);
  1595. }
  1596. function uroSegDataChanged(idx)
  1597. {
  1598. var segObj = uroSegWatchObjects[idx];
  1599. if(segObj.loaded === false) return false;
  1600. if(segObj.current.left != segObj.watch.left) return true;
  1601. if(segObj.current.right != segObj.watch.right) return true;
  1602. if(segObj.current.bottom != segObj.watch.bottom) return true;
  1603. if(segObj.current.top != segObj.watch.top) return true;
  1604. if(segObj.current.fromNode != segObj.watch.fromNode) return true;
  1605. if(segObj.current.toNode != segObj.watch.toNode) return true;
  1606. if(segObj.current.fwdDir != segObj.watch.fwdDir) return true;
  1607. if(segObj.current.revDir != segObj.watch.revDir) return true;
  1608. if(segObj.current.length != segObj.watch.length) return true;
  1609. if(segObj.current.level != segObj.watch.level) return true;
  1610. if(segObj.current.rank != segObj.watch.rank) return true;
  1611. if(segObj.current.roadType != segObj.watch.roadType) return true;
  1612. if(segObj.current.updatedOn != segObj.watch.updatedOn) return true;
  1613. return false;
  1614. }
  1615. function uroIsSegOnWatchList(fid)
  1616. {
  1617. for(var loop=0;loop<uroSegWatchObjects.length;loop++)
  1618. {
  1619. if(uroSegWatchObjects[loop].fid == fid) return loop;
  1620. }
  1621. return -1;
  1622. }
  1623. function uroAddCurrentSegWatchData(idx, left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn, server)
  1624. {
  1625. var segObj = uroSegWatchObjects[idx];
  1626. segObj.loaded = true;
  1627. segObj.server = server;
  1628. segObj.current = new uroSegWatchObjCheckProps(left, right, bottom, top, fromNode, toNode, fwdDir, revDir, length, level, rank, roadType, updatedOn);
  1629. return(uroSegDataChanged(idx));
  1630. }
  1631. function uroClearSegWatchList()
  1632. {
  1633. if(confirm('Removing all segments from the OWL cannot be undone\nAre you sure you want to do this?') === true)
  1634. {
  1635. uroSegWatchObjects = [];
  1636. uroOWLUpdateHTML();
  1637. }
  1638. }
  1639. function uroAddUpdateSegWatchList()
  1640. {
  1641. var selectedCount = W.selectionManager.selectedItems.length;
  1642. if(selectedCount === 0)
  1643. {
  1644. return;
  1645. }
  1646.  
  1647. for(var loop=0;loop < selectedCount; loop++)
  1648. {
  1649. var segObj = W.selectionManager.selectedItems[loop].model.attributes;
  1650. var fid = segObj.id;
  1651. var idx = uroIsSegOnWatchList(fid);
  1652. if(idx != -1)
  1653. {
  1654. 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);
  1655. uroAddLog('updated watchlist details for segment '+fid);
  1656. }
  1657. else
  1658. {
  1659. 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));
  1660. 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);
  1661. uroAddLog('added segment '+fid+' to watchlist');
  1662. }
  1663. }
  1664. //uroOWLUpdateHTML();
  1665. }
  1666. function uroRemoveSegFromWatchList()
  1667. {
  1668. var selectedCount = W.selectionManager.selectedItems.length;
  1669. if(selectedCount === 0)
  1670. {
  1671. return;
  1672. }
  1673.  
  1674. for(var loop=0;loop < selectedCount; loop++)
  1675. {
  1676. var fid = W.selectionManager.selectedItems[loop].model.attributes.id;
  1677. var idx = uroIsSegOnWatchList(fid);
  1678. if(idx != -1)
  1679. {
  1680. uroSegWatchObjects.splice(idx,1);
  1681. uroAddLog('removed segment '+fid+' from watchlist');
  1682. }
  1683. }
  1684. //uroOWLUpdateHTML();
  1685. }
  1686. function uroRetrieveSegments(lat, lon)
  1687. {
  1688. var pos = new OpenLayers.LonLat();
  1689. var changed = false;
  1690.  
  1691. pos.lon = lon;
  1692. pos.lat = lat;
  1693. pos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  1694.  
  1695. var URL = 'https://' + document.location.host;
  1696. URL += Waze.Config.api_base;
  1697. URL += '/Features?roadTypes=1%2C2%2C3%2C4%2C5%2C6%2C7%2C8%2C9%2C10%2C11%2C12%2C13%2C14%2C15%2C16%2C17%2C18%2C19%2C20%2C21';
  1698. URL += '&bbox=';
  1699. var latl = pos.lat - 0.25;
  1700. var latu = pos.lat + 0.25;
  1701. var lonl = pos.lon - 0.25;
  1702. var lonr = pos.lon + 0.25;
  1703. URL += lonl+','+latl+','+lonr+','+latu;
  1704. URL += '&language=en';
  1705. uroAddLog('retrieving segment data around '+pos.lon+','+pos.lat);
  1706.  
  1707. var req = new XMLHttpRequest();
  1708. req.open('GET',URL,false);
  1709. try
  1710. {
  1711. req.send();
  1712. uroAddLog('response '+req.status+' received');
  1713. if (req.status === 200)
  1714. {
  1715. var data = JSON.parse(req.responseText);
  1716. for(var idx = 0; idx < data.segments.objects.length; idx++)
  1717. {
  1718. var obj = data.segments.objects[idx];
  1719. var listIdx = uroIsSegOnWatchList(obj.id);
  1720. if(listIdx != -1)
  1721. {
  1722. //pos.lon = obj.geometry.coordinates[0];
  1723. //pos.lat = obj.geometry.coordinates[1];
  1724. //pos.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
  1725. //camPos.lon = uroTruncate(camPos.lon);
  1726. //camPos.lat = uroTruncate(camPos.lat);
  1727. //camChanged |= uroAddCurrentCamWatchData(listIdx, camPos.lat, camPos.lon, camObj.type, camObj.azymuth, camObj.speed, camObj.validated, W.location.code);
  1728. }
  1729. else if(obj.validated === false)
  1730. {
  1731.  
  1732. }
  1733. }
  1734. }
  1735. else
  1736. {
  1737. uroAddLog('request failed (status != 200)');
  1738. }
  1739. }
  1740. catch(err)
  1741. {
  1742. uroAddLog('segment load request failed (exception '+err+' caught)');
  1743. }
  1744. return changed;
  1745. }
  1746. function uroGetCurrentSegWatchListObjects()
  1747. {
  1748. var segChanged = false;
  1749. var segsChanged = [];
  1750. var segsDeleted = [];
  1751. var idx;
  1752. var segObj;
  1753.  
  1754. for(idx=0;idx<uroSegWatchObjects.length;idx++)
  1755. {
  1756. segObj = uroSegWatchObjects[idx];
  1757. if((segObj.loaded === false) && ((segObj.server == W.location.code) || (segObj.server == '??')))
  1758. {
  1759. var segLat = (segObj.watch.top + segObj.watch.bottom) / 2;
  1760. var segLon = (segObj.watch.right + segObj.watch.left) / 2;
  1761. if(typeof W.model.segments.objects[segObj.fid] == 'object')
  1762. {
  1763. if(W.model.segments.objects[segObj.fid].state != "Delete")
  1764. {
  1765. var wazeObj = W.model.segments.objects[segObj.fid];
  1766. 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);
  1767. }
  1768. else
  1769. {
  1770. segChanged |= uroRetrieveSegments(segLat, segLon);
  1771. }
  1772. }
  1773. else
  1774. {
  1775. segChanged |= uroRetrieveSegments(segLat, segLon);
  1776. }
  1777. }
  1778. }
  1779.  
  1780. if(segChanged)
  1781. {
  1782. for(idx=0;idx<uroSegWatchObjects.length;idx++)
  1783. {
  1784. if(uroSegDataChanged(idx))
  1785. {
  1786. segsChanged.push(uroSegWatchObjects[idx]);
  1787. }
  1788. }
  1789. }
  1790.  
  1791. for(idx=0;idx<uroSegWatchObjects.length;idx++)
  1792. {
  1793. segObj = uroSegWatchObjects[idx];
  1794. if((segObj.loaded === false) && (segObj.server == W.location.code))
  1795. {
  1796. segsDeleted.push(segObj);
  1797. }
  1798. }
  1799.  
  1800. if((segsChanged.length > 0) || (segsDeleted.length > 0))
  1801. {
  1802. var alertStr = 'Segment WatchList Alert!!!\r\n';
  1803. for(idx=0;idx<segsChanged.length;idx++)
  1804. {
  1805. alertStr += 'Segment ID '+segsChanged[idx].fid+' in group "'+uroFindCWLGroupByIdx(segsChanged[idx].groupID)+'" has been changed\r\n';
  1806. }
  1807. for(idx=0;idx<segsDeleted.length;idx++)
  1808. {
  1809. alertStr += 'Segment ID '+segsDeleted[idx].fid+' in group "'+uroFindCWLGroupByIdx(segsDeleted[idx].groupID)+'" has been deleted\r\n';
  1810. }
  1811. alert(alertStr);
  1812. }
  1813. }
  1814.  
  1815. // Places Functions
  1816. function uroPlaceWatchObjCheckProps(left, right, bottom, top, name, imageCount, residential, updatedOn)
  1817. {
  1818. if(left !== null) left = uroTruncate(uroTypeCast(left));
  1819. if(right !== null) right = uroTruncate(uroTypeCast(right));
  1820. if(bottom !== null) bottom = uroTruncate(uroTypeCast(bottom));
  1821. if(top !== null) top = uroTruncate(uroTypeCast(top));
  1822. if(imageCount !== null) imageCount = uroTypeCast(imageCount);
  1823. if(typeof residential == "string") residential = (residential == "true");
  1824. if(updatedOn !== null) updatedOn = uroTypeCast(updatedOn);
  1825.  
  1826. this.left = left;
  1827. this.right = right;
  1828. this.bottom = bottom;
  1829. this.top = top;
  1830. this.name = name;
  1831. this.imageCount = imageCount;
  1832. this.residential = residential;
  1833. this.updatedOn = updatedOn;
  1834. }
  1835. function uroPlaceWatchObj(persistent, fid, left, right, bottom, top, imageCount, name, residential, updatedOn, groupID, server)
  1836. {
  1837. groupID = uroTypeCast(groupID);
  1838. if(typeof persistent == "string") persistent = (persistent == "true");
  1839.  
  1840. this.fid = fid;
  1841. this.persistent = persistent;
  1842. this.loaded = false;
  1843. this.server = server;
  1844. this.groupID = groupID;
  1845. this.watch = new uroPlaceWatchObjCheckProps(left, right, bottom, top, name, imageCount, residential, updatedOn);
  1846. this.current = new uroPlaceWatchObjCheckProps(null, null, null, null, null, null, null, null);
  1847. }
  1848. function uroPlaceDataChanged(idx)
  1849. {
  1850. var placeObj = uroPlaceWatchObjects[idx];
  1851. if(placeObj.loaded === false) return false;
  1852. if(placeObj.current.left != placeObj.watch.left) return true;
  1853. if(placeObj.current.right != placeObj.watch.right) return true;
  1854. if(placeObj.current.bottom != placeObj.watch.bottom) return true;
  1855. if(placeObj.current.top != placeObj.watch.top) return true;
  1856. if(placeObj.current.name != placeObj.watch.name) return true;
  1857. if(placeObj.current.imageCount != placeObj.watch.imageCount) return true;
  1858. if(placeObj.current.residential != placeObj.watch.residential) return true;
  1859. if(placeObj.current.updatedOn != placeObj.watch.updatedOn) return true;
  1860. return false;
  1861. }
  1862. function uroIsPlaceOnWatchList(fid)
  1863. {
  1864. for(var loop=0;loop<uroPlaceWatchObjects.length;loop++)
  1865. {
  1866. if(uroPlaceWatchObjects[loop].fid == fid) return loop;
  1867. }
  1868. return -1;
  1869. }
  1870. function uroClearPlaceWatchList()
  1871. {
  1872. if(confirm('Removing all places from the OWL cannot be undone\nAre you sure you want to do this?') === true)
  1873. {
  1874. uroPlaceWatchObjects = [];
  1875. uroOWLUpdateHTML();
  1876. }
  1877. }
  1878.  
  1879.  
  1880. function uroHighlightCWLEntry()
  1881. {
  1882. this.style.backgroundColor = '#FFFFAA';
  1883. return false;
  1884. }
  1885. function uroUnhighlightCWLEntry()
  1886. {
  1887. var camidx = this.id.substr(8);
  1888. var changed = uroCamDataChanged(camidx);
  1889. var deleted = (uroCamWatchObjects[camidx].loaded === false);
  1890.  
  1891. if(uroCamWatchObjects[camidx].server != W.location.code)
  1892. {
  1893. if(uroCamWatchObjects[camidx].server == '??') this.style.backgroundColor = '#A0A0A0';
  1894. else this.style.backgroundColor = '#AAFFAA';
  1895. }
  1896. else if(changed) this.style.backgroundColor = '#AAAAFF';
  1897. else if(deleted) this.style.backgroundColor = '#FFAAAA';
  1898. else this.style.backgroundColor = '#FFFFFF';
  1899. return false;
  1900. }
  1901. function uroCWLIconHighlight()
  1902. {
  1903. var iconType = this.id.substr(11,1);
  1904. this.src = uroIcons[iconType][0];
  1905. return false;
  1906. }
  1907. function uroCWLIconLowlight()
  1908. {
  1909. var iconType = this.id.substr(11,1);
  1910. this.src = uroIcons[iconType][1];
  1911. return false;
  1912. }
  1913. function uroPopulateCWLGroupSelect()
  1914. {
  1915. var selector = document.getElementById('_uroCWLGroupSelect');
  1916. while(selector.options.length > 0)
  1917. {
  1918. selector.options.remove(0);
  1919. }
  1920. for(var loop=0;loop<uroCWLGroups.length;loop++)
  1921. {
  1922. var groupObj = uroCWLGroups[loop];
  1923. if(groupObj.groupID != -1)
  1924. {
  1925. selector.options.add(new Option(groupObj.groupName,groupObj.groupID));
  1926. }
  1927. }
  1928. }
  1929. function uroGetNextCWLGroupID()
  1930. {
  1931. var nextID = 1;
  1932. for(var loop=0;loop<uroCWLGroups.length;loop++)
  1933. {
  1934. if(uroCWLGroups[loop].groupID >= nextID)
  1935. {
  1936. nextID = uroCWLGroups[loop].groupID + 1;
  1937. }
  1938. }
  1939. return nextID;
  1940. }
  1941. function uroFindCWLGroupByName(groupName)
  1942. {
  1943. var groupID = -1;
  1944. for(var loop=0;loop<uroCWLGroups.length;loop++)
  1945. {
  1946. if((uroCWLGroups[loop].groupName == groupName) && (uroCWLGroups[loop].groupID != -1))
  1947. {
  1948. groupID = uroCWLGroups[loop].groupID;
  1949. break;
  1950. }
  1951. }
  1952. return groupID;
  1953. }
  1954. function uroAddCWLGroup()
  1955. {
  1956. var groupID = uroGetNextCWLGroupID();
  1957. var groupName = uroGetElmValue('_uroCWLGroupEntry');
  1958. if(uroFindCWLGroupByName(groupName) == -1)
  1959. {
  1960. uroCWLGroups.push(new uroOWLGroupObj(groupID,groupName,false));
  1961. uroPopulateCWLGroupSelect();
  1962. }
  1963. }
  1964. function uroRemoveCWLGroup()
  1965. {
  1966. var loop;
  1967. var selector = document.getElementById('_uroCWLGroupSelect');
  1968. var groupID = parseInt(selector.selectedOptions[0].value);
  1969. if(groupID === 0) return false; // prevent deletion of the default group
  1970.  
  1971. for(loop=0;loop<uroCamWatchObjects.length;loop++)
  1972. {
  1973. var cwObj = uroCamWatchObjects[loop];
  1974. if(cwObj.groupID == groupID)
  1975. {
  1976. cwObj.groupID = 0;
  1977. }
  1978. }
  1979. for(loop=0;loop<uroCWLGroups.length;loop++)
  1980. {
  1981. var groupObj = uroCWLGroups[loop];
  1982. if(groupObj.groupID == groupID)
  1983. {
  1984. groupObj.groupID = -1;
  1985. }
  1986. }
  1987. uroOWLUpdateHTML();
  1988. }
  1989. function uroAssignCameraToGroup()
  1990. {
  1991. var camidx = this.id.substr(13);
  1992. var selector = document.getElementById('_uroCWLGroupSelect');
  1993. uroCamWatchObjects[camidx].groupID = parseInt(selector.selectedOptions[0].value);
  1994. uroOWLUpdateHTML();
  1995. return false;
  1996. }
  1997. function uroAddBtnEvl(btnID, evlType, evlFunction)
  1998. {
  1999. var btnObj = document.getElementById(btnID);
  2000. if(btnObj !== null)
  2001. {
  2002. btnObj.addEventListener(evlType, evlFunction, true);
  2003. }
  2004. }
  2005. function uroCWLGroupCollapseExpand()
  2006. {
  2007. var groupidx = this.id.substr(18);
  2008. if(uroCWLGroups[groupidx].groupCollapsed === true) uroCWLGroups[groupidx].groupCollapsed = false;
  2009. else uroCWLGroups[groupidx].groupCollapsed = true;
  2010. uroOWLUpdateHTML();
  2011. return false;
  2012. }
  2013. function uroOWLUpdateHTML()
  2014. {
  2015. var camTypes = new Array("","","Speed", "Dummy", "Red Light");
  2016. var selectedGroup = 0;
  2017. var iHTML = '';
  2018. var camidx;
  2019. var groupidx;
  2020.  
  2021. if(document.getElementById('_uroCWLGroupSelect') !== null)
  2022. {
  2023. selectedGroup = document.getElementById('_uroCWLGroupSelect').selectedIndex;
  2024. }
  2025. iHTML = '<br><b>Camera Watchlist:</b><br><br>';
  2026. iHTML += '<div id="_uroCWLCamList" style="height:65%;overflow:auto;">';
  2027. if(uroCWLGroups.length > 0)
  2028. {
  2029. for(groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
  2030. {
  2031. var groupObj = uroCWLGroups[groupidx];
  2032. iHTML += '<div id="_uroCWLGroup-'+groupidx+'">';
  2033. if(groupObj.groupCollapsed === true)
  2034. {
  2035. iHTML += '<img src="'+uroIcons[0][1]+'" id="_uroCWLGroupState-'+groupidx+'">';
  2036. }
  2037. else
  2038. {
  2039. iHTML += '<img src="'+uroIcons[0][0]+'" id="_uroCWLGroupState-'+groupidx+'">';
  2040. }
  2041. iHTML += '<b>'+groupObj.groupName+'</b><br>';
  2042. groupObj.groupCount = 0;
  2043. if(uroCamWatchObjects.length > 0)
  2044. {
  2045. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  2046. {
  2047. var camObj = uroCamWatchObjects[camidx];
  2048. if(camObj.groupID == groupObj.groupID)
  2049. {
  2050. groupObj.groupCount++;
  2051. var changed = uroCamDataChanged(camidx);
  2052. var deleted = (camObj.loaded === false);
  2053. iHTML += '<div id="_uroCWL-'+camidx+'" style="padding:3px;border-width:2px;border-style:solid;border-color:#FFFFFF;background-color:';
  2054. if(camObj.server != W.location.code)
  2055. {
  2056. if(camObj.server == '??') iHTML += '#A0A0A0;';
  2057. else iHTML += '#AAFFAA;';
  2058. }
  2059. else if(changed) iHTML += '#AAAAFF;';
  2060. else if(deleted) iHTML += '#FFAAAA;';
  2061. else iHTML += '#FFFFFF;';
  2062.  
  2063. if(groupObj.groupCollapsed === true) iHTML += 'display:none;">';
  2064. else iHTML += 'display:block;">';
  2065.  
  2066. iHTML += 'ID: '+camObj.fid;
  2067. iHTML += ' ('+camObj.server+')';
  2068. iHTML += ' Type: '+camTypes[camObj.watch.type];
  2069. if(camObj.server != W.location.code)
  2070. {
  2071. if(camObj.server == '??')
  2072. {
  2073. iHTML += '<br><i>Unknown server</i>';
  2074. }
  2075. else
  2076. {
  2077. iHTML += '<br><i>Not on this server</i>';
  2078. }
  2079. }
  2080. else if(deleted)
  2081. {
  2082. iHTML += '<br>DELETED';
  2083. }
  2084. else if(changed)
  2085. {
  2086. if(camObj.current.type != camObj.watch.type)
  2087. {
  2088. iHTML += '<br>&nbsp;&nbsp;Type changed';
  2089. iHTML += ' ('+camObj.watch.type+' to '+camObj.current.type+')';
  2090. }
  2091. if(camObj.current.azymuth != camObj.watch.azymuth)
  2092. {
  2093. iHTML += '<br>&nbsp;&nbsp;Azimuth changed';
  2094. iHTML += ' ('+camObj.watch.azymuth+' to '+camObj.current.azymuth+')';
  2095. }
  2096. if(camObj.current.speed != camObj.watch.speed)
  2097. {
  2098. iHTML += '<br>&nbsp;&nbsp;Speed changed';
  2099. iHTML += ' ('+camObj.watch.speed+' to '+camObj.current.speed+')';
  2100. }
  2101. if(camObj.current.validated != camObj.watch.validated)
  2102. {
  2103. iHTML += '<br>&nbsp;&nbsp;Approval state changed';
  2104. iHTML += ' ('+camObj.watch.validated+' to '+camObj.current.validated+')';
  2105. }
  2106. if(camObj.current.lat != camObj.watch.lat)
  2107. {
  2108. iHTML += '<br>&nbsp;&nbsp;Latitude changed';
  2109. iHTML += ' ('+camObj.watch.lat+' to '+camObj.current.lat+')';
  2110. }
  2111. if(camObj.current.lon != camObj.watch.lon)
  2112. {
  2113. iHTML += '<br>&nbsp;&nbsp;Longitude changed';
  2114. iHTML += ' ('+camObj.watch.lon+' to '+camObj.current.lon+')';
  2115. }
  2116. }
  2117.  
  2118. if(camObj.server == W.location.code)
  2119. {
  2120. if(deleted === false)
  2121. {
  2122. iHTML += '&nbsp;<img id="_uroCWLIcon1-'+camidx+'" src="'+uroIcons[1][1]+'">';
  2123. }
  2124. iHTML += '&nbsp;<img id="_uroCWLIcon2-'+camidx+'" src="'+uroIcons[2][1]+'">';
  2125. }
  2126. iHTML += '</div>';
  2127. }
  2128. }
  2129. }
  2130. iHTML += '</div>';
  2131. }
  2132. }
  2133. iHTML += '</div><div id="_uroCWLControls">';
  2134. iHTML += '<hr>Group control:<br>';
  2135. iHTML += '<select id="_uroCWLGroupSelect" style="width:40%;height:22px;"></select>&nbsp;<input type="button" id="_btnCWLGroupDel" value="Delete group"><br>';
  2136. iHTML += '<input type="text" id="_uroCWLGroupEntry" style="width:40%;height:22px;">&nbsp;<input type="button" id="_btnCWLGroupAdd" value="Add group">';
  2137. iHTML += '<br><input type="button" id="_btnRescanCamWatchList" value="Refresh Camera Data"><br><br>';
  2138. iHTML += '<b>Remove cameras from OWL:</b><br>';
  2139. iHTML += '<input type="button" id="_btnRemoveDeletedCameras" value="Deleted">&nbsp;&nbsp;';
  2140. iHTML += '<input type="button" id="_btnRemoveUnknownServerCameras" value="Unknown Server">&nbsp;&nbsp;';
  2141. iHTML += '<input type="button" id="_btnClearCamWatchList" value="ALL Cameras">';
  2142. iHTML += '</div>';
  2143. uroOWL.innerHTML = iHTML;
  2144.  
  2145. if(uroCamWatchObjects.length > 0)
  2146. {
  2147. for(camidx=0;camidx<uroCamWatchObjects.length;camidx++)
  2148. {
  2149. document.getElementById("_uroCWL-"+camidx).onmouseover = uroHighlightCWLEntry;
  2150. document.getElementById("_uroCWL-"+camidx).onmouseleave = uroUnhighlightCWLEntry;
  2151.  
  2152. if(uroCamWatchObjects[camidx].server == W.location.code)
  2153. {
  2154. var icon1 = document.getElementById("_uroCWLIcon1-"+camidx);
  2155. var icon2 = document.getElementById("_uroCWLIcon2-"+camidx);
  2156. if(icon1 !== null)
  2157. {
  2158. icon1.onmouseover = uroCWLIconHighlight;
  2159. icon1.onmouseleave = uroCWLIconLowlight;
  2160. icon1.onclick = uroAssignCameraToGroup;
  2161. }
  2162. if(icon2 !== null)
  2163. {
  2164. icon2.onmouseover = uroCWLIconHighlight;
  2165. icon2.onmouseleave = uroCWLIconLowlight;
  2166. icon2.onclick = uroGotoCam;
  2167. }
  2168. }
  2169. }
  2170. }
  2171.  
  2172. uroAddBtnEvl('_btnClearCamWatchList', 'click', uroClearCamWatchList);
  2173. uroAddBtnEvl('_btnRemoveDeletedCameras', 'click', uroClearDeletedCameras);
  2174. uroAddBtnEvl('_btnRemoveUnknownServerCameras', 'click', uroClearUnknownServerCameras);
  2175. uroAddBtnEvl('_btnRescanCamWatchList', 'click', uroRescanCamWatchList);
  2176. uroAddBtnEvl('_btnCWLGroupDel', 'click', uroRemoveCWLGroup);
  2177. uroAddBtnEvl('_btnCWLGroupAdd', 'click', uroAddCWLGroup);
  2178. if(document.getElementById('_uroCWLGroupSelect') !== null)
  2179. {
  2180. uroAddLog('populating CWL group list');
  2181. uroPopulateCWLGroupSelect();
  2182. var selector = document.getElementById('_uroCWLGroupSelect');
  2183. if(selectedGroup >= selector.length)
  2184. {
  2185. selectedGroup = 0;
  2186. }
  2187. selector.selectedIndex = selectedGroup;
  2188. }
  2189.  
  2190. if(uroCWLGroups.length > 0)
  2191. {
  2192. for(groupidx=0;groupidx<uroCWLGroups.length;groupidx++)
  2193. {
  2194. if(uroCWLGroups[groupidx].groupCount === 0)
  2195. {
  2196. uroSetStyleDisplay('_uroCWLGroup-'+groupidx,'none');
  2197. }
  2198. else
  2199. {
  2200. uroSetOnClick('_uroCWLGroupState-'+groupidx,uroCWLGroupCollapseExpand);
  2201. }
  2202. }
  2203. }
  2204. }
  2205.  
  2206. // --------------------------------------------------------------------------------------------------------------------
  2207. // END OF WATCHLIST STUFF
  2208. // --------------------------------------------------------------------------------------------------------------------
  2209.  
  2210.  
  2211. function uroIsOnIgnoreList(fid)
  2212. {
  2213. if(sessionStorage.UROverview_FID_IgnoreList.indexOf('fid:'+fid) == -1) return false;
  2214. else return true;
  2215. }
  2216.  
  2217. function uroEnableIgnoreListControls()
  2218. {
  2219. var btnState = "visible";
  2220. if(sessionStorage.UROverview_FID_IgnoreList === '')
  2221. {
  2222. btnState = "hidden";
  2223. }
  2224. document.getElementById('_btnUndoLastHide').style.visibility = btnState;
  2225. document.getElementById('_btnClearSessionHides').style.visibility = btnState;
  2226. uroFilterItems();
  2227. }
  2228.  
  2229. function uroAddToIgnoreList()
  2230. {
  2231. if(!uroIsOnIgnoreList(uroShownFID))
  2232. {
  2233. sessionStorage.UROverview_FID_IgnoreList += 'fid:'+uroShownFID;
  2234. uroAddLog('added fid '+uroShownFID+' to ignore list');
  2235. uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
  2236. uroDiv.style.visibility = 'hidden';
  2237. uroEnableIgnoreListControls();
  2238. W.map.events.register("mousemove", null, uroFilterItemsOnMove);
  2239. }
  2240. return false;
  2241. }
  2242.  
  2243. /*
  2244. function uroRemoveFromIgnoreList(fid)
  2245. {
  2246. var ignorelist = sessionStorage.UROverview_FID_IgnoreList;
  2247. var fidpos = ignorelist.indexOf('fid:'+fid);
  2248. if(fidpos != -1)
  2249. {
  2250. var preFID = ignorelist.slice(0,fidpos);
  2251. ignorelist = ignorelist.slice(fidpos+1);
  2252. fidpos = ignorelist.indexOf('fid:');
  2253. if(fidpos == -1) ignorelist = '';
  2254. else ignorelist = ignorelist.slice(fidpos);
  2255. sessionStorage.UROverview_FID_IgnoreList = preFID + ignorelist;
  2256. uroAddLog('removed fid '+fid+' from ignore list');
  2257. uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
  2258. uroEnableIgnoreListControls();
  2259. }
  2260. }
  2261. */
  2262.  
  2263. function uroRemoveLastAddedIgnore()
  2264. {
  2265. var ignorelist = sessionStorage.UROverview_FID_IgnoreList;
  2266. var fidpos = ignorelist.lastIndexOf('fid:');
  2267. if(fidpos != -1)
  2268. {
  2269. ignorelist = ignorelist.slice(0,fidpos);
  2270. sessionStorage.UROverview_FID_IgnoreList = ignorelist;
  2271. uroAddLog('removed last fid from ignore list');
  2272. uroAddLog(sessionStorage.UROverview_FID_IgnoreList);
  2273. uroEnableIgnoreListControls();
  2274. }
  2275. }
  2276.  
  2277. function uroRemoveAllIgnores()
  2278. {
  2279. sessionStorage.UROverview_FID_IgnoreList = '';
  2280. uroEnableIgnoreListControls();
  2281. }
  2282.  
  2283.  
  2284. function uroKeywordPresent(desc, keyword)
  2285. {
  2286. var re;
  2287. if(uroGetCBChecked('_cbCaseInsensitive') === true) re = RegExp(keyword,'i');
  2288. else re = RegExp(keyword);
  2289.  
  2290. if(desc.search(re) != -1) return true;
  2291. else return false;
  2292. }
  2293.  
  2294.  
  2295. function uroClickify(desc)
  2296. {
  2297. var linkStartPos = desc.indexOf('http://');
  2298. if(linkStartPos == -1) linkStartPos = desc.indexOf('https://');
  2299. if(linkStartPos != -1)
  2300. {
  2301. var descPreLink = desc.slice(0,linkStartPos);
  2302. var descURL = desc.slice(linkStartPos);
  2303. var linkEndPos = descURL.indexOf(' ');
  2304. var descPostLink = '';
  2305. if(linkEndPos != -1)
  2306. {
  2307. descPostLink = descURL.slice(linkEndPos);
  2308. descURL = descURL.slice(0,linkEndPos);
  2309. }
  2310. var linkTarget = '';
  2311. if(descURL.indexOf('cryosphere') != -1) linkTarget = '_cryosphere';
  2312. else if(descURL.indexOf('waze.com') != -1) linkTarget = '_wazeUR';
  2313. desc = descPreLink + '<a target="'+linkTarget+'" href="'+descURL+'">here</a>' + descPostLink;
  2314. }
  2315. return desc;
  2316. }
  2317.  
  2318.  
  2319. function uroGetUpdateRequestSessions()
  2320. {
  2321. var idList = [];
  2322.  
  2323. while((idList.length < 50) && (uroPendingURSessionIDs.length))
  2324. {
  2325. var id = uroPendingURSessionIDs.shift();
  2326. idList.push(id);
  2327. }
  2328.  
  2329. if(idList.length > 0)
  2330. {
  2331. uroAddLog('grabbing '+idList.length+' updateRequestSessions, IDs: '+idList);
  2332. W.model.updateRequestSessions.get(idList);
  2333. }
  2334.  
  2335. if((uroPendingURSessionIDs.length) || (uroRequestedURSessionIDs.length))
  2336. {
  2337. setTimeout(uroGetUpdateRequestSessions,1000);
  2338. }
  2339. }
  2340.  
  2341. function uroRefreshUpdateRequestSessions()
  2342. {
  2343. var urcount = 0;
  2344. uroPendingURSessionIDs = [];
  2345. uroRequestedURSessionIDs = [];
  2346. for (var urID in W.model.mapUpdateRequests.objects)
  2347. {
  2348. if(W.model.mapUpdateRequests.objects.hasOwnProperty(urID))
  2349. {
  2350. if(W.model.updateRequestSessions.objects[urID] === undefined)
  2351. {
  2352. uroPendingURSessionIDs.push(urID);
  2353. }
  2354. urcount++;
  2355. }
  2356. }
  2357. uroGetUpdateRequestSessions();
  2358. }
  2359.  
  2360. function uroURHasMyComments(fid)
  2361. {
  2362. var nComments = W.model.updateRequestSessions.objects[fid].comments.length;
  2363. if(nComments === 0) return false;
  2364.  
  2365. for(var cidx=0; cidx<nComments; cidx++)
  2366. {
  2367. if(W.model.updateRequestSessions.objects[fid].comments[cidx].userID == uroUserID) return true;
  2368. }
  2369.  
  2370. return false;
  2371. }
  2372.  
  2373.  
  2374. function uroACMObj(urID, customType, hasMyComments, nComments)
  2375. {
  2376. this.urID = urID;
  2377. this.customType = customType;
  2378. this.hasMyComments = hasMyComments;
  2379. this.nComments = nComments;
  2380. }
  2381.  
  2382. function uroAddCustomMarkers(urID, customType, hasMyComments, nComments)
  2383. {
  2384. var useCustomMarker = false;
  2385. if(uroGetCBChecked('_cbMasterEnable') === true)
  2386. {
  2387. if(customType === 0) useCustomMarker = (uroGetCBChecked('_cbCustomRoadworksMarkers'));
  2388. else if(customType === 1) useCustomMarker = (uroGetCBChecked('_cbCustomConstructionMarkers'));
  2389. else if(customType === 2) useCustomMarker = (uroGetCBChecked('_cbCustomClosuresMarkers'));
  2390. else if(customType === 3) useCustomMarker = (uroGetCBChecked('_cbCustomEventsMarkers'));
  2391. else if(customType === 4) useCustomMarker = (uroGetCBChecked('_cbCustomNotesMarkers'));
  2392. else if(customType === 100) useCustomMarker = (uroGetCBChecked('_cbCustomElginMarkers'));
  2393. else if(customType === 101) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficCastMarkers'));
  2394. else if(customType === 102) useCustomMarker = (uroGetCBChecked('_cbCustomTrafficMasterMarkers'));
  2395. }
  2396. if(!useCustomMarker) customType = -1;
  2397. uroCustomMarkerList.push(new uroACMObj(urID, customType, hasMyComments, nComments));
  2398. }
  2399.  
  2400. function uroRenderCustomMarkers(markerType)
  2401. {
  2402. var urID;
  2403. var elmID;
  2404. var newSpan;
  2405. var divElem;
  2406. var objIdx;
  2407. var customType;
  2408. var customMarker;
  2409.  
  2410. if(markerType == 'ur')
  2411. {
  2412. var useDefaultConvoMarker = false;
  2413. var addCommentCount = false;
  2414.  
  2415. if(uroGetCBChecked('_cbMasterEnable') === true)
  2416. {
  2417. if((uroGetCBChecked('_cbNativeConvoMarkers')) && (uroBetaEditor === false)) useDefaultConvoMarker = true;
  2418. if((uroGetCBChecked('_cbNativeBetaConvoMarkers')) && (uroBetaEditor === true)) useDefaultConvoMarker = true;
  2419. if(uroGetCBChecked('_cbCommentCount')) addCommentCount = true;
  2420. }
  2421. else
  2422. {
  2423. useDefaultConvoMarker = true;
  2424. }
  2425.  
  2426. var uRCM_masterEnable = uroGetCBChecked('_cbMasterEnable');
  2427. divElem = document.getElementById(W.map.updateRequestLayer.id);
  2428. if(divElem.childNodes.length > 0)
  2429. {
  2430. for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
  2431. {
  2432. customType = -1;
  2433. var cmlObj = uroCustomMarkerList[objIdx];
  2434. if(uRCM_masterEnable === true)
  2435. {
  2436. customType = cmlObj.customType;
  2437. }
  2438. if(customType < 100)
  2439. {
  2440. urID = cmlObj.urID;
  2441. var nComments = cmlObj.nComments;
  2442. var iconObj = W.map.updateRequestLayer.markers[urID].icon;
  2443. newSpan = '';
  2444.  
  2445. if(nComments !== 0)
  2446. {
  2447. var classList = iconObj.imageDiv.classList;
  2448. elmID = "commentCount_"+urID;
  2449.  
  2450. if(addCommentCount)
  2451. {
  2452. // add a new comment count bubble if the UR doesn't already have one
  2453. if(document.getElementById(elmID) === null)
  2454. {
  2455. newSpan += '<span id="'+elmID+'" style="position:absolute;top:-9px;left:-11px;pointer-events:none;z-index:1">';
  2456. // define the comment-count holding span within the span used to hold the empty bubble image, and before the image is
  2457. // added to the HTML, to avoid z-indexing issues when adjacent comment count bubbles are overlapped...
  2458. newSpan += '<span id="'+elmID+"_inner"+'" style="position:absolute;top:4px;left:11px;font-size:11px;;pointer-events:none"></span>';
  2459. newSpan += '<img src="'+uroMarkers[0]+'">';
  2460. newSpan += '</span>';
  2461. }
  2462. }
  2463. else
  2464. {
  2465. // remove comment count bubble from this UR marker if one has previously been
  2466. // added and the user has now disabled the option...
  2467. if(document.getElementById(elmID) !== null)
  2468. {
  2469. document.getElementById(elmID).remove();
  2470. }
  2471. if(document.getElementById(elmID+"_inner") !== null)
  2472. {
  2473. document.getElementById(elmID+"_inner").remove();
  2474. }
  2475. }
  2476.  
  2477. elmID = "convoMarker_"+urID;
  2478. if(useDefaultConvoMarker === false)
  2479. {
  2480. if(document.getElementById(elmID) === null)
  2481. {
  2482. var hasMyComments = cmlObj.hasMyComments;
  2483. // 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
  2484. // conversation marker, the custom marker remains on top...
  2485. newSpan += '<span id="'+elmID+'" style="position:absolute;top:-9px;left:18px;pointer-events:none;z-index:1">';
  2486. if(hasMyComments) newSpan += '<img src="'+uroMarkers[2]+'">';
  2487. else newSpan += '<img src="'+uroMarkers[1]+'">';
  2488. newSpan += '</span>';
  2489. classList.remove("has-comments");
  2490. }
  2491. }
  2492. else
  2493. {
  2494. // remove custom conversation marker from this UR if one has previously been
  2495. // added and the user has now disabled this option
  2496. if(document.getElementById(elmID) !== null)
  2497. {
  2498. document.getElementById(elmID).remove();
  2499. }
  2500. if(nComments > 0)
  2501. {
  2502. // only replace the native marker class if the UR has comments - if we're just clearing the custom
  2503. // marker following a master enable switchoff, we don't then want to add native markers to URs which
  2504. // didn't have them in the first place...
  2505. classList.add("has-comments");
  2506. }
  2507. }
  2508. }
  2509. elmID = "customMarker_"+urID;
  2510. customMarker = '';
  2511. if(customType != -1)
  2512. {
  2513. if(document.getElementById(elmID) === null)
  2514. {
  2515. newSpan += '<span id="'+elmID+'" style="position:absolute;pointer-events:none;"></span>';
  2516. }
  2517. customType = uroGetCustomMarkerIdx(customType);
  2518. if(W.model.mapUpdateRequests.objects[urID] !== undefined)
  2519. {
  2520. if(W.model.mapUpdateRequests.objects[urID].attributes.open === false) customType += 1;
  2521. }
  2522. customMarker = '<img src="'+uroIcons[customType][0]+'">';
  2523. }
  2524. else
  2525. {
  2526. if(document.getElementById(elmID) !== null)
  2527. {
  2528. document.getElementById(elmID).remove();
  2529. }
  2530. }
  2531. if(newSpan !== '')
  2532. {
  2533. iconObj.$div.prepend(newSpan);
  2534. if(customMarker !== '')
  2535. {
  2536. if(document.getElementById(elmID) !== null)
  2537. {
  2538. document.getElementById(elmID).innerHTML = customMarker;
  2539. }
  2540. }
  2541.  
  2542. if(addCommentCount)
  2543. {
  2544. var styleLeft;
  2545. if(nComments < 10) styleLeft = '11px';
  2546. else if(nComments < 100) styleLeft = '8px';
  2547. else styleLeft = '5px';
  2548. elmID = "commentCount_"+urID;
  2549. if(document.getElementById(elmID+"_inner") !== null)
  2550. {
  2551. document.getElementById(elmID+"_inner").innerHTML = nComments;
  2552. document.getElementById(elmID+"_inner").style.left = styleLeft;
  2553. }
  2554. }
  2555. }
  2556. }
  2557. }
  2558. }
  2559. }
  2560.  
  2561. else if(markerType == 'mp')
  2562. {
  2563. divElem = document.getElementById(W.map.problemLayer.id);
  2564. if(divElem.childNodes.length > 0)
  2565. {
  2566. for(objIdx = 0; objIdx < uroCustomMarkerList.length; objIdx++)
  2567. {
  2568. customType = uroCustomMarkerList[objIdx].customType;
  2569. if((customType >= 100) || (customType == -1))
  2570. {
  2571. urID = uroCustomMarkerList[objIdx].urID;
  2572. elmID = "customMarker_"+urID;
  2573. if(customType != -1)
  2574. {
  2575. if(document.getElementById(elmID) === null)
  2576. {
  2577. newSpan = '<span id="'+elmID+'" style="position:absolute;pointer-events:none;"></span>';
  2578. if(W.map.problemLayer.markers[urID] !== undefined)
  2579. {
  2580. W.map.problemLayer.markers[urID].icon.$div.prepend(newSpan);
  2581. }
  2582. }
  2583. if(document.getElementById(elmID) !== null)
  2584. {
  2585. customType = uroGetCustomMarkerIdx(customType);
  2586. if(W.model.problems.objects[urID] !== undefined)
  2587. {
  2588. if(W.model.problems.objects[urID].attributes.open === false) customType += 1;
  2589. }
  2590. customMarker = '<img src="'+uroIcons[customType][0]+'">';
  2591. document.getElementById(elmID).innerHTML = customMarker;
  2592. }
  2593. }
  2594. else
  2595. {
  2596. if(document.getElementById(elmID) !== null)
  2597. {
  2598. document.getElementById(elmID).remove();
  2599. }
  2600. }
  2601. }
  2602. }
  2603. }
  2604. }
  2605. }
  2606.  
  2607.  
  2608. function uroChangeCustomMarkers(urID,isHighlighted,customType,markerType)
  2609. {
  2610. if(document.getElementById('customMarker_'+urID) !== null)
  2611. {
  2612. if(markerType == "ur")
  2613. {
  2614. if(W.model.updateRequestSessions.objects[urID].open === false) customType += 1;
  2615. }
  2616. else if(markerType == "mp")
  2617. {
  2618. if(W.model.problems.objects[urID].attributes.open === false) customType += 1;
  2619. }
  2620.  
  2621. if(isHighlighted === true)
  2622. {
  2623. document.getElementById('customMarker_'+urID).innerHTML = '<img src="'+uroIcons[customType][1]+'">';
  2624. }
  2625. else
  2626. {
  2627. document.getElementById('customMarker_'+urID).innerHTML = '<img src="'+uroIcons[customType][0]+'">';
  2628. }
  2629. }
  2630. }
  2631.  
  2632. function uroFilterPlaces()
  2633. {
  2634. if(uroFilterPreamble() === false) return;
  2635.  
  2636. if(uroPlaceSelected === true) return;
  2637.  
  2638. if(uroGetCBChecked('_cbDisablePlacesFiltering') === true) return;
  2639.  
  2640. var filterCats = [];
  2641. for(var i=0; i<W.Config.venues.categories.length; i++)
  2642. {
  2643. var parentCategory = W.Config.venues.categories[i];
  2644. var subCategory;
  2645.  
  2646. if(uroGetCBChecked('_cbPlacesFilter-'+parentCategory) === true)
  2647. {
  2648. filterCats.push(parentCategory);
  2649. for(var i1=0; i1<W.Config.venues.subcategories[parentCategory].length; i1++)
  2650. {
  2651. subCategory = W.Config.venues.subcategories[parentCategory][i1];
  2652. filterCats.push(subCategory);
  2653. }
  2654. }
  2655. else
  2656. {
  2657. for(var i2=0; i2<W.Config.venues.subcategories[parentCategory].length; i2++)
  2658. {
  2659. subCategory = W.Config.venues.subcategories[parentCategory][i2];
  2660. if(uroGetCBChecked('_cbPlacesFilter-'+subCategory) === true)
  2661. {
  2662. filterCats.push(subCategory);
  2663. }
  2664. }
  2665. }
  2666. }
  2667.  
  2668. var placeStyle;
  2669.  
  2670. var uFP_filterEditedLessThan = uroGetCBChecked('_cbPlaceFilterEditedLessThan');
  2671. var uFP_filterEditedMoreThan = uroGetCBChecked('_cbPlaceFilterEditedMoreThan');
  2672. var uFP_filterL0 = uroGetCBChecked('_cbHidePlacesL0');
  2673. var uFP_filterL1 = uroGetCBChecked('_cbHidePlacesL1');
  2674. var uFP_filterL2 = uroGetCBChecked('_cbHidePlacesL2');
  2675. var uFP_filterL3 = uroGetCBChecked('_cbHidePlacesL3');
  2676. var uFP_filterL4 = uroGetCBChecked('_cbHidePlacesL4');
  2677. var uFP_filterL5 = uroGetCBChecked('_cbHidePlacesL5');
  2678. var uFP_filterOnLockLevel = (uFP_filterL0 || uFP_filterL1 || uFP_filterL2 || uFP_filterL3 || uFP_filterL4 || uFP_filterL5);
  2679. var uFP_filterNoPhotos = uroGetCBChecked('_cbHideNoPhotoPlaces');
  2680. var uFP_filterWithPhotos = uroGetCBChecked('_cbHidePhotoPlaces');
  2681. var uFP_filterPrivate = uroGetCBChecked('_cbFilterPrivatePlaces');
  2682. var uFP_invertFilters = uroGetCBChecked('_cbInvertPlacesFilter');
  2683. var uFP_masterEnable = uroGetCBChecked('_cbMasterEnable');
  2684. var uFP_thresholdMinDays = document.getElementById('_inputFilterPlaceEditMinDays').value;
  2685. var uFP_thresholdMaxDays = document.getElementById('_inputFilterPlaceEditMaxDays').value;
  2686. for(var v=0; v<W.map.landmarkLayer.features.length; v++)
  2687. {
  2688. placeStyle = 'visible';
  2689. if(uFP_masterEnable === true)
  2690. {
  2691. var lmObj = W.map.landmarkLayer.features[v];
  2692.  
  2693. // when an area place is selected, the drag points for editing the place outline now get added as objects into W.map.landmarkLayer.features,
  2694. // however none of these objects have the .model property - we must therefore check each entry in features[] to see if it has .model before
  2695. // attempting to filter it...
  2696. if(lmObj.model !== undefined)
  2697. {
  2698. if(lmObj.model.attributes.id < 0)
  2699. {
  2700. // don't apply filtering to newly-created places - this allows the user to leave their filtering settings unchanged whilst
  2701. // adding a new place which, once saved, would then be hidden...
  2702. break;
  2703. }
  2704.  
  2705. if((uFP_filterEditedLessThan) || (uFP_filterEditedMoreThan))
  2706. {
  2707. var editDate = lmObj.model.attributes.updatedOn;
  2708. if(editDate === undefined)
  2709. {
  2710. // where a place has never been edited since its creation, use the creation date instead...
  2711. editDate = lmObj.model.attributes.createdOn;
  2712. }
  2713. if(editDate !== undefined)
  2714. {
  2715. var editDaysAgo = uroDateToDays(editDate);
  2716. if(uFP_filterEditedLessThan)
  2717. {
  2718. if(editDaysAgo < uFP_thresholdMinDays)
  2719. {
  2720. placeStyle = 'hidden';
  2721. }
  2722. }
  2723. if(uFP_filterEditedMoreThan)
  2724. {
  2725. if(editDaysAgo > uFP_thresholdMaxDays)
  2726. {
  2727. placeStyle = 'hidden';
  2728. }
  2729. }
  2730. }
  2731. }
  2732.  
  2733. if(placeStyle == 'visible')
  2734. {
  2735. if(uFP_filterOnLockLevel)
  2736. {
  2737. var lockLevel = lmObj.model.attributes.lockRank;
  2738. if ((uFP_filterL0) && (lockLevel === 0)) placeStyle = 'hidden';
  2739. if ((uFP_filterL1) && (lockLevel === 1)) placeStyle = 'hidden';
  2740. if ((uFP_filterL2) && (lockLevel === 2)) placeStyle = 'hidden';
  2741. if ((uFP_filterL3) && (lockLevel === 3)) placeStyle = 'hidden';
  2742. if ((uFP_filterL4) && (lockLevel === 4)) placeStyle = 'hidden';
  2743. if ((uFP_filterL5) && (lockLevel === 5)) placeStyle = 'hidden';
  2744. }
  2745. }
  2746.  
  2747. if(placeStyle == 'visible')
  2748. {
  2749. if(uFP_filterNoPhotos || uFP_filterWithPhotos)
  2750. {
  2751. var nPhotos = 0;
  2752. for(var loop=0; loop<lmObj.model.attributes.images.length; loop++)
  2753. {
  2754. if(lmObj.model.attributes.images[loop].attributes.approved) nPhotos++;
  2755. }
  2756. if((uFP_filterNoPhotos) && (nPhotos === 0)) placeStyle = 'hidden';
  2757. if((uFP_filterWithPhotos) && (nPhotos !== 0)) placeStyle = 'hidden';
  2758. }
  2759. }
  2760.  
  2761.  
  2762. if(placeStyle == 'visible')
  2763. {
  2764. if((uFP_filterPrivate === true) && (lmObj.model.attributes.residential === true))
  2765. {
  2766. placeStyle = 'hidden';
  2767. }
  2768. else
  2769. {
  2770. for(var cat=0; cat<filterCats.length; cat++)
  2771. {
  2772. if(lmObj.model.attributes.categories.contains(filterCats[cat]))
  2773. {
  2774. placeStyle = 'hidden';
  2775. break;
  2776. }
  2777. }
  2778. }
  2779. }
  2780. }
  2781.  
  2782. if(uFP_invertFilters === true)
  2783. {
  2784. if(placeStyle == 'hidden') placeStyle = 'visible';
  2785. else placeStyle = 'hidden';
  2786. }
  2787. }
  2788.  
  2789. var geoID = W.map.landmarkLayer.features[v].geometry.id;
  2790. if(document.getElementById(geoID) !== null)
  2791. {
  2792. document.getElementById(geoID).style.visibility = placeStyle;
  2793. }
  2794. }
  2795. var uFP_filterUneditable = uroGetCBChecked('_cbFilterUneditablePlaceUpdates');
  2796. var uFP_filterLockRanked = uroGetCBChecked('_cbFilterLockRankedPlaceUpdates');
  2797. var uFP_filterFlagged = uroGetCBChecked("_cbFilterFlaggedPUR");
  2798. var uFP_filterNewPlace = uroGetCBChecked("_cbFilterNewPlacePUR");
  2799. var uFP_filterUpdatedDetails = uroGetCBChecked("_cbFilterUpdatedDetailsPUR");
  2800. var uFP_filterNewPhoto = uroGetCBChecked("_cbFilterNewPhotoPUR");
  2801. var uFP_filterMinPURAge = uroGetCBChecked('_cbEnablePURMinAgeFilter');
  2802. var uFP_filterMaxPURAge = uroGetCBChecked('_cbEnablePURMaxAgeFilter');
  2803. var uFP_invertPURFilters = uroGetCBChecked('_cbInvertPURFilters');
  2804. var uFP_filterHighSeverity = uroGetCBChecked('_cbPURFilterHighSeverity');
  2805. var uFP_filterMedSeverity = uroGetCBChecked('_cbPURFilterMediumSeverity');
  2806. var uFP_filterLowSeverity = uroGetCBChecked('_cbPURFilterLowSeverity');
  2807. var uFP_leavePURGeos = uroGetCBChecked('_cbLeavePURGeos');
  2808. var uFP_thresholdMinPURDays = uroGetElmValue('_inputPURFilterMinDays');
  2809. var uFP_thresholdMaxPURDays = uroGetElmValue('_inputPURFilterMaxDays');
  2810. var uFP_isLoggedIn = W.loginManager.isLoggedIn();
  2811. var uFP_userRank = W.loginManager.user.rank;
  2812. var purAge = null;
  2813. for(var pu in W.map.placeUpdatesLayer.markers)
  2814. {
  2815. if(W.map.placeUpdatesLayer.markers.hasOwnProperty(pu))
  2816. {
  2817. var puObj = W.map.placeUpdatesLayer.markers[pu];
  2818. if(W.map.placeUpdatesLayer.getVisibility() === true)
  2819. {
  2820. placeStyle = 'visible';
  2821. if(uFP_masterEnable === true)
  2822. {
  2823. if(uFP_masterEnable === true)
  2824. {
  2825. if(uFP_filterUneditable === true)
  2826. {
  2827. if(puObj.model.attributes.permissions === 0)
  2828. {
  2829. placeStyle = 'hidden';
  2830. }
  2831. if((placeStyle == 'visible') && (uFP_isLoggedIn))
  2832. {
  2833. if(uFP_userRank < puObj.model.attributes.lockRank)
  2834. {
  2835. placeStyle = 'hidden';
  2836. }
  2837. }
  2838. if((placeStyle == 'visible') && (puObj.model.attributes.adLocked))
  2839. {
  2840. placeStyle = 'hidden';
  2841. }
  2842. }
  2843.  
  2844. if((placeStyle == 'visible') && (uFP_filterLockRanked === true))
  2845. {
  2846. if(puObj.model.attributes.lockRank !== 0)
  2847. {
  2848. placeStyle = 'hidden';
  2849. }
  2850. }
  2851.  
  2852. if((placeStyle == 'visible') && (uFP_filterFlagged === true))
  2853. {
  2854. if(puObj.icon.imageDiv.className.indexOf('flag') != -1)
  2855. {
  2856. placeStyle = 'hidden';
  2857. }
  2858. }
  2859. if((placeStyle == 'visible') && (uFP_filterNewPlace === true))
  2860. {
  2861. if(puObj.icon.imageDiv.className.indexOf('add_venue') != -1)
  2862. {
  2863. placeStyle = 'hidden';
  2864. }
  2865. }
  2866. if((placeStyle == 'visible') && (uFP_filterUpdatedDetails === true))
  2867. {
  2868. if((puObj.icon.imageDiv.className.indexOf('update_venue') != -1) || (puObj.icon.imageDiv.className.indexOf('multiple') != -1))
  2869. {
  2870. placeStyle = 'hidden';
  2871. }
  2872. }
  2873. if((placeStyle == 'visible') && (uFP_filterNewPhoto === true))
  2874. {
  2875. if(puObj.icon.imageDiv.className.indexOf('add_image') != -1)
  2876. {
  2877. placeStyle = 'hidden';
  2878. }
  2879. }
  2880.  
  2881. if(uFP_invertPURFilters === true)
  2882. {
  2883. if(placeStyle == 'hidden') placeStyle = 'visible';
  2884. else placeStyle = 'hidden';
  2885. }
  2886.  
  2887. if(uFP_filterMinPURAge || uFP_filterMaxPURAge)
  2888. {
  2889. purAge = uroGetPURAge(puObj.model);
  2890. if(uFP_filterMinPURAge === true)
  2891. {
  2892. if(purAge < uFP_thresholdMinPURDays) placeStyle = 'hidden';
  2893. }
  2894. if(uFP_filterMaxPURAge === true)
  2895. {
  2896. if(purAge > uFP_thresholdMaxPURDays) placeStyle = 'hidden';
  2897. }
  2898. }
  2899.  
  2900. if(placeStyle == 'visible')
  2901. {
  2902. var purSeverity = puObj._getSeverity();
  2903. if((uFP_filterHighSeverity) && (purSeverity == "high")) placeStyle = 'hidden';
  2904. if((placeStyle == 'visible') && (uFP_filterMedSeverity) && (purSeverity == "medium")) placeStyle = 'hidden';
  2905. if((placeStyle == 'visible') && (uFP_filterLowSeverity) && (purSeverity == "low")) placeStyle = 'hidden';
  2906. }
  2907. }
  2908. }
  2909.  
  2910. puObj.icon.imageDiv.style.visibility = placeStyle;
  2911.  
  2912. if(uFP_leavePURGeos === false)
  2913. {
  2914. if(puObj.model !== undefined)
  2915. {
  2916. if(puObj.model.geometry !== undefined)
  2917. {
  2918. var puGeo = document.getElementById(puObj.model.geometry.id);
  2919. if(puGeo !== null)
  2920. {
  2921. puGeo.style.visibility = placeStyle;
  2922. }
  2923. }
  2924. }
  2925. }
  2926. }
  2927. }
  2928. }
  2929. }
  2930.  
  2931. function uroFilterCameras()
  2932. {
  2933. if(uroFilterPreamble() === false) return;
  2934. var camLayer = document.getElementById(uroRootContainer+'_svgRoot');
  2935. if(camLayer === null)
  2936. {
  2937. if(uroNullCamLayer === false)
  2938. {
  2939. uroAddLog('caught null camLayer');
  2940. uroNullCamLayer = true;
  2941. }
  2942. return;
  2943. }
  2944. uroNullCamLayer = false;
  2945. if(uroMouseIsDown === false) W.map.camerasLayer.redraw();
  2946. if(uroGetCBChecked('_cbMasterEnable') === true)
  2947. {
  2948. for (var uroCamObj in W.model.cameras.objects)
  2949. {
  2950. if(W.model.cameras.objects.hasOwnProperty(uroCamObj))
  2951. {
  2952. var uroCamUpdater = '';
  2953. var uroCamUpdaterRank = -1;
  2954. var uroCamCreator = '';
  2955. var uroCamCreatorRank = -1;
  2956. var uroCam = W.model.cameras.objects[uroCamObj];
  2957. var uroCamStyle = 'visible';
  2958. if(uroCam.attributes.createdBy !== null)
  2959. {
  2960. if(W.model.users.objects[uroCam.attributes.createdBy] !== undefined)
  2961. {
  2962. uroCamCreator = W.model.users.objects[uroCam.attributes.createdBy].userName;
  2963. uroCamCreatorRank = W.model.users.objects[uroCam.attributes.createdBy].rank;
  2964. }
  2965. }
  2966.  
  2967. if(uroCam.attributes.updatedBy !== null)
  2968. {
  2969. if(W.model.users.objects[uroCam.attributes.updatedBy] !== undefined)
  2970. {
  2971. uroCamUpdater = W.model.users.objects[uroCam.attributes.updatedBy].userName;
  2972. uroCamUpdaterRank = W.model.users.objects[uroCam.attributes.updatedBy].rank;
  2973. }
  2974. }
  2975.  
  2976. var uroCamApproved = uroCam.attributes.validated;
  2977. var uroCamType = uroCam.attributes.type;
  2978.  
  2979. if(uroGetCBChecked('_cbShowOnlyMyCams') === true)
  2980. {
  2981. if((uroUserID != uroCam.attributes.createdBy)&&(uroUserID != uroCam.attributes.updatedBy)) uroCamStyle = 'hidden';
  2982. }
  2983.  
  2984. if((uroGetCBChecked('_cbShowWorldCams') === false) || (uroGetCBChecked('_cbShowUSACams') === false) || (uroGetCBChecked('_cbShowNonWorldCams') === false))
  2985. {
  2986. var posWorld = uroCamCreator.indexOf('world_');
  2987. var posUSA = uroCamCreator.indexOf('usa_');
  2988.  
  2989. if((uroGetCBChecked('_cbShowWorldCams') === false) && (posWorld === 0)) uroCamStyle = 'hidden';
  2990. if((uroGetCBChecked('_cbShowUSACams') === false) && (posUSA === 0)) uroCamStyle = 'hidden';
  2991. if((uroGetCBChecked('_cbShowNonWorldCams') === false) && (posWorld !== 0) && (posUSA !== 0)) uroCamStyle = 'hidden';
  2992. }
  2993.  
  2994. if((uroGetCBChecked('_cbShowApprovedCams') === false) || (uroGetCBChecked('_cbShowNonApprovedCams') === false))
  2995. {
  2996. if((uroGetCBChecked('_cbShowApprovedCams') === false) && (uroCamApproved === true)) uroCamStyle = 'hidden';
  2997. if((uroGetCBChecked('_cbShowNonApprovedCams') === false) && (uroCamApproved === false)) uroCamStyle = 'hidden';
  2998. }
  2999.  
  3000. if((uroGetCBChecked('_cbShowNonApprovedCams') === true) && (uroCamApproved === false))
  3001. {
  3002. if(((uroGetCBChecked('_cbShowOlderCreatedNonApproved') === true)) && (uroGetCameraAge(uroCam,1) <= uroGetElmValue('_inputCameraMinCreatedDays'))) uroCamStyle = 'hidden';
  3003. if(((uroGetCBChecked('_cbShowOlderUpdatedNonApproved') === true)) && (uroGetCameraAge(uroCam,0) <= uroGetElmValue('_inputCameraMinUpdatedDays'))) uroCamStyle = 'hidden';
  3004. }
  3005.  
  3006. if((uroGetCBChecked('_cbShowSpeedCams') === false) || (uroGetCBChecked('_cbShowRedLightCams') === false) || (uroGetCBChecked('_cbShowDummyCams') === false))
  3007. {
  3008. if((uroGetCBChecked('_cbShowSpeedCams') === false) && (uroCamType == 2)) uroCamStyle = 'hidden';
  3009. if((uroGetCBChecked('_cbShowRedLightCams') === false) && (uroCamType == 4)) uroCamStyle = 'hidden';
  3010. if((uroGetCBChecked('_cbShowDummyCams') === false) && (uroCamType == 3)) uroCamStyle = 'hidden';
  3011. }
  3012.  
  3013. if(uroGetCBChecked('_cbShowSpeedCams') === true)
  3014. {
  3015. if((uroGetCBChecked('_cbShowIfNoSpeedSet') === false) && (uroCam.attributes.speed === null)) uroCamStyle = 'hidden';
  3016. if((uroGetCBChecked('_cbShowIfSpeedSet') === false) && (uroCam.attributes.speed !== null)) uroCamStyle = 'hidden';
  3017. }
  3018.  
  3019. if(uroGetCBChecked('_cbHideCreatedByMe') === true)
  3020. {
  3021. if(uroUserID == uroCam.attributes.createdBy) uroCamStyle = 'hidden';
  3022. }
  3023. if((uroGetCBChecked('_cbHideCreatedByRank0') === true) && (uroCamCreatorRank === 0)) uroCamStyle = 'hidden';
  3024. if((uroGetCBChecked('_cbHideCreatedByRank1') === true) && (uroCamCreatorRank == 1)) uroCamStyle = 'hidden';
  3025. if((uroGetCBChecked('_cbHideCreatedByRank2') === true) && (uroCamCreatorRank == 2)) uroCamStyle = 'hidden';
  3026. if((uroGetCBChecked('_cbHideCreatedByRank3') === true) && (uroCamCreatorRank == 3)) uroCamStyle = 'hidden';
  3027. if((uroGetCBChecked('_cbHideCreatedByRank4') === true) && (uroCamCreatorRank == 4)) uroCamStyle = 'hidden';
  3028. if((uroGetCBChecked('_cbHideCreatedByRank5') === true) && (uroCamCreatorRank == 5)) uroCamStyle = 'hidden';
  3029.  
  3030. if(uroGetCBChecked('_cbHideUpdatedByMe') === true)
  3031. {
  3032. if(uroUserID == uroCam.attributes.updatedBy) uroCamStyle = 'hidden';
  3033. }
  3034. if((uroGetCBChecked('_cbHideUpdatedByRank0') === true) && (uroCamUpdaterRank === 0)) uroCamStyle = 'hidden';
  3035. if((uroGetCBChecked('_cbHideUpdatedByRank1') === true) && (uroCamUpdaterRank == 1)) uroCamStyle = 'hidden';
  3036. if((uroGetCBChecked('_cbHideUpdatedByRank2') === true) && (uroCamUpdaterRank == 2)) uroCamStyle = 'hidden';
  3037. if((uroGetCBChecked('_cbHideUpdatedByRank3') === true) && (uroCamUpdaterRank == 3)) uroCamStyle = 'hidden';
  3038. if((uroGetCBChecked('_cbHideUpdatedByRank4') === true) && (uroCamUpdaterRank == 4)) uroCamStyle = 'hidden';
  3039. if((uroGetCBChecked('_cbHideUpdatedByRank5') === true) && (uroCamUpdaterRank == 5)) uroCamStyle = 'hidden';
  3040.  
  3041. if((uroGetCBChecked('_cbHideCWLCams') === true) && (uroIsCamOnWatchList(uroCam.attributes.id) != -1)) uroCamStyle = 'hidden';
  3042.  
  3043. var uroCamGeometryID = uroCam.geometry.id;
  3044. if(camLayer.getElementById(uroCamGeometryID) !== null)
  3045. {
  3046. if(uroCamStyle == "hidden")
  3047. {
  3048. camLayer.getElementById(uroCamGeometryID).remove();
  3049. }
  3050. }
  3051. }
  3052. }
  3053. }
  3054. }
  3055.  
  3056. function uroFilterURs_onObjectsChanged()
  3057. {
  3058. if(uroBackfilling === false)
  3059. {
  3060. if(uroURDialogIsOpen === false)
  3061. {
  3062. uroURBackfill();
  3063. }
  3064. else
  3065. {
  3066. uroFilterURs();
  3067. }
  3068. }
  3069. }
  3070. function uroFilterURs_onObjectsAdded()
  3071. {
  3072. if(uroBackfilling === false)
  3073. {
  3074. uroURBackfill();
  3075. }
  3076. }
  3077. function uroFilterURs_onObjectsRemoved()
  3078. {
  3079. if(uroBackfilling === false)
  3080. {
  3081. uroURBackfill();
  3082. }
  3083. }
  3084.  
  3085.  
  3086. function uroBackfillQueueObj(lon, lat, blockSize)
  3087. {
  3088. this.lon = lon;
  3089. this.lat = lat;
  3090. this.blockSize = blockSize;
  3091. }
  3092.  
  3093. function uroURBackfill_GetData()
  3094. {
  3095. if(uroBackfillQueue.length === 0)
  3096. {
  3097. uroBackfilling = false;
  3098. uroFilterURs();
  3099. return;
  3100. }
  3101. var nextBFQueueObj = uroBackfillQueue.shift();
  3102. var lon = parseFloat(nextBFQueueObj.lon);
  3103. var lat = parseFloat(nextBFQueueObj.lat);
  3104. var blockSize = parseFloat(nextBFQueueObj.blockSize);
  3105. uroAddLog('Backfill square '+lon+','+lat);
  3106. var backfillReq = new XMLHttpRequest();
  3107. backfillReq.onreadystatechange = function ()
  3108. {
  3109. if (backfillReq.readyState == 4)
  3110. {
  3111. uroAddLog('backfill data request, response '+backfillReq.status+' received');
  3112. if (backfillReq.status == 200)
  3113. {
  3114. var tResp = JSON.parse(backfillReq.responseText);
  3115. var urCount = tResp.mapUpdateRequests.objects.length;
  3116.  
  3117. uroAddLog(urCount+' URs loaded for backfill processing');
  3118. if(urCount == 500)
  3119. {
  3120. uroAddLog('WARNING - backfill data may have been pre-filtered by server');
  3121. }
  3122.  
  3123. var backfilled = 0;
  3124. for(var i=0; i<urCount; i++)
  3125. {
  3126. var urID = tResp.mapUpdateRequests.objects[i].id;
  3127. if(W.model.mapUpdateRequests.objects[urID] === undefined)
  3128. {
  3129. var newUR = require('Waze/Feature/Vector/UpdateRequest');
  3130. var tUR = new newUR(tResp.mapUpdateRequests.objects[i]);
  3131. var tPoint = new OpenLayers.Geometry.Point();
  3132. tPoint.x = tResp.mapUpdateRequests.objects[i].geometry.coordinates[0];
  3133. tPoint.y = tResp.mapUpdateRequests.objects[i].geometry.coordinates[1];
  3134. tPoint.transform(new OpenLayers.Projection("EPSG:4326"),new OpenLayers.Projection("EPSG:900913"));
  3135. tUR.geometry = tPoint;
  3136. var tReqBounds = new OpenLayers.Geometry.Polygon();
  3137. var tBounds = new OpenLayers.Bounds();
  3138. tBounds.left = tPoint.x;
  3139. tBounds.right = tPoint.x;
  3140. tBounds.top = tPoint.y;
  3141. tBounds.bottom = tPoint.y;
  3142. tReqBounds.bounds = tBounds;
  3143. tUR.requestBounds = tReqBounds;
  3144. W.model.mapUpdateRequests.put(tUR);
  3145. backfilled++;
  3146. }
  3147. }
  3148. uroAddLog(backfilled+' URs backfilled');
  3149. }
  3150. uroURBackfill_GetData();
  3151. }
  3152. };
  3153. var tURL = 'https://' + document.location.host;
  3154. tURL += Waze.Config.api_base;
  3155. tURL += '/Features?language=en&mapUpdateRequestFilter=0';
  3156. tURL += '&bbox='+(lon)+','+(lat)+','+(lon + blockSize)+','+(lat + blockSize);
  3157. backfillReq.open('GET',tURL,true);
  3158. backfillReq.send();
  3159. }
  3160.  
  3161. function uroURBackfill()
  3162. {
  3163. if((uroGetCBChecked('_cbURBackfill') === false) || (uroGetCBChecked('_cbMasterEnable') === false))
  3164. {
  3165. uroFilterURs();
  3166. return;
  3167. }
  3168.  
  3169. var nativeURCount = Object.keys(W.model.mapUpdateRequests.objects).length;
  3170. if(nativeURCount < 500)
  3171. {
  3172. uroAddLog(nativeURCount+' URs loaded natively, no backfilling required');
  3173. uroFilterURs();
  3174. return;
  3175. }
  3176.  
  3177. uroAddLog('exactly 500 URs loaded, possible server-side filtering requiring backfill...');
  3178.  
  3179. var subSize = 0.1;
  3180. var vpWidth = W.map.getExtent().getWidth();
  3181. var vpHeight = W.map.getExtent().getHeight();
  3182. var vpCentre = W.map.getCenter();
  3183. var vpLL = new OpenLayers.LonLat();
  3184. var vpUR = new OpenLayers.LonLat();
  3185. vpLL.lon = vpCentre.lon - (vpWidth / 2);
  3186. vpLL.lat = vpCentre.lat - (vpHeight / 2);
  3187. vpUR.lon = vpCentre.lon + (vpWidth / 2);
  3188. vpUR.lat = vpCentre.lat + (vpHeight / 2);
  3189. vpLL = vpLL.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  3190. vpUR = vpUR.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  3191. vpLL.lon -= (subSize / 2);
  3192. vpLL.lat -= (subSize / 2);
  3193. vpUR.lon += (subSize / 2);
  3194. vpUR.lat += (subSize / 2);
  3195. vpLL.lon = +vpLL.lon.toFixed(1);
  3196. vpLL.lat = +vpLL.lat.toFixed(1);
  3197. vpUR.lon = +vpUR.lon.toFixed(1);
  3198. vpUR.lat = +vpUR.lat.toFixed(1);
  3199.  
  3200. uroBackfilling = true;
  3201. uroBackfillQueue = [];
  3202. for(var bfLat = vpLL.lat; bfLat <= vpUR.lat; bfLat += subSize)
  3203. {
  3204. for(var bfLon = vpLL.lon; bfLon <= vpUR.lon; bfLon += subSize)
  3205. {
  3206. uroBackfillQueue.push(new uroBackfillQueueObj(bfLon, bfLat, subSize));
  3207. }
  3208. }
  3209. uroURBackfill_GetData();
  3210. }
  3211.  
  3212.  
  3213. function uroFilterURs()
  3214. {
  3215. // compatibility fix for URComments - based on code supplied by RickZabel
  3216. var hasActiveURFilters = false;
  3217. if(uroGetCBChecked('_cbMasterEnable') === true)
  3218. {
  3219. var urTabInputs = document.getElementById('uroCtrlURs').getElementsByTagName('input');
  3220. for(var loop = 0; loop < urTabInputs.length; loop++)
  3221. {
  3222. if(urTabInputs[loop].type == 'checkbox')
  3223. {
  3224. var ignoreCB = false;
  3225. ignoreCB = ignoreCB || (urTabInputs[loop].id == '_cbCaseInsensitive');
  3226. ignoreCB = ignoreCB || (urTabInputs[loop].id == '_cbNoFilterForTaggedURs');
  3227. if((urTabInputs[loop].checked) && (ignoreCB === false))
  3228. {
  3229. hasActiveURFilters = true;
  3230. break;
  3231. }
  3232. }
  3233. }
  3234. }
  3235. sessionStorage.UROverview_hasActiveURFilters = hasActiveURFilters;
  3236. if(uroFilterPreamble() === false) return;
  3237. uroRefreshUpdateRequestSessions();
  3238. var selectorResolver = document.getElementById('_selectURResolverID');
  3239. var selectorCommentUser = document.getElementById('_selectURUserID');
  3240.  
  3241. if(uroGetCBChecked('_cbURResolverIDFilter') === false)
  3242. {
  3243. while(selectorResolver.options.length > 0)
  3244. {
  3245. selectorResolver.options.remove(0);
  3246. }
  3247. }
  3248. if(uroGetCBChecked('_cbURUserIDFilter') === false)
  3249. {
  3250. while(selectorCommentUser.options.length > 0)
  3251. {
  3252. selectorCommentUser.options.remove(0);
  3253. }
  3254. }
  3255. if(Object.keys(W.model.updateRequestSessions.objects).length === 0)
  3256. {
  3257. return;
  3258. }
  3259. var commenterUser = null;
  3260. if(uroGetCBChecked('_cbURUserIDFilter') === true)
  3261. {
  3262. if(selectorCommentUser.options.length === 0)
  3263. {
  3264. uroUpdateUserList();
  3265. }
  3266. if(selectorCommentUser.selectedOptions[0] !== undefined)
  3267. {
  3268. commenterUser = parseInt(selectorCommentUser.selectedOptions[0].value);
  3269. }
  3270. }
  3271. var resolverUser = null;
  3272. if(uroGetCBChecked('_cbURResolverIDFilter') === true)
  3273. {
  3274. if(selectorResolver.options.length === 0)
  3275. {
  3276. uroUpdateResolverList();
  3277. }
  3278. if(selectorResolver.selectedOptions[0] !== undefined)
  3279. {
  3280. resolverUser = parseInt(selectorResolver.selectedOptions[0].value);
  3281. }
  3282. }
  3283. uroCustomMarkerList = [];
  3284.  
  3285. var uFURs_masterEnable = uroGetCBChecked('_cbMasterEnable');
  3286. var filterOutsideEditableArea = uroGetCBChecked('_cbURFilterOutsideArea');
  3287. var filterSolved = uroGetCBChecked('_cbFilterSolved');
  3288. var filterUnidentified = uroGetCBChecked('_cbFilterUnidentified');
  3289. var filterClosed = uroGetCBChecked('_cbFilterClosedUR');
  3290. var filterOpen = uroGetCBChecked('_cbFilterOpenUR');
  3291. var filterDescMustBePresent = uroGetCBChecked('_cbURDescriptionMustBePresent');
  3292. var filterDescMustBeAbsent = uroGetCBChecked('_cbURDescriptionMustBeAbsent');
  3293. var filterKeywordMustBePresent = uroGetCBChecked('_cbEnableKeywordMustBePresent');
  3294. var filterKeywordMustBeAbsent = uroGetCBChecked('_cbEnableKeywordMustBeAbsent');
  3295. var filterMinURAge = uroGetCBChecked('_cbEnableMinAgeFilter');
  3296. var filterMaxURAge = uroGetCBChecked('_cbEnableMaxAgeFilter');
  3297. var filterMinComments = uroGetCBChecked('_cbEnableMinCommentsFilter');
  3298. var filterMaxComments = uroGetCBChecked('_cbEnableMaxCommentsFilter');
  3299. var filterReporterLastCommenter = uroGetCBChecked('_cbHideIfReporterLastCommenter');
  3300. var filterReporterNotLastCommenter = uroGetCBChecked('_cbHideIfReporterNotLastCommenter');
  3301. var filterHideAnyComments = uroGetCBChecked('_cbHideAnyComments');
  3302. var filterHideNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
  3303. var filterHideMyComments = uroGetCBChecked('_cbHideMyComments');
  3304. var filterIfLastCommenter = uroGetCBChecked('_cbHideIfLastCommenter');
  3305. var filterIfNotLastCommenter = uroGetCBChecked('_cbHideIfNotLastCommenter');
  3306. var filterCommentMinAge = uroGetCBChecked('_cbEnableCommentAgeFilter2');
  3307. var filterCommentMaxAge = uroGetCBChecked('_cbEnableCommentAgeFilter');
  3308. var filterUserID = uroGetCBChecked('_cbURUserIDFilter');
  3309. var filterMyFollowed = uroGetCBChecked('_cbHideMyFollowed');
  3310. var filterMyUnfollowed = uroGetCBChecked('_cbHideMyUnfollowed');
  3311. var filterWazeAuto = uroGetCBChecked('_cbFilterWazeAuto');
  3312. var filterRoadworks = uroGetCBChecked('_cbFilterRoadworks');
  3313. var filterConstruction = uroGetCBChecked('_cbFilterConstruction');
  3314. var filterClosure = uroGetCBChecked('_cbFilterClosure');
  3315. var filterEvent = uroGetCBChecked('_cbFilterEvent');
  3316. var filterNote = uroGetCBChecked('_cbFilterNote');
  3317. var filterIncorrectTurn = uroGetCBChecked('_cbFilterIncorrectTurn');
  3318. var filterIncorrectAddress = uroGetCBChecked('_cbFilterIncorrectAddress');
  3319. var filterIncorrectRoute = uroGetCBChecked('_cbFilterIncorrectRoute');
  3320. var filterMissingRoundabout = uroGetCBChecked('_cbFilterMissingRoundabout');
  3321. var filterGeneralError = uroGetCBChecked('_cbFilterGeneralError');
  3322. var filterTurnNotAllowed = uroGetCBChecked('_cbFilterTurnNotAllowed');
  3323. var filterIncorrectJunction = uroGetCBChecked('_cbFilterIncorrectJunction');
  3324. var filterMissingBridgeOverpass = uroGetCBChecked('_cbFilterMissingBridgeOverpass');
  3325. var filterWrongDrivingDirection = uroGetCBChecked('_cbFilterWrongDrivingDirection');
  3326. var filterMissingExit = uroGetCBChecked('_cbFilterMissingExit');
  3327. var filterMissingRoad = uroGetCBChecked('_cbFilterMissingRoad');
  3328. var filterMissingLandmark = uroGetCBChecked('_cbFilterMissingLandmark');
  3329. var filterBlockedRoad = uroGetCBChecked('_cbFilterBlockedRoad');
  3330. var filterUndefined = uroGetCBChecked('_cbFilterUndefined');
  3331. var invertURFilters = uroGetCBChecked('_cbInvertURFilter');
  3332. var noFilterTaggedURs = uroGetCBChecked('_cbNoFilterForTaggedURs');
  3333. var noFilterURInURL = uroGetCBChecked('_cbNoFilterForURInURL');
  3334. var keywordPresent = uroGetElmValue('_textKeywordPresent');
  3335. var keywordAbsent = uroGetElmValue('_textKeywordAbsent');
  3336. var thresholdMinAge = uroGetElmValue('_inputFilterMinDays');
  3337. var thresholdMaxAge = uroGetElmValue('_inputFilterMaxDays');
  3338. var thresholdMinComments = uroGetElmValue('_inputFilterMinComments');
  3339. var thresholdMaxComments = uroGetElmValue('_inputFilterMaxComments');
  3340. var thresholdMaxCommentAge = uroGetElmValue('_inputFilterCommentDays');
  3341. var thresholdMinCommentAge = uroGetElmValue('_inputFilterCommentDays2');
  3342. var urcFilteringIsActive = false;
  3343. var urcCB = document.getElementById('URCommentsFilterEnabled');
  3344. if(urcCB !== null)
  3345. {
  3346. if(urcCB.checked)
  3347. {
  3348. urcFilteringIsActive = true;
  3349. }
  3350. }
  3351. urcCB = document.getElementById('URCommentUROOnlyMyUR');
  3352. if(urcCB !== null)
  3353. {
  3354. if(urcCB.checked)
  3355. {
  3356. urcFilteringIsActive = true;
  3357. }
  3358. }
  3359. urcCB = document.getElementById('URCommentUROHideTagged');
  3360. if(urcCB !== null)
  3361. {
  3362. if(urcCB.checked)
  3363. {
  3364. urcFilteringIsActive = true;
  3365. }
  3366. }
  3367.  
  3368. for (var urobj in W.model.mapUpdateRequests.objects)
  3369. {
  3370. if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
  3371. {
  3372. var ureq = W.model.mapUpdateRequests.objects[urobj];
  3373. var ureqID = null;
  3374. if(ureq.fid === null) ureqID = ureq.attributes.id;
  3375. else ureqID = ureq.fid;
  3376. var urStyle = 'visible';
  3377. var inhibitFiltering = ((ureqID == uroURIDInURL) && (noFilterURInURL));
  3378. var hasMyComments = false;
  3379. var nComments = 0;
  3380. var customType = uroGetCustomType(ureqID, "ur");
  3381. if(W.model.updateRequestSessions.objects[ureqID] !== undefined)
  3382. {
  3383. nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
  3384. if((uFURs_masterEnable === false) && (nComments === 0))
  3385. {
  3386. // when master enable is turned off, we want to make sure that all URs, including ones that were previously hidden, are correctly
  3387. // displayed in their native form - i.e. no comment count or custom conversation bubbles. The easiest way to achieve this is to
  3388. // force the uroRenderCustomMarkers code to test for the presence of these bubbles on each UR, which we do by setting a non-zero
  3389. // comment count for each UR... For URs which genuinely do have no comments we use -1 to indicate that we're not really setting
  3390. // a comment count, but that we still need to do something that wouldn't be achieved by using 0.
  3391. nComments = -1;
  3392. }
  3393. }
  3394. if((uFURs_masterEnable === true) && (inhibitFiltering === false))
  3395. {
  3396. var wazeauto_ur = false;
  3397. var ukroadworks_ur = false;
  3398. var construction_ur = false;
  3399. var closure_ur = false;
  3400. var event_ur = false;
  3401. var note_ur = false;
  3402.  
  3403. var filterByNotIncludedKeyword = false;
  3404. var filterByIncludedKeyword = true;
  3405.  
  3406. var desc = '';
  3407. if(ureq.attributes.description !== null) desc = ureq.attributes.description.replace(/<\/?[^>]+(>|$)/g, "");
  3408.  
  3409. if(customType === 0) ukroadworks_ur = true;
  3410. else if(customType === 1) construction_ur = true;
  3411. else if(customType === 2) closure_ur = true;
  3412. else if(customType === 3) event_ur = true;
  3413. else if(customType === 4) note_ur = true;
  3414.  
  3415. // check UR against editable area...
  3416.  
  3417. if(filterOutsideEditableArea === true)
  3418. {
  3419. if(ureq.canEdit() === false) urStyle = 'hidden';
  3420. }
  3421. // check UR against current session ignore list...
  3422. if(uroIsOnIgnoreList(ureqID)) urStyle = 'hidden';
  3423.  
  3424. // check against closed/not identified filtering if enabled...
  3425. if(filterSolved === true)
  3426. {
  3427. if(ureq.attributes.resolution === 0) urStyle = 'hidden';
  3428. }
  3429. if(filterUnidentified === true)
  3430. {
  3431. if(ureq.attributes.resolution == 1) urStyle = 'hidden';
  3432. }
  3433.  
  3434. if((ureq.attributes.resolvedOn !== null) && (filterClosed === true))
  3435. {
  3436. urStyle = 'hidden';
  3437. }
  3438.  
  3439. if((ureq.attributes.resolvedOn === null) && (filterOpen === true))
  3440. {
  3441. urStyle = 'hidden';
  3442. }
  3443.  
  3444. if(urStyle == 'visible')
  3445. {
  3446. // check UR against keyword filtering if enabled...
  3447. if(filterDescMustBePresent === true)
  3448. {
  3449. if(desc === '') urStyle = 'hidden';
  3450. }
  3451. if(filterDescMustBeAbsent === true)
  3452. {
  3453. if(desc !== '') urStyle = 'hidden';
  3454. }
  3455.  
  3456. if(filterKeywordMustBePresent === true)
  3457. {
  3458. var keywordIsPresentInDesc = uroKeywordPresent(desc,keywordPresent);
  3459. filterByIncludedKeyword &= (!keywordIsPresentInDesc);
  3460. }
  3461. if(filterKeywordMustBeAbsent === true)
  3462. {
  3463. var keywordIsAbsentInDesc = uroKeywordPresent(desc,keywordAbsent);
  3464. filterByNotIncludedKeyword |= keywordIsAbsentInDesc;
  3465. }
  3466. }
  3467.  
  3468. if(urStyle == 'visible')
  3469. {
  3470. // do age-based filtering if enabled
  3471. if(filterMinURAge === true)
  3472. {
  3473. if(uroGetURAge(ureq,0,false) < thresholdMinAge) urStyle = 'hidden';
  3474. }
  3475. if(filterMaxURAge === true)
  3476. {
  3477. if(uroGetURAge(ureq,0,false) > thresholdMaxAge) urStyle = 'hidden';
  3478. }
  3479. }
  3480.  
  3481. if(urStyle == 'visible')
  3482. {
  3483. if(resolverUser !== null)
  3484. {
  3485. if(ureq.attributes.resolvedBy != resolverUser) urStyle = 'hidden';
  3486. }
  3487. }
  3488.  
  3489. if(urStyle == 'visible')
  3490. {
  3491. // do comments/following filtering
  3492. if(W.model.updateRequestSessions.objects[ureqID] !== undefined)
  3493. {
  3494. nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
  3495. var commentDaysOld = -1;
  3496.  
  3497.  
  3498. if(filterMinComments === true)
  3499. {
  3500. if(nComments < thresholdMinComments) urStyle = 'hidden';
  3501. }
  3502. if(filterMaxComments === true)
  3503. {
  3504. if(nComments > thresholdMaxComments) urStyle = 'hidden';
  3505. }
  3506.  
  3507.  
  3508. if(nComments > 0)
  3509. {
  3510. var reporterIsLastCommenter = false;
  3511. if(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1].userID == -1) reporterIsLastCommenter = true;
  3512.  
  3513. if(filterReporterLastCommenter === true)
  3514. {
  3515. if(reporterIsLastCommenter === true) urStyle = 'hidden';
  3516. }
  3517. else if(filterReporterNotLastCommenter === true)
  3518. {
  3519. if(reporterIsLastCommenter === false) urStyle = 'hidden';
  3520. }
  3521.  
  3522. hasMyComments = uroURHasMyComments(ureqID);
  3523. if(hasMyComments === false)
  3524. {
  3525. if(filterHideAnyComments === true) urStyle = 'hidden';
  3526. if(filterHideNotLastCommenter === true) urStyle = 'hidden';
  3527. }
  3528. else
  3529. {
  3530. if(filterHideMyComments === true) urStyle = 'hidden';
  3531.  
  3532. var userIsLastCommenter = false;
  3533. if(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1].userID == uroUserID) userIsLastCommenter = true;
  3534.  
  3535. if(filterIfLastCommenter === true)
  3536. {
  3537. if(userIsLastCommenter === true) urStyle = 'hidden';
  3538. }
  3539. else if(filterIfNotLastCommenter === true)
  3540. {
  3541. if(userIsLastCommenter === false) urStyle = 'hidden';
  3542. }
  3543. }
  3544. commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1]);
  3545. if((filterCommentMinAge === true) && (commentDaysOld != -1))
  3546. {
  3547. if(thresholdMinCommentAge > commentDaysOld) urStyle = 'hidden';
  3548. }
  3549. if((filterCommentMaxAge === true) && (commentDaysOld != -1))
  3550. {
  3551. if(thresholdMaxCommentAge < commentDaysOld) urStyle = 'hidden';
  3552. }
  3553.  
  3554. var cidx;
  3555. if((commenterUser !== null) && (urStyle != 'hidden'))
  3556. {
  3557. urStyle = 'hidden';
  3558. for(cidx=0; cidx<nComments; cidx++)
  3559. {
  3560. if(W.model.updateRequestSessions.objects[ureqID].comments[cidx].userID == commenterUser)
  3561. {
  3562. urStyle = 'visible';
  3563. break;
  3564. }
  3565. }
  3566. }
  3567.  
  3568. var commentText = '';
  3569. for(cidx=0; cidx<nComments; cidx++)
  3570. {
  3571. commentText += W.model.updateRequestSessions.objects[ureqID].comments[cidx].text;
  3572. }
  3573.  
  3574. if(filterKeywordMustBePresent === true)
  3575. {
  3576. var keywordIsPresentInComments = uroKeywordPresent(commentText,keywordPresent);
  3577. filterByIncludedKeyword &= (!keywordIsPresentInComments);
  3578. }
  3579. if(filterKeywordMustBeAbsent === true)
  3580. {
  3581. var keywordIsAbsentInComments = uroKeywordPresent(commentText,keywordAbsent);
  3582. filterByNotIncludedKeyword |= keywordIsAbsentInComments;
  3583. }
  3584. }
  3585. else
  3586. {
  3587. if(filterUserID === true)
  3588. {
  3589. urStyle = 'hidden';
  3590. }
  3591. }
  3592.  
  3593. filterByNotIncludedKeyword &= filterKeywordMustBeAbsent;
  3594. filterByIncludedKeyword &= filterKeywordMustBePresent;
  3595. if(filterByNotIncludedKeyword || filterByIncludedKeyword)
  3596. {
  3597. urStyle = 'hidden';
  3598. }
  3599.  
  3600.  
  3601. if(W.model.updateRequestSessions.objects[ureqID].isFollowing === true)
  3602. {
  3603. if(filterMyFollowed === true) urStyle = 'hidden';
  3604. }
  3605. else
  3606. {
  3607. if(filterMyUnfollowed === true) urStyle = 'hidden';
  3608. }
  3609. }
  3610. }
  3611.  
  3612. if(urStyle == 'visible')
  3613. {
  3614. // Test for Waze automatic URs before any others - these always (?) get inserted as General Error URs,
  3615. // so we can't filter them by type...
  3616. if(desc.indexOf('Waze Automatic:') != -1)
  3617. {
  3618. wazeauto_ur = true;
  3619. }
  3620.  
  3621. if(wazeauto_ur === true)
  3622. {
  3623. if(filterWazeAuto === true) urStyle = 'hidden';
  3624. }
  3625.  
  3626. else if(ukroadworks_ur === true)
  3627. {
  3628. if(filterRoadworks === true) urStyle = 'hidden';
  3629. }
  3630. else if(construction_ur === true)
  3631. {
  3632. if(filterConstruction === true) urStyle = 'hidden';
  3633. }
  3634. else if(closure_ur === true)
  3635. {
  3636. if(filterClosure === true) urStyle = 'hidden';
  3637. }
  3638. else if(event_ur === true)
  3639. {
  3640. if(filterEvent === true) urStyle = 'hidden';
  3641. }
  3642. else if(note_ur === true)
  3643. {
  3644. if(filterNote === true) urStyle = 'hidden';
  3645. }
  3646.  
  3647. else if(ureq.attributes.type == 6)
  3648. {
  3649. if(filterIncorrectTurn === true) urStyle = 'hidden';
  3650. }
  3651. else if(ureq.attributes.type == 7)
  3652. {
  3653. if (filterIncorrectAddress === true) urStyle = 'hidden';
  3654. }
  3655. else if(ureq.attributes.type == 8)
  3656. {
  3657. if(filterIncorrectRoute === true) urStyle = 'hidden';
  3658. }
  3659. else if(ureq.attributes.type == 9)
  3660. {
  3661. if(filterMissingRoundabout === true) urStyle = 'hidden';
  3662. }
  3663. else if(ureq.attributes.type == 10)
  3664. {
  3665. if(filterGeneralError === true) urStyle = 'hidden';
  3666. }
  3667. else if(ureq.attributes.type == 11)
  3668. {
  3669. if(filterTurnNotAllowed === true) urStyle = 'hidden';
  3670. }
  3671. else if(ureq.attributes.type == 12)
  3672. {
  3673. if(filterIncorrectJunction === true) urStyle = 'hidden';
  3674. }
  3675. else if(ureq.attributes.type == 13)
  3676. {
  3677. if(filterMissingBridgeOverpass === true) urStyle = 'hidden';
  3678. }
  3679. else if(ureq.attributes.type == 14)
  3680. {
  3681. if(filterWrongDrivingDirection === true) urStyle = 'hidden';
  3682. }
  3683. else if(ureq.attributes.type == 15)
  3684. {
  3685. if(filterMissingExit === true) urStyle = 'hidden';
  3686. }
  3687. else if(ureq.attributes.type == 16)
  3688. {
  3689. if(filterMissingRoad === true) urStyle = 'hidden';
  3690. }
  3691. else if(ureq.attributes.type == 18)
  3692. {
  3693. if(filterMissingLandmark === true) urStyle = 'hidden';
  3694. }
  3695. else if(ureq.attributes.type == 19)
  3696. {
  3697. if(filterBlockedRoad === true) urStyle = 'hidden';
  3698. }
  3699. else if(filterUndefined === true) urStyle = 'hidden';
  3700.  
  3701. if(invertURFilters === true)
  3702. {
  3703. if(urStyle == 'hidden') urStyle = 'visible';
  3704. else urStyle = 'hidden';
  3705. }
  3706. }
  3707.  
  3708. // filtering override for tagged URs
  3709. if(noFilterTaggedURs === true)
  3710. {
  3711. if(ukroadworks_ur === true)
  3712. {
  3713. if(filterRoadworks === false) urStyle = 'visible';
  3714. }
  3715. else if(construction_ur === true)
  3716. {
  3717. if(filterConstruction === false) urStyle = 'visible';
  3718. }
  3719. else if(closure_ur === true)
  3720. {
  3721. if(filterClosure === false) urStyle = 'visible';
  3722. }
  3723. else if(event_ur === true)
  3724. {
  3725. if(filterEvent === false) urStyle = 'visible';
  3726. }
  3727. else if(note_ur === true)
  3728. {
  3729. if(filterNote === false) urStyle = 'visible';
  3730. }
  3731. }
  3732. }
  3733. // only touch marker visibility if we've got active filter settings, or if URComments is not
  3734. // doing any filtering of its own
  3735. if((hasActiveURFilters === true) || (urcFilteringIsActive === false) || (uFURs_masterEnable === false))
  3736. {
  3737. W.map.updateRequestLayer.markers[urobj].icon.imageDiv.style.visibility = urStyle;
  3738. }
  3739. if(urStyle != 'hidden')
  3740. {
  3741. uroAddCustomMarkers(ureqID,customType, hasMyComments, nComments);
  3742. }
  3743. }
  3744. }
  3745. uroRenderCustomMarkers('ur');
  3746. }
  3747.  
  3748.  
  3749. function uroFilterProblems()
  3750. {
  3751. if(uroFilterPreamble() === false) return;
  3752. var selector;
  3753.  
  3754. if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === false) && (uroGetCBChecked('_cbMPClosedUserIDFilter') === false))
  3755. {
  3756. selector = document.getElementById('_selectMPUserID');
  3757. while(selector.options.length > 0)
  3758. {
  3759. selector.options.remove(0);
  3760. }
  3761. }
  3762.  
  3763. var solverUser = null;
  3764. if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === true) || (uroGetCBChecked('_cbMPClosedUserIDFilter') === true))
  3765. {
  3766. selector = document.getElementById('_selectMPUserID');
  3767. if(selector.options.length === 0)
  3768. {
  3769. uroUpdateMPSolverList();
  3770. }
  3771. if(selector.selectedOptions[0] !== undefined)
  3772. {
  3773. solverUser = parseInt(selector.selectedOptions[0].value);
  3774. }
  3775. }
  3776.  
  3777. var urobj;
  3778. var problem;
  3779. var problemStyle;
  3780. var problem_marker_img;
  3781.  
  3782. for (urobj in W.model.problems.objects)
  3783. {
  3784. if(W.model.problems.objects.hasOwnProperty(urobj))
  3785. {
  3786. problem = W.model.problems.objects[urobj];
  3787. problemStyle = 'visible';
  3788. var ureqID = null;
  3789. var customType = null;
  3790.  
  3791. if(uroGetCBChecked('_cbMasterEnable') === true)
  3792. {
  3793. ureqID = problem.attributes.id;
  3794. customType = uroGetCustomType(ureqID, "mp");
  3795.  
  3796. // check problem against current session ignore list...
  3797. if(uroIsOnIgnoreList(ureqID)) problemStyle = 'hidden';
  3798.  
  3799.  
  3800. if(uroGetCBChecked('_cbMPFilterOutsideArea') === true)
  3801. {
  3802. if(problem.canEdit() === false)
  3803. {
  3804. problemStyle = 'hidden';
  3805. }
  3806. }
  3807.  
  3808. // check against closed/not identified filtering if enabled...
  3809. problem_marker_img = '';
  3810. if(problem.geometry.id !== null)
  3811. {
  3812. if(document.getElementById(problem.geometry.id) !== null)
  3813. {
  3814. problem_marker_img = document.getElementById(problem.geometry.id).href.baseVal;
  3815. if(uroGetCBChecked('_cbMPFilterSolved') === true)
  3816. {
  3817. if(problem_marker_img.indexOf('_solved') != -1) problemStyle = 'hidden';
  3818. }
  3819. if(uroGetCBChecked('_cbMPFilterUnidentified') === true)
  3820. {
  3821. if(problem_marker_img.indexOf('_rejected') != -1) problemStyle = 'hidden';
  3822. }
  3823. }
  3824. }
  3825.  
  3826. if(uroGetCBChecked('_cbMPFilterClosed') === true)
  3827. {
  3828. if(problem.attributes.open === false)
  3829. {
  3830. problemStyle = 'hidden';
  3831. }
  3832. }
  3833.  
  3834. if(problemStyle == 'visible')
  3835. {
  3836. if(solverUser !== null)
  3837. {
  3838. if((uroGetCBChecked('_cbMPNotClosedUserIDFilter') === true) && (problem.attributes.resolvedBy == solverUser)) problemStyle = 'hidden';
  3839. if((uroGetCBChecked('_cbMPClosedUserIDFilter') === true) && (problem.attributes.resolvedBy != solverUser)) problemStyle = 'hidden';
  3840. }
  3841. }
  3842.  
  3843. if(problemStyle == 'visible')
  3844. {
  3845. var problemType = null;
  3846. if(uroDOMHasTurnProblems)
  3847. {
  3848. problemType = problem.attributes.problemType;
  3849. }
  3850. else
  3851. {
  3852. problemType = problem.attributes.subType;
  3853. }
  3854.  
  3855. if(problemType == 101)
  3856. {
  3857. if(uroGetCBChecked('_cbMPFilterDrivingDirectionMismatch') === true) problemStyle = 'hidden';
  3858. }
  3859. else if(problemType == 102)
  3860. {
  3861. if(uroGetCBChecked('_cbMPFilterMissingJunction') === true) problemStyle = 'hidden';
  3862. }
  3863. else if(problemType == 103)
  3864. {
  3865. if(uroGetCBChecked('_cbMPFilterMissingRoad') === true) problemStyle = 'hidden';
  3866. }
  3867. else if(problemType == 104)
  3868. {
  3869. if(uroGetCBChecked('_cbMPFilterCrossroadsJunctionMissing') === true) problemStyle = 'hidden';
  3870. }
  3871. else if(problemType == 105)
  3872. {
  3873. if(uroGetCBChecked('_cbMPFilterRoadTypeMismatch') === true) problemStyle = 'hidden';
  3874. }
  3875. else if(problemType == 106)
  3876. {
  3877. if(uroGetCBChecked('_cbMPFilterRestrictedTurn') === true) problemStyle = 'hidden';
  3878. }
  3879. else if(problemType == 200)
  3880. {
  3881. if(uroGetCBChecked('_cbMPFilterTurnProblem') === true) problemStyle = 'hidden';
  3882. }
  3883. else if(problemType == 300)
  3884. {
  3885. if(uroGetCBChecked('_cbMPFilterRoadClosureProblem') === true) problemStyle = 'hidden';
  3886. }
  3887. else if(uroGetCBChecked('_cbMPFilterUnknownProblem') === true) problemStyle = 'hidden';
  3888.  
  3889.  
  3890. if(uroGetCBChecked('_cbMPFilterReopenedProblem') === true)
  3891. {
  3892. if((problem.attributes.open === true) && (problem.attributes.resolvedOn !== null))
  3893. {
  3894. problemStyle = 'hidden';
  3895. }
  3896. }
  3897.  
  3898.  
  3899. if(uroGetCBChecked('_cbInvertMPFilter') === true)
  3900. {
  3901. if(problemStyle == 'hidden') problemStyle = 'visible';
  3902. else problemStyle = 'hidden';
  3903. }
  3904.  
  3905.  
  3906. if(problem.attributes.weight <= 3)
  3907. {
  3908. if(uroGetCBChecked('_cbMPFilterLowSeverity') === true) problemStyle = 'hidden';
  3909. }
  3910. else if(problem.attributes.weight <= 7)
  3911. {
  3912. if(uroGetCBChecked('_cbMPFilterMediumSeverity') === true) problemStyle = 'hidden';
  3913. }
  3914. else if(uroGetCBChecked('_cbMPFilterHighSeverity') === true) problemStyle = 'hidden';
  3915. }
  3916. }
  3917.  
  3918. W.map.problemLayer.markers[urobj].icon.imageDiv.style.visibility = problemStyle;
  3919.  
  3920. if((problemStyle != 'hidden') && (ureqID !== null) && (customType !== null))
  3921. {
  3922. uroAddCustomMarkers(ureqID,customType, false, 0);
  3923. }
  3924. }
  3925. }
  3926.  
  3927. if(uroDOMHasTurnProblems)
  3928. {
  3929. for (urobj in W.model.turnProblems.objects)
  3930. {
  3931. if(W.model.turnProblems.objects.hasOwnProperty(urobj))
  3932. {
  3933. problem = W.model.turnProblems.objects[urobj];
  3934. problemStyle = 'visible';
  3935.  
  3936. if(uroGetCBChecked('_cbMasterEnable') === true)
  3937. {
  3938. // check problem against current session ignore list...
  3939. if(uroIsOnIgnoreList(problem.attributes.id)) problemStyle = 'hidden';
  3940.  
  3941. // check against closed/not identified filtering if enabled...
  3942. problem_marker_img = '';
  3943. if(problem.geometry.id !== null)
  3944. {
  3945. if(document.getElementById(problem.geometry.id) !== null)
  3946. {
  3947. problem_marker_img = document.getElementById(problem.geometry.id).href.baseVal;
  3948. if(uroGetCBChecked('_cbMPFilterSolved') === true)
  3949. {
  3950. if(problem_marker_img.indexOf('_solved') != -1) problemStyle = 'hidden';
  3951. }
  3952. if(uroGetCBChecked('_cbMPFilterUnidentified') === true)
  3953. {
  3954. if(problem_marker_img.indexOf('_rejected') != -1) problemStyle = 'hidden';
  3955. }
  3956. }
  3957. }
  3958.  
  3959. if(uroGetCBChecked('_cbMPFilterClosed') === true)
  3960. {
  3961. if(problem.attributes.open === false)
  3962. {
  3963. problemStyle = 'hidden';
  3964. }
  3965. }
  3966.  
  3967. if(problemStyle == 'visible')
  3968. {
  3969. if(uroGetCBChecked('_cbMPFilterTurnProblem') === true) problemStyle = 'hidden';
  3970.  
  3971. if(uroGetCBChecked('_cbMPFilterReopenedProblem') === true)
  3972. {
  3973. if((problem.attributes.open === true) && (problem.attributes.resolvedOn !== null))
  3974. {
  3975. problemStyle = 'hidden';
  3976. }
  3977. }
  3978.  
  3979. if(uroGetCBChecked('_cbInvertMPFilter') === true)
  3980. {
  3981. if(problemStyle == 'hidden') problemStyle = 'visible';
  3982. else problemStyle = 'hidden';
  3983. }
  3984. }
  3985. }
  3986. W.map.problemLayer.markers[urobj].icon.imageDiv.style.visibility = problemStyle;
  3987. }
  3988. }
  3989. }
  3990. uroRenderCustomMarkers('mp');
  3991. }
  3992.  
  3993. function uroToHex(decValue,digits)
  3994. {
  3995. var modifier = 1;
  3996. for(var i=0; i<digits; i++)
  3997. {
  3998. modifier *= 16;
  3999. }
  4000. decValue = parseInt(decValue);
  4001. decValue += modifier;
  4002. var retval = decValue.toString(16);
  4003. retval = retval.substr(-digits);
  4004. retval = retval.toUpperCase();
  4005. return retval;
  4006. }
  4007.  
  4008. function uroFilterPreamble()
  4009. {
  4010. var mapviewport = document.getElementsByClassName("olMapViewport")[0];
  4011. if(mapviewport === null)
  4012. {
  4013. if(uroNullMapViewport === false)
  4014. {
  4015. uroAddLog('caught null mapviewport');
  4016. uroNullMapViewport = true;
  4017. }
  4018. return false;
  4019. }
  4020. uroNullMapViewport = false;
  4021.  
  4022. if((uroGetCBChecked('_cbWhiteBackground') === true) && (uroGetCBChecked('_cbMasterEnable') === true))
  4023. {
  4024. var customColour = '#' + uroToHex(uroGetElmValue('_inputCustomBackgroundRed'),2);
  4025. customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundGreen'),2);
  4026. customColour += uroToHex(uroGetElmValue('_inputCustomBackgroundBlue'),2);
  4027. mapviewport.style.backgroundColor = customColour;
  4028. }
  4029. else
  4030. {
  4031. mapviewport.style.backgroundColor = "#C2C2C2";
  4032. }
  4033.  
  4034. if((uroGetCBChecked('_cbHideAMLayer')) && (uroGetCBChecked('_cbMasterEnable') === true))
  4035. {
  4036. W.map.managedAreasLayer.setOpacity(0);
  4037. }
  4038. else
  4039. {
  4040. W.map.managedAreasLayer.setOpacity(1);
  4041. }
  4042.  
  4043. return true;
  4044. }
  4045.  
  4046. function uroFilterItems_URTabClick()
  4047. {
  4048. uroFilterURs();
  4049. }
  4050. function uroFilterItems_MPTabClick()
  4051. {
  4052. uroFilterProblems();
  4053. }
  4054. function uroFilterItems_PlacesTabClick()
  4055. {
  4056. uroFilterPlaces();
  4057. }
  4058. function uroFilterItems_CamerasTabClick()
  4059. {
  4060. uroFilterCameras();
  4061. }
  4062. function uroFilterItems_MiscTabClick()
  4063. {
  4064. uroFilterItems();
  4065. }
  4066. function uroFilterItems_MasterEnableClick()
  4067. {
  4068. if(uroGetCBChecked('_cbMasterEnable') === false)
  4069. {
  4070. uroHidePopup();
  4071. }
  4072. uroFilterItems();
  4073. }
  4074.  
  4075. function uroScaleTheScaleBar()
  4076. {
  4077. // adjust the scale bar to more accurately reflect true distances at all latitudes
  4078. var currLat = W.map.getCenter().transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326")).lat;
  4079.  
  4080. if((currLat < 85) && (currLat > -85))
  4081. {
  4082. var cosLat = Math.cos((currLat * Math.PI) / 180);
  4083. var scaleElm;
  4084. var elmWidth;
  4085. scaleElm = document.getElementsByClassName('olControlScaleLineTop')[0];
  4086. if(scaleElm.innerHTML.indexOf(' ') !== -1)
  4087. {
  4088. elmWidth = Math.round((scaleElm.clientWidth + 2) / cosLat);
  4089. scaleElm.innerHTML = scaleElm.innerHTML.replace(' ','');
  4090. scaleElm.style.width = elmWidth + 'px';
  4091. }
  4092. scaleElm = document.getElementsByClassName('olControlScaleLineBottom')[0];
  4093. if(scaleElm.innerHTML.indexOf(' ') !== -1)
  4094. {
  4095. elmWidth = Math.round((scaleElm.clientWidth + 2) / cosLat);
  4096. scaleElm.innerHTML = scaleElm.innerHTML.replace(' ','');
  4097. scaleElm.style.width = elmWidth + 'px';
  4098. }
  4099. }
  4100. }
  4101.  
  4102. function uroFilterItems()
  4103. {
  4104. uroScaleTheScaleBar();
  4105. uroFilterProblems();
  4106. uroFilterPlaces();
  4107. uroFilterCameras();
  4108. uroFilterURs();
  4109. }
  4110.  
  4111. function uroFilterItemsOnMove()
  4112. {
  4113. W.map.events.unregister('mousemove',null,uroFilterItemsOnMove);
  4114. uroFilterItems();
  4115. }
  4116.  
  4117.  
  4118. function uroDeleteObject()
  4119. {
  4120. uroAddLog('delete camera ID '+uroShownFID);
  4121. if(W.model.cameras.objects[uroShownFID] === null)
  4122. {
  4123. uroAddLog('camera object not found...');
  4124. return false;
  4125. }
  4126. uroRemoveCamFromWatchList();
  4127. var actionObj = require('Waze/Action/DeleteObject');
  4128. var deleteAction = new actionObj(W.model.cameras.objects[uroShownFID], null);
  4129. W.model.actionManager.add(deleteAction);
  4130. uroExitPopup();
  4131. uroHidePopup();
  4132. return false;
  4133. }
  4134.  
  4135.  
  4136. function uroGetUserNameAndRank(userID)
  4137. {
  4138. var userName;
  4139. var userLevel;
  4140. if(W.model.users.objects[userID] !== undefined)
  4141. {
  4142. userName = W.model.users.objects[userID].userName;
  4143. if(userName === undefined)
  4144. {
  4145. userName = userID;
  4146. }
  4147. userLevel = W.model.users.objects[userID].rank + 1;
  4148. }
  4149. else
  4150. {
  4151. userName = userID;
  4152. userLevel = '?';
  4153. }
  4154. return userName + ' (' + userLevel + ')';
  4155. }
  4156.  
  4157.  
  4158. function uroCheckCommentsForKeyword(idSrc, keyword)
  4159. {
  4160. var ursObj = W.model.updateRequestSessions.objects[idSrc];
  4161. if(typeof(ursObj) == 'undefined') return false;
  4162. if(ursObj.comments.length === 0) return false;
  4163.  
  4164. for(var idx=0; idx<ursObj.comments.length; idx++)
  4165. {
  4166. if(ursObj.comments[idx].text.indexOf(keyword) != -1)
  4167. {
  4168. return true;
  4169. }
  4170. }
  4171. return false;
  4172. }
  4173.  
  4174. function uroCheckCommentsForTag(idSrc)
  4175. {
  4176. var tags = ['[ROADWORKS]','[CONSTRUCTION]','[CLOSURE]','[EVENT]','[NOTE]'];
  4177. var ursObj = W.model.updateRequestSessions.objects[idSrc];
  4178. if(typeof(ursObj) == 'undefined') return -1;
  4179. if(ursObj.comments.length === 0) return -1;
  4180.  
  4181. for(var idx=ursObj.comments.length-1; idx>=0; idx--)
  4182. {
  4183. for(var tag=0; tag<tags.length; tag++)
  4184. {
  4185. var keyword = tags[tag];
  4186. if(ursObj.comments[idx].text.indexOf(keyword) != -1)
  4187. {
  4188. return tag;
  4189. }
  4190. }
  4191. }
  4192. return -1;
  4193. }
  4194.  
  4195. function uroGetCustomMarkerIdx(customType)
  4196. {
  4197. if(customType === 0) return 3;
  4198. if(customType === 1) return 3;
  4199. if(customType === 2) return 5;
  4200. if(customType === 3) return 7;
  4201. if(customType === 4) return 9;
  4202. if(customType === 100) return 11;
  4203. if(customType === 101) return 13;
  4204. if(customType === 102) return 15;
  4205. return -1;
  4206. }
  4207.  
  4208. function uroGetCustomType(idSrc, markerType)
  4209. {
  4210. var desc = '';
  4211. if(markerType == "ur")
  4212. {
  4213. var ureq = W.model.mapUpdateRequests.objects[idSrc];
  4214. // UR objects always have a .description attribute, which is set to null if empty...
  4215. if(ureq.attributes.description !== null)
  4216. {
  4217. desc = ureq.attributes.description;
  4218. }
  4219. }
  4220. else if(markerType == "mp")
  4221. {
  4222. var mp = W.model.problems.objects[idSrc];
  4223. // ...whereas MP objects with a blank description don't even get the attribute
  4224. if(mp.attributes.description !== undefined)
  4225. {
  4226. desc = mp.attributes.description;
  4227. }
  4228. }
  4229.  
  4230. if(desc !== '')
  4231. {
  4232. if(desc.indexOf('[ROADWORKS]') != -1) return 0;
  4233. if(desc.indexOf('[CONSTRUCTION]') != -1) return 1;
  4234. if(desc.indexOf('[CLOSURE]') != -1) return 2;
  4235. if(desc.indexOf('[EVENT]') != -1) return 3;
  4236. if(desc.indexOf('[NOTE]') != -1) return 4;
  4237. if(desc.indexOf('[Elgin]') != -1) return 100;
  4238. if(desc.indexOf('[TrafficCast]') != -1) return 101;
  4239. if(desc.indexOf('[TM]') != -1) return 102;
  4240. }
  4241. if(markerType == "ur")
  4242. {
  4243. return uroCheckCommentsForTag(idSrc);
  4244. }
  4245.  
  4246. return -1;
  4247. }
  4248.  
  4249.  
  4250. function uroFormatRestriction(restObj)
  4251. {
  4252. var retval = '<tr>';
  4253. retval += '<td style="text-align:center;">';
  4254. if((restObj.days & 1) == 1) retval += 'S';
  4255. else retval += '-';
  4256. retval += '</td><td style="text-align:center;">';
  4257. if((restObj.days & 2) == 2) retval += 'M';
  4258. else retval += '-';
  4259. retval += '</td><td style="text-align:center;">';
  4260. if((restObj.days & 4) == 4) retval += 'T';
  4261. else retval += '-';
  4262. retval += '</td><td style="text-align:center;">';
  4263. if((restObj.days & 8) == 8) retval += 'W';
  4264. else retval += '-';
  4265. retval += '</td><td style="text-align:center;">';
  4266. if((restObj.days & 16) == 16) retval += 'T';
  4267. else retval += '-';
  4268. retval += '</td><td style="text-align:center;">';
  4269. if((restObj.days & 32) == 32) retval += 'F';
  4270. else retval += '-';
  4271. retval += '</td><td style="text-align:center;">';
  4272. if((restObj.days & 64) == 64) retval += 'S';
  4273. else retval += '-';
  4274.  
  4275. retval += '</td><td>';
  4276.  
  4277. if(restObj.fromDate === null) retval += 'All dates';
  4278. else retval += restObj.fromDate+' to '+restObj.toDate;
  4279.  
  4280. retval += '</td><td>';
  4281.  
  4282. if(restObj.allDay === true) retval += 'All day';
  4283. else retval += restObj.fromTime+' to '+restObj.toTime;
  4284.  
  4285. retval += '</td><td>';
  4286.  
  4287. if(restObj.allVehicleTypes == restObj.vehicleTypes) retval += 'All vehicles';
  4288. else retval += 'Some vehicles';
  4289.  
  4290. retval += '</td><td>';
  4291.  
  4292. if(restObj.description !== null)
  4293. {
  4294. var desc = restObj.description.replace(/<\/?[^>]+(>|$)/g, "");
  4295. desc = uroClickify(desc);
  4296. retval += desc;
  4297. }
  4298.  
  4299. retval += '</td></tr>';
  4300.  
  4301. return retval;
  4302. }
  4303.  
  4304. function uroHidePopup()
  4305. {
  4306. if(uroPopupShown)
  4307. {
  4308. uroDiv.style.visibility = 'hidden';
  4309. uroPopupShown = false;
  4310. uroPopupTimer = -2;
  4311. uroShownFID = -1;
  4312. }
  4313. }
  4314.  
  4315. function uroRecentreSessionOnUR()
  4316. {
  4317. W.map.updateRequestLayer.markers[uroShownFID].icon.imageDiv.click();
  4318. W.map.moveTo(W.map.updateRequestLayer.markers[uroShownFID].lonlat, 5);
  4319. uroHidePopup();
  4320. return false;
  4321. }
  4322.  
  4323. function uroRecentreSessionOnMP()
  4324. {
  4325. W.map.problemLayer.markers[uroShownFID].icon.imageDiv.click();
  4326. W.map.moveTo(W.map.problemLayer.markers[uroShownFID].lonlat, 5);
  4327. uroHidePopup();
  4328. return false;
  4329. }
  4330.  
  4331. function uroRecentreSessionOnPUR()
  4332. {
  4333. W.map.placeUpdatesLayer.markers[uroShownFID].icon.imageDiv.click();
  4334. W.map.moveTo(W.map.placeUpdatesLayer.markers[uroShownFID].lonlat, 5);
  4335. uroHidePopup();
  4336. return false;
  4337. }
  4338.  
  4339. function uroRecentreSessionOnVenueNavPoint()
  4340. {
  4341. W.map.moveTo(uroGetVenueNavPoint(uroShownFID), 5);
  4342. uroHidePopup();
  4343. return false;
  4344. }
  4345.  
  4346. function uroGetDateTimeString(ts)
  4347. {
  4348. var tDateObj = new Date(ts);
  4349. var dateLocale;
  4350. var timeLocale;
  4351. if(uroGetCBChecked('_cbDateFmtDDMMYY')) dateLocale = 'en-gb';
  4352. if(uroGetCBChecked('_cbDateFmtMMDDYY')) dateLocale = 'en-us';
  4353. if(uroGetCBChecked('_cbDateFmtYYMMDD')) dateLocale = 'ja';
  4354. if(uroGetCBChecked('_cbTimeFmt24H')) timeLocale = 'en-gb';
  4355. if(uroGetCBChecked('_cbTimeFmt12H')) timeLocale = 'en-us';
  4356. return tDateObj.toLocaleDateString(dateLocale) + ' ' + tDateObj.toLocaleTimeString(timeLocale);
  4357. }
  4358.  
  4359. function uroParsePxString(pxString)
  4360. {
  4361. return parseInt(pxString.split("px")[0]);
  4362. }
  4363.  
  4364. function uroStackListObj(fid,x,y)
  4365. {
  4366. this.fid = fid;
  4367. this.x = uroTypeCast(x);
  4368. this.y = uroTypeCast(y);
  4369. }
  4370.  
  4371. function uroRestackMarkers()
  4372. {
  4373. if(uroStackList.length === 0) return;
  4374. var markerCollection = null;
  4375. if(uroStackType == 1) markerCollection = W.map.updateRequestLayer.markers;
  4376. else if(uroStackType == 2) markerCollection = W.map.problemLayer.markers;
  4377. else if(uroStackType == 3) markerCollection = W.map.placeUpdatesLayer.markers;
  4378.  
  4379. if(markerCollection !== null)
  4380. {
  4381. // strip off the .realX attributes from any UR object we've previously added it to, to allow
  4382. // the native recentering to work again...
  4383. for(var marker in markerCollection)
  4384. {
  4385. if(markerCollection.hasOwnProperty(marker))
  4386. {
  4387. var testMarkerObj = markerCollection[marker];
  4388. if(testMarkerObj.model.attributes.geometry.realX !== undefined)
  4389. {
  4390. testMarkerObj.model.attributes.geometry.x = testMarkerObj.model.attributes.geometry.realX;
  4391. delete(testMarkerObj.model.attributes.geometry.realX);
  4392. }
  4393. }
  4394. }
  4395. // now restack any markers that were repositioned...
  4396. for(var idx=0; idx<uroStackList.length; idx++)
  4397. {
  4398. var orig_x = uroStackList[idx].x + 'px';
  4399. var orig_y = uroStackList[idx].y + 'px';
  4400. var fid = uroStackList[idx].fid;
  4401.  
  4402. if(markerCollection[fid] !== undefined)
  4403. {
  4404. markerCollection[fid].icon.imageDiv.style.left = orig_x;
  4405. markerCollection[fid].icon.imageDiv.style.top = orig_y;
  4406. }
  4407. }
  4408. uroStackList = [];
  4409. uroUnstackedMasterID = null;
  4410. uroStackType = null;
  4411. }
  4412. }
  4413.  
  4414. function uroIsIDAlreadyUnstacked(idSrc)
  4415. {
  4416. if(uroStackList.length === 0) return false;
  4417. for(var idx=0; idx<uroStackList.length; idx++)
  4418. {
  4419. if(uroStackList[idx].fid == idSrc) return true;
  4420. }
  4421. return false;
  4422. }
  4423.  
  4424. function uroCheckStacking(stackType, masterID, unstackedX, unstackedY)
  4425. {
  4426. if(uroIsIDAlreadyUnstacked(masterID) === true) return;
  4427. if(uroStackType !== null) return;
  4428. if(uroPopupDwellTimer > 0) return;
  4429.  
  4430. uroAddLog('checking for marker stack, type '+stackType+'...');
  4431. var stackList = [];
  4432. stackList.push(masterID);
  4433. var threshSquared = uroGetElmValue('_inputUnstackSensitivity');
  4434. threshSquared *= threshSquared;
  4435.  
  4436. var markerCollection = null;
  4437. var marker;
  4438. var tempX = 1000000000;
  4439. if(stackType == 1) markerCollection = W.map.updateRequestLayer.markers;
  4440. else if(stackType == 2) markerCollection = W.map.problemLayer.markers;
  4441. else if(stackType == 3) markerCollection = W.map.placeUpdatesLayer.markers;
  4442.  
  4443. if(markerCollection !== null)
  4444. {
  4445. for(marker in markerCollection)
  4446. {
  4447. if(markerCollection.hasOwnProperty(marker))
  4448. {
  4449. var testMarkerObj = markerCollection[marker];
  4450. var includeInStack = (testMarkerObj.icon.imageDiv.style.visibility != 'hidden');
  4451. var suppressClosed = (testMarkerObj.icon.imageDiv.classList.contains("recently-closed") & (W.map.updateRequestLayer.showHidden === false));
  4452.  
  4453. if(testMarkerObj.model.attributes.geometry.realX === undefined)
  4454. {
  4455. testMarkerObj.model.attributes.geometry.realX = testMarkerObj.model.attributes.geometry.x;
  4456. testMarkerObj.model.attributes.geometry.x = tempX;
  4457. tempX++;
  4458. }
  4459.  
  4460. if((includeInStack) && (!suppressClosed))
  4461. {
  4462. if(testMarkerObj.id != masterID)
  4463. {
  4464. var xdiff = unstackedX - uroParsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.left);
  4465. var ydiff = unstackedY - uroParsePxString(markerCollection[testMarkerObj.id].icon.imageDiv.style.top);
  4466. var distSquared = ((xdiff * xdiff) + (ydiff * ydiff));
  4467. if(distSquared < threshSquared)
  4468. {
  4469. stackList.push(testMarkerObj.id);
  4470. }
  4471. }
  4472. }
  4473. }
  4474. }
  4475. }
  4476.  
  4477. // 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
  4478. // allow this to occur even if unstacking isn't required at this zoom level. To then reenable recentering when clicking on the crosshairs
  4479. // 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
  4480. // to do all of the unstacking *except* for actually unstacking the stack and hiding the other markers...
  4481. var inhibitUnstacking = (W.map.getZoom() < uroGetElmValue('_inputUnstackZoomLevel'));
  4482. // 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
  4483. // doesn't need to be unstacked, and where we're only using the .geometry.x fiddle to prevent autocentering...
  4484. inhibitUnstacking |= (stackList.length == 1);
  4485. uroStackType = stackType;
  4486. if(stackList.length > 0)
  4487. {
  4488. uroAddLog('markers are stacked!');
  4489. if(uroUnstackedMasterID != masterID)
  4490. {
  4491. uroAddLog('unstacked ID mismatch, relocating markers...');
  4492. uroRestackMarkers();
  4493. uroUnstackedMasterID = masterID;
  4494. uroStackList = [];
  4495.  
  4496. // push the highlighted marker onto the stacklist so uroIsIDAlreadyUnstacked() will return true
  4497. uroStackList.push(new uroStackListObj(masterID,unstackedX,unstackedY));
  4498.  
  4499. for(var shoveIdx=0; shoveIdx < stackList.length; shoveIdx++)
  4500. {
  4501. var fid = stackList[shoveIdx];
  4502. var x = uroParsePxString(markerCollection[fid].icon.imageDiv.style.left);
  4503. var y = uroParsePxString(markerCollection[fid].icon.imageDiv.style.top);
  4504. // store the unstacked marker positions so they can be reinstated later
  4505. uroStackList.push(new uroStackListObj(fid,x,y));
  4506. if(!inhibitUnstacking)
  4507. {
  4508. markerCollection[fid].icon.imageDiv.style.left = unstackedX + 'px';
  4509. markerCollection[fid].icon.imageDiv.style.top = unstackedY + 'px';
  4510. unstackedX += 10;
  4511. unstackedY -= 30;
  4512. }
  4513. }
  4514.  
  4515. if(!inhibitUnstacking)
  4516. {
  4517. // hide other markers to prevent confusion with the unstacked markers
  4518. for(marker in markerCollection)
  4519. {
  4520. if(markerCollection.hasOwnProperty(marker))
  4521. {
  4522. var toHideID = markerCollection[marker].id;
  4523. if(uroIsIDAlreadyUnstacked(toHideID) === false)
  4524. {
  4525. markerCollection[toHideID].icon.imageDiv.style.visibility = 'hidden';
  4526. }
  4527. }
  4528. }
  4529. }
  4530. }
  4531. }
  4532. else
  4533. {
  4534. uroRestackMarkers();
  4535. }
  4536. }
  4537.  
  4538. function uroGetVenueNavPoint(uroFID)
  4539. {
  4540. for(var vObj in W.model.venues.objects)
  4541. {
  4542. if(W.model.venues.objects.hasOwnProperty(vObj))
  4543. {
  4544. if(uroFID == vObj)
  4545. {
  4546. return W.model.venues.objects[vObj].getNavigationPoint().point.toLonLat();
  4547. }
  4548. }
  4549. }
  4550. // just in case... return a safe value if the requested venue object wasn't found
  4551. return W.map.getCenter();
  4552. }
  4553.  
  4554. function uroOpenNewTab()
  4555. {
  4556. // flush the current settings into localStorage before the new tab opens, so that when its instance of
  4557. // URO+ fires up it'll have the same settings as this one
  4558. uroSaveSettings();
  4559. return true;
  4560. }
  4561.  
  4562. function uroEditTBR()
  4563. {
  4564. if(uroTBRObj === null)
  4565. {
  4566. return;
  4567. }
  4568. uroTBRObj.click();
  4569. return false;
  4570. }
  4571.  
  4572. function uroNewLookHighlightedItemsCheck(e)
  4573. {
  4574. if(e == 'dwellTimeout')
  4575. {
  4576. }
  4577. else
  4578. {
  4579. if((uroMouseIsDown) && (e.buttons === 0))
  4580. {
  4581. uroAddLog('trapped erroneous mousedown state');
  4582. uroMouseIsDown = false;
  4583. }
  4584. }
  4585. if(uroMouseIsDown)
  4586. {
  4587. return;
  4588. }
  4589.  
  4590. if(OpenLayers === null)
  4591. {
  4592. if(uroNullOpenLayers === false)
  4593. {
  4594. uroAddLog('caught null OpenLayers');
  4595. uroNullOpenLayers = true;
  4596. }
  4597. return;
  4598. }
  4599. uroNullOpenLayers = false;
  4600.  
  4601. var rc = document.getElementById(uroRootContainer);
  4602. if(rc === null)
  4603. {
  4604. if(uroNullRootContainer === false)
  4605. {
  4606. uroAddLog('caught null rootContainer');
  4607. uroNullRootContainer = true;
  4608. }
  4609. return;
  4610. }
  4611. uroNullRootContainer = false;
  4612.  
  4613. if(W.map.updateRequestLayer === null)
  4614. {
  4615. if(uroNullURLayer === false)
  4616. {
  4617. uroAddLog('caught null UR layer');
  4618. uroNullURLayer = true;
  4619. }
  4620. return;
  4621. }
  4622. uroNullURLayer = false;
  4623.  
  4624. if(W.map.problemLayer === null)
  4625. {
  4626. if(uroNullProblemLayer === false)
  4627. {
  4628. uroAddLog('caught null problem layer');
  4629. uroNullProblemLayer = true;
  4630. }
  4631. return;
  4632. }
  4633. uroNullProblemLayer = false;
  4634.  
  4635. if(uroGetCBChecked('_cbMasterEnable') === false)
  4636. {
  4637. return;
  4638. }
  4639.  
  4640. var mouseX;
  4641. var mouseY;
  4642. if(e == 'dwellTimeout')
  4643. {
  4644. mouseX = uroPrevMouseX;
  4645. mouseY = uroPrevMouseY;
  4646. }
  4647. else
  4648. {
  4649. mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
  4650. mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top;
  4651.  
  4652. var maxJitter = uroGetElmValue('_inputMaxJitter');
  4653. if((Math.abs(uroPrevMouseX - mouseX) > maxJitter) || (Math.abs(uroPrevMouseY - mouseY) > maxJitter))
  4654. {
  4655. uroPopupDwellTimer = uroGetElmValue('_inputPopupDwellTimeout');
  4656. }
  4657. uroPrevMouseX = mouseX;
  4658. uroPrevMouseY = mouseY;
  4659. }
  4660.  
  4661. var result = '';
  4662. var rw;
  4663. var rh;
  4664.  
  4665. var popupXOffset = uroParsePxString(window.getComputedStyle(document.getElementById('sidebar')).getPropertyValue("width"));
  4666. var popupYOffset = $(document.getElementById("WazeMap")).offset().top - 80;
  4667. var uroPopupX = mouseX + popupXOffset + 10;
  4668. var uroPopupY = mouseY + popupYOffset + 10;
  4669.  
  4670. var objHasIgnoreLink = false;
  4671. var objHasDeleteLink = false;
  4672. var objHasAddWatchLink = false;
  4673. var objHasRemoveWatchLink = false;
  4674. var objHasUpdateWatchLink = false;
  4675. var objHasRecentreSessionLink = false;
  4676. var objHasOpenInNewTabLink = false;
  4677.  
  4678. var isVenue = false;
  4679. var newPopupType = null;
  4680. var markerObj;
  4681. var markerPos;
  4682. var markerImg;
  4683. var ureq;
  4684. var idx;
  4685. var hovered;
  4686.  
  4687. // popup for segment restrictions
  4688. if(uroGetCBChecked('_cbInhibitSegPopup') === false)
  4689. {
  4690. for(var slIdx=0; slIdx < W.map.segmentLayer.features.length; slIdx++)
  4691. {
  4692. if(W.map.segmentLayer.features[slIdx].renderIntent == 'highlight')
  4693. {
  4694. var doPopUp = false;
  4695. var segObj;
  4696. var restObj;
  4697. if(W.map.segmentLayer.features[slIdx].fid === null) segObj = W.map.segmentLayer.features[slIdx].model;
  4698. else segObj = W.map.segmentLayer.features[slIdx];
  4699. result += '<table cellpadding=4 border=1">';
  4700. if(segObj.attributes.fwdRestrictions.length > 0)
  4701. {
  4702. doPopUp = true;
  4703. result += '<tr><td colspan=11><b>A-B restrictions:</b></td></tr>';
  4704. for(idx = 0; idx < segObj.attributes.fwdRestrictions.length; idx++)
  4705. {
  4706. restObj = segObj.attributes.fwdRestrictions[idx];
  4707. result += uroFormatRestriction(restObj);
  4708. }
  4709. }
  4710. if (segObj.attributes.revRestrictions.length > 0)
  4711. {
  4712. doPopUp = true;
  4713. result += '<tr><td colspan=11><b>B-A restrictions:</b></td></tr>';
  4714. for(idx = 0; idx < segObj.attributes.revRestrictions.length; idx++)
  4715. {
  4716. restObj = segObj.attributes.revRestrictions[idx];
  4717. result += uroFormatRestriction(restObj);
  4718. }
  4719. }
  4720. result += '</table>';
  4721. if(W.map.closuresMarkerLayer.getVisibility() === true)
  4722. {
  4723. result += '<table cellpadding=4 border=1" width="100%">';
  4724. if(segObj.attributes.hasClosures === true)
  4725. {
  4726. var hasFwd = false;
  4727. var hasRev = false;
  4728. var rcObj;
  4729. var roadClosure;
  4730. for(roadClosure in W.model.roadClosures.objects)
  4731. {
  4732. if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
  4733. {
  4734. rcObj = W.model.roadClosures.objects[roadClosure];
  4735. if(rcObj.segID == segObj.attributes.id)
  4736. {
  4737. if(rcObj.forward === true)
  4738. {
  4739. if(hasFwd === false)
  4740. {
  4741. result += '<tr><td colspan=3><b>A-B closures:</b></td></tr>';
  4742. hasFwd = true;
  4743. }
  4744.  
  4745. if(rcObj.active === true)
  4746. {
  4747. result += '<tr>';
  4748. }
  4749. else
  4750. {
  4751. result += '<tr bgcolor="#C0C0C0">';
  4752. }
  4753.  
  4754. var startDate = rcObj.startDate;
  4755. var endDate = "unknown";
  4756. if(rcObj.endDate !== null)
  4757. {
  4758. endDate = rcObj.endDate;
  4759. }
  4760. var provider = "---";
  4761. if(rcObj.provider !== null)
  4762. {
  4763. provider = rcObj.provider;
  4764. }
  4765. var reason = "---";
  4766. if(rcObj.reason !== null)
  4767. {
  4768. reason = rcObj.reason;
  4769. }
  4770. result += '<td>' + startDate + ' to ' + endDate + '</td>';
  4771. result += '<td>' + provider + '</td>';
  4772. result += '<td>' + reason + '</td>';
  4773. result += '</td></tr>';
  4774. }
  4775. else
  4776. {
  4777. hasRev = true;
  4778. }
  4779. }
  4780. }
  4781. }
  4782. if(hasRev === true)
  4783. {
  4784. result += '<tr><td colspan=3><b>B-A closures:</b></td></tr>';
  4785. for(roadClosure in W.model.roadClosures.objects)
  4786. {
  4787. if(W.model.roadClosures.objects.hasOwnProperty(roadClosure))
  4788. {
  4789. rcObj = W.model.roadClosures.objects[roadClosure];
  4790. if(rcObj.segID == segObj.attributes.id)
  4791. {
  4792. if(rcObj.forward === false)
  4793. {
  4794. if(rcObj.active === true)
  4795. {
  4796. result += '<tr>';
  4797. }
  4798. else
  4799. {
  4800. result += '<tr bgcolor="#C0C0C0">';
  4801. }
  4802.  
  4803. result += '<td>' + rcObj.startDate + ' to ' + rcObj.endDate + '</td>';
  4804. result += '<td>' + rcObj.provider + '</td>';
  4805. result += '<td>' + rcObj.reason + '</td>';
  4806. result += '</td></tr>';
  4807. }
  4808. }
  4809. }
  4810. }
  4811. }
  4812. if((hasFwd === true) || (hasRev === true))
  4813. {
  4814. doPopUp = true;
  4815. }
  4816. }
  4817. result += '</table>';
  4818. }
  4819.  
  4820. if(doPopUp === true)
  4821. {
  4822. if(segObj.attributes.id === null) uroFID = segObj.id;
  4823. else uroFID = segObj.attributes.id;
  4824. newPopupType = 'segment_restriction';
  4825. uroAddDebug(newPopupType+' '+uroFID);
  4826. }
  4827. break;
  4828. }
  4829. }
  4830. }
  4831.  
  4832. // popup for restricted turns
  4833. if(newPopupType === null)
  4834. {
  4835. if(uroGetCBChecked('_cbInhibitTurnsPopup') === false)
  4836. {
  4837. var turnMarkerCount = W.map.layers[uroTurnsLayerIdx].markers.length;
  4838. if(turnMarkerCount > 0)
  4839. {
  4840. for(idx=0; idx<turnMarkerCount; idx++)
  4841. {
  4842. markerObj = W.map.layers[uroTurnsLayerIdx].markers[idx];
  4843. var arrowElm = markerObj.icon.imageDiv.childNodes[0];
  4844. markerImg = window.getComputedStyle(arrowElm).getPropertyValue("background-image");
  4845. markerPos = window.getComputedStyle(arrowElm).getPropertyValue("background-position");
  4846. markerPos = markerPos.split(' ');
  4847. markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
  4848.  
  4849. hovered = false;
  4850.  
  4851. if((markerImg.indexOf('turns-sa7bd56a5e6.png') != -1) || (markerImg.indexOf('turns-sd65f776611.png') != -1))
  4852. {
  4853. if(markerPos == -222)
  4854. {
  4855. hovered = true;
  4856. }
  4857. }
  4858. if(hovered === true)
  4859. {
  4860. uroAddLog('hover over restricted turn marker');
  4861. uroTBRObj = arrowElm.childNodes[0];
  4862. var trObj = ($(arrowElm).data('model'));
  4863. var resObj = null;
  4864. if(trObj.fromSeg.attributes.fromRestrictions !== undefined)
  4865. {
  4866. resObj = trObj.fromSeg.attributes.fromRestrictions[trObj.toSeg.attributes.id];
  4867. }
  4868. if(resObj === null)
  4869. {
  4870. if(trObj.fromSeg.attributes.toRestrictions !== undefined)
  4871. {
  4872. resObj = trObj.fromSeg.attributes.toRestrictions[trObj.toSeg.attributes.id];
  4873. }
  4874. }
  4875.  
  4876. result += '<label id="_editTBR">Click to edit</label><br>';
  4877. result += '<table cellpadding=4 border=1">';
  4878. for(var resIdx=0; resIdx < resObj.length; resIdx++)
  4879. {
  4880. result += uroFormatRestriction(resObj[resIdx]);
  4881. }
  4882. result += '</table>';
  4883. uroFID = markerObj.icon.imageDiv._eventCacheID;
  4884. newPopupType = 'turn_restriction';
  4885. uroPopupY -= 20;
  4886. uroAddDebug(newPopupType+' '+uroFID);
  4887. break;
  4888. }
  4889. }
  4890. }
  4891. }
  4892. }
  4893.  
  4894. var targetTab = '';
  4895. // popup for landmarks
  4896. if((newPopupType === null) && (uroGetCBChecked('_cbInhibitLandmarkPopup') === false))
  4897. {
  4898. uroPlaceSelected = false;
  4899. for(var llFeatureIdx=0; llFeatureIdx < W.map.landmarkLayer.features.length; llFeatureIdx++)
  4900. {
  4901. var renderIntent = W.map.landmarkLayer.features[llFeatureIdx].renderIntent;
  4902. if(renderIntent == 'highlight')
  4903. {
  4904. var venueObj;
  4905. if(W.map.landmarkLayer.features[llFeatureIdx].fid === null) venueObj = W.map.landmarkLayer.features[llFeatureIdx].model;
  4906. else venueObj = W.map.landmarkLayer.features[llFeatureIdx];
  4907.  
  4908. if(venueObj.attributes.id === null) uroFID = venueObj.id;
  4909. else uroFID = venueObj.attributes.id;
  4910.  
  4911. var navpointPos=new OpenLayers.LonLat();
  4912. navpointPos = uroGetVenueNavPoint(uroFID);
  4913. navpointPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  4914.  
  4915. result += '<b>';
  4916. if(venueObj.attributes.name === '') result += 'Unnamed landmark';
  4917. else result += venueObj.attributes.name;
  4918. result += '</b><br>';
  4919.  
  4920. result += '<ul>';
  4921. for(idx = 0; idx < venueObj.attributes.categories.length; idx++)
  4922. {
  4923. result += '<li>' + I18n.lookup("venues.categories")[venueObj.attributes.categories[idx]];
  4924. }
  4925. result += '</ul>';
  4926.  
  4927. if(venueObj.attributes.residential === true)
  4928. {
  4929. result += '<i>Residential</i>';
  4930. }
  4931.  
  4932. var npLink = document.location.href;
  4933. var npLayers = '&layers='+W.map.mapState.getLayerVisibilityBitmask();
  4934. npLink = npLink.substr(0,npLink.indexOf('?zoom'));
  4935. npLink += '?zoom=5&lat='+navpointPos.lat+'&lon='+navpointPos.lon+npLayers;
  4936.  
  4937. targetTab = "_uroTab_" + Math.round(Math.random()*1000000);
  4938. result += '<hr>Jump to nav point: <a href="'+npLink+'" id="_openInNewTab" target="'+targetTab+'">in new tab</a> - ';
  4939. objHasOpenInNewTabLink = true;
  4940. result += '<a href="#" id="_recentreSession">in this tab</a>';
  4941. objHasRecentreSessionLink = true;
  4942.  
  4943. uroPopupY -= 20;
  4944. newPopupType = 'venue';
  4945. uroAddDebug(newPopupType+' '+uroFID);
  4946. isVenue = true;
  4947. break;
  4948. }
  4949. else if((renderIntent == 'select') || (renderIntent == 'highlightselected'))
  4950. {
  4951. uroPlaceSelected = true;
  4952. }
  4953. }
  4954. }
  4955.  
  4956. var unstackedX;
  4957. var unstackedY;
  4958. var ureqID = null;
  4959. var isUR = false;
  4960. var isProblem = false;
  4961. var isTurnProb = false;
  4962. var isPlaceUpdate = false;
  4963. // look for URs, place updates and problems
  4964. if(newPopupType === null)
  4965. {
  4966. var customIdx;
  4967. var idSrc = null;
  4968. if(uroGetCBChecked('_cbInhibitURPopup') === false)
  4969. {
  4970. hovered = false;
  4971. for(var markerURL in W.map.updateRequestLayer.markers)
  4972. {
  4973. if(W.map.updateRequestLayer.markers.hasOwnProperty(markerURL))
  4974. {
  4975. markerObj = W.map.updateRequestLayer.markers[markerURL];
  4976. markerImg = markerObj.icon.$div.css('background-image');
  4977. markerPos = markerObj.icon.$div.css('background-position');
  4978. markerPos = markerPos.split(' ');
  4979. markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
  4980. var urIDSrc = markerObj.id;
  4981. if
  4982. (
  4983. (markerImg.indexOf('problems-s8f369ca968.png') != -1) ||
  4984. (markerImg.indexOf('problems-sc703e224cf.png') != -1)
  4985. )
  4986. {
  4987. if((markerPos == -403) || (markerPos == -483) || (markerPos == -563) || (markerPos == -643))
  4988. {
  4989. hovered = true;
  4990. uroAddLog('UR image type 1');
  4991. }
  4992. }
  4993. else if(markerImg.indexOf('problems-se224ab677e.png') != -1)
  4994. {
  4995. if((markerPos == -40) || (markerPos == -160) || (markerPos == -200) || (markerPos == -240))
  4996. {
  4997. hovered = true;
  4998. uroAddLog('UR image type 2');
  4999. }
  5000. }
  5001. else
  5002. {
  5003. if(markerPos > -200)
  5004. {
  5005. hovered = true;
  5006. uroAddLog('UR image type 3');
  5007. }
  5008. }
  5009. if(hovered === true)
  5010. {
  5011. isUR = true;
  5012. newPopupType = 'ur';
  5013. uroAddLog('hover over UR ID '+urIDSrc);
  5014.  
  5015. unstackedX = uroParsePxString(W.map.updateRequestLayer.markers[urIDSrc].icon.imageDiv.style.left);
  5016. unstackedY = uroParsePxString(W.map.updateRequestLayer.markers[urIDSrc].icon.imageDiv.style.top);
  5017.  
  5018. // override popup base position
  5019. uroPopupX = unstackedX + popupXOffset + 6;
  5020. uroPopupY = unstackedY + popupYOffset + 46;
  5021. uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
  5022. uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
  5023.  
  5024. // check for stacking...
  5025. if(uroShownFID != idSrc)
  5026. {
  5027. uroCheckStacking(1,urIDSrc, unstackedX, unstackedY);
  5028. }
  5029.  
  5030. if(urIDSrc != uroCustomMarkerFID)
  5031. {
  5032. if(uroCustomMarkerFID !== null)
  5033. {
  5034. customIdx = uroGetCustomMarkerIdx(uroGetCustomType(uroCustomMarkerFID, uroCustomMarkerType));
  5035. uroChangeCustomMarkers(uroCustomMarkerFID,false,customIdx,uroCustomMarkerType);
  5036. uroCustomMarkerFID = null;
  5037. }
  5038. customIdx = uroGetCustomMarkerIdx(uroGetCustomType(urIDSrc, "ur"));
  5039. if(customIdx != -1)
  5040. {
  5041. uroCustomMarkerFID = urIDSrc;
  5042. uroCustomMarkerType = "ur";
  5043. uroChangeCustomMarkers(urIDSrc,hovered,customIdx,uroCustomMarkerType);
  5044. }
  5045. }
  5046. idSrc = urIDSrc;
  5047. break;
  5048. }
  5049. }
  5050. }
  5051.  
  5052. if((hovered === false) && (uroStackType == 1))
  5053. {
  5054. uroRestackMarkers();
  5055. uroFilterURs();
  5056. }
  5057.  
  5058. if((idSrc === null) && (uroCustomMarkerFID !== null))
  5059. {
  5060. customIdx = uroGetCustomMarkerIdx(uroGetCustomType(uroCustomMarkerFID, uroCustomMarkerType));
  5061. uroChangeCustomMarkers(uroCustomMarkerFID,false, customIdx,uroCustomMarkerType);
  5062. uroCustomMarkerFID = null;
  5063. uroCustomMarkerType = null;
  5064. }
  5065. }
  5066.  
  5067. if((newPopupType === null) && (uroGetCBChecked('_cbInhibitPUPopup') === false))
  5068. {
  5069. hovered = false;
  5070. for(var markerPUL in W.map.placeUpdatesLayer.markers)
  5071. {
  5072. if(W.map.placeUpdatesLayer.markers.hasOwnProperty(markerPUL))
  5073. {
  5074. markerObj = W.map.placeUpdatesLayer.markers[markerPUL];
  5075. markerImg = markerObj.icon.$div.css('background-image');
  5076. markerPos = markerObj.icon.$div.css('background-position');
  5077. markerPos = markerPos.split(' ');
  5078. markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
  5079.  
  5080. if(markerImg.indexOf('placeUpdates-sb30471988c.png') != -1)
  5081. {
  5082. // absolute offsets: 0 = new place, -120 = flagged, -240 = new photo, -360 = updated details
  5083. // relative offsets: 0 = green, -40 = highlighted, -80 = default
  5084. if(((markerPos + 40) % 120) === 0)
  5085. {
  5086. hovered = true;
  5087. uroAddLog('PUR marker type 1');
  5088. }
  5089. }
  5090. else if((markerImg.indexOf('placeUpdates-s2e8d9c5ce4.png') != -1)||(markerImg.indexOf('placeUpdates-s58d24ce3b0.png') != -1))
  5091. {
  5092. if
  5093. (
  5094. (markerObj.icon.$div.css("filter") == "brightness(110%)") ||
  5095. (markerObj.icon.$div.css("webkit-filter") == "brightness(1.1)")
  5096. )
  5097. {
  5098. hovered = true;
  5099. uroAddLog('PUR marker type 2');
  5100. }
  5101. }
  5102. if(hovered === true)
  5103. {
  5104. idSrc = markerObj.id;
  5105. unstackedX = uroParsePxString(W.map.placeUpdatesLayer.markers[idSrc].icon.imageDiv.style.left);
  5106. unstackedY = uroParsePxString(W.map.placeUpdatesLayer.markers[idSrc].icon.imageDiv.style.top);
  5107.  
  5108. // override popup base position
  5109. uroPopupX = unstackedX + popupXOffset + 6;
  5110. uroPopupY = unstackedY + popupYOffset + 46;
  5111. uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
  5112. uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
  5113. if(uroShownFID != idSrc)
  5114. {
  5115. // check for stacking...
  5116. uroCheckStacking(3,idSrc, unstackedX, unstackedY);
  5117. }
  5118.  
  5119. isPlaceUpdate = true;
  5120. newPopupType = 'pur';
  5121. uroAddLog('hover over placeUpdate ID '+idSrc);
  5122. break;
  5123. }
  5124. }
  5125. }
  5126. if((hovered === false) && (uroStackType == 3))
  5127. {
  5128. uroRestackMarkers();
  5129. uroFilterPlaces();
  5130. }
  5131. }
  5132.  
  5133. if((newPopupType === null) && (uroGetCBChecked('_cbInhibitMPPopup') === false))
  5134. {
  5135. hovered = false;
  5136. for(var markerPL in W.map.problemLayer.markers)
  5137. {
  5138. if(W.map.problemLayer.markers.hasOwnProperty(markerPL))
  5139. {
  5140. markerObj = W.map.problemLayer.markers[markerPL];
  5141. markerImg = markerObj.icon.$div.css('background-image');
  5142. markerPos = markerObj.icon.$div.css('background-position');
  5143. markerPos = markerPos.split(' ');
  5144. markerPos = parseInt(markerPos[1].substr(0,markerPos[1].length-2));
  5145.  
  5146. if((markerImg.indexOf('problems-s8f369ca968.png') != -1)||(markerImg.indexOf('problems-sc703e224cf.png') != -1))
  5147. {
  5148. if((markerPos == -65) || (markerPos == -145) || (markerPos == -225) || (markerPos == -305))
  5149. {
  5150. hovered = true;
  5151. uroAddLog('Problem image type 1');
  5152. }
  5153. }
  5154. else if(markerImg.indexOf('problems-se224ab677e.png') != -1)
  5155. {
  5156. if((markerPos == -320) || (markerPos == -560) || (markerPos == -520) || (markerPos == -440))
  5157. {
  5158. hovered = true;
  5159. uroAddLog('Problem image type 2');
  5160. }
  5161. }
  5162. else
  5163. {
  5164. if(markerPos > -200)
  5165. {
  5166. hovered = true;
  5167. uroAddLog('Problem image type 3');
  5168. }
  5169. }
  5170. if(hovered === true)
  5171. {
  5172. idSrc = null;
  5173. if(markerObj.model.fid === null) idSrc = markerObj.id;
  5174. else idSrc = markerObj.model.fid;
  5175.  
  5176. unstackedX = uroParsePxString(W.map.problemLayer.markers[idSrc].icon.imageDiv.style.left);
  5177. unstackedY = uroParsePxString(W.map.problemLayer.markers[idSrc].icon.imageDiv.style.top);
  5178.  
  5179. // override popup base position
  5180. uroPopupX = unstackedX + popupXOffset + 6;
  5181. uroPopupY = unstackedY + popupYOffset + 46;
  5182. uroPopupX -= uroParsePxString(W.map.segmentLayer.div.style.left);
  5183. uroPopupY -= uroParsePxString(W.map.segmentLayer.div.style.top);
  5184.  
  5185. // check for stacking...
  5186. if(uroShownFID != idSrc)
  5187. {
  5188. uroCheckStacking(2,idSrc, unstackedX, unstackedY);
  5189. }
  5190.  
  5191. if(idSrc != uroCustomMarkerFID)
  5192. {
  5193. if(uroCustomMarkerFID !== null)
  5194. {
  5195. customIdx = uroGetCustomMarkerIdx(uroGetCustomType(uroCustomMarkerFID, uroCustomMarkerType));
  5196. uroChangeCustomMarkers(uroCustomMarkerFID,false,customIdx,uroCustomMarkerType);
  5197. uroCustomMarkerFID = null;
  5198. uroCustomMarkerType = null;
  5199. }
  5200. customIdx = uroGetCustomMarkerIdx(uroGetCustomType(idSrc, "mp"));
  5201. if(customIdx != -1)
  5202. {
  5203. uroCustomMarkerFID = idSrc;
  5204. uroCustomMarkerType = "mp";
  5205. uroChangeCustomMarkers(idSrc,hovered,customIdx,uroCustomMarkerType);
  5206. }
  5207. }
  5208.  
  5209.  
  5210. isProblem = true;
  5211. newPopupType = 'map_problem';
  5212. uroAddLog('hover over problem ID '+idSrc);
  5213.  
  5214. break;
  5215. }
  5216. }
  5217. }
  5218. if((hovered === false) && (uroStackType == 2))
  5219. {
  5220. uroRestackMarkers();
  5221. uroFilterProblems();
  5222. }
  5223. if((idSrc === null) && (uroCustomMarkerFID !== null))
  5224. {
  5225. var newCustomIdx = uroGetCustomMarkerIdx(uroGetCustomType(uroCustomMarkerFID, uroCustomMarkerType));
  5226. uroChangeCustomMarkers(uroCustomMarkerFID, false, newCustomIdx, uroCustomMarkerType);
  5227. uroCustomMarkerFID = null;
  5228. uroCustomMarkerType = null;
  5229. }
  5230. }
  5231.  
  5232. if (idSrc !== null)
  5233. {
  5234. ureq = null;
  5235. if(isUR) ureq = W.model.mapUpdateRequests.objects[idSrc];
  5236. else if(isProblem)
  5237. {
  5238. ureq = W.model.problems.objects[idSrc];
  5239. if(ureq === undefined)
  5240. {
  5241. if(uroDOMHasTurnProblems)
  5242. {
  5243. ureq = W.model.turnProblems.objects[idSrc];
  5244. if(ureq !== undefined) isTurnProb = true;
  5245. }
  5246. }
  5247. }
  5248. else if(isPlaceUpdate) ureq = W.map.placeUpdatesLayer.markers[idSrc].model;
  5249.  
  5250. if(ureq.fid !== null) ureqID = ureq.fid;
  5251. else if(ureq.id !== null) ureqID = ureq.id;
  5252. else if(ureq.attributes.id !== null) ureqID = ureq.attributes.id;
  5253.  
  5254. uroFID = ureqID;
  5255. uroAddDebug(newPopupType+' '+uroFID);
  5256. }
  5257. else
  5258. {
  5259. if(uroFID != -1)
  5260. {
  5261. uroFID = -1;
  5262. uroAddDebug('uroFID=-1');
  5263. }
  5264. }
  5265.  
  5266. if(uroFID != -1)
  5267. {
  5268. var uroDaysResolved;
  5269. if(isUR)
  5270. {
  5271. uroAddLog('building popup for UR '+idSrc);
  5272. result = '<b>Update Request ('+idSrc+'): ' + I18n.lookup("update_requests.types")[ureq.attributes.type] + '</b><br>';
  5273. if(ureq.attributes.description !== null)
  5274. {
  5275. var desc = ureq.attributes.description.replace(/<\/?[^>]+(>|$)/g, "");
  5276. if(desc != "null")
  5277. {
  5278. desc = uroClickify(desc);
  5279. result += desc + '<br>';
  5280. }
  5281. }
  5282. var uroDaysOld = uroGetURAge(ureq,0,false);
  5283. var uroSubmittedTS = uroGetURAge(ureq,0,true);
  5284. if(uroSubmittedTS != -1)
  5285. {
  5286. uroSubmittedTS = uroGetDateTimeString(uroSubmittedTS);
  5287. }
  5288. if(uroDaysOld != -1)
  5289. {
  5290. result += '<i>Submitted ' + uroParseDaysAgo(uroDaysOld) + ' ';
  5291. if(uroSubmittedTS != -1) result += '(' + uroSubmittedTS + ') ';
  5292. if(ureq.attributes.guestUserName !== undefined)
  5293. {
  5294. result += 'via Livemap';
  5295. if(ureq.attributes.guestUserName !== '')
  5296. {
  5297. result += ' by '+ureq.attributes.guestUserName.replace(/<\/?[^>]+(>|$)/g, "");
  5298. }
  5299. }
  5300. result += '</i>';
  5301. }
  5302. if(ureq.attributes.resolvedOn !== null)
  5303. {
  5304. uroDaysResolved = uroGetURAge(ureq,1,false);
  5305. var uroResolvedTS = uroGetURAge(ureq,1,true);
  5306. if(uroResolvedTS != -1)
  5307. {
  5308. uroResolvedTS = uroGetDateTimeString(uroResolvedTS);
  5309. }
  5310.  
  5311. if(uroDaysResolved != -1)
  5312. {
  5313. result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' ';
  5314. if(uroResolvedTS != -1) result += '(' + uroResolvedTS + ')</i>';
  5315.  
  5316. result += '<br><i>Marked as ';
  5317. if(ureq.attributes.resolution === 0) result += 'solved';
  5318. else if(ureq.attributes.resolution == 1) result += 'not identified';
  5319. else result += 'unknown';
  5320. if(ureq.attributes.resolvedBy !== null)
  5321. {
  5322. result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy);
  5323. }
  5324. result += '</i>';
  5325. }
  5326. }
  5327. if(W.model.updateRequestSessions.objects[ureqID] !== undefined)
  5328. {
  5329. var hasMyComments = uroURHasMyComments(ureqID);
  5330. var nComments = W.model.updateRequestSessions.objects[ureqID].comments.length;
  5331. result += '<br>' + nComments + ' comment';
  5332. if(nComments != 1) result += 's';
  5333. if((hasMyComments === false) && (nComments > 0)) result += ' (none by me)';
  5334. if(nComments > 0)
  5335. {
  5336. var commentDaysOld = uroGetCommentAge(W.model.updateRequestSessions.objects[ureqID].comments[nComments-1]);
  5337. if(commentDaysOld != -1)
  5338. {
  5339. result += ', last update '+uroParseDaysAgo(commentDaysOld);
  5340. }
  5341. }
  5342. }
  5343. }
  5344. else if(isProblem)
  5345. {
  5346. uroAddLog('building popup for problem '+idSrc);
  5347. if(isTurnProb) result = '<b>Turn Problem ('+idSrc+'): ' + I18n.lookup("problems.types").turn.title;
  5348. else
  5349. {
  5350. result = '<b>Map Problem ('+idSrc+'): ';
  5351.  
  5352. var problemType = null;
  5353. if(uroDOMHasTurnProblems)
  5354. {
  5355. problemType = ureq.attributes.problemType;
  5356. }
  5357. else
  5358. {
  5359. problemType = ureq.attributes.subType;
  5360. }
  5361.  
  5362. if(problemType == 300)
  5363. {
  5364. result += I18n.lookup("problems.panel.closure.title");
  5365. }
  5366. else
  5367. {
  5368. if(I18n.lookup("problems.types")[problemType] === undefined) result += 'Unknown problem type ('+problemType+')';
  5369. else result += I18n.lookup("problems.types")[problemType].title;
  5370. }
  5371. }
  5372. result += '</b><br>';
  5373. if(ureq.attributes.description !== undefined)
  5374. {
  5375. result += 'Description: ' + ureq.attributes.description + '<br>';
  5376. }
  5377. if(ureq.attributes.extraInfo !== undefined)
  5378. {
  5379. result += 'ExtraInfo: ' + ureq.attributes.extraInfo + '<br>';
  5380. }
  5381. if(ureq.attributes.provider !== undefined)
  5382. {
  5383. result += 'Provider: ' + ureq.attributes.provider + '<br>';
  5384. }
  5385. if(ureq.attributes.resolvedOn !== undefined)
  5386. {
  5387. uroDaysResolved = uroGetURAge(ureq,1,false);
  5388. if(uroDaysResolved != -1)
  5389. {
  5390. result += '<br><i>Closed ' + uroParseDaysAgo(uroDaysResolved) + ' ';
  5391. if(ureq.attributes.resolvedBy !== undefined)
  5392. {
  5393. result += ' by '+uroGetUserNameAndRank(ureq.attributes.resolvedBy);
  5394. }
  5395.  
  5396. if((ureq.attributes.open === true) && (ureq.attributes.resolvedOn !== undefined))
  5397. {
  5398. result += '<br>Reopened by Waze';
  5399. }
  5400. result += '</i>';
  5401. }
  5402. }
  5403. }
  5404. else if(isPlaceUpdate)
  5405. {
  5406. uroAddLog('building popup for placeUpdate '+idSrc);
  5407. result = '<b>';
  5408. if(ureq.attributes.name === '') result += 'Unnamed landmark';
  5409. else result += ureq.attributes.name;
  5410. result += '</b><br>';
  5411.  
  5412. result += '<ul>';
  5413. for(idx = 0; idx < ureq.attributes.categories.length; idx++)
  5414. {
  5415. result += '<li>' + I18n.lookup("venues.categories")[ureq.attributes.categories[idx]];
  5416. }
  5417. result += '</ul>';
  5418.  
  5419. if(ureq.attributes.residential === true)
  5420. {
  5421. result += '<i>Residential</i>';
  5422. }
  5423.  
  5424. var daysOld = uroGetPURAge(ureq);
  5425. if(daysOld != -1)
  5426. {
  5427. result += '<br><i>Submitted '+uroParseDaysAgo(daysOld)+'</i>';
  5428. }
  5429. }
  5430.  
  5431. // add "open new WME tab" link
  5432. var urPos=new OpenLayers.LonLat();
  5433. if(isPlaceUpdate)
  5434. {
  5435. urPos=ureq.geometry.bounds.centerLonLat.clone();
  5436. }
  5437. else
  5438. {
  5439. if(ureq.geometry.realX === undefined)
  5440. {
  5441. urPos.lon = ureq.geometry.x;
  5442. }
  5443. else
  5444. {
  5445. urPos.lon = ureq.geometry.realX;
  5446. }
  5447. urPos.lat = ureq.geometry.y;
  5448. }
  5449. urPos.transform(new OpenLayers.Projection("EPSG:900913"),new OpenLayers.Projection("EPSG:4326"));
  5450. var urLink = document.location.href;
  5451. var urLayers = '&layers='+W.map.mapState.getLayerVisibilityBitmask();
  5452. urLink = urLink.substr(0,urLink.indexOf('?zoom'));
  5453. urLink += '?zoom=5&lat='+urPos.lat+'&lon='+urPos.lon+urLayers;
  5454.  
  5455. if(isUR) urLink += '&mapUpdateRequest='+idSrc;
  5456. else if(isTurnProb) urLink += '&showturn='+idSrc+'&endshow';
  5457. else if(isProblem) urLink += '&mapProblem='+idSrc;
  5458. else if(isPlaceUpdate) urLink += '&showpur='+idSrc+'&endshow';
  5459.  
  5460. targetTab = "_uroTab_" + Math.round(Math.random()*1000000);
  5461. result += '<hr><ul><li><a href="'+urLink+'" id="_openInNewTab" target="'+targetTab+'">Open in new tab</a> - ';
  5462. objHasOpenInNewTabLink = true;
  5463. result += '<a href="#" id="_recentreSession">centre in current tab</a>';
  5464. objHasRecentreSessionLink = true;
  5465.  
  5466. // add "open new livemap tab" link
  5467. var lmLink = null;
  5468. if(document.getElementsByClassName("waze-header-menu").length === 0)
  5469. {
  5470. uroAddLog('Livemap link in livemap element');
  5471. lmLink = document.getElementById('livemap').href;
  5472. }
  5473. else
  5474. {
  5475. uroAddLog('Livemap link in header menu, locating...');
  5476. var menuItems = document.getElementsByClassName("waze-header-menu")[0];
  5477. for(var miloop = 0; miloop<menuItems.childElementCount; miloop++)
  5478. {
  5479. if(menuItems.children[miloop].innerHTML.indexOf('livemap') != -1)
  5480. {
  5481. uroAddLog('found link in menu entry '+miloop);
  5482. lmLink = menuItems.children[miloop].getElementsByTagName('a')[0].href;
  5483. uroAddLog(lmLink);
  5484. }
  5485. }
  5486. }
  5487. if(lmLink !== null)
  5488. {
  5489. var zpos = lmLink.indexOf('?');
  5490. if(zpos > -1) lmLink = lmLink.substr(0,zpos);
  5491. lmLink += '?zoom=17&lat='+urPos.lat+'&lon='+urPos.lon+'&layers=BTTTT';
  5492. result += '<li><a href="'+lmLink+'" target="_lmTab">Open in new livemap tab</a>';
  5493. }
  5494.  
  5495. if(!isPlaceUpdate)
  5496. {
  5497. // add "ignore for this session" link
  5498. result += '<li><a href="#" id="_addtoignore">Hide for this session</a></ul>';
  5499. objHasIgnoreLink = true;
  5500. }
  5501. }
  5502. }
  5503.  
  5504. // look for cameras
  5505. if((newPopupType === null) && (uroGetCBChecked('_cbInhibitCamPopup') === false))
  5506. {
  5507. for(var clFeatureIdx = 0; clFeatureIdx < W.map.camerasLayer.features.length; clFeatureIdx++)
  5508. {
  5509. if(W.map.camerasLayer.features[clFeatureIdx].renderIntent == 'highlight')
  5510. {
  5511. if(W.map.camerasLayer.features[clFeatureIdx].fid === null) ureq = W.map.camerasLayer.features[clFeatureIdx].model;
  5512. else ureq = W.map.camerasLayer.features[clFeatureIdx];
  5513.  
  5514. if(ureq.fid === null) ureqID = ureq.attributes.id;
  5515. else ureqID = ureq.fid;
  5516.  
  5517. // test isSelected() so that we only do overview data on cameras that are being hovered over
  5518. if(ureq.isSelected() === false)
  5519. {
  5520. uroPopupY -= 20;
  5521. newPopupType = 'camera';
  5522. uroFID = ureqID;
  5523. uroAddDebug(newPopupType+' '+uroFID);
  5524. uroAddLog('generating popup for camera '+uroFID);
  5525. if(I18n.lookup("edit.camera.fields.type") === undefined)
  5526. {
  5527. result += '<b>Camera: ' + ureq.TYPES[ureq.attributes.type] + '</b><br>';
  5528. }
  5529. else
  5530. {
  5531. result += '<b>Camera: ' + I18n.lookup("edit.camera.fields.type")[ureq.attributes.type] + '</b><br>';
  5532. }
  5533. result += 'ID: '+uroFID+'<br>';
  5534. result += 'Created by ';
  5535. var userID;
  5536. if(W.model.users.get(ureq.attributes.createdBy) !== undefined)
  5537. {
  5538. userID = ureq.attributes.createdBy;
  5539. result += uroGetUserNameAndRank(userID);
  5540. }
  5541. else result += 'unknown';
  5542. result += ', ';
  5543. var camAge = uroGetCameraAge(ureq,1);
  5544. if(camAge != -1)
  5545. {
  5546. result += uroParseDaysAgo(camAge);
  5547. }
  5548. else result += 'unknown days ago';
  5549. result += '<br>Updated by ';
  5550. if(W.model.users.get(ureq.attributes.updatedBy) !== undefined)
  5551. {
  5552. userID = ureq.attributes.updatedBy;
  5553. var userName = W.model.users.objects[userID].userName;
  5554. var userLevel = W.model.users.objects[userID].rank + 1;
  5555. result += userName + ' (' + userLevel + ')';
  5556. }
  5557. else result += 'unknown';
  5558. result += ', ';
  5559. camAge = uroGetCameraAge(ureq,0);
  5560. if(camAge != -1)
  5561. {
  5562. result += uroParseDaysAgo(camAge);
  5563. }
  5564. else result += 'unknown days ago';
  5565. result += '<br>Speed data: ';
  5566. result += uroGetCameraSpeedString(ureq.attributes.speed);
  5567. result += '<hr><ul>';
  5568. if(uroIsCamOnWatchList(uroFID) != -1)
  5569. {
  5570. result += '<li><a href="#" id="_updatewatchlist">Update watchlist entry</a>';
  5571. result += '<li><a href="#" id="_removefromwatchlist">Remove from watchlist</a>';
  5572. objHasUpdateWatchLink = true;
  5573. objHasRemoveWatchLink = true;
  5574. }
  5575. else
  5576. {
  5577. result += '<li><a href="#" id="_addtowatchlist">Add to watchlist</a>';
  5578. objHasAddWatchLink = true;
  5579. }
  5580. if(ureq.attributes.permissions !== 0)
  5581. {
  5582. result += '<li><a href="#" id="_deleteobject">Delete Camera</a>';
  5583. objHasDeleteLink = true;
  5584. }
  5585. result += '</ul>';
  5586. }
  5587. break;
  5588. }
  5589. }
  5590. }
  5591.  
  5592.  
  5593. if((newPopupType !== null) && (uroPopupDwellTimer === 0))
  5594. {
  5595. if((uroFID != uroShownFID) || (newPopupType != uroShownPopupType))
  5596. {
  5597. if(uroFID != uroShownFID) uroAddLog('FID mismatch, show popup: '+uroFID+'/'+uroShownFID);
  5598. else uroAddLog('Popup type mismatch: '+newPopupType+'/'+uroShownPopupType);
  5599. uroShownFID = uroFID;
  5600. uroShownPopupType = newPopupType;
  5601. uroPopupShown = false;
  5602. }
  5603.  
  5604. if(uroPopupShown === false)
  5605. {
  5606. uroAddLog('display popup at '+uroPopupX+','+uroPopupY);
  5607. uroPopupShown = true;
  5608. uroDiv.style.height = "auto";
  5609. uroDiv.style.width = "auto";
  5610. uroDiv.innerHTML = result;
  5611. if((uroFID != -1) && (objHasIgnoreLink === true))
  5612. {
  5613. uroAddEventListener('_addtoignore','click', uroAddToIgnoreList, true);
  5614. }
  5615. if(objHasDeleteLink === true)
  5616. {
  5617. uroAddEventListener('_deleteobject','click', uroDeleteObject, true);
  5618. }
  5619. if(objHasRemoveWatchLink === true)
  5620. {
  5621. uroAddEventListener('_removefromwatchlist','click', uroRemoveCamFromWatchList, true);
  5622. }
  5623. if(objHasAddWatchLink === true)
  5624. {
  5625. uroAddEventListener('_addtowatchlist','click', uroAddCamToWatchList, true);
  5626. }
  5627. if(objHasUpdateWatchLink === true)
  5628. {
  5629. uroAddEventListener('_updatewatchlist','click', uroUpdateCamWatchList, true);
  5630. }
  5631. if(objHasOpenInNewTabLink === true)
  5632. {
  5633. uroAddEventListener('_openInNewTab','mouseup', uroOpenNewTab, true);
  5634. }
  5635. if(objHasRecentreSessionLink === true)
  5636. {
  5637. if(isUR) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnUR, true);
  5638. else if((isProblem)||(isTurnProb)) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnMP, true);
  5639. else if(isPlaceUpdate) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnPUR, true);
  5640. else if(isVenue) uroAddEventListener('_recentreSession', 'click', uroRecentreSessionOnVenueNavPoint, true);
  5641. }
  5642. if(newPopupType == 'turn_restriction')
  5643. {
  5644. uroAddEventListener('_editTBR','click', uroEditTBR, true);
  5645. }
  5646.  
  5647. // restrict the popup width to be no wider than half the window width to avoid it completely
  5648. // overlapping the marker it's associated with - by keeping it to half the window width we
  5649. // guarantee that it'll fit either to the left or the right of the marker no matter how far
  5650. // across the screen the marker is located...
  5651. rw = parseInt(uroDiv.clientWidth);
  5652. if(rw > (window.innerWidth / 2))
  5653. {
  5654. rw = (window.innerWidth / 2);
  5655. uroDiv.style.width = rw+'px';
  5656. }
  5657. // get the div height after any adjustment of the width above, to account for whatever content
  5658. // reflow may have occurred as a result of reducing the width...
  5659. rh = parseInt(uroDiv.clientHeight);
  5660.  
  5661. if((uroPopupX + rw) > window.innerWidth)
  5662. {
  5663. // where the popup would be off the right hand side of the screen, move it completely over to the
  5664. // other side of the mouse pointer
  5665. uroPopupX -= (rw + 20);
  5666. if(uroPopupX < 0) uroPopupX = 0;
  5667. }
  5668. if((uroPopupY + rh) > window.innerHeight)
  5669. {
  5670. // where the popup would be off the bottom of the screen, shift it up just far enough to be
  5671. // fully visible
  5672. uroPopupY -= (((uroPopupY + rh) - window.innerHeight) + 30);
  5673. if(uroPopupY < 0) uroPopupY = 0;
  5674. }
  5675. uroDiv.style.top = uroPopupY+'px';
  5676. uroDiv.style.left = uroPopupX+'px';
  5677. uroDiv.style.visibility = 'visible';
  5678. }
  5679. uroPopupTimer = -1;
  5680. }
  5681. else if((newPopupType === null) && (uroPopupDwellTimer !== 0) && (uroPopupShown === true))
  5682. {
  5683. uroHidePopup();
  5684. }
  5685. else
  5686. {
  5687. if((uroPopupTimer == -1) && (uroFID != uroShownFID))
  5688. {
  5689. uroPopupTimer = uroGetElmValue('_inputPopupEntryTimeout');
  5690. }
  5691. }
  5692. }
  5693.  
  5694. function uroRestyleWMETabs()
  5695. {
  5696. if(!uroGetCBChecked('_cbDisableTabStyling'))
  5697. {
  5698. // The nav-tabs class is now also used for the General/Closures tabs on the segment edit panel, so we have
  5699. // to restrict the scope of this code to just those nav-tab classed elements within the user-tabs element.
  5700. var navTabs = document.getElementById('user-tabs').getElementsByClassName("nav-tabs")[0].children;
  5701. for(var loop = 0; loop<navTabs.length; loop++)
  5702. {
  5703. navTabs[loop].children[0].style.padding = "4px";
  5704. }
  5705. }
  5706. var panelDisplay = '';
  5707. if(uroGetCBChecked('_cbHideEditorInfo'))
  5708. {
  5709. panelDisplay = "none";
  5710. }
  5711. document.getElementById("user-details").style.display = panelDisplay;
  5712. setTimeout(uroRestyleWMETabs,1000);
  5713. }
  5714.  
  5715. function uroExclusiveCB()
  5716. {
  5717. var cbChecked = uroGetCBChecked(this.id);
  5718.  
  5719. if(cbChecked === true)
  5720. {
  5721. var pairedList = this.attributes.pairedWith.value.split(',');
  5722. for(var i=0; i<pairedList.length; i++)
  5723. {
  5724. uroSetCBChecked(pairedList[i], false);
  5725. }
  5726. }
  5727. }
  5728.  
  5729.  
  5730. function uroGetAMs(e)
  5731. {
  5732. var amList = '';
  5733. if(W.map.managedAreasLayer.getVisibility() === true)
  5734. {
  5735. var mouseX = e.pageX - document.getElementById('map').getBoundingClientRect().left;
  5736. var mouseY = e.pageY - document.getElementById('map').getBoundingClientRect().top - document.getElementById('toolbar').clientHeight;
  5737. var mousePixel = new OL.Pixel(mouseX, mouseY);
  5738. var mousePoint = W.map.getLonLatFromPixel(mousePixel).toPoint();
  5739.  
  5740. for(var amObj in W.model.managedAreas.objects)
  5741. {
  5742. if(W.model.managedAreas.objects[amObj].geometry.containsPoint(mousePoint))
  5743. {
  5744. if(amList !== '') amList += ', ';
  5745. amList += uroGetUserNameAndRank(W.model.managedAreas.objects[amObj].userID);
  5746. }
  5747. }
  5748. if(amList === '')
  5749. {
  5750. amList = 'none';
  5751. }
  5752. amList = "<b>Area Managers:</b> "+amList;
  5753. }
  5754. document.getElementById("uroAMList").innerHTML = amList;
  5755. }
  5756.  
  5757.  
  5758. function uroMouseDown()
  5759. {
  5760. uroMouseIsDown = true;
  5761. }
  5762.  
  5763. function uroMouseUp()
  5764. {
  5765. uroMouseIsDown = false;
  5766. }
  5767.  
  5768. function uroUREvent_onObjectsChanged()
  5769. {
  5770. }
  5771.  
  5772. function uroUREvent_onObjectsAdded()
  5773. {
  5774. if(uroGetCBChecked('_cbURResolverIDFilter') === true)
  5775. {
  5776. uroUpdateResolverList();
  5777. }
  5778. uroFilterURs();
  5779. }
  5780.  
  5781. function uroUREvent_onObjectsRemoved()
  5782. {
  5783. }
  5784.  
  5785. function uroGetSelectedURCommentCount()
  5786. {
  5787. if(W.model.updateRequestSessions.objects[uroSelectedURID] !== undefined)
  5788. {
  5789. var cachedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
  5790. uroAddLog(uroSelectedURID+':'+cachedCommentCount+' '+uroExpectedCommentCount);
  5791.  
  5792. // if there aren't the same number of cached comments as there are comments in the UR dialog list, initiate
  5793. // a refresh of the comment data...
  5794. if(cachedCommentCount != uroExpectedCommentCount)
  5795. {
  5796. if(uroPendingCommentDataRefresh === true)
  5797. {
  5798. if(cachedCommentCount > 0)
  5799. {
  5800. uroCachedLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].comments[cachedCommentCount-1].id;
  5801. }
  5802. else
  5803. {
  5804. uroCachedLastCommentID = null;
  5805. }
  5806. uroAddLog('updateRequestSessions refresh required for UR '+uroSelectedURID);
  5807. if(uroCachedLastCommentID !== null)
  5808. {
  5809. uroAddLog('last comment ID for this UR is '+uroCachedLastCommentID);
  5810. }
  5811. else
  5812. {
  5813. uroAddLog('first comment for this UR, no previous comment to ID');
  5814. }
  5815. var idList = [];
  5816. idList.push(uroSelectedURID);
  5817. // need to delete the existing cache object first, as .get() is only capable of creating new objects,
  5818. // it doesn't seem able to update an existing object with new data
  5819. W.model.updateRequestSessions.remove(W.model.updateRequestSessions.objects[uroSelectedURID]);
  5820. W.model.updateRequestSessions.get(idList);
  5821. // the call to .get() initiates a XMLHttpRequest for the data, so we now need to switch modes - the
  5822. // refresh process has started so we're no longer pending, but we are now waiting for the XMLHttpRequest
  5823. // to return something...
  5824. uroPendingCommentDataRefresh = false;
  5825. uroWaitingCommentDataRefresh = true;
  5826. }
  5827. else
  5828. {
  5829. if(cachedCommentCount > 0)
  5830. {
  5831. var currentLastCommentID = W.model.updateRequestSessions.objects[uroSelectedURID].comments[cachedCommentCount-1].id;
  5832. if(currentLastCommentID == uroCachedLastCommentID)
  5833. {
  5834. // most recent comment loaded for this UR is the same one that was present at the start of this
  5835. // refresh process, so kick back into pending mode so we can retry the .get()...
  5836. uroAddLog('latest comment ID still the same, reverting to pending mode...');
  5837. uroPendingCommentDataRefresh = true;
  5838. }
  5839. else
  5840. {
  5841. // something may have gone awry here - the most recent comment loaded for this UR doesn't have the
  5842. // same ID as the one present at the start of the refresh process, yet the comment counts still don't
  5843. // match up, which suggests either a comment got lost along the way or someone else has commented on
  5844. // the same UR at almost the same time. To get out of the loop this would create, assume that a
  5845. // mismatch in the IDs means the .get() has completed successfully no matter what the new comment
  5846. // count is, and take this new count to be the count we were expecting all along...
  5847. uroAddLog('latest comment ID different, but expected count not correct...');
  5848. uroExpectedCommentCount = cachedCommentCount;
  5849. }
  5850. }
  5851. else
  5852. {
  5853. uroAddLog('first comment on this UR not received yet, reverting to pending mode...');
  5854. uroPendingCommentDataRefresh = true;
  5855. }
  5856. }
  5857.  
  5858. }
  5859. else
  5860. {
  5861. // if the WME session is loaded with a UR already selected, such that WME has opened the UR dialog as part
  5862. // of the session startup process, adding new comments to the UR cause the cached data to be updated immediately.
  5863. // This prevents URO+ from switching into waiting mode in the above block of code, so we have to instead do
  5864. // it here by comparing the cached count against the expected count following the Send click event.
  5865. if(cachedCommentCount >= uroExpectedCommentCount)
  5866. {
  5867. uroPendingCommentDataRefresh = false;
  5868. uroWaitingCommentDataRefresh = true;
  5869. uroExpectedCommentCount = null;
  5870. }
  5871.  
  5872. // once the cached data has been updated, refilter the URs so that the new comment count is taken into account
  5873. // immediately for filtering and display purposes
  5874. if(uroWaitingCommentDataRefresh === true)
  5875. {
  5876. uroWaitingCommentDataRefresh = false;
  5877. uroFilterURs();
  5878. uroAddLog('refresh complete');
  5879. }
  5880. }
  5881. }
  5882. }
  5883.  
  5884. function uroAddedComment()
  5885. {
  5886. // when the user clicks the Send button to submit a new UR comment, this event handler fires before the new comment is
  5887. // posted to the server and thus also before the comment list gets updated in the UR dialog. So we take the current
  5888. // comment count and, if the new comment edit box isn't empty, increment it by 1 to get the expected count. Then we
  5889. // set the pending flag true to initiate a session refresh on the next 100ms tick
  5890. uroExpectedCommentCount = W.map.panelRegion.currentView.conversationView.conversation.comments.length;
  5891. if(document.getElementsByClassName('new-comment-text')[0].value !== '')
  5892. {
  5893. uroExpectedCommentCount++;
  5894. uroAddLog('new comment added to UR '+uroSelectedURID+', cache refresh required...');
  5895. uroPendingCommentDataRefresh = true;
  5896. }
  5897. else
  5898. {
  5899. uroPendingCommentDataRefresh = false;
  5900. }
  5901. }
  5902.  
  5903. function uroInhibitNextUpdateRequestButton(e)
  5904. {
  5905. e.stopPropagation();
  5906. document.getElementsByClassName('close-panel')[0].click();
  5907. }
  5908.  
  5909. function uroCloneClosure()
  5910. {
  5911. console.debug(this.id);
  5912. var closureOffset = parseInt(this.id.split('-')[1]);
  5913. console.debug(closureOffset);
  5914. // grab the current closure details from the UI...
  5915. document.getElementsByClassName('closure-item')[closureOffset].children[0].children[0].children[1].click();
  5916. var cLocation = document.getElementsByName('closure_location')[0].value;
  5917. var cReason = document.getElementsByName('closure_reason')[0].value;
  5918. var cDirection = document.getElementsByName('closure_direction')[0].selectedIndex;
  5919. var cHasStartDate = document.getElementsByName('closure_hasStartDate')[0].checked;
  5920. var cStartDate = document.getElementsByName('closure_startDate')[0].value;
  5921. var cStartTime = document.getElementsByName('closure_startTime')[0].value;
  5922. var cEndDate = document.getElementsByName('closure_endDate')[0].value;
  5923. var cEndTime = document.getElementsByName('closure_endTime')[0].value;
  5924. var cIgnoreTraffic = document.getElementsByName('closure_permanent')[0].checked;
  5925. document.getElementsByClassName('closures')[0].getElementsByClassName('cancel-button')[0].click();
  5926. // now open the add closure UI and populate it with the values we've just grabbed...
  5927. document.getElementsByClassName('add-closure-button')[0].click();
  5928. document.getElementsByName('closure_location')[0].value = cLocation;
  5929. // need to generate a change event on each of the form fields, because WME appears to be silently populating some hidden
  5930. // closure object with the details as they're entered manually, and if we just set the form values without then forcing
  5931. // the change event as well then WME will end up using its default values instead of the ones we've so lovingly copied...
  5932. document.getElementsByName('closure_location')[0].dispatchEvent(new Event('change', { 'bubbles': true }))
  5933. document.getElementsByName('closure_reason')[0].value = cReason;
  5934. document.getElementsByName('closure_reason')[0].dispatchEvent(new Event('change', { 'bubbles': true }))
  5935. document.getElementsByName('closure_direction')[0].selectedIndex = cDirection;
  5936. document.getElementsByName('closure_direction')[0].dispatchEvent(new Event('change', { 'bubbles': true }))
  5937. document.getElementsByName('closure_hasStartDate')[0].checked = cHasStartDate;
  5938. document.getElementsByName('closure_hasStartDate')[0].dispatchEvent(new Event('change', { 'bubbles': true }))
  5939. document.getElementsByName('closure_startDate')[0].value = cStartDate;
  5940. document.getElementsByName('closure_startDate')[0].dispatchEvent(new Event('change', { 'bubbles': true }));
  5941. document.getElementsByName('closure_startTime')[0].value = cStartTime;
  5942. document.getElementsByName('closure_startTime')[0].dispatchEvent(new Event('change', { 'bubbles': true }))
  5943. document.getElementsByName('closure_endDate')[0].value = cEndDate;
  5944. document.getElementsByName('closure_endDate')[0].dispatchEvent(new Event('change', { 'bubbles': true }))
  5945. document.getElementsByName('closure_endTime')[0].value = cEndTime;
  5946. document.getElementsByName('closure_endTime')[0].dispatchEvent(new Event('change', { 'bubbles': true }))
  5947. document.getElementsByName('closure_permanent')[0].checked = cIgnoreTraffic;
  5948. document.getElementsByName('closure_permanent')[0].dispatchEvent(new Event('change', { 'bubbles': true }))
  5949. }
  5950. function uroTenthSecondTick()
  5951. {
  5952. if(uroSetupListeners)
  5953. {
  5954. if(W.loginManager.isLoggedIn())
  5955. {
  5956. uroSetupListeners = false;
  5957.  
  5958. // filter markers when the marker objects are modified (this happens whenever WME needs to load fresh marker data
  5959. // due to having panned/zoomed the map beyond the extents of the previously loaded data)
  5960. W.model.mapUpdateRequests.events.register("objectschanged", null, uroFilterURs_onObjectsChanged);
  5961. W.model.mapUpdateRequests.events.register("objectsadded", null, uroFilterURs_onObjectsAdded);
  5962. W.model.mapUpdateRequests.events.register("objectsremoved", null, uroFilterURs_onObjectsRemoved);
  5963.  
  5964. W.model.updateRequestSessions.events.register("objectschanged", null, uroUREvent_onObjectsChanged);
  5965. W.model.updateRequestSessions.events.register("objectsadded", null, uroUREvent_onObjectsAdded);
  5966. W.model.updateRequestSessions.events.register("objectsremoved", null, uroUREvent_onObjectsRemoved);
  5967.  
  5968. W.model.cameras.events.register("objectschanged", null, uroFilterCameras);
  5969. W.model.cameras.events.register("objectsadded", null, uroFilterCameras);
  5970. W.model.cameras.events.register("objectsremoved", null, uroFilterCameras);
  5971. W.model.problems.events.register("objectschanged", null, uroFilterProblems);
  5972. W.model.problems.events.register("objectsadded", null, uroFilterProblems);
  5973. W.model.problems.events.register("objectsremoved", null, uroFilterProblems);
  5974. W.model.venues.events.register("objectschanged", null, uroFilterPlaces);
  5975. W.model.venues.events.register("objectsadded", null, uroFilterPlaces);
  5976. W.model.venues.events.register("objectsremoved", null, uroFilterPlaces);
  5977.  
  5978. var userTabs = document.getElementById(uroUserTabId);
  5979. var tabContent = null;
  5980.  
  5981. var navTabs = userTabs.getElementsByClassName('nav-tabs')[0];
  5982. tabContent = document.getElementById('user-info').getElementsByClassName('tab-content')[0];
  5983. var newtabUR = document.createElement('li');
  5984. newtabUR.innerHTML = '<a href="#sidepanel-uroverview" data-toggle="tab">URO+</a>';
  5985. navTabs.appendChild(newtabUR);
  5986. uroControls.id = "sidepanel-uroverview";
  5987. uroControls.className = "tab-pane";
  5988. tabContent.appendChild(uroControls);
  5989.  
  5990. uroAddEventListener('_btnUndoLastHide',"click", uroRemoveLastAddedIgnore, true);
  5991. uroAddEventListener('_btnClearSessionHides',"click", uroRemoveAllIgnores, true);
  5992. uroEnableIgnoreListControls();
  5993.  
  5994. uroAddEventListener('_btnClearCamWatchList',"click", uroClearCamWatchList, true);
  5995. uroAddEventListener('_btnSettingsToText',"click", uroSettingsToText, true);
  5996. uroAddEventListener('_btnTextToSettings',"click", uroTextToSettings, true);
  5997. uroAddEventListener('_btnResetSettings',"click", uroDefaultSettings, true);
  5998. uroAddEventListener('_btnClearSettingsText',"click", uroClearSettingsText, true);
  5999. uroAddEventListener('_cbMasterEnable',"click", uroFilterItems_MasterEnableClick, true);
  6000. uroAddEventListener('_btnDebugToScreen',"click", uroDumpDebug, true);
  6001.  
  6002. uroSetOnClick("_linkSelectUserRequests",uroShowURTab);
  6003. uroSetOnClick("_linkSelectMapProblems",uroShowMPTab);
  6004. uroSetOnClick("_linkSelectPlaces",uroShowPlacesTab);
  6005. uroSetOnClick("_linkSelectCameras",uroShowCameraTab);
  6006. uroSetOnClick("_linkSelectMisc",uroShowMiscTab);
  6007. uroSetOnClick("_linkSelectOWL",uroShowOWLTab);
  6008. for(var idx=0;idx<W.Config.venues.categories.length;idx++)
  6009. {
  6010. uroSetOnClick('_uroPlacesGroupState-'+idx,uroPlacesGroupCollapseExpand);
  6011. }
  6012. // add exclusiveCB click handlers to all checkboxes with a pairedWith attribute
  6013. var cbList = document.getElementsByTagName('input');
  6014. for (var optIdx=0;optIdx<cbList.length;optIdx++)
  6015. {
  6016. if((cbList[optIdx].id.indexOf('_cb') === 0) && (cbList[optIdx].attributes.pairedWith !== undefined))
  6017. {
  6018. uroSetOnClick(cbList[optIdx].id,uroExclusiveCB);
  6019. }
  6020. }
  6021.  
  6022. uroAddLog('finalise onload');
  6023. uroLoadSettings();
  6024. uroNewLookCheckDetailsRequest();
  6025. if(uroGetCBChecked('_cbEnableDTE'))
  6026. {
  6027. if(dteControlsIdx != -1)
  6028. {
  6029. dteSetNewTabLength();
  6030. }
  6031. else
  6032. {
  6033. uroAddLog('ERROR - archive panel not found!');
  6034. uroSetStyleDisplay(uroUserTabId,'');
  6035. }
  6036. }
  6037.  
  6038. // filter markers as and when the map is moved
  6039. W.map.events.register("moveend", null, uroFilterItems);
  6040. W.map.events.register("mousemove", null, uroGetAMs);
  6041. W.map.events.register("mousemove", null, uroNewLookHighlightedItemsCheck);
  6042. W.map.events.registerPriority("mousedown", null, uroMouseDown);
  6043.  
  6044. // trap mousedown on Streetview marker drag
  6045. document.getElementsByClassName('street-view-control')[0].onmousedown = uroMouseDown;
  6046.  
  6047. W.map.events.register("mouseup", null, uroMouseUp);
  6048.  
  6049. uroSetStyles(uroCtrlURs);
  6050. uroSetStyles(uroCtrlMPs);
  6051. uroSetStyles(uroCtrlPlaces);
  6052. uroSetStyles(uroCtrlCameras);
  6053. uroSetStyles(uroCtrlMisc);
  6054. uroSetStyles(uroOWL);
  6055.  
  6056. uroShowURTab();
  6057. uroRestyleWMETabs();
  6058. uroUserID = W.loginManager.getLoggedInUser().id;
  6059. uroFilterItems();
  6060. uroShowDebugOutput = uroPersistentDebugOutput;
  6061. var dbgMode = "none";
  6062. if(uroShowDebugOutput)
  6063. {
  6064. dbgMode = "inline";
  6065. }
  6066. document.getElementById('_uroDebugMode').style.display = dbgMode;
  6067. uroAddEventListener('_uroVersion',"click", uroToggleDebug, true);
  6068. }
  6069. }
  6070. else
  6071. {
  6072. var mousePos = document.getElementsByClassName('mouse-position')[0].innerHTML;
  6073.  
  6074. if(document.getElementsByClassName('panel')[0] === undefined)
  6075. {
  6076. uroHidePopupOnPanelOpen = true;
  6077. }
  6078.  
  6079. if(uroPopupShown === true)
  6080. {
  6081. var hidePopup = false;
  6082. if((mousePos == '00.00000, 00.00000') && (uroMouseInPopup === false))
  6083. {
  6084. hidePopup = true;
  6085. }
  6086. if(document.getElementsByClassName('panel')[0] !== undefined)
  6087. {
  6088. if(uroHidePopupOnPanelOpen === true)
  6089. {
  6090. hidePopup = true;
  6091. uroHidePopupOnPanelOpen = false;
  6092. }
  6093. }
  6094.  
  6095. if(hidePopup === true)
  6096. {
  6097. uroHidePopup();
  6098. }
  6099. }
  6100.  
  6101. if((uroAreaNameHoverObj !== null) && (uroAreaNameHoverTime != -1) && (uroAreaNameOverlayShown === false))
  6102. {
  6103. if(++uroAreaNameHoverTime > 5)
  6104. {
  6105. uroAreaNameOverlaySetup();
  6106. }
  6107. }
  6108. uroReplaceAreaNames(false);
  6109.  
  6110. if(uroPopupTimer > 0)
  6111. {
  6112. if(uroMouseInPopup === false)
  6113. {
  6114. uroPopupTimer--;
  6115. }
  6116. }
  6117. if(uroPopupTimer === 0)
  6118. {
  6119. uroHidePopup();
  6120. }
  6121.  
  6122. if(uroPopupDwellTimer > 0)
  6123. {
  6124. uroPopupDwellTimer--;
  6125. if(uroPopupDwellTimer === 0)
  6126. {
  6127. uroNewLookHighlightedItemsCheck('dwellTimeout');
  6128. }
  6129. }
  6130.  
  6131. if(document.getElementsByClassName("archive-panel")[0] === undefined)
  6132. {
  6133. if(dteClearHighlightsOnPanelClose)
  6134. {
  6135. dteClearListHighlight();
  6136. dteClearHighlightsOnPanelClose = false;
  6137. }
  6138. }
  6139. else
  6140. {
  6141. if(dteArmClearHighlightsOnPanelClose)
  6142. {
  6143. dteArmClearHighlightsOnPanelClose = false;
  6144. dteClearHighlightsOnPanelClose = true;
  6145. }
  6146. }
  6147.  
  6148. // replace the "next xxx" button on UR, MP and PUR editing UIs
  6149. if(W.map.panelRegion.hasView() === true)
  6150. {
  6151. var doneString = I18n.lookup('problems.panel.done');
  6152. var updateButton = false;
  6153.  
  6154. var nurButton = document.getElementsByClassName('btn btn-block next')[0];
  6155. if(nurButton !== undefined)
  6156. {
  6157. var nextURString = I18n.lookup('update_requests.panel.next');
  6158. var nextMPString = I18n.lookup('problems.panel.next');
  6159. updateButton = ((nurButton.innerHTML.indexOf(nextURString) !== -1) && (uroGetCBChecked('_cbInhibitNURButton') === true));
  6160. if(updateButton === false)
  6161. {
  6162. updateButton = ((nurButton.innerHTML.indexOf(nextMPString) !== -1) && (uroGetCBChecked('_cbInhibitNMPButton') === true));
  6163. }
  6164. }
  6165. else
  6166. {
  6167. nurButton = document.getElementsByClassName('btn btn-block next-venue')[0];
  6168. if(nurButton !== undefined)
  6169. {
  6170. var nextPURString = I18n.lookup('venues.update_requests.panel.next_venue');
  6171. updateButton = ((nurButton.innerHTML.indexOf(nextPURString) !== -1) && (uroGetCBChecked('_cbInhibitNPURButton') === true));
  6172. }
  6173. }
  6174.  
  6175. if(updateButton === true)
  6176. {
  6177. // first change its label...
  6178. nurButton.innerHTML = doneString;
  6179. // ...then add a new click handler that'll inhibit the native ones
  6180. nurButton.addEventListener("click", uroInhibitNextUpdateRequestButton, false);
  6181. uroAddLog('inhibit Next UR/MP/PUR button');
  6182. }
  6183. }
  6184. // test for the opening or closing of the UR editing dialog so we can detect when a new comment is added
  6185. var URDialogIsOpen = (document.getElementsByClassName('new-comment-form').length == 1);
  6186. if(URDialogIsOpen)
  6187. {
  6188. var thisSelectedURID = W.map.panelRegion.currentView.conversationView.conversation.getID();
  6189. if(thisSelectedURID != uroSelectedURID)
  6190. {
  6191. // if the user selects a new UR whilst the editing dialog is still open, treat it in the
  6192. // same way as if the user had selected that UR with the dialog closed
  6193. uroURDialogIsOpen = false;
  6194. }
  6195. if(uroURDialogIsOpen === false)
  6196. {
  6197. // user is editing a new UR
  6198. uroSelectedURID = thisSelectedURID;
  6199. // add our own click event handler to the Send button, so we can do stuff whenever a new comment is added
  6200. document.getElementsByClassName('new-comment-form')[0].getElementsByClassName('btn')[0].addEventListener("click", uroAddedComment, false);
  6201. uroAddLog('user is editing UR '+uroSelectedURID);
  6202. uroExpectedCommentCount = W.model.updateRequestSessions.objects[uroSelectedURID].comments.length;
  6203. }
  6204. }
  6205. else if(uroURDialogIsOpen === true)
  6206. {
  6207. // dialog was open and has now been closed
  6208. uroSelectedURID = null;
  6209. }
  6210. uroURDialogIsOpen = URDialogIsOpen;
  6211.  
  6212. if(((uroPendingCommentDataRefresh === true) || (uroWaitingCommentDataRefresh === true)) && (uroSelectedURID !== null))
  6213. {
  6214. uroAddLog('check completion of comment data refresh for UR '+uroSelectedURID+' ('+uroPendingCommentDataRefresh+','+uroWaitingCommentDataRefresh+')');
  6215. uroGetSelectedURCommentCount();
  6216. }
  6217.  
  6218. var selectedTotal = W.selectionManager.selectedItems.length;
  6219. if((selectedTotal > 0) && (document.getElementById('_uroDivOWLBtns') === null))
  6220. {
  6221. var selectedClass = W.selectionManager.selectedItems[0].model.CLASS_NAME;
  6222. var displayAddToOWLBtn = false;
  6223. var displayUpdateOWLBtn = false;
  6224. var displayRemoveFromOWLBtn = false;
  6225. var selectedSegments = false;
  6226. var selectedLandmarks = false;
  6227. var fid;
  6228. var loop;
  6229.  
  6230. // WME only seems to allow multi-object selections for segments, so testing the class of the first object in the
  6231. // selection list tells us the class of any other objects in the list too...
  6232. if(selectedClass == "Waze.Feature.Vector.Segment")
  6233. {
  6234. selectedSegments = true;
  6235. for(loop=0; loop<selectedTotal; loop++)
  6236. {
  6237. fid = W.selectionManager.selectedItems[loop].model.attributes.id;
  6238. var segIdx = uroIsSegOnWatchList(fid);
  6239. if(segIdx == -1)
  6240. {
  6241. displayAddToOWLBtn = true;
  6242. }
  6243. else
  6244. {
  6245. if(uroSegDataChanged(segIdx))
  6246. {
  6247. displayUpdateOWLBtn = true;
  6248. }
  6249. displayRemoveFromOWLBtn = true;
  6250. }
  6251. }
  6252. }
  6253.  
  6254. else if(selectedClass == "Waze.Feature.Vector.Landmark")
  6255. {
  6256. selectedLandmarks = true;
  6257. for(loop=0; loop<selectedTotal; loop++)
  6258. {
  6259. fid = W.selectionManager.selectedItems[loop].model.attributes.id;
  6260. var placeIdx = uroIsPlaceOnWatchList(fid);
  6261. if(placeIdx == -1)
  6262. {
  6263. displayAddToOWLBtn = true;
  6264. }
  6265. else
  6266. {
  6267. if(uroPlaceDataChanged(placeIdx))
  6268. {
  6269. displayUpdateOWLBtn = true;
  6270. }
  6271. displayRemoveFromOWLBtn = true;
  6272. }
  6273. }
  6274. }
  6275.  
  6276. var btnHTML = '<div id="_uroDivOWLBtns">';
  6277. if((displayAddToOWLBtn === true) && (displayUpdateOWLBtn === false))
  6278. {
  6279. btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Add to OWL</button>';
  6280. }
  6281. else if((displayUpdateOWLBtn === true) && (displayAddToOWLBtn === false))
  6282. {
  6283. btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Update OWL</button>';
  6284. }
  6285. else if((displayAddToOWLBtn === true) && (displayUpdateOWLBtn === true))
  6286. {
  6287. btnHTML += '<button class="btn btn-default" id="_btnAddUpdateOWL">Add to & Update OWL</button>';
  6288. }
  6289.  
  6290. if(displayRemoveFromOWLBtn === true)
  6291. {
  6292. btnHTML += '<button class="btn btn-default" id="_btnRemoveOWL">Remove from OWL</button>';
  6293. }
  6294. btnHTML += '</div>';
  6295. /*
  6296. // once we get around to enabling these again, remember that altering the inner HTML of the
  6297. // segment-edit-general panel when the selected segment is part of a roundabout then disables
  6298. // the onclick handler for the select roundabout button...
  6299. //
  6300. // also remember that the current WME beta has yet another different side panel arrangement
  6301. if(selectedSegments === true)
  6302. {
  6303. document.getElementById("segment-edit-general").innerHTML += btnHTML;
  6304. }
  6305. else if(selectedLandmarks === true)
  6306. {
  6307. document.getElementById("landmark-edit-general").innerHTML += btnHTML;
  6308. }
  6309.  
  6310. if((displayAddToOWLBtn === true)||(displayUpdateOWLBtn === true))
  6311. {
  6312. if(selectedSegments === true)
  6313. {
  6314. uroAddEventListener('_btnAddUpdateOWL','click', uroAddUpdateSegWatchList, true);
  6315. }
  6316. else
  6317. {
  6318. uroAddEventListener('_btnAddUpdateOWL','click', uroAddUpdatePlaceWatchList, true);
  6319. }
  6320. }
  6321.  
  6322. if(displayRemoveFromOWLBtn === true)
  6323. {
  6324. if(selectedSegments === true)
  6325. {
  6326. uroAddEventListener('_btnRemoveOWL','click', uroRemoveSegFromWatchList, true);
  6327. }
  6328. else
  6329. {
  6330. uroAddEventListener('_btnRemoveOWL','click', uroRemovePlaceFromWatchList, true);
  6331. }
  6332. }
  6333. */
  6334. }
  6335. // fix the livemap link in WME beta...
  6336. if(uroBetaEditor === true)
  6337. {
  6338. var lmLink = document.getElementsByClassName('waze-header-menu')[0].children[0].children[0].getAttribute('href');
  6339. if(lmLink.indexOf('https:') === -1)
  6340. {
  6341. uroAddLog('fixing livemap link...');
  6342. lmLink = 'https://www.waze.com' + lmLink;
  6343. document.getElementsByClassName('waze-header-menu')[0].children[0].children[0].setAttribute('href',lmLink);
  6344. }
  6345. }
  6346.  
  6347. // closure cloning support...
  6348. //
  6349. // has the closures tab been generated?
  6350. if(document.getElementById('segment-edit-closures') !== null)
  6351. {
  6352. // and is it active?
  6353. if(document.getElementById('segment-edit-closures').className === 'tab-pane active')
  6354. {
  6355. // and are there any closures defined for all of the selected segment(s)...
  6356. var nClosures = document.getElementsByClassName('full-closures')[0].childNodes.length;
  6357. if(nClosures > 0)
  6358. {
  6359. // and last but by no means least, have we already added the clone icon to this closure?
  6360. for(var cLoop = 0; cLoop < nClosures; cLoop++)
  6361. {
  6362. var btnElm = document.getElementsByClassName('full-closures')[0].childNodes[cLoop].children[0].children[0];
  6363. if(uroHasNewFontAwesome == null)
  6364. {
  6365. uroHasNewFontAwesome = (document.getElementsByClassName('full-closures')[0].childNodes[0].children[0].children[0].getElementsByClassName("fa-trash").length == 1);
  6366. }
  6367. if(btnElm.innerHTML.indexOf('_uroCloneClosure-') == -1)
  6368. {
  6369. var newAnchor = document.createElement('a');
  6370. var anchorID = '_uroCloneClosure-'+cLoop;
  6371. newAnchor.href="#";
  6372. if(uroHasNewFontAwesome)
  6373. {
  6374. newAnchor.innerHTML="<i class='fa fa-copy'></i>";
  6375. }
  6376. else
  6377. {
  6378. newAnchor.innerHTML="<i class='icon-copy'></i>";
  6379. }
  6380. newAnchor.id = anchorID;
  6381. btnElm.appendChild(newAnchor);
  6382. document.getElementById(anchorID).addEventListener("click", uroCloneClosure, false);
  6383. }
  6384. }
  6385. }
  6386. }
  6387. }
  6388. }
  6389. }
  6390.  
  6391.  
  6392. function uroActiveTab(_id)
  6393. {
  6394. var e = document.getElementById(_id);
  6395. e.style.backgroundColor = "aliceblue";
  6396. e.style.borderTop = "1px solid";
  6397. e.style.borderLeft = "1px solid";
  6398. e.style.borderRight = "1px solid";
  6399. e.style.borderBottom = "0px solid";
  6400. }
  6401.  
  6402. function uroInactiveTab(_id)
  6403. {
  6404. var e = document.getElementById(_id);
  6405. e.style.backgroundColor = "white";
  6406. e.style.borderTop = "0px solid";
  6407. e.style.borderLeft = "0px solid";
  6408. e.style.borderRight = "0px solid";
  6409. e.style.borderBottom = "1px solid";
  6410. }
  6411.  
  6412.  
  6413. function uroInactiveAllTabs()
  6414. {
  6415. uroInactiveTab("_tabSelectCameras");
  6416. uroInactiveTab("_tabSelectMapProblems");
  6417. uroInactiveTab("_tabSelectMisc");
  6418. uroInactiveTab("_tabSelectUserRequests");
  6419. uroInactiveTab("_tabSelectCWL");
  6420. uroInactiveTab("_tabSelectPlaces");
  6421.  
  6422. if(!uroCtrlsHidden)
  6423. {
  6424. uroSetStyleDisplay('uroCtrlURs','none');
  6425. uroSetStyleDisplay('uroCtrlMPs','none');
  6426. uroSetStyleDisplay('uroCtrlCameras','none');
  6427. uroSetStyleDisplay('uroCtrlMisc','none');
  6428. uroSetStyleDisplay('uroOWL','none');
  6429. uroSetStyleDisplay('uroCtrlPlaces','none');
  6430. }
  6431. }
  6432.  
  6433.  
  6434. function uroShowURTab()
  6435. {
  6436. uroInactiveAllTabs();
  6437. uroActiveTab("_tabSelectUserRequests");
  6438. uroCurrentTab = 1;
  6439. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlURs','block');
  6440. return false;
  6441. }
  6442.  
  6443.  
  6444. function uroShowMPTab()
  6445. {
  6446. uroInactiveAllTabs();
  6447. uroActiveTab("_tabSelectMapProblems");
  6448. uroCurrentTab = 2;
  6449. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMPs','block');
  6450. return false;
  6451. }
  6452.  
  6453. function uroShowPlacesTab()
  6454. {
  6455. uroInactiveAllTabs();
  6456. uroActiveTab("_tabSelectPlaces");
  6457. uroCurrentTab = 3;
  6458. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlPlaces','block');
  6459. for(var idx=0;idx<uroPlacesGroupsCollapsed.length;idx++)
  6460. {
  6461. uroPlacesGroupCEHandler(idx);
  6462. }
  6463. return false;
  6464. }
  6465.  
  6466. function uroShowCameraTab()
  6467. {
  6468. uroInactiveAllTabs();
  6469. uroActiveTab("_tabSelectCameras");
  6470. uroCurrentTab = 4;
  6471. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlCameras','block');
  6472. return false;
  6473. }
  6474.  
  6475. function uroShowOWLTab()
  6476. {
  6477. uroInactiveAllTabs();
  6478. uroActiveTab("_tabSelectCWL");
  6479. uroCurrentTab = 5;
  6480. if(!uroCtrlsHidden) uroSetStyleDisplay('uroOWL','block');
  6481. uroOWLUpdateHTML();
  6482. return false;
  6483. }
  6484.  
  6485. function uroShowMiscTab()
  6486. {
  6487. uroInactiveAllTabs();
  6488. uroActiveTab("_tabSelectMisc");
  6489. uroCurrentTab = 6;
  6490. if(!uroCtrlsHidden) uroSetStyleDisplay('uroCtrlMisc','block');
  6491. return false;
  6492. }
  6493.  
  6494.  
  6495. function uroNewLookCheckDetailsRequest()
  6496. {
  6497. var thisurl = document.location.href;
  6498. var doRetry = true;
  6499. var urID;
  6500. var endmarkerpos = thisurl.indexOf('&endshow');
  6501. var showmarkerpos = thisurl.indexOf('&showturn=');
  6502. if((endmarkerpos != -1) && (showmarkerpos != -1))
  6503. {
  6504. showmarkerpos += 10;
  6505. uroAddLog('showturn tab opened');
  6506. urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
  6507. uroAddLog(' turn problem ID = '+urID);
  6508.  
  6509. try
  6510. {
  6511. W.map.problemLayer.markers[urID].icon.imageDiv.click();
  6512. doRetry = false;
  6513. }
  6514. catch(err)
  6515. {
  6516. uroAddLog('problems not fully loaded, retrying...');
  6517. }
  6518.  
  6519. if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
  6520. }
  6521. else
  6522. {
  6523. showmarkerpos = thisurl.indexOf('&showpur=');
  6524. if((endmarkerpos != -1) && (showmarkerpos != -1))
  6525. {
  6526. showmarkerpos += 9;
  6527. uroAddLog('showPUR tab opened');
  6528. urID = thisurl.substr(showmarkerpos,endmarkerpos-showmarkerpos);
  6529. uroAddLog(' PUR ID = '+urID);
  6530.  
  6531. try
  6532. {
  6533. W.map.placeUpdatesLayer.markers[urID].icon.imageDiv.click();
  6534. doRetry = false;
  6535. }
  6536. catch(err)
  6537. {
  6538. uroAddLog('PURs not fully loaded, retrying...');
  6539. }
  6540.  
  6541. if(doRetry) setTimeout(uroNewLookCheckDetailsRequest,500);
  6542. }
  6543. }
  6544.  
  6545. }
  6546.  
  6547.  
  6548. function uroUpdateMPSolverList()
  6549. {
  6550. if(Object.keys(W.model.problems.objects).length === 0)
  6551. {
  6552. return;
  6553. }
  6554.  
  6555. var resolverList = [];
  6556. var selector = document.getElementById('_selectMPUserID');
  6557. var selectedUser = null;
  6558. if(selector.selectedOptions[0] !== undefined)
  6559. {
  6560. selectedUser = parseInt(selector.selectedOptions[0].value);
  6561. }
  6562. while(selector.options.length > 0)
  6563. {
  6564. selector.options.remove(0);
  6565. }
  6566. var selectedIdx = 0;
  6567. var idx = 0;
  6568.  
  6569. for (var mpobj in W.model.problems.objects)
  6570. {
  6571. if(W.model.problem.objects.hasOwnProperty(mpobj))
  6572. {
  6573. var prob = W.model.problems.objects[mpobj];
  6574. if(prob.attributes.resolvedBy !== null)
  6575. {
  6576. var userID = prob.attributes.resolvedBy;
  6577. var userName = W.model.users.objects[userID].userName;
  6578. if(resolverList.indexOf(userName) == -1)
  6579. {
  6580. resolverList.push(userName);
  6581. selector.options.add(new Option(userName, userID));
  6582. if(userID == selectedUser)
  6583. {
  6584. selectedIdx = idx;
  6585. }
  6586. idx++;
  6587. }
  6588. }
  6589. }
  6590. }
  6591.  
  6592. if(selectedIdx !== null)
  6593. {
  6594. selector.selectedIndex = selectedIdx;
  6595. }
  6596. }
  6597.  
  6598.  
  6599. function uroUpdateResolverList()
  6600. {
  6601. if(Object.keys(W.model.mapUpdateRequests.objects).length === 0)
  6602. {
  6603. return;
  6604. }
  6605.  
  6606. var resolverList = [];
  6607. var selector = document.getElementById('_selectURResolverID');
  6608. var selectedUser = null;
  6609. if(selector.selectedOptions[0] !== undefined)
  6610. {
  6611. selectedUser = parseInt(selector.selectedOptions[0].value);
  6612. }
  6613. while(selector.options.length > 0)
  6614. {
  6615. selector.options.remove(0);
  6616. }
  6617. var selectedIdx = 0;
  6618. var idx = 0;
  6619.  
  6620. for (var urobj in W.model.mapUpdateRequests.objects)
  6621. {
  6622. if(W.model.mapUpdateRequests.objects.hasOwnProperty(urobj))
  6623. {
  6624. var ureq = W.model.mapUpdateRequests.objects[urobj];
  6625. if(ureq.attributes.resolvedBy !== null)
  6626. {
  6627. var userID = ureq.attributes.resolvedBy;
  6628. var userName = W.model.users.objects[userID].userName;
  6629. if(resolverList.indexOf(userName) == -1)
  6630. {
  6631. resolverList.push(userName);
  6632. selector.options.add(new Option(userName, userID));
  6633. if(userID == selectedUser)
  6634. {
  6635. selectedIdx = idx;
  6636. }
  6637. idx++;
  6638. }
  6639. }
  6640. }
  6641. }
  6642. if(selectedIdx !== null)
  6643. {
  6644. selector.selectedIndex = selectedIdx;
  6645. }
  6646. }
  6647.  
  6648. function uroUpdateUserList()
  6649. {
  6650. if(Object.keys(W.model.updateRequestSessions.objects).length === 0) return;
  6651.  
  6652. var selector = document.getElementById('_selectURUserID');
  6653.  
  6654. var selectedUser = null;
  6655. if(selector.selectedOptions[0] !== undefined)
  6656. {
  6657. selectedUser = parseInt(selector.selectedOptions[0].value);
  6658. }
  6659.  
  6660. while(selector.options.length > 0)
  6661. {
  6662. selector.options.remove(0);
  6663. }
  6664.  
  6665. var selectedIdx = null;
  6666.  
  6667. var listedIDs = [];
  6668. for(var ursIdx in W.model.updateRequestSessions.objects)
  6669. {
  6670. if(W.model.updateRequestSessions.objects.hasOwnProperty(ursIdx))
  6671. {
  6672. var ursObj = W.model.updateRequestSessions.objects[ursIdx];
  6673. if(ursObj.comments.length > 0)
  6674. {
  6675. for(var cidx=0; cidx < ursObj.comments.length; cidx++)
  6676. {
  6677. var userID = ursObj.comments[cidx].userID;
  6678. if((listedIDs.indexOf(userID) == -1) && (userID != -1))
  6679. {
  6680. listedIDs.push(userID);
  6681. }
  6682. }
  6683. }
  6684. }
  6685. }
  6686.  
  6687. if(listedIDs.length > 0)
  6688. {
  6689. var users = W.model.users.getByIds(listedIDs);
  6690. for(var idx=0; idx<listedIDs.length; idx++)
  6691. {
  6692. selector.options.add(new Option(users[idx].userName, listedIDs[idx]));
  6693. if(listedIDs[idx] == selectedUser)
  6694. {
  6695. selectedIdx = idx;
  6696. }
  6697. }
  6698. }
  6699.  
  6700.  
  6701. if(selectedIdx !== null)
  6702. {
  6703. selector.selectedIndex = selectedIdx;
  6704. }
  6705. }
  6706.  
  6707.  
  6708. function uroSetStyles(obj)
  6709. {
  6710. obj.style.fontSize = '12px';
  6711. obj.style.lineHeight = '100%';
  6712. obj.style.overflow = 'auto';
  6713. obj.style.height = (window.innerHeight * 0.55) + 'px';
  6714. }
  6715.  
  6716. function uroPlacesGroupCEHandler(groupidx)
  6717. {
  6718. if(uroPlacesGroupsCollapsed[groupidx] === false)
  6719. {
  6720. document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "block";
  6721. document.getElementById('_uroPlacesGroupState-'+groupidx).src = uroIcons[0][0];
  6722. }
  6723. else
  6724. {
  6725. document.getElementById('_uroPlacesGroup-'+groupidx).style.display = "none";
  6726. document.getElementById('_uroPlacesGroupState-'+groupidx).src = uroIcons[0][1];
  6727. }
  6728. }
  6729. function uroPlacesGroupCollapseExpand()
  6730. {
  6731. var groupidx = this.id.substr(21);
  6732. if(uroPlacesGroupsCollapsed[groupidx] === true) uroPlacesGroupsCollapsed[groupidx] = false;
  6733. else uroPlacesGroupsCollapsed[groupidx] = true;
  6734. uroPlacesGroupCEHandler(groupidx);
  6735. return false;
  6736. }
  6737. function uroPopulatePlacesTab()
  6738. {
  6739. var tHTML = '';
  6740. tHTML += '<b>Filter PURs by category/status:</b><br>';
  6741. tHTML += '<input type="checkbox" id="_cbFilterUneditablePlaceUpdates">Ones I can\'t edit</input><br>';
  6742. tHTML += '<input type="checkbox" id="_cbFilterLockRankedPlaceUpdates">Ones with non-zero lockRanks</input><br>';
  6743. tHTML += '<input type="checkbox" id="_cbFilterNewPlacePUR">Ones for new places</input><br>';
  6744. tHTML += '<input type="checkbox" id="_cbFilterUpdatedDetailsPUR">Ones for updated place details</input><br>';
  6745. tHTML += '<input type="checkbox" id="_cbFilterNewPhotoPUR">Ones for new photos</input><br>';
  6746. tHTML += '<input type="checkbox" id="_cbFilterFlaggedPUR">Ones flagged for attention</input><br>';
  6747. tHTML += '<br><input type="checkbox" id="_cbLeavePURGeos">Don\'t hide place polygons/points</input><br>';
  6748. tHTML += '<br><input type="checkbox" id="_cbInvertPURFilters">Invert PUR filters</input><br>';
  6749.  
  6750. tHTML += '<br><b>Filter PURs by severity:</b><br>';
  6751. tHTML += '<input type="checkbox" id="_cbPURFilterLowSeverity">Low</input>&nbsp;&nbsp;';
  6752. tHTML += '<input type="checkbox" id="_cbPURFilterMediumSeverity">Medium</input>&nbsp;&nbsp;';
  6753. tHTML += '<input type="checkbox" id="_cbPURFilterHighSeverity">High</input>';
  6754.  
  6755. tHTML += '<br><b>Filter PURs by age of submission:</b><br>';
  6756. tHTML += '<input type="checkbox" id="_cbEnablePURMinAgeFilter">Hide PURs less than </input>';
  6757. tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPURFilterMinDays"> days old<br>';
  6758. tHTML += '<input type="checkbox" id="_cbEnablePURMaxAgeFilter">Hide PURs more than </input>';
  6759. tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputPURFilterMaxDays"> days old<br>';
  6760.  
  6761. tHTML += '<hr>';
  6762.  
  6763. tHTML += '<br><b>Filter Places by state:</b><br>';
  6764. tHTML += 'Hide if last edited<br>';
  6765. tHTML += '<input type="checkbox" id="_cbPlaceFilterEditedLessThan"> less than </input>';
  6766. tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterPlaceEditMinDays"> days ago<br>';
  6767. tHTML += '<input type="checkbox" id="_cbPlaceFilterEditedMoreThan"> more than </input>';
  6768. tHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterPlaceEditMaxDays"> days ago<br>';
  6769.  
  6770. tHTML += '<br>Hide if locked at level:<br>';
  6771. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL0">1</input>';
  6772. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL1">2</input>';
  6773. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL2">3</input>';
  6774. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL3">4</input>';
  6775. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL4">5</input>';
  6776. tHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHidePlacesL5">6</input>';
  6777.  
  6778. tHTML += '<br><br><input type="checkbox" id="_cbHidePhotoPlaces" pairedWith="_cbHideNoPhotoPlaces">Hide or </input>';
  6779. tHTML += '<input type="checkbox" id="_cbHideNoPhotoPlaces" pairedWith="_cbHidePhotoPlaces">show ones with photos</input><br>';
  6780.  
  6781. tHTML += '<br><br><b>Filter Places by category:</b><br>';
  6782.  
  6783. var nCategories = W.Config.venues.categories.length;
  6784. var i;
  6785. if(uroPlacesGroupsCollapsed.length != nCategories)
  6786. {
  6787. for(i=0; i<nCategories; i++)
  6788. {
  6789. uroPlacesGroupsCollapsed.push(false);
  6790. }
  6791. }
  6792.  
  6793. for(i=0; i<nCategories; i++)
  6794. {
  6795. var parentCategory = W.Config.venues.categories[i];
  6796. var localisedName = I18n.lookup("venues.categories")[parentCategory];
  6797.  
  6798. if(uroPlacesGroupsCollapsed[i] === true)
  6799. {
  6800. tHTML += '<img src="'+uroIcons[0][1]+'" id="_uroPlacesGroupState-'+i+'">';
  6801. }
  6802. else
  6803. {
  6804. tHTML += '<img src="'+uroIcons[0][0]+'" id="_uroPlacesGroupState-'+i+'">';
  6805. }
  6806.  
  6807. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPlacesFilter-'+parentCategory+'"><b>'+localisedName+'</b></input><br>';
  6808. tHTML += '<div id="_uroPlacesGroup-'+i+'" style="padding:3px;border-width:2px;border-style:solid;border-color:#FFFFFF">';
  6809.  
  6810. for(var ii=0; ii<W.Config.venues.subcategories[parentCategory].length; ii++)
  6811. {
  6812. var subCategory = W.Config.venues.subcategories[parentCategory][ii];
  6813. localisedName = I18n.lookup("venues.categories")[subCategory];
  6814. tHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbPlacesFilter-'+subCategory+'">'+localisedName+'</input><br>';
  6815. }
  6816. tHTML += '</div>';
  6817. }
  6818. tHTML += '<input type="checkbox" id="_cbFilterPrivatePlaces"><b>Residential Places</b></input><br>';
  6819. tHTML += '<br><input type="checkbox" id="_cbInvertPlacesFilter">Invert Place filters?</input>';
  6820.  
  6821. uroCtrlPlaces.innerHTML = tHTML;
  6822. }
  6823.  
  6824. function uroWazeBits()
  6825. {
  6826. // "fake" uroWazeBits() function which only performs layer scan, to stop the uroWazeBits() call in WMETB from
  6827. // messing around with other stuff in the actual uroWazeBits() function (now renamed uroRealWazeBits...) that
  6828. // really only ought to be called once.
  6829. var i;
  6830. for(i=0;i<W.map.layers.length;i++)
  6831. {
  6832. if(W.map.layers[i].name == 'Spotlight') uroMaskLayer = i;
  6833. if(W.map.layers[i].name.indexOf('Waze.Control.SelectHighlightFeature') != -1) uroRootContainer = W.map.layers[i].div.id;
  6834. if(W.map.layers[i].name == 'Node Connections') uroTurnsLayerIdx = i;
  6835. }
  6836. uroPlacesRoot = W.map.landmarkLayer.id + '_vroot';
  6837.  
  6838. for(i=0;i<W.map.controls.length;i++)
  6839. {
  6840. if(W.map.controls[i].CLASS_NAME == 'Waze.View.ArchivePanel') dteControlsIdx = i;
  6841. else if(W.map.controls[i].CLASS_NAME == 'Waze.Control.Archive') dteControlsIdx = i;
  6842.  
  6843. if(W.map.controls[i].id !== null)
  6844. {
  6845. if(W.map.controls[i].id.indexOf('UpdateRequests') != -1) uroURControlsIdx = i;
  6846. if(W.map.controls[i].id.indexOf('MapProblems') != -1) uroProblemControlsIdx = i;
  6847. }
  6848. }
  6849. uroAddLog('uroMaskLayer at idx '+uroMaskLayer);
  6850. uroAddLog('Turns layer at idx '+uroTurnsLayerIdx);
  6851. uroAddLog('uroRootContainer = '+uroRootContainer);
  6852. uroAddLog('Places root layer = '+uroPlacesRoot);
  6853. }
  6854.  
  6855.  
  6856. function uroRealWazeBits()
  6857. {
  6858. if(document.getElementsByClassName("sandbox").length > 0)
  6859. {
  6860. uroAddLog('WME practice mode detected, script is disabled...');
  6861. return;
  6862. }
  6863.  
  6864. uroAddLog('adding WazeBits...'+uroToHex(uroWazeBitsPresent,4));
  6865. if((uroWazeBitsPresent & 0x0001) === 0)
  6866. {
  6867. if(typeof W != "undefined")
  6868. {
  6869. if(typeof W.map != "undefined")
  6870. {
  6871. uroAddLog(' W.map OK');
  6872. uroWazeBitsPresent |= 0x0001;
  6873. }
  6874. }
  6875. }
  6876. if((uroWazeBitsPresent & 0x0002) === 0)
  6877. {
  6878. if(typeof W != "undefined")
  6879. {
  6880. if(typeof W.model != "undefined")
  6881. {
  6882. uroAddLog(' W.model OK');
  6883. uroWazeBitsPresent |= 0x0002;
  6884. }
  6885. }
  6886. }
  6887. if((uroWazeBitsPresent & 0x0004) === 0)
  6888. {
  6889. if(typeof W != "undefined")
  6890. {
  6891. if(typeof W.loginManager != "undefined")
  6892. {
  6893. uroAddLog(' loginManager OK');
  6894. uroWazeBitsPresent |= 0x0004;
  6895. }
  6896. }
  6897. }
  6898. if((uroWazeBitsPresent & 0x0008) === 0)
  6899. {
  6900. if(typeof W != "undefined")
  6901. {
  6902. if(typeof W.selectionManager != "undefined")
  6903. {
  6904. uroAddLog(' selectionManager OK');
  6905. uroWazeBitsPresent |= 0x0008;
  6906. }
  6907. }
  6908. }
  6909. if((uroWazeBitsPresent & 0x0010) === 0)
  6910. {
  6911. if(typeof OpenLayers != "undefined")
  6912. {
  6913. uroAddLog(' OpenLayers OK');
  6914. uroWazeBitsPresent |= 0x0010;
  6915. }
  6916. }
  6917. if((uroWazeBitsPresent & 0x0020) === 0)
  6918. {
  6919. if(typeof Waze != "undefined")
  6920. {
  6921. uroAddLog(' Waze OK');
  6922. uroWazeBitsPresent |= 0x0020;
  6923. }
  6924. }
  6925. if((uroWazeBitsPresent & 0x0040) === 0)
  6926. {
  6927. if(document.getElementById('user-tabs') !== null)
  6928. {
  6929. uroUserTabId = 'user-tabs';
  6930. uroAddLog(' user-tabs OK');
  6931. uroWazeBitsPresent |= 0x0040;
  6932. }
  6933. }
  6934. if((uroWazeBitsPresent & 0x0080) === 0)
  6935. {
  6936. if(document.getElementById('sidepanel-drives') !== null)
  6937. {
  6938. uroAddLog(' sidepanel-drives OK');
  6939. uroWazeBitsPresent |= 0x0080;
  6940. }
  6941. }
  6942. if((uroWazeBitsPresent & 0x0100) === 0)
  6943. {
  6944. if(typeof I18n != "undefined")
  6945. {
  6946. uroAddLog(' I18n OK');
  6947. uroWazeBitsPresent |= 0x0100;
  6948. }
  6949. }
  6950.  
  6951. if(uroWazeBitsPresent !== 0x01FF)
  6952. {
  6953. setTimeout(uroRealWazeBits,250);
  6954. }
  6955. else if(W.loginManager.isLoggedIn() === false)
  6956. {
  6957. uroAddLog('Waiting for user log-in...');
  6958. setTimeout(uroRealWazeBits,1000);
  6959. }
  6960. else
  6961. {
  6962. uroAddLog('All WazeBits present and correct...');
  6963.  
  6964. uroDOMHasTurnProblems = (W.model.turnProblems !== undefined);
  6965.  
  6966. uroPopulatePlacesTab();
  6967.  
  6968. uroControls.appendChild(uroCtrlURs);
  6969. uroControls.appendChild(uroCtrlMPs);
  6970. uroControls.appendChild(uroCtrlPlaces);
  6971. uroControls.appendChild(uroCtrlCameras);
  6972. uroControls.appendChild(uroOWL);
  6973. uroControls.appendChild(uroCtrlMisc);
  6974. uroControls.appendChild(uroCtrlHides);
  6975. uroControls.appendChild(uroAMList);
  6976.  
  6977. uroCtrlURs.onclick = uroFilterItems_URTabClick;
  6978. uroCtrlMPs.onclick = uroFilterItems_MPTabClick;
  6979. uroCtrlPlaces.onclick = uroFilterItems_PlacesTabClick;
  6980. uroCtrlCameras.onclick = uroFilterItems_CamerasTabClick;
  6981. uroCtrlMisc.onclick = uroFilterItems_MiscTabClick;
  6982.  
  6983. uroWazeBits();
  6984.  
  6985. uroDiv.addEventListener("mouseover", uroEnterPopup, false);
  6986. uroDiv.addEventListener("mouseout", uroExitPopup, false);
  6987.  
  6988. if(sessionStorage.UROverview_FID_IgnoreList === undefined) sessionStorage.UROverview_FID_IgnoreList = '';
  6989. if(sessionStorage.UROverview_FID_WatchList === undefined) sessionStorage.UROverview_FID_WatchList = '';
  6990. if(uroConfirmIntercepted === false) uroAddInterceptor();
  6991. if(uroBetaEditor === true)
  6992. {
  6993. uroAddLog('fixing header links...');
  6994. var nLinks = document.getElementsByClassName('waze-header-menu')[0].children.length;
  6995. for(var link=0; link<nLinks; link++)
  6996. {
  6997. var relLink = document.getElementsByClassName('waze-header-menu')[0].children[link].children[0].getAttribute('href');
  6998. relLink = 'https://www.waze.com' + relLink;
  6999. document.getElementsByClassName('waze-header-menu')[0].children[link].children[0].setAttribute('href',relLink);
  7000. }
  7001. }
  7002. setInterval(uroTenthSecondTick,100);
  7003. }
  7004. }
  7005.  
  7006.  
  7007. function uroAddInterceptor()
  7008. {
  7009. uroAddLog('Adding interceptor function...');
  7010. // add interceptor function for confirm(), so that we can auto-select the "OK" option when solving URs
  7011. // which have pending question...
  7012.  
  7013. var _confirm = confirm;
  7014. confirm = function(msg)
  7015. {
  7016. if((I18n.lookup("update_requests.panel.confirm") == msg) && (uroGetCBChecked('_cbDisablePendingQuestions') === true))
  7017. {
  7018. uroAddLog('Intercepted pending comments confirmation...');
  7019. return true;
  7020. }
  7021. else if(typeof(msg) == 'undefined')
  7022. {
  7023. uroAddLog('Intercepted blank confirmation...');
  7024. return true;
  7025. }
  7026. else
  7027. {
  7028. return _confirm(msg);
  7029. }
  7030. };
  7031.  
  7032. uroConfirmIntercepted = true;
  7033. }
  7034.  
  7035.  
  7036. function uroEnterPopup()
  7037. {
  7038. uroMouseInPopup = true;
  7039. }
  7040.  
  7041. function uroExitPopup()
  7042. {
  7043. uroMouseInPopup = false;
  7044. }
  7045.  
  7046. function uroToggleDebug()
  7047. {
  7048. uroShowDebugOutput = !uroShowDebugOutput;
  7049. var dbgMode = "none";
  7050. if(uroShowDebugOutput)
  7051. {
  7052. dbgMode = "inline";
  7053. }
  7054. document.getElementById('_uroDebugMode').style.display = dbgMode;
  7055. }
  7056.  
  7057. function uroInitialise()
  7058. {
  7059. if(document.URL.indexOf('editor-beta') != -1) uroBetaEditor = true;
  7060. var urlBits = document.URL.split("&mapUpdateRequest=");
  7061. if(urlBits.length == 2)
  7062. {
  7063. uroURIDInURL = parseInt(urlBits[1].split('&')[0]);
  7064. uroAddLog('found UR ID '+uroURIDInURL+' in URL');
  7065. }
  7066.  
  7067. // create a new div to display the UR details floaty-box
  7068. uroDiv = document.createElement('div');
  7069. uroDiv.id = "uroDiv";
  7070. uroDiv.style.position = 'absolute';
  7071. uroDiv.style.visibility = 'hidden';
  7072. uroDiv.style.top = '0';
  7073. uroDiv.style.left = '0';
  7074. uroDiv.style.zIndex = 100;
  7075. uroDiv.style.backgroundColor = 'aliceblue';
  7076. uroDiv.style.borderWidth = '3px';
  7077. uroDiv.style.borderStyle = 'solid';
  7078. uroDiv.style.borderRadius = '10px';
  7079. uroDiv.style.boxShadow = '5px 5px 10px Silver';
  7080. uroDiv.style.padding = '4px';
  7081. document.body.appendChild(uroDiv);
  7082.  
  7083.  
  7084. uroControls = document.createElement('section');
  7085. uroControls.style.fontSize = '12px';
  7086. uroControls.id = 'uroControls';
  7087. var updateURL;
  7088. if(navigator.userAgent.indexOf('Chrome') == -1)
  7089. {
  7090. updateURL = 'https://greasyfork.org/scripts/1952-uroverview-plus-uro';
  7091. }
  7092. else
  7093. {
  7094. updateURL = 'https://chrome.google.com/webstore/detail/uroverview/amdamgkgchnbaopmphhjapmjcdghdphi';
  7095. }
  7096. var tabbyHTML = '<b><a href="'+updateURL+'" target="_blank">UROverview Plus</a></b> <label id="_uroVersion">'+uroVersion+'</label>';
  7097. tabbyHTML += '<label id="_uroDebugMode">(dbg)</label>';
  7098. tabbyHTML += '&nbsp;<input type="checkbox" id="_cbMasterEnable" checked>Enabled</input>';
  7099. tabbyHTML += '<p><table border=0 width="100%"><tr>';
  7100. tabbyHTML += '<td valign="center" align="center" id="_tabSelectUserRequests"><a href="#" id="_linkSelectUserRequests" style="text-decoration:none;font-size:12px">URs</a></td>';
  7101. tabbyHTML += '<td valign="center" align="center" id="_tabSelectMapProblems"><a href="#" id="_linkSelectMapProblems" style="text-decoration:none;font-size:12px">MPs</a></td>';
  7102. tabbyHTML += '<td valign="center" align="center" id="_tabSelectPlaces"><a href="#" id="_linkSelectPlaces" style="text-decoration:none;font-size:12px">Places</a></td>';
  7103. tabbyHTML += '<td valign="center" align="center" id="_tabSelectCameras"><a href="#" id="_linkSelectCameras" style="text-decoration:none;font-size:12px">Cams</a></td>';
  7104. tabbyHTML += '<td valign="center" align="center" id="_tabSelectCWL"><a href="#" id="_linkSelectOWL" style="text-decoration:none;font-size:12px">OWL</a></td>';
  7105. tabbyHTML += '<td valign="center" align="center" id="_tabSelectMisc"><a href="#" id="_linkSelectMisc" style="text-decoration:none;font-size:12px">Misc</a></td>';
  7106. tabbyHTML += '</tr></table>';
  7107. uroControls.innerHTML = tabbyHTML;
  7108.  
  7109.  
  7110. uroCtrlURs = document.createElement('p');
  7111. uroCtrlMPs = document.createElement('p');
  7112. uroCtrlCameras = document.createElement('p');
  7113. uroOWL = document.createElement('p');
  7114. uroCtrlMisc = document.createElement('p');
  7115. uroAMList = document.createElement('div');
  7116. uroCtrlHides = document.createElement('div');
  7117. uroCtrlPlaces = document.createElement('p');
  7118.  
  7119. // UR controls tab
  7120. uroCtrlURs.id = "uroCtrlURs";
  7121. uroCtrlURs.innerHTML = '<br>';
  7122.  
  7123. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURFilterOutsideArea">Hide URs outside my editable area</input><br>';
  7124. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbNoFilterForURInURL">Don\'t filter UR in URL</input><br><br>';
  7125. uroCtrlURs.innerHTML += '<b>Filter by type:</b><br>';
  7126. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterWazeAuto">Waze Automatic</input><br>';
  7127. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectTurn">Incorrect turn</input><br>';
  7128. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectAddress">Incorrect address</input><br>';
  7129. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectRoute">Incorrect route</input><br>';
  7130. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingRoundabout">Missing roundabout</input><br>';
  7131. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterGeneralError">General error</input><br>';
  7132. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterTurnNotAllowed">Turn not allowed</input><br>';
  7133. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterIncorrectJunction">Incorrect junction</input><br>';
  7134. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingBridgeOverpass">Missing bridge overpass</input><br>';
  7135. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterWrongDrivingDirection">Wrong driving direction</input><br>';
  7136. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingExit">Missing exit</input><br>';
  7137. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingRoad">Missing road</input><br>';
  7138. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterBlockedRoad">Blocked road</input><br>';
  7139. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterMissingLandmark">Missing Landmark</input><br>';
  7140. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterUndefined">Undefined</input><br>';
  7141.  
  7142. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<i>Specially tagged types</i><br>';
  7143. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterRoadworks">[ROADWORKS]</input><br>';
  7144. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterConstruction">[CONSTRUCTION]</input><br>';
  7145. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterClosure">[CLOSURE]</input><br>';
  7146. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterEvent">[EVENT]</input><br>';
  7147. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbFilterNote">[NOTE]</input><br><br>';
  7148. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbInvertURFilter">Invert operation of type filters?</input><br>';
  7149.  
  7150. uroCtrlURs.innerHTML += '<hr>';
  7151.  
  7152. uroCtrlURs.innerHTML += '<br><b>Hide by state:</b><br>';
  7153. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterOpenUR">Open</input><br>';
  7154. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterClosedUR">Closed</input><br>';
  7155. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterSolved">Solved</input><br>';
  7156. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbFilterUnidentified">Not identified</input><br><br>';
  7157.  
  7158.  
  7159. uroCtrlURs.innerHTML += '<br><b>Filter by age of submission:</b><br>';
  7160. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMinAgeFilter">Hide URs less than </input>';
  7161. 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>';
  7162. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMaxAgeFilter">Hide URs more than </input>';
  7163. 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>';
  7164.  
  7165. uroCtrlURs.innerHTML += '<br><b>Filter by description/comments/following:</b><br>';
  7166. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURDescriptionMustBePresent" pairedWith="_cbURDescriptionMustBeAbsent">Hide</input> or ';
  7167. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURDescriptionMustBeAbsent" pairedWith="_cbURDescriptionMustBePresent">show</input> URs with no description<br>';
  7168. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableKeywordMustBePresent">Hide URs not including </input>';
  7169. uroCtrlURs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordPresent"><br>';
  7170. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableKeywordMustBeAbsent">Hide URs including </input>';
  7171. uroCtrlURs.innerHTML += '<input type="text" style="font-size:14px; line-height:16px; height:22px; margin-bottom:4px;" id="_textKeywordAbsent"><br>';
  7172. uroCtrlURs.innerHTML += '&nbsp;&nbsp;<input type="checkbox" id="_cbCaseInsensitive"><i>Case-insensitive matches?</i></input><br><br>';
  7173.  
  7174. uroCtrlURs.innerHTML += 'With comments from me?<br>';
  7175. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyComments" pairedWith="_cbHideAnyComments">Yes </input>';
  7176. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideAnyComments" pairedWith="_cbHideMyComments">No</input><br>';
  7177.  
  7178. uroCtrlURs.innerHTML += 'If last comment made by me?<br>';
  7179. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfLastCommenter" pairedWith="_cbHideIfNotLastCommenter">Yes </input>';
  7180. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfNotLastCommenter" pairedWith="_cbHideIfLastCommenter">No </input><br>';
  7181.  
  7182. uroCtrlURs.innerHTML += 'If last comment made by UR reporter?<br>';
  7183. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfReporterLastCommenter" pairedWith="_cbHideIfReporterNotLastCommenter">Yes </input>';
  7184. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideIfReporterNotLastCommenter" pairedWith="_cbHideIfReporterLastCommenter">No</input><br>';
  7185.  
  7186. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMinCommentsFilter">With less than </input>';
  7187. uroCtrlURs.innerHTML += '<input type="number" min="1" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMinComments"> comments<br>';
  7188. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableMaxCommentsFilter">With more than </input>';
  7189. uroCtrlURs.innerHTML += '<input type="number" min="0" size="3" style="width:50px;line-height:14px;height:22px;margin-bottom:4px;" id="_inputFilterMaxComments"> comments<br>';
  7190.  
  7191. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyFollowed" pairedWith="_cbHideMyUnfollowed">Ones I am or </input>';
  7192. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbHideMyUnfollowed" pairedWith="_cbHideMyFollowed">am not following</input><br>';
  7193.  
  7194. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter2">Last comment less than </input>';
  7195. 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>';
  7196. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbEnableCommentAgeFilter">Last comment more than </input>';
  7197. 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>';
  7198.  
  7199. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURUserIDFilter">Without comments from user</input>';
  7200. uroCtrlURs.innerHTML += '<select id="_selectURUserID" style="width:80%; height:22px;"></select><br>';
  7201. uroCtrlURs.innerHTML += '<input type="checkbox" id="_cbURResolverIDFilter">Not resolved by user</input>';
  7202. uroCtrlURs.innerHTML += '<select id="_selectURResolverID" style="width:80%; height:22px;"></select>';
  7203.  
  7204. uroCtrlURs.innerHTML += '<br><br><input type="checkbox" id="_cbNoFilterForTaggedURs"><b>Don\'t apply state/age filters to tagged URs</b></input><br>';
  7205.  
  7206.  
  7207. // Map problems controls tab
  7208. uroCtrlMPs.id = "uroCtrlMPs";
  7209. uroCtrlMPs.innerHTML = '<br>';
  7210.  
  7211. uroCtrlMPs.innerHTML += '<b>Filter MPs by type:</b><br>';
  7212. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMissingJunction">Missing junction</input><br>';
  7213. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMissingRoad">Missing road</input><br>';
  7214. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterCrossroadsJunctionMissing">Missing crossroads</input><br>';
  7215. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterDrivingDirectionMismatch">Driving direction mismatch</input><br>';
  7216. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRoadTypeMismatch">Road type mismatch</input><br>';
  7217. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRestrictedTurn">Restricted turn might be allowed</input><br>';
  7218. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterRoadClosureProblem">Road closure</input><br>';
  7219. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterUnknownProblem">Unknown problem type</input><br><br>';
  7220. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterTurnProblem">Turn Problems</input><br><br>';
  7221.  
  7222. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterReopenedProblem">Reopened Problems</input><br><br>';
  7223.  
  7224. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbInvertMPFilter">Invert operation of type filters?</input><br>';
  7225.  
  7226. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterOutsideArea">Hide MPs outside my editable area</input><br>';
  7227.  
  7228. uroCtrlMPs.innerHTML += '<br><b>Hide closed/solved/unidentified Problems:</b><br>';
  7229. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterClosed">Closed</input><br>';
  7230. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterSolved">Solved</input><br>';
  7231. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterUnidentified">Not identified</input><br><br>';
  7232.  
  7233. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPClosedUserIDFilter" pairedWith="_cbMPNotClosedUserIDFilter">Closed</input> or ';
  7234. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPNotClosedUserIDFilter" pairedWith="_cbMPClosedUserIDFilter">Not Closed</input> by user';
  7235. uroCtrlMPs.innerHTML += '<select id="_selectMPUserID" style="width:80%; height:22px;"></select><br>';
  7236.  
  7237. uroCtrlMPs.innerHTML += '<br><b>Hide problems (not turn) by severity:</b><br>';
  7238. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterLowSeverity">Low</input>&nbsp;&nbsp;';
  7239. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterMediumSeverity">Medium</input>&nbsp;&nbsp;';
  7240. uroCtrlMPs.innerHTML += '<input type="checkbox" id="_cbMPFilterHighSeverity">High</input><br>';
  7241.  
  7242.  
  7243. // Places filtering tab
  7244. uroCtrlPlaces.id = "uroCtrlPlaces";
  7245. uroCtrlPlaces.innerHTML = 'Places filter list being populated, please wait...';
  7246.  
  7247.  
  7248. // Camera controls tab
  7249. uroCtrlCameras.id = "uroCtrlCameras";
  7250. uroCtrlCameras.innerHTML = '<br><b>Show Cameras by creator:</b><br>';
  7251. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowWorldCams" checked>world_* users</input><br>';
  7252. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowUSACams" checked>usa_* users</input><br>';
  7253. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowNonWorldCams" checked>other users</input><br>';
  7254. uroCtrlCameras.innerHTML += '<br><input type="checkbox" id="_cbShowOnlyMyCams">Show ONLY cameras created/edited by me</input><br>';
  7255.  
  7256.  
  7257. uroCtrlCameras.innerHTML += '<br><b>Show Cameras by approval status:</b><br>';
  7258. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowApprovedCams" checked>approved</input><br>';
  7259. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowNonApprovedCams" checked>non-approved</input><br>';
  7260. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowOlderCreatedNonApproved"> if created more than </input>';
  7261. 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>';
  7262. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowOlderUpdatedNonApproved"> if updated more than </input>';
  7263. 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>';
  7264.  
  7265. uroCtrlCameras.innerHTML += '<br><b>Show Cameras by type:</b><br>';
  7266. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowSpeedCams" checked>Speed</input><br>';
  7267. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowIfSpeedSet" checked> with speed data</input><br>';
  7268. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbShowIfNoSpeedSet" checked> with no speed data</input><br>';
  7269. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowRedLightCams" checked>Red Light</input><br>';
  7270. uroCtrlCameras.innerHTML += '<input type="checkbox" id="_cbShowDummyCams" checked>Dummy</input><br>';
  7271.  
  7272. uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by creator:</b><br>';
  7273. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByMe">me</input>';
  7274. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank0">L1</input>';
  7275. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank1">L2</input>';
  7276. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank2">L3</input>';
  7277. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank3">L4</input>';
  7278. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank4">L5</input>';
  7279. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideCreatedByRank5">L6</input>';
  7280.  
  7281. uroCtrlCameras.innerHTML += '<br><b>Hide Cameras by updater:</b><br>';
  7282. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByMe">me</input>';
  7283. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank0">L1</input>';
  7284. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank1">L2</input>';
  7285. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank2">L3</input>';
  7286. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank3">L4</input>';
  7287. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank4">L5</input>';
  7288. uroCtrlCameras.innerHTML += '&nbsp;&nbsp;&nbsp;<input type="checkbox" id="_cbHideUpdatedByRank5">L6</input>';
  7289.  
  7290. uroCtrlCameras.innerHTML += '<br><br><b><input type="checkbox" id="_cbHideCWLCams">Hide cameras on watchlist</input></b><br>';
  7291.  
  7292.  
  7293. // Object watchlist tab
  7294. uroOWL.id = "uroOWL";
  7295. uroOWLUpdateHTML();
  7296.  
  7297.  
  7298. // Misc controls tab
  7299. uroCtrlMisc.id = "uroCtrlMisc";
  7300. uroCtrlMisc.innerHTML = '<br><b>Use default conversation markers:</b><br>';
  7301. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbNativeConvoMarkers" checked>in public WME</input><br>';
  7302. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbNativeBetaConvoMarkers" checked>in beta WME</input><br>';
  7303.  
  7304. uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbCommentCount">Show comment count on UR markers</input></b><br>';
  7305.  
  7306. uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbURBackfill">Backfill UR data</input></b><br>';
  7307.  
  7308. uroCtrlMisc.innerHTML += '<br><br><b>Marker Unstacking:</b><br>';
  7309. 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>';
  7310. 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>';
  7311.  
  7312. uroCtrlMisc.innerHTML += '<br><br><b>Use custom marker for URs tagged as:</b><br>';
  7313. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomRoadworksMarkers">[ROADWORKS]</input><br>';
  7314. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomConstructionMarkers">[CONSTRUCTION]</input><br>';
  7315. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomClosuresMarkers">[CLOSURE]</input><br>';
  7316. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomEventsMarkers">[EVENT]</input><br>';
  7317. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomNotesMarkers">[NOTE]</input><br>';
  7318.  
  7319. uroCtrlMisc.innerHTML += '<br><br><b>Use custom marker for MPs tagged as:</b><br>';
  7320. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomElginMarkers">[Elgin]</input><br>';
  7321. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTrafficMasterMarkers">[TM]</input><br>';
  7322. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbCustomTrafficCastMarkers">[TrafficCast]</input><br>';
  7323.  
  7324. uroCtrlMisc.innerHTML += '<br><br><b>Popup mouse behaviour:</b><br>';
  7325. 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>';
  7326. 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>';
  7327. 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>';
  7328.  
  7329. uroCtrlMisc.innerHTML += '<br><br><b>Disable popup for:</b><br>';
  7330. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitURPopup">URs</input><br>';
  7331. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitMPPopup">MPs</input><br>';
  7332. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitCamPopup">Cameras</input><br>';
  7333. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitSegPopup">Segments</input><br>';
  7334. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitTurnsPopup">Restricted Turns</input><br>';
  7335. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitLandmarkPopup">Places</input><br>';
  7336. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitPUPopup">Place Updates</input><br>';
  7337.  
  7338. uroCtrlMisc.innerHTML += '<br><br><b>Date/Time formatting for popups:</b><br>';
  7339. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtDDMMYY" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtYYMMDD" checked>day/month/year</input><br>';
  7340. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtMMDDYY" pairedWith="_cbDateFmtDDMMYY,_cbDateFmtYYMMDD">month/day/year</input><br>';
  7341. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbDateFmtYYMMDD" pairedWith="_cbDateFmtMMDDYY,_cbDateFmtDDMMYY">year/month/day</input><br><br>';
  7342. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbTimeFmt24H" pairedWith="_cbTimeFmt12H" checked>24 hour</input><br>';
  7343. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbTimeFmt12H" pairedWith="_cbTimeFmt24H">12 hour</input><br><br>';
  7344. uroCtrlMisc.innerHTML += '<i>Unticked uses browser default setting</i>';
  7345.  
  7346. uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbWhiteBackground">Use custom background colour</input></b><br>';
  7347. 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">';
  7348. 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">';
  7349. 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>';
  7350. /*
  7351. uroCtrlMisc.innerHTML += '<br><br><b>Disable map move when selecting:</b><br>';
  7352. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitURCentering">URs</input><br>';
  7353. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitMPCentering">MPs</input><br>';
  7354. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitPURCentering">PURs</input><br>';
  7355. */
  7356. uroCtrlMisc.innerHTML += '<br><br><b>Replace "Next ..." button with "Done" for:</b><br>';
  7357. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNURButton">URs</input><br>';
  7358. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNMPButton">MPs</input><br>';
  7359. uroCtrlMisc.innerHTML += '<input type="checkbox" id="_cbInhibitNPURButton">PURs</input><br>';
  7360. uroCtrlMisc.innerHTML += '<br><br><b><input type="checkbox" id="_cbHideAMLayer">Hide Area Manager polygons</input></b><br>';
  7361. uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisablePlacesFiltering">Disable Places filtering</input></b><br>';
  7362. ////uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisablePendingQuestions">Disable UR Pending Questions confirmation</input></b><br>';
  7363. uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbDisableTabStyling">Use default tab styling</input></b><br>';
  7364. uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbHideEditorInfo">Hide sidebar editor info</input></b><br>';
  7365. uroCtrlMisc.innerHTML += '<b><input type="checkbox" id="_cbEnableDTE">Drive Tab Enhancement (DTE)</input></b><br>';
  7366.  
  7367. uroCtrlMisc.innerHTML += '<br><br><b>Settings backup/restore/reset:</b><br>';
  7368. uroCtrlMisc.innerHTML += '<input type="button" id="_btnSettingsToText" value="Backup">&nbsp;&nbsp;&nbsp;';
  7369. uroCtrlMisc.innerHTML += '<input type="button" id="_btnTextToSettings" value="Restore">&nbsp;&nbsp;|&nbsp;&nbsp;';
  7370. uroCtrlMisc.innerHTML += '<input type="button" id="_btnResetSettings" value="Reset"><br><br>';
  7371. uroCtrlMisc.innerHTML += '<textarea id="_txtSettings" value="" /><br>';
  7372. uroCtrlMisc.innerHTML += '<input type="button" id="_btnClearSettingsText" value="Clear"><br>';
  7373.  
  7374. uroCtrlMisc.innerHTML += '<br><br><b>Debug:</b><br>';
  7375. uroCtrlMisc.innerHTML += '<input type="button" id="_btnDebugToScreen" value="Show debug data">';
  7376. // footer for tabs container
  7377. uroCtrlHides.id = 'uroCtrlHides';
  7378. uroCtrlHides.innerHTML = '<input type="button" id="_btnUndoLastHide" value="Undo last hide">&nbsp;&nbsp;&nbsp;';
  7379. uroCtrlHides.innerHTML += '<input type="button" id="_btnClearSessionHides" value="Undo all hides"><p>';
  7380.  
  7381. // footer for AM list
  7382. uroAMList.id = 'uroAMList';
  7383. window.addEventListener("beforeunload", uroSaveSettings, false);
  7384.  
  7385. uroRealWazeBits();
  7386. }
  7387.  
  7388.  
  7389. function dteAddHeader()
  7390. {
  7391. var rlcObj = document.getElementsByClassName("result-list-container");
  7392. if(typeof rlcObj == "undefined") return;
  7393. if(typeof rlcObj[0].children[0] == "undefined") return;
  7394. if(typeof rlcObj[0].children[0].innerHTML == "undefined") return;
  7395.  
  7396. var thtml = rlcObj[0].children[0].innerHTML;
  7397. if(thtml.indexOf('Full drive history') == -1)
  7398. {
  7399. thtml += '<br><br><i><small>Full drive history goes back to '+dteOldestFullDrive.toDateString()+'</small></i>';
  7400. rlcObj[0].children[0].innerHTML = thtml;
  7401. }
  7402. }
  7403.  
  7404. function dteGetData()
  7405. {
  7406. var loc = 'https://'+window.location.hostname+Waze.Config.api_base+'/Archive/MyList?minDistance=1000';
  7407. loc += '&offset='+dteOffset+'&count=5';
  7408. var dteReq = new XMLHttpRequest();
  7409. dteReq.onreadystatechange = function()
  7410. {
  7411. var foundMissingDrive = false;
  7412. if(dteReq.readyState == 4)
  7413. {
  7414. uroAddLog('drive data request, response '+dteReq.status+' received');
  7415. if(dteReq.status == 200)
  7416. {
  7417. if(dteReq.responseText !== "")
  7418. {
  7419. var drives = JSON.parse(dteReq.responseText);
  7420. var loadedDrives = drives.archives.objects.length;
  7421. uroAddLog('received '+loadedDrives+' drives');
  7422. if(loadedDrives != 5) foundMissingDrive = true;
  7423.  
  7424. for(var loop=0; loop < loadedDrives; loop++)
  7425. {
  7426. if(drives.archives.objects[loop].hasFullSession === false)
  7427. {
  7428. foundMissingDrive = true;
  7429. }
  7430. else
  7431. {
  7432. dteOffset++;
  7433. dteOldestFullDrive = new Date(drives.archives.objects[loop].startTime);
  7434. }
  7435. }
  7436. }
  7437. else
  7438. {
  7439. foundMissingDrive = true;
  7440. }
  7441. }
  7442. if(foundMissingDrive === false)
  7443. {
  7444. dteGetData();
  7445. }
  7446. else
  7447. {
  7448. uroAddLog(dteOffset+' full drives in history');
  7449. uroAddLog('oldest drives are on '+dteOldestFullDrive.toDateString());
  7450. if(dteOffset < 5)
  7451. {
  7452. dteOffset = 5;
  7453. uroAddLog('insufficient full drives, using standard drives tab');
  7454. }
  7455. else if(dteOffset > 50)
  7456. {
  7457. var nPages = Math.ceil(dteOffset / 50);
  7458. uroAddLog('too many full drives for a single tab page, splitting over '+nPages+' pages...');
  7459. dteOffset = Math.ceil(dteOffset/nPages);
  7460. }
  7461.  
  7462. if((dteOldestFullDrive - dteEpoch) > 0)
  7463. {
  7464. var totalDrives = 0;
  7465. if(W.model.archives.additionalInfo !== null)
  7466. {
  7467. totalDrives = W.model.archives.additionalInfo.totalSessions;
  7468. }
  7469. if(totalDrives !== null)
  7470. {
  7471. uroAddLog('updating drives tab...');
  7472. W.map.controls[dteControlsIdx].sidePanelView.ResultsPerPage = dteOffset;
  7473. uroAddLog(totalDrives+' drives in history');
  7474. W.map.controls[dteControlsIdx].sidePanelView.setSessions(totalDrives);
  7475. W.map.controls[dteControlsIdx].loadSessions(0);
  7476. }
  7477. setInterval(dteAddHeader,250);
  7478. setInterval(dteCheckDriveListChanges,250);
  7479. }
  7480. }
  7481. }
  7482. };
  7483. dteReq.open('GET',loc,true);
  7484. dteReq.send();
  7485.  
  7486. }
  7487.  
  7488. function dteSetNewTabLength()
  7489. {
  7490. uroAddLog('altering ResultsPerPage parameter...');
  7491.  
  7492. var t = document.getElementById('sidepanel-drives');
  7493. t.style.overflow = 'auto';
  7494. t.style.height = (window.innerHeight * 0.6) + 'px';
  7495. dteOffset = 0;
  7496. dteGetData();
  7497. }
  7498.  
  7499. function dteListClick()
  7500. {
  7501. dteClearListHighlight();
  7502. this.style.backgroundColor = "lightgreen";
  7503. dteArmClearHighlightsOnPanelClose = true;
  7504. }
  7505.  
  7506. function dteClearListHighlight()
  7507. {
  7508. var drivesShown = document.getElementById('sidepanel-drives').getElementsByClassName('result session').length;
  7509. if(drivesShown > 0)
  7510. {
  7511. for(var loop = 0;loop < drivesShown; loop++)
  7512. {
  7513. var listEntry = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[loop];
  7514. listEntry.style.backgroundColor = "";
  7515. }
  7516. }
  7517. }
  7518.  
  7519. function dteCheckDriveListChanges()
  7520. {
  7521. var drivesShown = document.getElementById('sidepanel-drives').getElementsByClassName('result session').length;
  7522. if(drivesShown > 0)
  7523. {
  7524. var topID = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[0].getAttribute('data-id');
  7525. if(topID != dteTopID)
  7526. {
  7527. dteTopID = topID;
  7528. for(var loop = 0;loop < drivesShown; loop++)
  7529. {
  7530. var listEntry = document.getElementById('sidepanel-drives').getElementsByClassName('result session')[loop];
  7531. var driveID = listEntry.getAttribute('data-id');
  7532. var driveObj = W.model.archives.objects[driveID];
  7533. var driveSecs = Math.floor((driveObj.endTime - driveObj.startTime) / 1000);
  7534. var driveHours = Math.floor(driveSecs / 3600);
  7535. driveSecs -= (driveHours * 3600);
  7536. var driveMins = Math.floor(driveSecs / 60);
  7537. driveSecs -= (driveMins * 60);
  7538. var trueTime = (driveHours+':'+("0"+driveMins).slice(-2)+'.'+("0"+driveSecs).slice(-2));
  7539. listEntry.getElementsByTagName('span')[1].innerHTML = trueTime;
  7540. listEntry.addEventListener("click", dteListClick, false);
  7541. }
  7542. }
  7543. }
  7544. }
  7545.  
  7546. uroInitialise();
  7547.