WME 🇺🇦 E87 Inconsistent direction

Дозволяє вирішувати проблему різнонаправленних сегментів

  1. // ==UserScript==
  2. // @name WME E87 Inconsistent direction
  3. // @name:uk WME 🇺🇦 E87 Inconsistent direction
  4. // @version 0.1.2
  5. // @description Solves the inconsistent direction problem
  6. // @description:uk Дозволяє вирішувати проблему різнонаправленних сегментів
  7. // @license MIT License
  8. // @author Anton Shevchuk
  9. // @namespace https://greatest.deepsurf.us/users/227648-anton-shevchuk
  10. // @supportURL https://github.com/AntonShevchuk/wme-template/issues
  11. // @match https://*.waze.com/editor*
  12. // @match https://*.waze.com/*/editor*
  13. // @exclude https://*.waze.com/user/editor*
  14. // @icon 
  15. // @grant none
  16. // @require https://update.greatest.deepsurf.us/scripts/389765/1090053/CommonUtils.js
  17. // @require https://update.greatest.deepsurf.us/scripts/450160/1218867/WME-Bootstrap.js
  18. // @require https://update.greatest.deepsurf.us/scripts/452563/1218878/WME.js
  19. // @require https://update.greatest.deepsurf.us/scripts/450221/1137043/WME-Base.js
  20. // @require https://update.greatest.deepsurf.us/scripts/450320/1555446/WME-UI.js
  21. // ==/UserScript==
  22.  
  23. /* jshint esversion: 8 */
  24.  
  25. /* global require */
  26. /* global $, jQuery */
  27. /* global W */
  28. /* global I18n */
  29. /* global OpenLayers */
  30. /* global WME, WMEBase */
  31. /* global WMEUI, WMEUIHelper, WMEUIHelperPanel, WMEUIHelperModal, WMEUIHelperTab, WMEUIShortcut, WMEUIHelperFieldset */
  32. /* global Container, Settings, SimpleCache, Tools */
  33.  
  34. (function () {
  35. 'use strict'
  36.  
  37. // Script name, uses as unique index
  38. const NAME = 'E87'
  39.  
  40. // Translations
  41. const TRANSLATION = {
  42. 'en': {
  43. title: 'Direction →',
  44. description: 'Plugin WME E87 solves the inconsistent direction problem.<br/>Choose one or more segment to change direction.',
  45. buttons: {
  46. toggle: 'Change direction',
  47. forward: 'A → B',
  48. reverse: 'B → A',
  49. },
  50. },
  51. 'uk': {
  52. title: 'Напрямки →',
  53. description: 'Плагін WME E87 для вирішиння проблеми різно направленних вулиць.<br/>Оберіть один або декілька сегментів щоб застосувати зміни.',
  54. buttons: {
  55. toggle: 'Змінити напрямок',
  56. forward: 'A → B',
  57. reverse: 'B → A',
  58. },
  59. },
  60. 'ru': {
  61. title: 'Направления →',
  62. description: 'Плагин WME E87 для решения проблемы разнонаправленных улиц.<br/>Выберите один или несколько сегментов, чтобы внести изменения.',
  63. buttons: {
  64. toggle: 'Изменить направление',
  65. forward: 'A → B',
  66. reverse: 'B → A',
  67. },
  68. }
  69. }
  70.  
  71. const STYLE =
  72. '.lanes-tab div.e87 { border: 1px solid var(--hairline); border-radius: 6px; margin-bottom: 16px; padding: 8px 16px 18px; } ' +
  73. 'button.waze-btn.e87 { background: #f2f4f7; border: 1px solid #ccc; margin: 2px; } ' +
  74. 'button.waze-btn.e87:hover { background: #ffffff; transition: background-color 100ms linear; box-shadow: 0 2px 8px 0 rgba(0, 0, 0, 0.1), inset 0 0 100px 100px rgba(255, 255, 255, 0.3); } ' +
  75. 'button.waze-btn.e87:focus { background: #f2f4f7; } ' +
  76. 'button.e87-forward, button.e87-reverse { margin: 2px 8px; }' +
  77. 'div.e87-container { display: flex; flex: auto; justify-content: space-evenly; } ' +
  78. 'p.e87-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }'
  79.  
  80. WMEUI.addTranslation(NAME, TRANSLATION)
  81. WMEUI.addStyle(STYLE)
  82.  
  83. const BUTTONS = {
  84. toggle: {
  85. title: I18n.t(NAME).buttons.toggle,
  86. description: I18n.t(NAME).buttons.toggle,
  87. shortcut: '',
  88. },
  89. }
  90.  
  91. // Default settings
  92. const SETTINGS = {}
  93.  
  94. class E87 extends WMEBase {
  95. constructor (name, settings = null) {
  96. super(name, settings)
  97.  
  98. /** @type {WMEUIHelper} */
  99. this.helper = new WMEUIHelper(this.name)
  100.  
  101. /** @type {WMEUIHelperTab} */
  102. this.tab = this.helper.createTab(I18n.t(this.name).title, { image: GM_info.script.icon })
  103. this.tab.addText('description', I18n.t(this.name).description)
  104. this.tab.addText('info', '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version)
  105. this.tab.inject()
  106.  
  107. /** @type {WMEUIHelperPanel} */
  108. this.panel = this.helper.createPanel(I18n.t(name).title)
  109. }
  110.  
  111. /**
  112. * Init button for selection of the segment
  113. * @param buttons
  114. */
  115. init (buttons) {
  116. buttons.toggle.callback = (e) => {
  117. e.preventDefault()
  118. WME.getSelectedSegments().forEach(
  119. segment => this.invert(segment.getID())
  120. )
  121. }
  122. this.panel.addButtons(buttons)
  123. }
  124.  
  125. /**
  126. * Handler for `segment.wme` event
  127. * @param {jQuery.Event} event
  128. * @param {HTMLElement} element
  129. * @param {W.model} model
  130. * @return {void}
  131. */
  132. onSegment (event, element, model) {
  133. // Skip for walking trails and blocked roads
  134. if (model.isWalkingRoadType()
  135. || model.isLockedByHigherRank()
  136. || !model.isGeometryEditable()) {
  137. return
  138. }
  139.  
  140. element
  141. //.parentNode.parentNode
  142. //.querySelector('.lanes-tab')
  143. .prepend(this.panel.html())
  144. }
  145.  
  146. /**
  147. * Handler for `segments.wme` event
  148. * @param {jQuery.Event} event
  149. * @param {HTMLElement} element
  150. * @param {Array} models
  151. * @return {void}
  152. */
  153. onSegments (event, element, models) {
  154. // Skip for walking trails or locked roads
  155. if (models.filter((model) => model.isWalkingRoadType() || model.isLockedByHigherRank() || !model.isGeometryEditable()).length > 0) {
  156. element.querySelector('div.form-group.e87')?.remove()
  157. return
  158. }
  159.  
  160. let reversed = W.selectionManager.getReversedSegments()
  161.  
  162. if (reversed.numReversed === 0) {
  163. // you can reverse all selected segments
  164. element
  165. //.parentNode.parentNode
  166. //.querySelector('.lanes-tab')
  167. .prepend(this.panel.html())
  168. return
  169. }
  170.  
  171. let result = this.detect(reversed)
  172.  
  173. if (result.forward.length && result.reverse.length) {
  174. this.log('Inconsistent direction detected: forward = ' + result.forward.length + ' backward = ' + result.reverse.length)
  175.  
  176. let buttonToForward = document.createElement('button')
  177. buttonToForward.type = 'button'
  178. buttonToForward.title = I18n.t(NAME).buttons.toggle
  179. buttonToForward.className = 'waze-btn waze-btn-small waze-btn-white e87 e87-forward'
  180. buttonToForward.innerText = I18n.t(NAME).buttons.forward + ' (' + result.reverse.length + ')'
  181. buttonToForward.onclick = (e) => {
  182. e.preventDefault()
  183. result.reverse.forEach(el => this.invert(el))
  184. buttonToForward.innerText = I18n.t(NAME).buttons.forward + ' (0)'
  185. buttonToForward.disabled = true
  186. }
  187. let buttonToReverse = document.createElement('button')
  188. buttonToReverse.type = 'button'
  189. buttonToReverse.title = I18n.t(NAME).buttons.toggle
  190. buttonToReverse.className = 'waze-btn waze-btn-small waze-btn-white e87 e87-reverse'
  191. buttonToReverse.innerText = I18n.t(NAME).buttons.reverse + ' (' + result.forward.length + ')'
  192. buttonToReverse.onclick = (e) => {
  193. e.preventDefault()
  194. result.forward.forEach(el => this.invert(el))
  195. buttonToReverse.innerText = I18n.t(NAME).buttons.reverse + ' (0)'
  196. buttonToReverse.disabled = true
  197. }
  198.  
  199. this.container?.remove();
  200.  
  201. this.container = document.createElement('div')
  202. this.container.className = 'e87-container'
  203. this.container.append(buttonToForward)
  204. this.container.append(buttonToReverse)
  205.  
  206. $('wz-alert.sidebar-alert.inconsistent-direction-alert .sidebar-alert-content')
  207. .after(this.container)
  208. }
  209. }
  210.  
  211. /**
  212. * Detect directions
  213. * @param {Object} segments information
  214. * @return {Object}
  215. */
  216. detect (segments) {
  217. let forward = [], reverse = []
  218.  
  219. for (let el in segments) {
  220. el = Number.parseInt(el)
  221. if (Number.isNaN(el)) {
  222. continue
  223. }
  224. if (segments[el]) {
  225. reverse.push(el)
  226. } else {
  227. forward.push(el)
  228. }
  229. }
  230.  
  231. return {
  232. forward: forward,
  233. reverse: reverse
  234. }
  235. }
  236.  
  237. /**
  238. * Invert direction of the segment
  239. * @param {Number} id of the segment
  240. */
  241. invert (id) {
  242. let segment = W.model.segments.getObjectById(id)
  243. if (segment.isLockedByHigherRank()) {
  244. this.log('Locked by higher rank')
  245. return
  246. }
  247. this.group('invert segment ' + id)
  248. this.log('segment', segment)
  249.  
  250. // setup and reverse attributes
  251. let attributes = {}
  252. attributes.fwdDirection = segment.attributes.revDirection
  253. attributes.revDirection = segment.attributes.fwdDirection
  254. let fwdTurnsLocked = segment.attributes.fwdTurnsLocked
  255. let revTurnsLocked = segment.attributes.revTurnsLocked
  256. // attributes.fwdTurnsLocked = segment.attributes.revTurnsLocked // ???
  257. // attributes.revTurnsLocked = segment.attributes.fwdTurnsLocked // ???
  258. // segment.setAttribute("revTurnsLocked", segment.attributes.fwdTurnsLocked)}
  259. // segment.setAttribute("fwdTurnsLocked", segment.attributes.revTurnsLocked)}
  260. attributes.fwdMaxSpeed = segment.attributes.revMaxSpeed
  261. attributes.revMaxSpeed = segment.attributes.fwdMaxSpeed
  262. attributes.fwdMaxSpeedUnverified = segment.attributes.revMaxSpeedUnverified
  263. attributes.revMaxSpeedUnverified = segment.attributes.fwdMaxSpeedUnverified
  264. attributes.fwdLaneCount = segment.attributes.revLaneCount
  265. attributes.revLaneCount = segment.attributes.fwdLaneCount
  266.  
  267. attributes.restrictions = []
  268. for (let i = 0; i < segment.attributes.restrictions.length; i++) {
  269. attributes.restrictions[i] = segment.attributes.restrictions[i].withReverseDirection()
  270. }
  271.  
  272. this.log('attributes', attributes)
  273.  
  274. let fromNode = segment.getFromNode()
  275. let toNode = segment.getToNode()
  276.  
  277. let onA = {}
  278. let toConnections = {}
  279. fromNode.getSegmentIds().forEach(segId => {
  280. // incoming directions
  281. if (segId !== id) {
  282. onA[segId] = W.model.getTurnGraph().getTurnThroughNode(fromNode, W.model.segments.getObjectById(segId), segment)
  283. onA[segId].toVertex.direction = onA[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
  284. }
  285. // outgoing directions
  286. toConnections[segId] = W.model.getTurnGraph().getTurnThroughNode(fromNode, segment, W.model.segments.getObjectById(segId))
  287. toConnections[segId].fromVertex.direction = toConnections[segId].fromVertex.direction === 'fwd' ? 'rev' : 'fwd'
  288. // u-turn
  289. if (segId === id) {
  290. toConnections[segId].toVertex.direction = toConnections[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
  291. }
  292. })
  293.  
  294. let onB = {}
  295. let fromConnections = {}
  296. toNode.getSegmentIds().forEach(segId => {
  297. if (segId !== id) {
  298. onB[segId] = W.model.getTurnGraph().getTurnThroughNode(toNode, W.model.segments.getObjectById(segId), segment)
  299. onB[segId].toVertex.direction = onB[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
  300. }
  301.  
  302. fromConnections[segId] = W.model.getTurnGraph().getTurnThroughNode(toNode, segment, W.model.segments.getObjectById(segId))
  303. fromConnections[segId].fromVertex.direction = fromConnections[segId].fromVertex.direction === 'fwd' ? 'rev' : 'fwd'
  304.  
  305. // u-turn
  306. if (segId === id) {
  307. fromConnections[segId].toVertex.direction = fromConnections[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
  308. }
  309. })
  310.  
  311. // invert the geometry of the segment
  312. let geometry = segment.getOLGeometry().clone()
  313. geometry.components.reverse()
  314.  
  315. if (!geometry.components[0].equals(toNode.getOLGeometry())) {
  316. let delta = { x: 0, y: 0 }
  317. delta.x = toNode.getOLGeometry().x - geometry.components[0].x
  318. delta.y = toNode.getOLGeometry().y - geometry.components[0].y
  319. geometry.components[0].move(delta.x, delta.y)
  320. }
  321. let points = geometry.components.length - 1
  322. if (!geometry.components[points].equals(fromNode.getOLGeometry())) {
  323. let delta = { x: 0, y: 0 }
  324. delta.x = fromNode.getOLGeometry().x - geometry.components[points].x
  325. delta.y = fromNode.getOLGeometry().y - geometry.components[points].y
  326. geometry.components[points].move(delta.x, delta.y)
  327. }
  328.  
  329. // disconnect the segment
  330. let disconnect = new WazeActionMultiAction([new WazeActionDisconnectSegment(segment, fromNode), new WazeActionDisconnectSegment(segment, toNode)])
  331. disconnect._description = I18n.t('save.changes_log.actions.DisconnectSegment.default')
  332. W.model.actionManager.add(disconnect)
  333.  
  334. // update geometry of the segment
  335. W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(segment, segment.getGeometry(), W.userscripts.toGeoJSONGeometry(geometry)))
  336.  
  337. // update attributes
  338. W.model.actionManager.add(new WazeActionUpdateObject(segment, attributes))
  339.  
  340. // connect the segment
  341. let connect = new WazeActionMultiAction([new WazeActionConnectSegment(toNode, segment), new WazeActionConnectSegment(fromNode, segment)])
  342. connect._description = I18n.t('save.changes_log.actions.ConnectSegment.default')
  343. W.model.actionManager.add(connect)
  344.  
  345. // update Turn's attributes
  346. segment.setAttribute('fwdTurnsLocked', revTurnsLocked)
  347. segment.setAttribute('revTurnsLocked', fwdTurnsLocked)
  348. // W.model.actionManager.add(new WazeActionUpdateObject(segment, segment.getAttributes()))
  349.  
  350. // allow all connections
  351. // W.model.actionManager.add(new WazeActionModifyAllConnections(segment.getToNode(), true));
  352. // W.model.actionManager.add(new WazeActionModifyAllConnections(segment.getFromNode(), true));
  353.  
  354. this.applyTurns(fromConnections)
  355. this.applyTurns(toConnections)
  356. this.applyTurns(onA)
  357. this.applyTurns(onB)
  358.  
  359. this.groupEnd()
  360. }
  361.  
  362. /**
  363. * Apply turns for segments
  364. * @param segments
  365. */
  366. applyTurns (segments) {
  367. let actions = []
  368. for (let sid in segments) {
  369. let segment = segments[sid]
  370. let turn
  371. switch (segment.turnData.state) {
  372. case 0 :
  373. case 1 :
  374. turn = WazeModelGraphTurnData.create()
  375. turn = turn.withState(segment.turnData.state)
  376. .withRestrictions(segment.turnData.restrictions)
  377. .withInstructionOpcode(segment.turnData.instructionOpcode)
  378. .withLanes(segment.turnData.lanes)
  379.  
  380. actions.push(new WazeModelGraphActionsSetTurn(W.model.getTurnGraph(), segment.withTurnData(turn)))
  381. break
  382. }
  383. }
  384. let multiAction = new WazeActionMultiAction(actions)
  385. multiAction._description = I18n.t('save.changes_log.actions.SetTurn.update')
  386. W.model.actionManager.add(multiAction)
  387. }
  388. }
  389.  
  390. let WazeActionConnectSegment
  391. let WazeActionDisconnectSegment
  392. let WazeActionModifyAllConnections
  393. let WazeActionMultiAction
  394. let WazeActionUpdateObject
  395. let WazeActionUpdateSegmentGeometry
  396. let WazeModelGraphTurnData
  397. let WazeModelGraphActionsSetTurn
  398.  
  399. $(document).on('bootstrap.wme', () => {
  400. let Instance = new E87(NAME, SETTINGS)
  401. Instance.init(BUTTONS)
  402.  
  403. WazeActionConnectSegment = require('Waze/Action/ConnectSegment')
  404. WazeActionDisconnectSegment = require('Waze/Action/DisconnectSegment')
  405. WazeActionModifyAllConnections = require('Waze/Action/ModifyAllConnections')
  406. WazeActionMultiAction = require('Waze/Action/MultiAction')
  407. WazeActionUpdateObject = require('Waze/Action/UpdateObject')
  408. WazeActionUpdateSegmentGeometry = require('Waze/Action/UpdateSegmentGeometry')
  409. WazeModelGraphTurnData = require('Waze/Model/Graph/TurnData')
  410. WazeModelGraphActionsSetTurn = require('Waze/Model/Graph/Actions/SetTurn')
  411. })
  412. })()