WME E87 Inconsistent direction

Solves the inconsistent direction problem

当前为 2023-01-06 提交的版本,查看 最新版本

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