WME E85 Simplify Street Geometry

Simplify Street Geometry, looks like fork

Instalar este script¿?
Script recomendado por el autor

Puede que también te guste WME E50 Fetch POI Data.

Instalar este script
// ==UserScript==
// @name         WME E85 Simplify Street Geometry
// @name:uk      WME 🇺🇦 E85 Simplify Street Geometry
// @version      0.2.9
// @description  Simplify Street Geometry, looks like fork
// @description:uk Спрощуємо та вирівнюємо геометрію вулиць
// @license      MIT License
// @author       Anton Shevchuk
// @namespace    https://greasyforks.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://update.greasyforks.org/scripts/389765/1090053/CommonUtils.js
// @require      https://update.greasyforks.org/scripts/450160/1619452/WME-Bootstrap.js
// @require      https://update.greasyforks.org/scripts/452563/1218878/WME.js
// @require      https://update.greasyforks.org/scripts/450221/1137043/WME-Base.js
// @require      https://update.greasyforks.org/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 */
/* 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',
      },
      settings: {
        simplify: {
          title: 'Settings',
          description: 'Settings for simplifying segments',
          short: 'Remove a fragment shorter than',
          angle: 'If the angle is bigger than',
          twoShort: 'and fragments shorter than',
        },
        buttons:{
          title: 'Buttons',
          description: 'Set the angle of the buttons',
          C: '1st Button',
          D: '2nd Button',
          E: '3rd Button',
          F: '4th Button',
        }
      },
    },
    'uk': {
      title: 'Геометрія вулиць',
      description: 'Спрощуйте та вирівнюйте вулиці',
      buttons: {
        A: 'Спростити',
        B: 'Вирівняти',
      },
      settings: {
        simplify: {
          title: 'Налаштування',
          description: 'Для спрощення сегментів будуть враховані наступні параметри',
          short: 'Видаляти фрагменти менші ніж',
          angle: 'Або якщо кут більше ніж',
          twoShort: 'та фрагменти меньші ніж',
        },
        buttons: {
          title: 'Кнопки',
          description: 'Налаштуйте кут для кнопок',
          C: 'Для першої',
          D: 'Для другої',
          E: 'Для третьої',
          F: 'Для четвертої',
        }
      },
    },
    'ru': {
      title: 'Геометрия улиц',
      description: 'Упрощайте и выравнивайте геометрию улиц',
      buttons: {
        A: 'Упростить',
        B: 'Выровнять',
      },
      settings: {
        simplify: {
          title: 'Настройки',
          description: 'Параметры для упрощения геометрии сегмента',
          short: 'Если фрагмент короче, чем',
          angle: 'Или угол больше чем',
          twoShort: 'и фрагменты меньше, чем',
        },
        buttons: {
          title: 'Кнопки',
          description: 'Настройте угол для кнопок',
          C: 'Для 1-ой кнопки',
          D: 'Для 2-ой кнопки',
          E: 'Для 3-ей кнопки',
          F: 'Для 4-ой кнопки',
        }
      },
    }
  }

  const STYLE =
    'button.e85.e85-A { background-color: #0f9; margin-right: 2px }' +
    'button.e85.e85-B { background-color: #09f; margin-right: 20px; color: #fff }' +
    'button.e85.e85-C { background-color: #fdd; margin: 2px 2px 0 0}' +
    'button.e85.e85-D { background-color: #fbb; margin: 2px 2px 0 0 }' +
    'button.e85.e85-E { background-color: #f99; margin: 2px 2px 0 0 }' +
    'button.e85.e85-F { background-color: #f77; margin: 2px 2px 0 0 }' +
    '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: '',
    },
  }

  // Default settings
  const SETTINGS = {
    simplify: {
      short: 5,
      angle: 176,
      twoShort: 50,
    },
    buttons: {
      C:90,
      D:60,
      E:40,
      F:30
    }
  }

  let WazeActionAddNode
  let WazeActionMoveNode
  let WazeActionMultiAction
  let WazeActionUpdateSegmentGeometry

  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,
        {
          image: GM_info.script.icon
        }
      )

      // Setup options for the script
      let fieldset = this.helper.createFieldset(I18n.t(NAME).settings.simplify.title)
      fieldset.addText('description', I18n.t(NAME).settings.simplify.description)

      let simplify = this.settings.get('simplify')
      for (let item in simplify) {
        if (simplify.hasOwnProperty(item)) {
          fieldset.addNumber(
            'settings-simplify-' + item,
            I18n.t(NAME).settings.simplify[item],
            event => this.settings.set(['simplify', item], event.target.value),
            this.settings.get('simplify', item),
            (item === 'angle') ? 150 : 0,
            (item === 'angle') ? 180 : 200,
            1
          )
        }
      }

      this.tab.addElement(fieldset)


      // Setup options for the script
      let fieldsetButtons = this.helper.createFieldset(I18n.t(NAME).settings.buttons.title)
      fieldsetButtons.addText('description', I18n.t(NAME).settings.buttons.description)

      let settingsButtons = this.settings.get('buttons')
      for (let item in settingsButtons) {
        if (settingsButtons.hasOwnProperty(item)) {
          fieldsetButtons.addNumber(
            'settings-buttons-' + item,
            I18n.t(NAME).settings.buttons[item],
            event => this.settings.set(['buttons', item], event.target.value),
            this.settings.get('buttons', item),
            10,
            180,
            (item === 'F') ? 1 : 5
          )
        }
      }

      this.tab.addElement(fieldsetButtons)

      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) {
      // Skip for blocked roads
      if (model.isLockedByHigherRank() || !model.isGeometryEditable()) {
        return
      }

      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.getGeometry().coordinates.length < 3) {
        simplifyButton.html().disabled = true
        straightenButton.html().disabled = true
      }

      const existingFormGroup = element.querySelector('div.form-group.e85');
      if (existingFormGroup) {
        existingFormGroup.replaceWith(panel.html());
      } else {
        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) {
      // Skip for locked roads
      if (models.filter((model) => model.isLockedByHigherRank() || !model.isGeometryEditable()).length > 0) {
        element.querySelector('div.form-group.e85')?.remove()
        return
      }

      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
      )

      // Don't straighten multiple components
      let straightenButton = panel.addButton(
        'B',
        BUTTONS.B.title,
        BUTTONS.B.description,
        () => this.straightenStreetGeometry(models),
        BUTTONS.B.shortcut
      )

      let modelWithComponents = models.filter(model => model.getGeometry().coordinates.length > 2)

      if (modelWithComponents.length === 0) {
        simplifyButton.html().disabled = true
      }

      if (W.selectionManager.getSegmentSelection().multipleConnectedComponents) {
        straightenButton.html().disabled = true
      }

      if (!W.selectionManager.getSegmentSelection().multipleConnectedComponents
        && models.length === 2) {
        panel.addDiv('align-by-angle')

        for (let key of ['C','D','E','F']) {
          let angle = this.settings.get('buttons', key)
          panel.addButton(
            key,
            `∡${angle}°`,
            `∡${angle}°`,
            () => this.alignStreetGeometry(models[0], models[1], angle),
            ''
          )
        }
      }

      const existingFormGroup = element.querySelector('div.form-group.e85');
      if (existingFormGroup) {
        existingFormGroup.replaceWith(panel.html());
      } else {
        element.prepend(panel.html());
      }
    }

    /**
     * Remove geometry nodes on the target segment
     * @param {Object} model
     * @return {void}
     */
    simplifySegmentGeometry (model) {
      this.log('check geometry of the segment with ID ' + model.getID())

      if (model.getGeometry().coordinates.length < 3) {
        this.log('geometry is simple, skipped')
        return
      }

      this.group('simplify segment geometry')
      let nodes = []

      // calculate angles for every inside point
      for (let i = 0; i < model.getGeometry().coordinates.length - 2; i++) {
        let nodeStart = model.getGeometry().coordinates[i],
          nodeCenter = model.getGeometry().coordinates[i + 1],
          nodeEnd = model.getGeometry().coordinates[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 a short START segment
        if (node.start < this.settings.get('simplify', 'short')) {
          this.log('found too short segment: ' + node.start + 'm')
          removeNodes.push(i+1)
          continue // skip the next rule
        }
        // mark to remove a node with a short END segment and big ANGLE
        if (node.angle >= this.settings.get('simplify', 'angle')
          && node.end < this.settings.get('simplify', 'short')) {
          this.log('found too short fragment: ' + node.end + 'm')
          removeNodes.push(i+1)
          i++ // skip next node
          continue // skip the next rule
        }
        // mark to remove a node with a big angle and short segments
        if (node.angle >= this.settings.get('simplify', 'angle')
          && node.start + node.end < this.settings.get('simplify', 'twoShort')) {
          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.getGeometry() }
        let coordinates = []
        for (let i = 0; i < newGeometry.coordinates.length; i++) {
          if (removeNodes.indexOf(i) === -1) {
            coordinates.push(newGeometry.coordinates[i])
          }
        }
        newGeometry.coordinates = coordinates
        W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(model, model.getGeometry(), 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[0] - start[0], 2) + Math.pow(center[1] - start[1], 2),
        a = Math.pow(center[0] - end[0], 2) + Math.pow(center[1] - end[1], 2),
        c = Math.pow(end[0] - start[0], 2) + Math.pow(end[1] - start[1], 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 {Array<number,number>} start point
     * @param {Array<number,number>} end point
     * @return {Number} length in meters
     */
    findLength (start, end) {
      return distance(start[0], start[1], end[0], end[1])
    }

    /**
     * Remove geometry nodes on all segments on the screen
     * @return {void}
     */
    simplifyOnScreen () {
      this.group('simplify on screen segments')
      this.simplifyStreetGeometry(
        WME.getSegments()
      )
      this.groupEnd()
    }

    /**
     * Remove geometry nodes on the selected segments
     * @return {void}
     */
    simplifySelected () {
      this.group('simplify selected segments')
      this.simplifyStreetGeometry(
        WME.getSelectedSegments()
      )
      this.groupEnd()
    }

    /**
     * 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()
    }

    /**
     * Aligns the segments into a straight line by moving the intermediate
     * nodes to the intersection points of the perpendiculars with
     * the calculated line passing through the start and end nodes of the selection.
     *
     * A, B, and C are the parameters of the calculated line equation:
     *   Ax + 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 segmentSelection = W.selectionManager.getSegmentSelection()

      if (segmentSelection.multipleConnectedComponents) {
        this.log('don\'t try to straighten multiple segments without connection')
      }

      let
        allNodeIds = [], // all nodes for selected segments
        dupNodeIds = [], // only nodes inside connections
        virtualNodes = [] // virtual nodes of segments

      models.forEach(segment => {
        this.log('straighten segment #' + segment.getID())

        // simplify a segment to straight
        this.straightenSegmentGeometry(segment)

        // collect the nodes
        allNodeIds.push(segment.getFromNode().getID())
        allNodeIds.push(segment.getToNode().getID())
        virtualNodes = virtualNodes.concat(segment.getVirtualNodes())
      })

      if (virtualNodes.length ) {
        this.log('⚠️ virtual nodes are present, please disconnect all trails and rails from the segments and try again')

        // doesn't work, but why? what is wrong with this code?
        // virtualNodes.forEach(node => {
        //   let element = document.getElementById(node.getOLGeometry.id)
        //   element.setAttribute("fill","#dd7700")
        //
        //   element.addEventListener("click", () => {
        //     element.setAttribute("fill","#00ece3")
        //   });
        // })

        return
      }

      allNodeIds.forEach((nodeId, idx) => {
        if (allNodeIds.indexOf(nodeId, idx + 1) > -1) {
          if (!dupNodeIds.includes(nodeId))
            dupNodeIds.push(nodeId);
        }
      });

      let distinctNodeIds = [...new Set(allNodeIds)];
      let endPointNodeIds = distinctNodeIds.filter((nodeId) => !dupNodeIds.includes(nodeId));
      let endPointNodes = W.model.nodes.getByIds(endPointNodeIds),
        endPointNode1Geo = endPointNodes[0].getGeometry().coordinates,
        endPointNode2Geo = endPointNodes[1].getGeometry().coordinates

      const a = endPointNode2Geo[1] - endPointNode1Geo[1],
        b = endPointNode1Geo[0] - endPointNode2Geo[0],
        c = endPointNode2Geo[0] * endPointNode1Geo[1] - endPointNode1Geo[0] * endPointNode2Geo[1];

      dupNodeIds.forEach((nodeId) => {
        const node = W.model.nodes.getObjectById(nodeId),
          nodeCoordinates = node.getGeometry().coordinates;
        const d = nodeCoordinates[1] * a - nodeCoordinates[0] * b,
          newCoordinates = getIntersectCoordinates(a, b, c, d);

        this.log('move node #' + nodeId + ' to [' + newCoordinates[0] + ';' + newCoordinates[1] + ']')
        this.moveNode(node, newCoordinates)
      });

      // I don't understand why doesn't it work, in the WME all looks good, but it fails when try to save changes
      // virtualNodes.forEach((node) => {
      //   const nodeCoordinates = node.getGeometry().coordinates;
      //   const d = nodeCoordinates[1] * a - nodeCoordinates[0] * b,
      //     newCoordinates = getIntersectCoordinates(a, b, c, d);
      //
      //   this.log('move node #' + node.getID() + ' to [' + newCoordinates[0] + ';' + newCoordinates[1] + ']')
      //   this.moveNode(node, newCoordinates)
      // });

      this.groupEnd()
    }


    /**
     * Align two segments by angle
     * This method moves the node to new point
     *
     * @param {Object} segment1
     * @param {Object} segment2
     * @param {Number} angle
     * @return {void}
     */
    alignStreetGeometry (segment1, segment2, angle = 90) {
      this.log('align street geometry ∡' + angle + '°')

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

      /**
       * Extract coordinates from components
       * @param {Object} segment
       * @param {'first'|'second'|'last-but-one'|'last'} position
       * @return {*[]}
       */
      function getCoordinatesFromComponent(segment, position) {
        let pos = 0
        switch (position) {
          case 'first':
            pos = 0
            break
          case 'second':
            pos = 1
            break
          case 'last-but-one':
            pos = segment.getOLGeometry().components.length - 2
            break
          case 'last':
            pos = segment.getOLGeometry().components.length - 1
            break
        }
        return [
          segment.getOLGeometry().components[pos].x,
          segment.getOLGeometry().components[pos].y,
        ]
      }

      let A, B, C, commonNode

      if (segment1.getToNode().getID() === segment2.getFromNode().getID()) {
        // A → B → C
        commonNode = segment1.getToNode()
        A = getCoordinatesFromComponent(segment1, 'last-but-one')
        B = getCoordinatesFromComponent(segment1, 'last')
        C = getCoordinatesFromComponent(segment2, 'second')
      } else if (segment1.getFromNode().getID() === segment2.getFromNode().getID()) {
        // B ← A → C
        commonNode = segment1.getFromNode()
        A = getCoordinatesFromComponent(segment1, 'second')
        B = getCoordinatesFromComponent(segment1, 'first')
        C = getCoordinatesFromComponent(segment2, 'second')
      } else if (segment1.getToNode().getID() === segment2.getToNode().getID()) {
        // A → B ← C
        commonNode = segment1.getToNode()
        A = getCoordinatesFromComponent(segment1, 'last-but-one')
        B = getCoordinatesFromComponent(segment1, 'last')
        C = getCoordinatesFromComponent(segment2, 'last-but-one')
      } else if (segment1.getFromNode().getID() === segment2.getToNode().getID()) {
        // B ← A ← C
        commonNode = segment1.getFromNode()
        A = getCoordinatesFromComponent(segment1, 'second')
        B = getCoordinatesFromComponent(segment1, 'first')
        C = getCoordinatesFromComponent(segment2, 'last-but-one')
      }

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

      this.log('common node coords [' + B[0] + ';' + B[1] + ']')

      // Coordinates of points A, B and C
      // First selected segment uses it as line for calculation
      let intersection = findIntersectionWithAngle(A, B, C, angle)

      // Uses OpenLayers with convertation, because
      intersection = W.userscripts.toGeoJSONGeometry(new OpenLayers.Geometry.Point( ...intersection ))

      this.log('point of the intersection is [' + intersection[0] + ', ' + intersection[1] +']')

      this.moveNode(commonNode, intersection.coordinates)
    }

    /**
     * Straighten up segment, remove all geometry nodes except first and last
     * @param {Object} segment
     */
    straightenSegmentGeometry (segment) {
      this.group('straighten segment geometry')
      if (segment.getGeometry().coordinates.length > 2) {
        let multiAction = new WazeActionMultiAction()
        let newGeometry = structuredClone(segment.attributes.geoJSONGeometry)
        // just left the first and last elements
        newGeometry.coordinates.splice(1, newGeometry.coordinates.length - 2)
        // W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(segment, segment.getGeometry(), newGeometry))
        let updateSegmentGeometry = new WazeActionUpdateSegmentGeometry(segment, segment.attributes.geoJSONGeometry, newGeometry)
        updateSegmentGeometry.generateDescription();
        multiAction.doSubAction(W.model, updateSegmentGeometry);
        W.model.actionManager.add(multiAction);
      }
    }

    /**
     * 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 nodeGeometry = node.getGeometry()
      nodeGeometry.coordinates = coords

      let connectedSegObjs = {}
      let emptyObj = {}

      node.getSegmentIds().forEach((id) => {
        connectedSegObjs[id] = { ...W.model.segments.getObjectById(id).getGeometry() }
      })

      W.model.actionManager.add(
        new WazeActionMoveNode(
          node,
          node.getGeometry(),
          nodeGeometry,
          connectedSegObjs,
          emptyObj
        )
      )
    }
  }

  /**
   * Finds the intersection of a line (passing through A and B) and a second line
   * that passes through point C and intersects the first line at a given angle.
   *
   * Note: For any angle, there are two possible intersection points. This function
   * calculates one of them.
   *
   * @param {Array<number,number>} A A point on the primary line.
   * @param {Array<number,number>} B A second point on the primary line.
   * @param {Array<number,number>} C The point through which the second line passes.
   * @param {number} angleDegrees The desired angle between the two lines in degrees.
   * @returns {(Array<number,number>|null)} The coordinates of the intersection point, or null if the lines are parallel.
   */
  function findIntersectionWithAngle(A, B, C, angleDegrees) {
    const [Ax, Ay] = A;
    const [Bx, By] = B;
    const [Cx, Cy] = C;

    // Convert the desired angle from degrees to radians for trigonometric functions
    const angleRadians = angleDegrees * (Math.PI / 180);

    // If the angle is 0 or 180, the lines are parallel. No unique intersection unless C is on line AB.
    // We check for floating point inaccuracies by seeing if sin(angle) is very close to 0.
    if (Math.abs(Math.sin(angleRadians)) < 1e-9) {
      console.error("Angle is 0 or 180 degrees; lines are parallel. No unique intersection.");
      return null;
    }

    let m1; // Slope of line AB
    let c1; // Y-intercept of line AB

    // --- Handle Vertical Line Case for AB ---
    if (Math.abs(Ax - Bx) < 1e-9) {
      m1 = Infinity;
      c1 = Infinity; // No y-intercept
    } else {
      m1 = (By - Ay) / (Bx - Ax);
      c1 = Ay - m1 * Ax;
    }

    // Calculate the slope of the second line (m2) using the tangent formula
    const tanAngle = Math.tan(angleRadians);
    let m2;

    if (m1 === Infinity) {
      // If line AB is vertical, its angle is 90 degrees. The new line's slope is tan(90 ± angle).
      // tan(90 - angle) = cot(angle) = 1 / tan(angle)
      m2 = 1 / tanAngle;
    } else {
      // Using the formula: m2 = (m1 - tan(angle)) / (1 + m1 * tan(angle))
      // This corresponds to one of the two possible lines.
      // The other uses + in the numerator and - in the denominator.
      const denominator = 1 + m1 * tanAngle;
      if (Math.abs(denominator) < 1e-9) {
        // This occurs when 1 + m1*tan(angle) = 0 => m1 = -1/tan(angle), i.e., the lines are perpendicular
        // The slope m2 would be infinite (a vertical line).
        m2 = Infinity;
      } else {
        m2 = (m1 - tanAngle) / denominator;
      }
    }

    // Now we have the slope m2 of the line passing through C.
    const c2 = Cy - m2 * Cx;

    // --- Find the final intersection point ---
    let x, y;

    // Check for near-parallel lines which can cause massive floating point errors
    if (Math.abs(m1 - m2) < 1e-9) {
      console.error("Lines are parallel, no unique intersection.");
      return null;
    }

    if (m1 === Infinity) { // Line AB is vertical
      x = Ax;
      y = m2 * x + c2;
    } else if (m2 === Infinity) { // The new line is vertical
      x = Cx;
      y = m1 * x + c1;
    } else { // General case
      x = (c2 - c1) / (m1 - m2);
      y = m1 * x + c1;
    }

    return [x, y];
  }

  /**
   * 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 the direction
   * @param {Number} A
   * @param {Number} B
   * @return {Number}
   */
  function getDeltaDirect (A, B) {
    if (A < B) {
      return 1.0
    } else if (A > B) {
      return -1.0
    }

    return 0.0
  }

  /**
   * Calculate the approximate distance between two coordinates (lat/lon)
   *
   * © Chris Veness, MIT-licensed,
   * http://www.movable-type.co.uk/scripts/latlong.html#equirectangular
   *
   * @param {number} λ1 first point latitude
   * @param {number} φ1 first point longitude
   * @param {number} λ2 second point latitude
   * @param {number} φ2 second point longitude
   */
  function distance (λ1, φ1, λ2, φ2) {
    let R = 6371000;
    let Δλ = (λ2 - λ1) * Math.PI / 180;
    φ1 = φ1 * Math.PI / 180;
    φ2 = φ2 * Math.PI / 180;
    let x = Δλ * Math.cos((φ1+φ2)/2);
    let y = (φ2-φ1);
    let d = Math.sqrt(x*x + y*y);
    return R * d;
  };

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

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

    let Instance = new E85(NAME, SETTINGS)

    Instance.init(BUTTONS)

    // Bind shortcut
    WMEUI.addShortcut(
      NAME,
      I18n.t(NAME).description,
      NAME,
      I18n.t(NAME).title,
      'A+E',
      () => Instance.simplifySelected()
    )

    // Bind shortcut
    WMEUI.addShortcut(
      NAME + '-all',
      I18n.t(NAME).description + ' [*]',
      NAME,
      I18n.t(NAME).title + ' [*]',
      'A+R',
      () => Instance.simplifyOnScreen()
    )

    // setup name for a shortcut section
    WMEUIShortcut.setGroupTitle(NAME, I18n.t(NAME).title)
  })
})()
长期地址
遇到问题?请前往 GitHub 提 Issues。