WME Switch Uturns

Switches U-turns for selected node or segment. Forked and improved "WME Add Uturn from node" script.

  1. // ==UserScript==
  2. // @name WME Switch Uturns
  3. // @version 2025.03.17.002
  4. // @description Switches U-turns for selected node or segment. Forked and improved "WME Add Uturn from node" script.
  5. // @author ixxvivxxi, uranik, turbopirate, AntonShevchuk
  6. // @namespace https://greatest.deepsurf.us/users/160654-waze-ukraine
  7. // @match https://*.waze.com/editor*
  8. // @match https://*.waze.com/*/editor*
  9. // @exclude https://*.waze.com/user/editor*
  10. // @icon 
  11. // @grant none
  12. // @require https://update.greatest.deepsurf.us/scripts/450160/1218867/WME-Bootstrap.js
  13. // @require https://update.greatest.deepsurf.us/scripts/452563/1218878/WME.js
  14. // @require https://update.greatest.deepsurf.us/scripts/450221/1137043/WME-Base.js
  15. // @require https://update.greatest.deepsurf.us/scripts/450320/1555446/WME-UI.js
  16. // ==/UserScript==
  17.  
  18. /* jshint esversion: 8 */
  19. /* global require */
  20. /* global $, jQuery */
  21. /* global W */
  22. /* global I18n */
  23. /* global WME, WMEBase, WMEUI, WMEUIHelper, WMEUIHelperTab, WMEUIShortcut */
  24.  
  25. (function () {
  26. 'use strict'
  27.  
  28. // Script name, uses as unique index
  29. const NAME = 'SWITCH-UTURNS'
  30.  
  31. // Translations
  32. const TRANSLATION = {
  33. 'en': {
  34. title: 'Switch U-Turns',
  35. description: 'Choose a segment or a node to switch u-turns with <a href="#keyboard-dialog" target="_blank" rel="noopener noreferrer" data-toggle="modal">Keyboard shortcuts</a> or buttons',
  36. count: 'Count nodes and U-Turns',
  37. switch: 'Switch U-turn at point',
  38. nodes: 'Nodes',
  39. allowed: 'Allowed',
  40. disallowed: 'Disallowed',
  41. allow: 'Allow all U-turns',
  42. disallow: 'Disallow all U-turns',
  43. },
  44. 'uk': {
  45. title: 'Керування розворотами',
  46. description: 'Оберіть сегмент або вузол щоб змінити розвороти за допомогою <a href="#keyboard-dialog" target="_blank" rel="noopener noreferrer" data-toggle="modal">гарячих клавіш</a> або кнопок',
  47. count: 'Порахувати вузли та розвороти',
  48. switch: 'Змінити розворот у вузлі',
  49. nodes: 'Вузли',
  50. allowed: 'Дозволено',
  51. disallowed: 'Заборонено',
  52. allow: 'Дозволити всі розвороти',
  53. disallow: 'Заборонити всі розвороти',
  54. },
  55. 'ru': {
  56. title: 'Управление разворотами',
  57. description: 'Выберите сегмент или узел для изменения разворотов с помощью <a href="#keyboard-dialog" target="_blank" rel="noopener noreferrer" data-toggle="modal">комбинаций клавиш</a> или кнопок',
  58. count: 'Посчитать узлы и развороты',
  59. switch: 'Изменить разворот в узле',
  60. nodes: 'Узлы',
  61. allowed: 'Разрешено',
  62. disallowed: 'Запрещено',
  63. allow: 'Разрешить все развороты',
  64. disallow: 'Запретить все развороты',
  65. }
  66. }
  67.  
  68. const STYLE =
  69. 'p.switch-uturns-counter { margin-top: 15px; padding-left: 15px; }' +
  70. 'p.switch-uturns-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }'
  71.  
  72. WMEUI.addTranslation(NAME, TRANSLATION)
  73. WMEUI.addStyle(STYLE)
  74.  
  75. const ALLOW = 1
  76. const DISALLOW = 0
  77.  
  78. class UTurns extends WMEBase {
  79. constructor (name, settings = null) {
  80. super(name, settings)
  81.  
  82. /** @type {WMEUIHelper} */
  83. this.helper = new WMEUIHelper(this.name)
  84.  
  85. /** @type {WMEUIHelperTab} */
  86. this.tab = this.helper.createTab(
  87. I18n.t(this.name).title,
  88. {
  89. image: GM_info.script.icon
  90. }
  91. )
  92. this.tab.addText('description', I18n.t(this.name).description)
  93. let button = this.tab.addButton(NAME, I18n.t(name).count, '', () => this.updateTabUI(this.countUturns()))
  94. button.html().className += ' waze-btn-blue'
  95.  
  96. this.tab.addText('counter', '')
  97. this.tab.addText(
  98. 'info',
  99. '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
  100. )
  101. // Inject custom HTML to container in the WME interface
  102. this.tab.inject()
  103. }
  104. /**
  105. * Handler for `node.wme` event
  106. * @param {jQuery.Event} event
  107. * @param {HTMLElement} element
  108. * @param {W.model} model
  109. * @return {void}
  110. */
  111. onNode (event, element, model) {
  112. if (model.getSegmentIds().length < 2 || !model.isGeometryEditable()) {
  113. return
  114. }
  115. this.removePanel(element)
  116. this.createPanel(element)
  117. this.updateNodeUI()
  118. }
  119.  
  120. /**
  121. * Added controls
  122. * @param {HTMLElement} element
  123. */
  124. createPanel (element) {
  125. // Container
  126. let container = document.createElement('div')
  127. container.id = NAME
  128. // Separator space
  129. container.append(document.createElement('hr'))
  130. // Title
  131. let title = document.createElement('p')
  132. title.innerText = I18n.t(NAME).title
  133. container.append(title)
  134. // Text
  135. this.text = document.createElement('p')
  136. container.append(this.text)
  137. // Allow button
  138. this.allow = document.createElement('wz-button')
  139. this.allow.color = 'shadowed'
  140. this.allow.innerText = I18n.t(NAME).allow
  141. this.allow.onclick = () => this.switchNodeUturn(ALLOW)
  142. container.append(this.allow)
  143. // Disallow button
  144. this.disallow = document.createElement('wz-button')
  145. this.disallow.color = 'shadowed'
  146. this.disallow.innerText = I18n.t(NAME).disallow
  147. this.disallow.onclick = () => this.switchNodeUturn(DISALLOW)
  148. container.append(this.disallow)
  149.  
  150. element.parentNode.append(container)
  151. }
  152.  
  153. /**
  154. * Remove controls
  155. * @param {HTMLElement} element
  156. */
  157. removePanel(element) {
  158. element.parentNode.querySelector('#' + NAME)?.remove()
  159. }
  160.  
  161. /**
  162. * Update counter for the plugin tab
  163. * @param {Object} counter
  164. */
  165. updateTabUI (counter) {
  166. this.tab.html().querySelector('p.switch-uturns-counter').innerHTML = '' +
  167. I18n.t(NAME).nodes + ': ' + counter.nodes + '<br/>' +
  168. I18n.t(NAME).allowed + ': ' + counter.allowed + '<br/>' +
  169. I18n.t(NAME).disallowed + ': ' + counter.disallowed
  170. }
  171.  
  172. /**
  173. * Updated buttons status and counters
  174. */
  175. updateNodeUI () {
  176. let node = WME.getSelectedNode()
  177. if (!node
  178. || node.getSegmentIds().length < 2
  179. || !node.isGeometryEditable()) {
  180. return
  181. }
  182. let counter = this.countNodeUturns(node)
  183.  
  184. // Change display properties of the buttons
  185. this.allow.style.display = counter.disallowed ? 'flex' : 'none'
  186. this.disallow.style.display = counter.allowed ? 'flex' : 'none'
  187.  
  188. // Change text
  189. this.text.innerHTML =
  190. I18n.t(NAME).allowed + ': ' + counter.allowed + '<br/>' +
  191. I18n.t(NAME).disallowed + ': ' + counter.disallowed
  192. }
  193.  
  194. /**
  195. * @return {{nodes: number, allowed: number, disallowed: number}}
  196. */
  197. countUturns () {
  198. let counters = {
  199. nodes: 0,
  200. allowed: 0,
  201. disallowed: 0
  202. }
  203. for (let node in W.model.nodes.objects) {
  204. let counter = this.countNodeUturns(W.model.nodes.objects[node])
  205. counters.nodes++
  206. counters.allowed += counter.allowed
  207. counters.disallowed += counter.disallowed
  208. }
  209. return counters
  210. }
  211.  
  212. /**
  213. * @param {Object} node
  214. * @return {{allowed: number, disallowed: number}}
  215. */
  216. countNodeUturns (node) {
  217. let counter = {
  218. allowed: 0,
  219. disallowed: 0
  220. }
  221.  
  222. let segmentsIds = node.getSegmentIds()
  223.  
  224. for (let i = 0; i < segmentsIds.length; i++) {
  225. let segment = W.model.segments.getObjectById(segmentsIds[i])
  226. if (!segment) {
  227. continue
  228. }
  229. if (segment.isTurnAllowed(segment, node)) {
  230. counter.allowed++
  231. } else {
  232. counter.disallowed++
  233. }
  234. }
  235. return counter
  236. }
  237.  
  238. /**
  239. * Handler for selected node
  240. * @param {Number} status ALLOW or DISALLOW
  241. */
  242. switchNodeUturn (status) {
  243. let node = WME.getSelectedNode()
  244. if (!node) {
  245. return
  246. }
  247. let segmentsIds = node.getSegmentIds()
  248. if (segmentsIds.length < 2) {
  249. return
  250. }
  251. for (let i = 0; i < segmentsIds.length; i++) {
  252. let segment = W.model.segments.getObjectById(segmentsIds[i])
  253. if (segment.isOneWay()) {
  254. continue;
  255. }
  256. let turn = W.model.getTurnGraph().getTurnThroughNode(node, segment, segment)
  257. W.model.actionManager.add(
  258. new WazeActionSetTurn(
  259. W.model.getTurnGraph(),
  260. turn.withTurnData(turn.getTurnData().withState(status)))
  261. )
  262. }
  263. this.updateNodeUI()
  264.  
  265. this.log('u-turns in the node with ID ' + node.getID() + ' switched to ' + (status ? 'ALLOW' : 'DISALLOW'))
  266. }
  267.  
  268. /**
  269. * Handler for selected segments
  270. * @param direction
  271. */
  272. switchSegmentUturn (direction = 'A') {
  273. let segments = WME.getSelectedSegments()
  274. for (let i = 0, total = segments.length; i < total; i++) {
  275. let segment = segments[i]
  276. if (segment.isOneWay()) {
  277. continue;
  278. }
  279. let node = (direction === 'A') ? segment.getFromNode() : segment.getToNode()
  280. let status = segment.isTurnAllowed(segment, node) ? DISALLOW : ALLOW
  281. let turn = W.model.getTurnGraph().getTurnThroughNode(node, segment, segment)
  282. W.model.actionManager.add(
  283. new WazeActionSetTurn(
  284. W.model.getTurnGraph(),
  285. turn.withTurnData(turn.getTurnData().withState(status)) // enable it
  286. )
  287. )
  288.  
  289. this.log('u-turn in the point ' + direction + ' switched to ' + (status ? 'ALLOW' : 'DISALLOW'))
  290. }
  291. }
  292. }
  293.  
  294. let WazeActionSetTurn
  295.  
  296. $(document)
  297. .on('bootstrap.wme', () => {
  298. // Require Waze components
  299. WazeActionSetTurn = require('Waze/Model/Graph/Actions/SetTurn')
  300.  
  301. let UTurnsInstance = new UTurns(NAME)
  302.  
  303. // Hotkeys for node manipulation
  304. WMEUI.addShortcut(NAME + '-node-allow', I18n.t(NAME).allow, NAME, I18n.t(NAME).title, 'A+A', () => UTurnsInstance.switchNodeUturn(1))
  305. WMEUI.addShortcut(NAME + '-node-disallow', I18n.t(NAME).disallow, NAME, I18n.t(NAME).title, 'A+S', () => UTurnsInstance.switchNodeUturn(0))
  306. // Hotkeys for segment manipulation
  307. WMEUI.addShortcut(NAME + '-segment-a', I18n.t(NAME).switch + ' A', NAME, I18n.t(NAME).title, 'A+Q', () => UTurnsInstance.switchSegmentUturn('A'))
  308. WMEUI.addShortcut(NAME + '-segment-b', I18n.t(NAME).switch + ' B', NAME, I18n.t(NAME).title, 'A+W', () => UTurnsInstance.switchSegmentUturn('B'))
  309. // Update count of UTurns on events
  310. W.model.actionManager.events.register('afterundoaction', null, () => UTurnsInstance.updateNodeUI())
  311. W.model.actionManager.events.register('afterclearactions', null, () => UTurnsInstance.updateNodeUI())
  312. W.model.actionManager.events.register('afteraction', null, () => UTurnsInstance.updateNodeUI())
  313. })
  314. })()