WME E85 Simplify Street Geometry

Simplify Street Geometry, looks like fork

目前為 2023-02-23 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 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.1.5
// @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-e85/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=1153357
// @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 */
/* 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 simplifying segments',
        simplifyShort: 'Remove a fragment shorter than',
        simplifyAngle: 'If the angle is bigger than',
        simplifyTwoShort: 'and fragments shorter than',
      },
    },
    'uk': {
      title: 'Геометрія вулиць',
      description: 'Спрощуйте та вирівнюйте вулиці',
      buttons: {
        A: 'Спростити',
        B: 'Вирівняти',
        C: '∡90°',
      },
      settings: {
        title: 'Налаштування',
        description: 'Для спрощення сегментів будуть враховані наступні параметри',
        simplifyShort: 'Видаляти фрагменти менші ніж',
        simplifyAngle: 'Або якщо кут більше ніж',
        simplifyTwoShort: 'та фрагменти меньші ніж',
      },
    },
    'ru': {
      title: 'Геометрия улиц',
      description: 'Упрощайте и выравнивайте геометрию улиц',
      buttons: {
        A: 'Упростить',
        B: 'Выровнять',
        C: '∡90°',
      },
      settings: {
        title: 'Настройки',
        description: 'Параметры для упрощения геометрии сегмента',
        simplifyShort: 'Если фрагмент короче, чем',
        simplifyAngle: 'Или угол больше чем',
        simplifyTwoShort: 'и фрагменты меньше, чем',
      },
    }
  }

  const STYLE =
    'button.e85.e85-A { background-color: #0f9; margin-right: 2px }' +
    'button.e85.e85-B { background-color: #09f; color: #fff }' +
    'button.e85.e85-C { background-color: #f99; margin-left: 2px }' +
    'button.e85.e85-A:disabled, button.e85.e85-B:disabled { background-color: #ccc }' +
    '.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: 8px; }' +
    '.e85 fieldset.e85 div.controls label { white-space: normal; font-weight: normal; line-height: 32px; font-size: 13px; }' +
    '.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: 5,
    simplifyAngle: 176,
    simplifyTwoShort: 50,
  }

  let WazeActionUpdateSegmentGeometry
  let WazeActionMoveNode
  let WazeActionAddNode

  class E85 extends WMEBase {
    /**
     * 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,
        {
          'icon': 'route'
        }
      )

      // 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],
            event => this.settings.set([item], event.target.value),
            this.settings.get(item),
            (item === 'simplifyAngle') ? 150 : 0,
            (item === 'simplifyAngle') ? 180 : 200,
            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 `segment.wme` event
     * @param {jQuery.Event} event
     * @param {HTMLElement} element
     * @param {W.model} model
     * @return {void}
     */
    onSegment (event, element, model) {
      let panel = this.helper.createPanel(I18n.t(this.name).title)
      let simplifyButton = panel.addButton(
        'A',
        BUTTONS.A.title,
        BUTTONS.A.description,
        () => this.simplifySegmentGeometry(model),
        BUTTONS.A.shortcut
      )

      let straightenButton = panel.addButton(
        'B',
        BUTTONS.B.title,
        BUTTONS.B.description,
        () => this.straightenSegmentGeometry(model),
        BUTTONS.B.shortcut
      )
      if (model.geometry.components.length < 3) {
        simplifyButton.html().disabled = true
        straightenButton.html().disabled = true
      }
      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) {
      let panel = this.helper.createPanel(I18n.t(this.name).title)
      let simplifyButton = 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
      )

      let modelWithComponents = models.filter(model => model.geometry.components.length > 2)

      if (modelWithComponents.length === 0) {
        simplifyButton.html().disabled = true
      }
      if (models.length === 2) {
        panel.addButton(
          'C',
          BUTTONS.C.title,
          BUTTONS.C.description,
          () => this.orthogonalizeStreetGeometry(models[0], models[1]),
          BUTTONS.C.shortcut
        )
      }

      element.prepend(panel.html())
    }

    /**
     * Remove geometry nodes on the target segment
     * @param {Object} model
     * @return {void}
     */
    simplifySegmentGeometry (model) {
      if (model.geometry.components.length < 3) {
        return
      }

      this.group('simplify segment geometry')
      this.log('check geometry of the segment with ID ' + model.getID())
      let nodes = []

      // calculate angles for every inside point
      for (let i = 0; i < model.geometry.components.length - 2; i++) {
        let nodeStart = model.geometry.components[i],
          nodeCenter = model.geometry.components[i + 1],
          nodeEnd = model.geometry.components[i + 2]

        nodes[i] = {
          angle: Math.round(this.findAngle(nodeStart, nodeCenter, nodeEnd)),
          start: Math.round(this.findLength(nodeStart, nodeCenter)),
          end: Math.round(this.findLength(nodeCenter, nodeEnd)),
        }
        this.log('point ' + (i+1) + ' : ' + nodes[i].angle + '°, ' + nodes[i].start + 'm, ' + nodes[i].end + 'm')
      }

      let removeNodes = []

      for (let i = 0; i < nodes.length; i++) {
        let node = nodes[i]

        // mark to remove a node with short START segment
        if (node.start < this.settings.get('simplifyShort')) {
          this.log('found too short segment: ' + node.start + 'm')
          removeNodes.push(i+1)
          continue // skip next rule
        }
        // mark to remove a node with short END segment and big ANGLE
        if (node.angle >= this.settings.get('simplifyAngle')
          && node.end < this.settings.get('simplifyShort')) {
          this.log('found too short fragment: ' + node.end + 'm')
          removeNodes.push(i+1)
          i++ // skip next node
          continue // skip next rule
        }
        // mark to remove a node with big angle and short segments
        if (node.angle >= this.settings.get('simplifyAngle')
          && node.start + node.end < this.settings.get('simplifyTwoShort')) {
          this.log(
            'found point with short fragment: ' + node.start + ' + ' + node.end + ' = ' +
            (node.start + node.end) + 'm and angle equal to ' + node.angle + '°'
          )
          removeNodes.push(i+1)
          // continue // skip next rule
        }
      }

      // 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))
      }
      this.groupEnd()
    }

    /**
     * Calculates the angle (in radians) between two vectors pointing outward from one center
     *
     * @param {Object} start first point
     * @param {Object} center second point
     * @param {Object} end third point
     */
    findAngle (start, center, end) {
      let b = Math.pow(center.x - start.x, 2) + Math.pow(center.y - start.y, 2),
        a = Math.pow(center.x - end.x, 2) + Math.pow(center.y - end.y, 2),
        c = Math.pow(end.x - start.x, 2) + Math.pow(end.y - start.y, 2)
      return Math.acos((a + b - c) / Math.sqrt(4 * a * b)) * (180 / Math.PI)
    }

    /**
     * Get the length of the line by point coordinates
     * @param {Object} start
     * @param {Object} end
     * @return {Number} length in meters
     */
    findLength (start, end) {
      let line = new OpenLayers.Geometry.LineString([start, end])
      return line.getGeodesicLength('EPSG:900913')
    }

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

    /**
     * Выравнивает сегменты в прямую линию, перемещая промежуточные узлы
     * в точки пересечения перпендикуляров к вычисленной прямой, проходящей через
     * начальный и конечный узлы выделения
     * A,B,C - параметры вычисленной прямой уравнения Аx + By + C = 0
     *
     * @param {Array} models
     * @return {void}
     */
    straightenStreetGeometry (models) {
      this.group('straighten street geometry')
      this.log('calculating the formula for the straight line')

      let T1, T2,
        t,
        A = 0.0,
        B = 0.0,
        C = 0.0

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

        let geometry = segment.geometry

        if (geometry.components.length < 2) {
          continue
        }

        // определяем формулу наклонной прямой
        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)

        // looks very strange
        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)
        }

        // looks very strange
        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.group('straighten 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 = getIntersectCoordinates(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 = getIntersectCoordinates(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] + ')')
        this.groupEnd()
      }
    }

    /**
     * Orthogonalize two segments
     * This method move the node to new point
     *
     * @param {Object} segment1
     * @param {Object} segment2
     * @return {void}
     */
    orthogonalizeStreetGeometry (segment1, segment2) {
      this.log('orthogonalize street geometry')

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

      let seg1Attrs = segment1.attributes,
        seg2Attrs = segment2.attributes
      let commonNodeID

      // find ID of the common node
      let node = {}
      if (seg1Attrs.fromNodeID === seg2Attrs.fromNodeID) commonNodeID = seg1Attrs.fromNodeID
      else if (seg1Attrs.fromNodeID === seg2Attrs.toNodeID) commonNodeID = seg1Attrs.fromNodeID
      else if (seg1Attrs.toNodeID === seg2Attrs.fromNodeID) commonNodeID = seg1Attrs.toNodeID
      else 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(segment1)
      this.straightenSegmentGeometry(segment2)

      // вычислим новое положение общего узла
      // координаты концов первого сегмента
      let x1 = segment1.getFromNode().attributes.geometry.x,
        y1 = segment1.getFromNode().attributes.geometry.y,
        x2 = segment1.getToNode().attributes.geometry.x,
        y2 = segment1.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, getIntersectCoordinates(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))
    }
  }

  /**
   * Find intersection point
   * @param {Number} A
   * @param {Number} B
   * @param {Number} C
   * @param {Number} D
   * @return {Number[]}
   */
  function getIntersectCoordinates (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
  }

  /**
   * Detect direction
   * @param {Number} A
   * @param {Number} B
   * @return {Number}
   */
  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)

    // setup name for shortcut section
    WMEUIShortcut.setGroupTitle(NAME, I18n.t(NAME).title)
  })
})()