Greasy Fork is available in English.

WME Switch Uturns

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

// ==UserScript==
// @name         WME Switch Uturns
// @version      2023.12.14.001
// @description  Switches U-turns for selected node or segment. Forked and improved "WME Add Uturn from node" script.
// @author       ixxvivxxi, uranik, turbopirate, AntonShevchuk
// @namespace    https://greasyfork.org/users/160654-waze-ukraine
// @match        https://*.waze.com/editor*
// @match        https://*.waze.com/*/editor*
// @exclude      https://*.waze.com/user/editor*
// @icon         
// @grant        none
// @require      https://update.greasyfork.org/scripts/450160/1218867/WME-Bootstrap.js
// @require      https://update.greasyfork.org/scripts/452563/1218878/WME.js
// @require      https://update.greasyfork.org/scripts/450221/1137043/WME-Base.js
// @require      https://update.greasyfork.org/scripts/450320/1281847/WME-UI.js
// ==/UserScript==

/* jshint esversion: 8 */
/* global require */
/* global $, jQuery */
/* global W */
/* global I18n */
/* global WME, WMEBase, WMEUI, WMEUIHelper, WMEUIHelperTab, WMEUIShortcut */

(function () {
  'use strict'

  // Script name, uses as unique index
  const NAME = 'SWITCH-UTURNS'

  // Translations
  const TRANSLATION = {
    'en': {
      title: 'Switch U-Turns',
      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',
      count: 'Count nodes and U-Turns',
      switch: 'Switch U-turn at point',
      nodes: 'Nodes',
      allowed: 'Allowed',
      disallowed: 'Disallowed',
      allow: 'Allow all U-turns',
      disallow: 'Disallow all U-turns',
    },
    'uk': {
      title: 'Керування розворотами',
      description: 'Оберіть сегмент або вузол щоб змінити розвороти за допомогою <a href="#keyboard-dialog" target="_blank" rel="noopener noreferrer" data-toggle="modal">гарячих клавіш</a> або кнопок',
      count: 'Порахувати вузли та розвороти',
      switch: 'Змінити розворот у вузлі',
      nodes: 'Вузли',
      allowed: 'Дозволено',
      disallowed: 'Заборонено',
      allow: 'Дозволити всі розвороти',
      disallow: 'Заборонити всі розвороти',
    },
    'ru': {
      title: 'Управление разворотами',
      description: 'Выберите сегмент или узел для изменения разворотов с помощью <a href="#keyboard-dialog" target="_blank" rel="noopener noreferrer" data-toggle="modal">комбинаций клавиш</a> или кнопок',
      count: 'Посчитать узлы и развороты',
      switch: 'Изменить разворот в узле',
      nodes: 'Узлы',
      allowed: 'Разрешено',
      disallowed: 'Запрещено',
      allow: 'Разрешить все развороты',
      disallow: 'Запретить все развороты',
    }
  }

  const STYLE =
    'p.switch-uturns-counter { margin-top: 15px; padding-left: 15px; }' +
    'p.switch-uturns-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 ALLOW = 1
  const DISALLOW = 0

  class UTurns 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)
      let button = this.tab.addButton(NAME, I18n.t(name).count, '', () => this.updateTabUI(this.countUturns()))
      button.html().className += ' waze-btn-blue'

      this.tab.addText('counter', '')
      this.tab.addText(
        'info',
        '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
      )
      // Inject custom HTML to container in the WME interface
      this.tab.inject()
    }
    /**
     * Handler for `node.wme` event
     * @param {jQuery.Event} event
     * @param {HTMLElement} element
     * @param {W.model} model
     * @return {void}
     */
    onNode (event, element, model) {
      if (model.getSegmentIds().length < 2 || !model.isGeometryEditable()) {
        return
      }
      this.removePanel(element)
      this.createPanel(element)
      this.updateNodeUI()
    }

    /**
     * Added controls
     * @param {HTMLElement} element
     */
    createPanel (element) {
      // Container
      let container = document.createElement('div')
      container.id = NAME
      // Separator space
      container.append(document.createElement('hr'))
      // Title
      let title = document.createElement('p')
      title.innerText = I18n.t(NAME).title
      container.append(title)
      // Text
      this.text = document.createElement('p')
      container.append(this.text)
      // Allow button
      this.allow = document.createElement('wz-button')
      this.allow.color = 'shadowed'
      this.allow.innerText = I18n.t(NAME).allow
      this.allow.onclick = () => this.switchNodeUturn(ALLOW)
      container.append(this.allow)
      // Disallow button
      this.disallow = document.createElement('wz-button')
      this.disallow.color = 'shadowed'
      this.disallow.innerText = I18n.t(NAME).disallow
      this.disallow.onclick = () => this.switchNodeUturn(DISALLOW)
      container.append(this.disallow)

      element.parentNode.append(container)
    }

    /**
     * Remove controls
     * @param {HTMLElement} element
     */
    removePanel(element) {
      element.parentNode.querySelector('#' + NAME)?.remove()
    }

    /**
     * Update counter for the plugin tab
     * @param {Object} counter
     */
    updateTabUI (counter) {
      this.tab.html().querySelector('p.switch-uturns-counter').innerHTML = '' +
        I18n.t(NAME).nodes + ': ' + counter.nodes + '<br/>' +
        I18n.t(NAME).allowed + ': ' + counter.allowed + '<br/>' +
        I18n.t(NAME).disallowed + ': ' + counter.disallowed
    }

    /**
     * Updated buttons status and counters
     */
    updateNodeUI () {
      let node = WME.getSelectedNode()
      if (!node
        || node.getSegmentIds().length < 2
        || !node.isGeometryEditable()) {
        return
      }
      let counter = this.countNodeUturns(node)

      // Change display properties of the buttons
      this.allow.style.display = counter.disallowed ? 'flex' : 'none'
      this.disallow.style.display = counter.allowed ? 'flex' : 'none'

      // Change text
      this.text.innerHTML =
        I18n.t(NAME).allowed + ': ' + counter.allowed + '<br/>' +
        I18n.t(NAME).disallowed + ': ' + counter.disallowed
    }

    /**
     * @return {{nodes: number, allowed: number, disallowed: number}}
     */
    countUturns () {
      let counters = {
        nodes: 0,
        allowed: 0,
        disallowed: 0
      }
      for (let node in W.model.nodes.objects) {
        let counter = this.countNodeUturns(W.model.nodes.objects[node])
        counters.nodes++
        counters.allowed += counter.allowed
        counters.disallowed += counter.disallowed
      }
      return counters
    }

    /**
     * @param {Object} node
     * @return {{allowed: number, disallowed: number}}
     */
    countNodeUturns (node) {
      let counter = {
        allowed: 0,
        disallowed: 0
      }

      let segmentsIds = node.getSegmentIds()

      for (let i = 0; i < segmentsIds.length; i++) {
        let segment = W.model.segments.getObjectById(segmentsIds[i])
        if (!segment) {
          continue
        }
        if (segment.isTurnAllowed(segment, node)) {
          counter.allowed++
        } else {
          counter.disallowed++
        }
      }
      return counter
    }

    /**
     * Handler for selected node
     * @param {Number} status ALLOW or DISALLOW
     */
    switchNodeUturn (status) {
      let node = WME.getSelectedNode()
      if (!node) {
        return
      }
      let segmentsIds = node.getSegmentIds()
      if (segmentsIds.length < 2) {
        return
      }
      for (let i = 0; i < segmentsIds.length; i++) {
        let segment = W.model.segments.getObjectById(segmentsIds[i])
        if (segment.isOneWay()) {
          continue;
        }
        let turn = W.model.getTurnGraph().getTurnThroughNode(node, segment, segment)
        W.model.actionManager.add(
          new WazeActionSetTurn(
            W.model.getTurnGraph(),
            turn.withTurnData(turn.getTurnData().withState(status)))
        )
      }
      this.updateNodeUI()

      this.log('u-turns in the node with ID ' + node.getID() + ' switched to ' + (status ? 'ALLOW' : 'DISALLOW'))
    }

    /**
     * Handler for selected segments
     * @param direction
     */
    switchSegmentUturn (direction = 'A') {
      let segments = WME.getSelectedSegments()
      for (let i = 0, total = segments.length; i < total; i++) {
        let segment = segments[i]
        if (segment.isOneWay()) {
          continue;
        }
        let node = (direction === 'A') ? segment.getFromNode() : segment.getToNode()
        let status = segment.isTurnAllowed(segment, node) ? DISALLOW : ALLOW
        let turn = W.model.getTurnGraph().getTurnThroughNode(node, segment, segment)
        W.model.actionManager.add(
          new WazeActionSetTurn(
            W.model.getTurnGraph(),
            turn.withTurnData(turn.getTurnData().withState(status)) // enable it
          )
        )

        this.log('u-turn in the point ' + direction + ' switched to ' + (status ? 'ALLOW' : 'DISALLOW'))
      }
    }
  }

  let WazeActionSetTurn

  $(document)
    .on('bootstrap.wme', () => {
      // Require Waze components
      WazeActionSetTurn = require('Waze/Model/Graph/Actions/SetTurn')

      let UTurnsInstance = new UTurns(NAME)

      // Hotkeys for node manipulation
      WMEUI.addShortcut(NAME + '-node-allow', I18n.t(NAME).allow, NAME, I18n.t(NAME).title, 'A+A', () => UTurnsInstance.switchNodeUturn(1))
      WMEUI.addShortcut(NAME + '-node-disallow', I18n.t(NAME).disallow, NAME, I18n.t(NAME).title, 'A+S', () => UTurnsInstance.switchNodeUturn(0))
      // Hotkeys for segment manipulation
      WMEUI.addShortcut(NAME + '-segment-a', I18n.t(NAME).switch + ' A', NAME, I18n.t(NAME).title, 'A+Q', () => UTurnsInstance.switchSegmentUturn('A'))
      WMEUI.addShortcut(NAME + '-segment-b', I18n.t(NAME).switch + ' B', NAME, I18n.t(NAME).title, 'A+W', () => UTurnsInstance.switchSegmentUturn('B'))
      // Update count of UTurns on events
      W.model.actionManager.events.register('afterundoaction', null, () => UTurnsInstance.updateNodeUI())
      W.model.actionManager.events.register('afterclearactions', null, () => UTurnsInstance.updateNodeUI())
      W.model.actionManager.events.register('afteraction', null, () => UTurnsInstance.updateNodeUI())
    })
})()