WME E87 Inconsistent direction

Solves the inconsistent direction problem

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

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         WME E87 Inconsistent direction
// @version      0.0.8
// @description  Solves the inconsistent direction problem
// @license      MIT License
// @author       Anton Shevchuk
// @namespace    https://greasyfork.org/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://greasyfork.org/scripts/389765-common-utils/code/CommonUtils.js?version=1090053
// @require      https://greasyfork.org/scripts/450160-wme-bootstrap/code/WME-Bootstrap.js?version=1135567
// @require      https://greasyfork.org/scripts/452563-wme/code/WME.js?version=1101598
// @require      https://greasyfork.org/scripts/450221-wme-base/code/WME-Base.js?version=1137043
// @require      https://greasyfork.org/scripts/450320-wme-ui/code/WME-UI.js?version=1137289
// ==/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 =
    '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) {
      element.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) {
      let reversed = W.selectionManager.getReversedSegments()

      if (reversed.numReversed === 0) {
        // you can reverse all selected segments
        element.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
        }

        let container = document.createElement('div')
        container.className = 'e87-container'
        container.append(buttonToForward)
        container.append(buttonToReverse)

        $('wz-alert.sidebar-alert.inconsistent-direction-alert .sidebar-alert-content')
          .after(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.geometry.clone()
      geometry.components.reverse()

      if (!geometry.components[0].equals(toNode.attributes.geometry)) {
        let delta = { x: 0, y: 0 }
        delta.x = toNode.attributes.geometry.x - geometry.components[0].x
        delta.y = toNode.attributes.geometry.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.attributes.geometry)) {
        let delta = { x: 0, y: 0 }
        delta.x = fromNode.attributes.geometry.x - geometry.components[points].x
        delta.y = fromNode.attributes.geometry.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.geometry, 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')
  })
})()