WME E85 Simplify Street Geometry

Simplify Street Geometry, looks like fork

目前為 2022-12-12 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         WME E85 Simplify Street Geometry
// @version      0.0.1
// @description  Simplify Street Geometry, looks like fork
// @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=1126584
// @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=1101617
// @require      https://greasyfork.org/scripts/450320-wme-ui/code/WME-UI.js?version=1127621
// ==/UserScript==

/* jshint esversion: 8 */

/* global require */
/* global $, jQuery */
/* global W */
/* global I18n */
/* global OpenLayers */
/* global WME, WMEBase */
/* global WMEUI, WMEUIHelper, WMEUIHelperPanel, WMEUIHelperModal, WMEUIHelperTab, WMEUIShortcut */
/* global Container, Settings, SimpleCache, Tools  */

(function () {
  'use strict'

  // Script name, uses as unique index
  const NAME = 'E85'

  // Translations
  const TRANSLATION = {
    'en': {
      title: 'Street Geometry',
      description: 'Simplify and straighten up streets',
      buttons: {
        A: 'Simplify',
        B: 'Straighten',
        C: '∡90°',
      },
      settings: {
        title: 'Settings',
        description: 'Settings for simplify function',
        simplifyShort: 'Remove segment shorter than',
        simplifyTwoShort: 'Join segments shorter than',
      },
    },
    'uk': {
      title: 'Геометрія вулиць',
      description: 'Спрощуйте та вирівнюйте вулиці',
      buttons: {
        A: 'Спростити',
        B: 'Вирівняти',
        C: '∡90°',
      },
      settings: {
        title: 'Налаштування',
        description: 'Для спрощення сегментів будуть враховані наступні параметри',
        simplifyShort: 'Видаляти сегменти менші ніж',
        simplifyTwoShort: 'Об’єднувати сегменти меньші ніж',
      },
    },
    'ru': {
      title: 'Геометрия улиц',
      description: 'Упрощайте и выравнивайте геометрию улиц',
      buttons: {
        A: 'Упростить',
        B: 'Выровнять',
        C: '∡90°',
      },
      settings: {
        title: 'Настройки',
        description: 'Параметры для упрощения геометрии сегмента',
        simplifyShort: 'Если сегмент короче, чем',
        simplifyTwoShort: 'Если сегменты меньше, чем',
      },
    }
  }

  const STYLE =
    '.e85 legend { cursor:pointer; font-size: 12px; font-weight: bold; width: auto; text-align: right; border: 0; margin: 0; padding: 0 8px; }' +
    '.e85 fieldset { border: 1px solid #ddd; padding: 4px; }' +
    '.e85 fieldset.e85 div.controls label { white-space: normal; font-weight: normal; line-height: 32px; }' +
    '.e85 fieldset.e85 div.controls input[type="number"] { float:right; wight: 32px }' +
    'p.e85-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 = {
    A: {
      title: I18n.t(NAME).buttons.A,
      description: I18n.t(NAME).buttons.A,
      shortcut: '',
    },
    B: {
      title: I18n.t(NAME).buttons.B,
      description: I18n.t(NAME).buttons.B,
      shortcut: '',
    },
    C: {
      title: I18n.t(NAME).buttons.C,
      description: I18n.t(NAME).buttons.C,
      shortcut: '',
    },
  }

  // Default settings
  const SETTINGS = {
    simplifyShort: 3,
    simplifyTwoShort: 40
  }

  let WazeActionUpdateSegmentGeometry
  let WazeActionMoveNode
  let WazeActionAddNode

  class E85 extends WMEBase {
    constructor (name, settings) {
      super(name)
      this.settings = new Settings(name, settings)
    }

    /**
     * Initial UI elements
     * @param {Object} buttons
     */
    init (buttons) {
      /** @type {WMEUIHelper} */
      this.helper = new WMEUIHelper(this.name)

      /** @type {WMEUIHelperTab} */
      this.tab = this.helper.createTab(
        I18n.t(this.name).title,
        I18n.t(this.name).description,
        {
          'icon': '<i class="w-icon panel-header-component-icon w-icon-route"></i>'
        }
      )

      // Setup options for script
      let fieldset = this.helper.createFieldset(I18n.t(NAME).settings.title)
      fieldset.addText('description', I18n.t(NAME).settings.description)
      let settings = this.settings.get()
      for (let item in settings) {
        if (settings.hasOwnProperty(item)) {
          fieldset.addNumber(
            'settings-' + item,
            I18n.t(NAME).settings[item],
            I18n.t(NAME).settings[item],
            event => this.settings.set([item], event.target.value),
            this.settings.get(item),
            1,
            100,
            1
          )
        }
      }
      this.tab.addElement(fieldset)
      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 window `beforeunload` event
     * @param {jQuery.Event} event
     * @return {Null}
     */
    onBeforeUnload (event) {
      this.settings.save()
    }

    /**
     * Handler for `segment.wme` event
     * @param {jQuery.Event} event
     * @param {HTMLElement} element
     * @param {W.model} model
     * @return {void}
     */
    onSegment (event, element, model) {
      this.log('Selected one segment')

      let panel = this.helper.createPanel(I18n.t(this.name).title)
      panel.addButton(
        'A',
        BUTTONS.A.title,
        BUTTONS.A.description,
        () => this.simplifySegmentGeometry(model),
        BUTTONS.A.shortcut
      )

      panel.addButton(
        'B',
        BUTTONS.B.title,
        BUTTONS.B.description,
        () => this.straightenSegmentGeometry(model),
        BUTTONS.B.shortcut
      )

      element.prepend(panel.html())
    }

    /**
     * Handler for `segments.wme` event
     * @param {jQuery.Event} event
     * @param {HTMLElement} element
     * @param {Array} models
     * @return {void}
     */
    onSegments (event, element, models) {
      this.log('Selected some segments')

      let panel = this.helper.createPanel(I18n.t(this.name).title)
      panel.addButton(
        'A',
        BUTTONS.A.title,
        BUTTONS.A.description,
        () => this.simplifyStreetGeometry(models),
        BUTTONS.A.shortcut
      )
      panel.addButton(
        'B',
        BUTTONS.B.title,
        BUTTONS.B.description,
        () => this.straightenStreetGeometry(models),
        BUTTONS.B.shortcut
      )

      if (models.length === 2) {
        panel.addButton(
          'C',
          BUTTONS.C.title,
          BUTTONS.C.description,
          () => this.orthogonalizeStreetGeometry(models),
          BUTTONS.C.shortcut
        )
      }

      element.prepend(panel.html())
    }

    /**
     * Remove geometry nodes on the target segment
     * @param {Object} model
     * @return {void}
     */
    simplifySegmentGeometry (model) {
      this.log(
        'try to simplify segment geometry: remove segments shorten than ' + this.settings.get('simplifyShort') + 'm ' +
        'and join segments shorten than ' + this.settings.get('simplifyTwoShort') + 'm'
      )
      if (model.geometry.components.length < 3) {
        return
      }

      // calculate every segment length
      let segmentsLength = []
      for (let i = 0; i < model.geometry.components.length - 1; i++) {
        let nodeStart = model.geometry.components[i],
          nodeEnd = model.geometry.components[i + 1]

        let line = new OpenLayers.Geometry.LineString([nodeStart, nodeEnd])
        segmentsLength.push(Math.round(line.getGeodesicLength('EPSG:900913')))
      }

      this.log('length of the segments: ' + segmentsLength.join(', '))

      // find nodes with short segments around
      let removeNodes = []
      for (let i = 0; i < segmentsLength.length - 1; i++) {
        if (segmentsLength[i] < this.settings.get('simplifyShort')) {
          this.log('found too short segment: ' + segmentsLength[i] + 'm')
          removeNodes.push(i+1)
        } else if (segmentsLength[i] + segmentsLength[i+1] < this.settings.get('simplifyTwoShort')) {
          this.log(
            'found node with short segments: ' + segmentsLength[i] + ' + ' + segmentsLength[i+1] +' = '+
            (segmentsLength[i] + segmentsLength[i+1]) + 'm'
          )
          removeNodes.push(i+1)
        }
      }

      // remove nodes from geometry
      if (removeNodes.length) {
        let newGeometry = model.geometry.clone()
        let components = []
        for (let i = 0; i < newGeometry.components.length; i++) {
          if (removeNodes.indexOf(i) === -1) {
            components.push(newGeometry.components[i])
          }
        }
        newGeometry.components = components
        W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(model, model.geometry, newGeometry))
      }
    }

    /**
     * Remove geometry nodes on the target segments
     * @param {Array} models
     * @return {void}
     */
    simplifyStreetGeometry(models) {
      for (let i = 0; i < models.length; i++) {
        this.simplifySegmentGeometry(models[i])
      }
    }

    /**
     * Выравнивает сегменты в прямую линию, перемещая промежуточные узлы
     * в точки пересечения перпендикуляров к вычисленной прямой, проходящей через
     * начальный и конечный узлы выделения
     * A,B,C - параметры вычисленной прямой уравнения Аx + By + C = 0
     *
     * @param {Array} models
     * @return {void}
     */
    straightenStreetGeometry (models) {
      this.log('simplify street geometry')
      let T1, T2,
        t,
        A = 0.0,
        B = 0.0,
        C = 0.0,
        D = 0.0

      // определим линию выравнивания
      this.log('calculating the formula for the line')

      for (let i = 0; i < models.length; i++) {
        let segment = models[i]

        let geometry = segment.geometry

        // определяем формулу наклонной прямой
        if (geometry.components.length > 1) {
          let A1 = geometry.components[0].clone(),
            A2 = geometry.components[geometry.components.length - 1].clone()

          let dX = getDeltaDirect(A1.x, A2.x)
          let dY = getDeltaDirect(A1.y, A2.y)

          let tX = i > 0 ? getDeltaDirect(T1.x, T2.x) : 0
          let tY = i > 0 ? getDeltaDirect(T1.y, T2.y) : 0

          this.log('vector of the line - tX=' + tX + ', tY=' + tY)
          this.log('segment #' + (i + 1) + ' (' + A1.x + '; ' + A1.y + ') - (' + A2.x + '; ' + A2.y + '), dX=' + dX + ', dY=' + dY)

          if (dX < 0) {
            t = A1.x
            A1.x = A2.x
            A2.x = t

            t = A1.y
            A1.y = A2.y
            A2.y = t

            dX = getDeltaDirect(A1.x, A2.x)
            dY = getDeltaDirect(A1.y, A2.y)

            this.log('rotate segment #' + (i + 1) + ' (' + A1.x + '; ' + A1.y + ') - (' + A2.x + '; ' + A2.y + '), dX=' + dX + ', dY=' + dY)
          }

          if (i === 0) {
            T1 = A1.clone()
            T2 = A2.clone()
          } else {
            if (A1.x < T1.x) {
              T1.x = A1.x
              T1.y = A1.y
            }

            if (A2.x > T2.x) {
              T2.x = A2.x
              T2.y = A2.y
            }
          }

          this.log('calculated straight line (' + T1.x + '; ' + T1.y + ') - (' + T2.x + '; ' + T2.y + ')')
        }
      }

      A = T2.y - T1.y
      B = T1.x - T2.x
      C = T2.x * T1.y - T1.x * T2.y

      this.log('line coords: (' + T1.x + ';' + T1.y + ') - (' + T2.x + ';' + T2.y + ')')
      this.log('line formula: ' + A + 'x + ' + B + 'y + ' + C)

      for (let i = 0; i < models.length; i++) {
        let segment = models[i]

        this.log('simplify segment #' + i)

        // упрощаем сегмент, если нужно
        this.straightenSegmentGeometry(segment)

        // работа с узлом
        let node = W.model.nodes.getObjectById(segment.attributes.fromNodeID)
        let D = node.attributes.geometry.y * A - node.attributes.geometry.x * B
        let r1 = getIntersectCoord(A, B, C, D)
        this.log('move node A')
        this.moveNode(node, r1)

        let node2 = W.model.nodes.getObjectById(segment.attributes.toNodeID)
        let D2 = node2.attributes.geometry.y * A - node2.attributes.geometry.x * B
        let r2 = getIntersectCoord(A, B, C, D2)
        this.log('move node B')
        this.moveNode(node2, r2)

        this.log('segment #' + (i + 1) + ' (' + r1[0] + ';' + r1[1] + ') - (' + r2[0] + ';' + r2[1] + ')')
      }
    }

    /**
     * выстраивает два выбранных сегмента перпендикулярно друг другу
     * перемещая их общий узел
     *
     * @param {Array} models
     * @return {void}
     */
    orthogonalizeStreetGeometry (models) {
      this.log('orthogonalize street geometry')

      let seg1 = models[0],
        seg2 = models[1],
        seg1Attrs = seg1.attributes,
        seg2Attrs = seg2.attributes
      let commonNodeID

      if (seg1.type != 'segment' || seg2.type != 'segment') {
        this.log('only segments must be selected')
        return
      }

      // ID общего узла
      let node = {}
      if (seg1Attrs.fromNodeID === seg2Attrs.fromNodeID) commonNodeID = seg1Attrs.fromNodeID
      if (seg1Attrs.fromNodeID === seg2Attrs.toNodeID) commonNodeID = seg1Attrs.fromNodeID
      if (seg1Attrs.toNodeID === seg2Attrs.fromNodeID) commonNodeID = seg1Attrs.toNodeID
      if (seg1Attrs.toNodeID === seg2Attrs.toNodeID) commonNodeID = seg1Attrs.toNodeID

      if (!commonNodeID) {
        this.log('segments does not have common node.')
        return
      }

      this.log('common node ID: ' + commonNodeID)
      node = W.model.nodes.getObjectById(commonNodeID)

      // ID другого узла второго сегмента. От него будем строить перпендикуляр
      let otherNodeID = commonNodeID == seg2Attrs.fromNodeID ? seg2Attrs.toNodeID : seg2Attrs.fromNodeID
      let otherNode = W.model.nodes.getObjectById(otherNodeID)

      // упростим оба сегмента
      // TODO: подумать, можно ли использовать координаты промежуточных узлов и не упрощать сегменты
      this.straightenSegmentGeometry(seg1)
      this.straightenSegmentGeometry(seg2)

      // вычислим новое положение общего узла
      // координаты концов первого сегмента
      let x1 = seg1.getFromNode().attributes.geometry.x,
        y1 = seg1.getFromNode().attributes.geometry.y,
        x2 = seg1.getToNode().attributes.geometry.x,
        y2 = seg1.getToNode().attributes.geometry.y

      // коэффициенты в формуле прямой, проходящей через концы первого сегмента
      let A = y1 - y2,
        B = x2 - x1,
        C = x1 * y2 - x2 * y1,
        // что такое D ???
        D = otherNode.attributes.geometry.y * A - otherNode.attributes.geometry.x * B

      // move node and its segments to calculated position
      this.moveNode(node, getIntersectCoord(A, B, C, D))
    }

    /**
     * Straighten up segment, remove all geometry nodes except first and last
     * @param {Object} segment
     */
    straightenSegmentGeometry (segment) {
      if (segment.geometry.components.length > 2) {
        let newGeometry = segment.geometry.clone()
        newGeometry.components.splice(1, newGeometry.components.length - 2)
        W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(segment, segment.geometry, newGeometry))
      }
    }

    /**
     * Move node to new position
     * @param {Object} node target
     * @param {Array<2>} coords of the new position, array of the wo elements
     */
    moveNode (node, coords) {
      let nodeGeo = node.geometry.clone()
      nodeGeo.x = coords[0]
      nodeGeo.y = coords[1]
      nodeGeo.calculateBounds()

      let connectedSegObjs = {}
      let emptyObj = {}
      for (let j = 0; j < node.attributes.segIDs.length; j++) {
        let segId = node.attributes.segIDs[j]
        connectedSegObjs[segId] = W.model.segments.getObjectById(segId).geometry.clone()
      }
      W.model.actionManager.add(new WazeActionMoveNode(node, node.geometry, nodeGeo, connectedSegObjs, emptyObj))
    }
  }

  // рассчитаем пересечение перпендикуляра точки с наклонной прямой
  function getIntersectCoord (A, B, C, D) {
    //  http://rsdn.ru/forum/alg/2589531.hot
    let r = [2]
    r[1] = -1.0 * (C * B - A * D) / (A * A + B * B)
    r[0] = (-r[1] * (B + A) - C + D) / (A - B)

    return r
  }

  // определим направляющие
  function getDeltaDirect (A, B) {
    let d = 0.0

    if (A < B)
      d = 1.0
    else if (A > B)
      d = -1.0

    return d
  }

  $(document).on('bootstrap.wme', () => {

    WazeActionUpdateSegmentGeometry = require('Waze/Action/UpdateSegmentGeometry')
    WazeActionMoveNode = require('Waze/Action/MoveNode')
    WazeActionAddNode = require('Waze/Action/AddNode')

    let Instance = new E85(NAME, SETTINGS)
    Instance.init(BUTTONS)

    // rename shortcut section
    WMEUIShortcut.setGroupTitle(NAME, I18n.t(NAME).title)
  })

})()