- // ==UserScript==
- // @name WME E87 Inconsistent direction
- // @name:uk WME 🇺🇦 E87 Inconsistent direction
- // @version 0.1.2
- // @description Solves the inconsistent direction problem
- // @description:uk Дозволяє вирішувати проблему різнонаправленних сегментів
- // @license MIT License
- // @author Anton Shevchuk
- // @namespace https://greatest.deepsurf.us/users/227648-anton-shevchuk
- // @supportURL https://github.com/AntonShevchuk/wme-template/issues
- // @match https://*.waze.com/editor*
- // @match https://*.waze.com/*/editor*
- // @exclude https://*.waze.com/user/editor*
- // @icon 
- // @grant none
- // @require https://update.greatest.deepsurf.us/scripts/389765/1090053/CommonUtils.js
- // @require https://update.greatest.deepsurf.us/scripts/450160/1218867/WME-Bootstrap.js
- // @require https://update.greatest.deepsurf.us/scripts/452563/1218878/WME.js
- // @require https://update.greatest.deepsurf.us/scripts/450221/1137043/WME-Base.js
- // @require https://update.greatest.deepsurf.us/scripts/450320/1555446/WME-UI.js
- // ==/UserScript==
-
- /* jshint esversion: 8 */
-
- /* global require */
- /* global $, jQuery */
- /* global W */
- /* global I18n */
- /* global OpenLayers */
- /* global WME, WMEBase */
- /* global WMEUI, WMEUIHelper, WMEUIHelperPanel, WMEUIHelperModal, WMEUIHelperTab, WMEUIShortcut, WMEUIHelperFieldset */
- /* global Container, Settings, SimpleCache, Tools */
-
- (function () {
- 'use strict'
-
- // Script name, uses as unique index
- const NAME = 'E87'
-
- // Translations
- const TRANSLATION = {
- 'en': {
- title: 'Direction →',
- description: 'Plugin WME E87 solves the inconsistent direction problem.<br/>Choose one or more segment to change direction.',
- buttons: {
- toggle: 'Change direction',
- forward: 'A → B',
- reverse: 'B → A',
- },
- },
- 'uk': {
- title: 'Напрямки →',
- description: 'Плагін WME E87 для вирішиння проблеми різно направленних вулиць.<br/>Оберіть один або декілька сегментів щоб застосувати зміни.',
- buttons: {
- toggle: 'Змінити напрямок',
- forward: 'A → B',
- reverse: 'B → A',
- },
- },
- 'ru': {
- title: 'Направления →',
- description: 'Плагин WME E87 для решения проблемы разнонаправленных улиц.<br/>Выберите один или несколько сегментов, чтобы внести изменения.',
- buttons: {
- toggle: 'Изменить направление',
- forward: 'A → B',
- reverse: 'B → A',
- },
- }
- }
-
- const STYLE =
- '.lanes-tab div.e87 { border: 1px solid var(--hairline); border-radius: 6px; margin-bottom: 16px; padding: 8px 16px 18px; } ' +
- 'button.waze-btn.e87 { background: #f2f4f7; border: 1px solid #ccc; margin: 2px; } ' +
- '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); } ' +
- 'button.waze-btn.e87:focus { background: #f2f4f7; } ' +
- 'button.e87-forward, button.e87-reverse { margin: 2px 8px; }' +
- 'div.e87-container { display: flex; flex: auto; justify-content: space-evenly; } ' +
- 'p.e87-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }'
-
- WMEUI.addTranslation(NAME, TRANSLATION)
- WMEUI.addStyle(STYLE)
-
- const BUTTONS = {
- toggle: {
- title: I18n.t(NAME).buttons.toggle,
- description: I18n.t(NAME).buttons.toggle,
- shortcut: '',
- },
- }
-
- // Default settings
- const SETTINGS = {}
-
- class E87 extends WMEBase {
- constructor (name, settings = null) {
- super(name, settings)
-
- /** @type {WMEUIHelper} */
- this.helper = new WMEUIHelper(this.name)
-
- /** @type {WMEUIHelperTab} */
- this.tab = this.helper.createTab(I18n.t(this.name).title, { image: GM_info.script.icon })
- this.tab.addText('description', I18n.t(this.name).description)
- this.tab.addText('info', '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version)
- this.tab.inject()
-
- /** @type {WMEUIHelperPanel} */
- this.panel = this.helper.createPanel(I18n.t(name).title)
- }
-
- /**
- * Init button for selection of the segment
- * @param buttons
- */
- init (buttons) {
- buttons.toggle.callback = (e) => {
- e.preventDefault()
- WME.getSelectedSegments().forEach(
- segment => this.invert(segment.getID())
- )
- }
- this.panel.addButtons(buttons)
- }
-
- /**
- * Handler for `segment.wme` event
- * @param {jQuery.Event} event
- * @param {HTMLElement} element
- * @param {W.model} model
- * @return {void}
- */
- onSegment (event, element, model) {
- // Skip for walking trails and blocked roads
- if (model.isWalkingRoadType()
- || model.isLockedByHigherRank()
- || !model.isGeometryEditable()) {
- return
- }
-
- element
- //.parentNode.parentNode
- //.querySelector('.lanes-tab')
- .prepend(this.panel.html())
- }
-
- /**
- * Handler for `segments.wme` event
- * @param {jQuery.Event} event
- * @param {HTMLElement} element
- * @param {Array} models
- * @return {void}
- */
- onSegments (event, element, models) {
- // Skip for walking trails or locked roads
- if (models.filter((model) => model.isWalkingRoadType() || model.isLockedByHigherRank() || !model.isGeometryEditable()).length > 0) {
- element.querySelector('div.form-group.e87')?.remove()
- return
- }
-
- let reversed = W.selectionManager.getReversedSegments()
-
- if (reversed.numReversed === 0) {
- // you can reverse all selected segments
- element
- //.parentNode.parentNode
- //.querySelector('.lanes-tab')
- .prepend(this.panel.html())
- return
- }
-
- let result = this.detect(reversed)
-
- if (result.forward.length && result.reverse.length) {
- this.log('Inconsistent direction detected: forward = ' + result.forward.length + ' backward = ' + result.reverse.length)
-
- let buttonToForward = document.createElement('button')
- buttonToForward.type = 'button'
- buttonToForward.title = I18n.t(NAME).buttons.toggle
- buttonToForward.className = 'waze-btn waze-btn-small waze-btn-white e87 e87-forward'
- buttonToForward.innerText = I18n.t(NAME).buttons.forward + ' (' + result.reverse.length + ')'
- buttonToForward.onclick = (e) => {
- e.preventDefault()
- result.reverse.forEach(el => this.invert(el))
- buttonToForward.innerText = I18n.t(NAME).buttons.forward + ' (0)'
- buttonToForward.disabled = true
- }
- let buttonToReverse = document.createElement('button')
- buttonToReverse.type = 'button'
- buttonToReverse.title = I18n.t(NAME).buttons.toggle
- buttonToReverse.className = 'waze-btn waze-btn-small waze-btn-white e87 e87-reverse'
- buttonToReverse.innerText = I18n.t(NAME).buttons.reverse + ' (' + result.forward.length + ')'
- buttonToReverse.onclick = (e) => {
- e.preventDefault()
- result.forward.forEach(el => this.invert(el))
- buttonToReverse.innerText = I18n.t(NAME).buttons.reverse + ' (0)'
- buttonToReverse.disabled = true
- }
-
- this.container?.remove();
-
- this.container = document.createElement('div')
- this.container.className = 'e87-container'
- this.container.append(buttonToForward)
- this.container.append(buttonToReverse)
-
- $('wz-alert.sidebar-alert.inconsistent-direction-alert .sidebar-alert-content')
- .after(this.container)
- }
- }
-
- /**
- * Detect directions
- * @param {Object} segments information
- * @return {Object}
- */
- detect (segments) {
- let forward = [], reverse = []
-
- for (let el in segments) {
- el = Number.parseInt(el)
- if (Number.isNaN(el)) {
- continue
- }
- if (segments[el]) {
- reverse.push(el)
- } else {
- forward.push(el)
- }
- }
-
- return {
- forward: forward,
- reverse: reverse
- }
- }
-
- /**
- * Invert direction of the segment
- * @param {Number} id of the segment
- */
- invert (id) {
- let segment = W.model.segments.getObjectById(id)
- if (segment.isLockedByHigherRank()) {
- this.log('Locked by higher rank')
- return
- }
- this.group('invert segment ' + id)
- this.log('segment', segment)
-
- // setup and reverse attributes
- let attributes = {}
- attributes.fwdDirection = segment.attributes.revDirection
- attributes.revDirection = segment.attributes.fwdDirection
- let fwdTurnsLocked = segment.attributes.fwdTurnsLocked
- let revTurnsLocked = segment.attributes.revTurnsLocked
- // attributes.fwdTurnsLocked = segment.attributes.revTurnsLocked // ???
- // attributes.revTurnsLocked = segment.attributes.fwdTurnsLocked // ???
- // segment.setAttribute("revTurnsLocked", segment.attributes.fwdTurnsLocked)}
- // segment.setAttribute("fwdTurnsLocked", segment.attributes.revTurnsLocked)}
- attributes.fwdMaxSpeed = segment.attributes.revMaxSpeed
- attributes.revMaxSpeed = segment.attributes.fwdMaxSpeed
- attributes.fwdMaxSpeedUnverified = segment.attributes.revMaxSpeedUnverified
- attributes.revMaxSpeedUnverified = segment.attributes.fwdMaxSpeedUnverified
- attributes.fwdLaneCount = segment.attributes.revLaneCount
- attributes.revLaneCount = segment.attributes.fwdLaneCount
-
- attributes.restrictions = []
- for (let i = 0; i < segment.attributes.restrictions.length; i++) {
- attributes.restrictions[i] = segment.attributes.restrictions[i].withReverseDirection()
- }
-
- this.log('attributes', attributes)
-
- let fromNode = segment.getFromNode()
- let toNode = segment.getToNode()
-
- let onA = {}
- let toConnections = {}
- fromNode.getSegmentIds().forEach(segId => {
- // incoming directions
- if (segId !== id) {
- onA[segId] = W.model.getTurnGraph().getTurnThroughNode(fromNode, W.model.segments.getObjectById(segId), segment)
- onA[segId].toVertex.direction = onA[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
- }
- // outgoing directions
- toConnections[segId] = W.model.getTurnGraph().getTurnThroughNode(fromNode, segment, W.model.segments.getObjectById(segId))
- toConnections[segId].fromVertex.direction = toConnections[segId].fromVertex.direction === 'fwd' ? 'rev' : 'fwd'
- // u-turn
- if (segId === id) {
- toConnections[segId].toVertex.direction = toConnections[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
- }
- })
-
- let onB = {}
- let fromConnections = {}
- toNode.getSegmentIds().forEach(segId => {
- if (segId !== id) {
- onB[segId] = W.model.getTurnGraph().getTurnThroughNode(toNode, W.model.segments.getObjectById(segId), segment)
- onB[segId].toVertex.direction = onB[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
- }
-
- fromConnections[segId] = W.model.getTurnGraph().getTurnThroughNode(toNode, segment, W.model.segments.getObjectById(segId))
- fromConnections[segId].fromVertex.direction = fromConnections[segId].fromVertex.direction === 'fwd' ? 'rev' : 'fwd'
-
- // u-turn
- if (segId === id) {
- fromConnections[segId].toVertex.direction = fromConnections[segId].toVertex.direction === 'fwd' ? 'rev' : 'fwd'
- }
- })
-
- // invert the geometry of the segment
- let geometry = segment.getOLGeometry().clone()
- geometry.components.reverse()
-
- if (!geometry.components[0].equals(toNode.getOLGeometry())) {
- let delta = { x: 0, y: 0 }
- delta.x = toNode.getOLGeometry().x - geometry.components[0].x
- delta.y = toNode.getOLGeometry().y - geometry.components[0].y
- geometry.components[0].move(delta.x, delta.y)
- }
- let points = geometry.components.length - 1
- if (!geometry.components[points].equals(fromNode.getOLGeometry())) {
- let delta = { x: 0, y: 0 }
- delta.x = fromNode.getOLGeometry().x - geometry.components[points].x
- delta.y = fromNode.getOLGeometry().y - geometry.components[points].y
- geometry.components[points].move(delta.x, delta.y)
- }
-
- // disconnect the segment
- let disconnect = new WazeActionMultiAction([new WazeActionDisconnectSegment(segment, fromNode), new WazeActionDisconnectSegment(segment, toNode)])
- disconnect._description = I18n.t('save.changes_log.actions.DisconnectSegment.default')
- W.model.actionManager.add(disconnect)
-
- // update geometry of the segment
- W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(segment, segment.getGeometry(), W.userscripts.toGeoJSONGeometry(geometry)))
-
- // update attributes
- W.model.actionManager.add(new WazeActionUpdateObject(segment, attributes))
-
- // connect the segment
- let connect = new WazeActionMultiAction([new WazeActionConnectSegment(toNode, segment), new WazeActionConnectSegment(fromNode, segment)])
- connect._description = I18n.t('save.changes_log.actions.ConnectSegment.default')
- W.model.actionManager.add(connect)
-
- // update Turn's attributes
- segment.setAttribute('fwdTurnsLocked', revTurnsLocked)
- segment.setAttribute('revTurnsLocked', fwdTurnsLocked)
- // W.model.actionManager.add(new WazeActionUpdateObject(segment, segment.getAttributes()))
-
- // allow all connections
- // W.model.actionManager.add(new WazeActionModifyAllConnections(segment.getToNode(), true));
- // W.model.actionManager.add(new WazeActionModifyAllConnections(segment.getFromNode(), true));
-
- this.applyTurns(fromConnections)
- this.applyTurns(toConnections)
- this.applyTurns(onA)
- this.applyTurns(onB)
-
- this.groupEnd()
- }
-
- /**
- * Apply turns for segments
- * @param segments
- */
- applyTurns (segments) {
- let actions = []
- for (let sid in segments) {
- let segment = segments[sid]
- let turn
- switch (segment.turnData.state) {
- case 0 :
- case 1 :
- turn = WazeModelGraphTurnData.create()
- turn = turn.withState(segment.turnData.state)
- .withRestrictions(segment.turnData.restrictions)
- .withInstructionOpcode(segment.turnData.instructionOpcode)
- .withLanes(segment.turnData.lanes)
-
- actions.push(new WazeModelGraphActionsSetTurn(W.model.getTurnGraph(), segment.withTurnData(turn)))
- break
- }
- }
- let multiAction = new WazeActionMultiAction(actions)
- multiAction._description = I18n.t('save.changes_log.actions.SetTurn.update')
- W.model.actionManager.add(multiAction)
- }
- }
-
- let WazeActionConnectSegment
- let WazeActionDisconnectSegment
- let WazeActionModifyAllConnections
- let WazeActionMultiAction
- let WazeActionUpdateObject
- let WazeActionUpdateSegmentGeometry
- let WazeModelGraphTurnData
- let WazeModelGraphActionsSetTurn
-
- $(document).on('bootstrap.wme', () => {
- let Instance = new E87(NAME, SETTINGS)
- Instance.init(BUTTONS)
-
- WazeActionConnectSegment = require('Waze/Action/ConnectSegment')
- WazeActionDisconnectSegment = require('Waze/Action/DisconnectSegment')
- WazeActionModifyAllConnections = require('Waze/Action/ModifyAllConnections')
- WazeActionMultiAction = require('Waze/Action/MultiAction')
- WazeActionUpdateObject = require('Waze/Action/UpdateObject')
- WazeActionUpdateSegmentGeometry = require('Waze/Action/UpdateSegmentGeometry')
- WazeModelGraphTurnData = require('Waze/Model/Graph/TurnData')
- WazeModelGraphActionsSetTurn = require('Waze/Model/Graph/Actions/SetTurn')
- })
- })()