WME E85 Simplify Street Geometry

Simplify Street Geometry, looks like fork

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         WME E85 Simplify Street Geometry
// @name:uk      WME 🇺🇦 E85 Simplify Street Geometry
// @name:ru      WME 🇺🇦 E85 Simplify Street Geometry
// @version      0.4.2
// @description  Simplify Street Geometry, looks like fork
// @description:uk Спрощуємо та вирівнюємо геометрію вулиць
// @description:ru Упрощаем и выравниваем геометрию улиц
// @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://update.greasyfork.org/scripts/389765/1090053/CommonUtils.js
// @require      https://update.greasyfork.org/scripts/450160/1704233/WME-Bootstrap.js
// @require      https://update.greasyfork.org/scripts/450221/1691071/WME-Base.js
// @require      https://update.greasyfork.org/scripts/450320/1688694/WME-UI.js
// ==/UserScript==

/* jshint esversion: 8 */

/* global require */
/* global $, jQuery */
/* global I18n */
/* global WMEBase */
/* global WMEUI, WMEUIHelper, WMEUIHelperPanel, WMEUIHelperModal, WMEUIHelperTab */
/* global Container, Settings, SimpleCache, Tools  */
/* global Node$1, Segment, Venue, VenueAddress, WmeSDK */

(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',
          G: '5th Button',
        }
      },
    },
    'uk': {
      title: 'Геометрія вулиць',
      description: 'Спрощуйте та вирівнюйте вулиці',
      buttons: {
        A: 'Спростити',
        B: 'Вирівняти',
      },
      settings: {
        simplify: {
          title: 'Налаштування',
          description: 'Для спрощення сегментів будуть враховані наступні параметри',
          short: 'Видаляти фрагменти менші ніж',
          angle: 'Або якщо кут більше ніж',
          twoShort: 'та фрагменти меньші ніж',
        },
        buttons: {
          title: 'Кнопки',
          description: 'Налаштуйте кут для кнопок',
          C: 'Для першої',
          D: 'Для другої',
          E: 'Для третьої',
          F: 'Для четвертої',
          G: 'Для п\'ятої',
        }
      },
    },
    'ru': {
      title: 'Геометрия улиц',
      description: 'Упрощайте и выравнивайте геометрию улиц',
      buttons: {
        A: 'Упростить',
        B: 'Выровнять',
      },
      settings: {
        simplify: {
          title: 'Настройки',
          description: 'Параметры для упрощения геометрии сегмента',
          short: 'Если фрагмент короче, чем',
          angle: 'Или угол больше чем',
          twoShort: 'и фрагменты меньше, чем',
        },
        buttons: {
          title: 'Кнопки',
          description: 'Настройте угол для кнопок',
          C: 'Для 1-ой кнопки',
          D: 'Для 2-ой кнопки',
          E: 'Для 3-ей кнопки',
          F: 'Для 4-ой кнопки',
          G: 'Для 5-ой кнопки',
        }
      },
    }
  }

  WMEUI.addTranslation(NAME, TRANSLATION)

  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; }' +
    '#sidebar p.e85-blue { background-color:#0057B8;color:white;height:32px;text-align:center;line-height:32px;font-size:24px;margin:0; }' +
    '#sidebar p.e85-yellow { background-color:#FFDD00;color:black;height:32px;text-align:center;line-height:32px;font-size:24px;margin:0; }'

  WMEUI.addStyle(STYLE)

  const BUTTONS = {
    A: {
      title: I18n.t(NAME).buttons.A,
      description: I18n.t(NAME).buttons.A,
      shortcut: null,
    },
    B: {
      title: I18n.t(NAME).buttons.B,
      description: I18n.t(NAME).buttons.B,
      shortcut: null,
    },
  }

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

  class E85 extends WMEBase {

    constructor (name, settings, buttons) {
      super(name, settings)

      this.buttons = buttons

      this.initHelper()

      this.initTab()

      this.initShortcuts()
    }

    initHelper() {
      /** @type {WMEUIHelper} */
      this.helper = new WMEUIHelper(this.name)
    }

    /**
     * Initial UI elements
     */
    initTab () {
      /** @type {WMEUIHelperTab} */
      let tab = this.helper.createTab(
        I18n.t(this.name).title,
        {
          sidebar: this.wmeSDK.Sidebar,
          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
          )
        }
      }

      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
          )
        }
      }

      tab.addElement(fieldsetButtons)

      tab.addText(
        'info',
        '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
      )

      tab.addText('blue', 'made in')
      tab.addText('yellow', 'Ukraine')

      // Inject custom HTML to container in the WME interface
      tab.inject()
    }

    /**
     * Initial shortcuts
     */
    initShortcuts () {
      let shortcuts = [
        {
          description: I18n.t(NAME).description,
          shortcutId: this.id,
          shortcutKeys: 'A+E',
          callback: () => this.simplifySelected()
        },
        {
          description: I18n.t(NAME).description + ' [*]',
          shortcutId: this.id + '-all',
          shortcutKeys: 'A+R',
          callback: () => this.simplifyAll()
        },
      ]

      for (let shortcut of shortcuts) {
        if (this.wmeSDK.Shortcuts.areShortcutKeysInUse({ shortcutKeys: shortcut.shortcutKeys })) {
          this.log('Shortcut already in use')
          shortcut.shortcutKeys = null
        }
        this.wmeSDK.Shortcuts.createShortcut(shortcut);
      }
    }

    /**
     * Handler for `segment.wme` event
     * @param {jQuery.Event} event
     * @param {HTMLElement} element
     * @param {Segment} model
     * @return {void}
     */
    onSegment (event, element, model) {
      // Skip for blocked roads
      if (
        this.wmeSDK.DataModel.Segments.hasPermissions({ segmentId: model.id })
      ) {
        // Panel can be already exists
        let panel = this.helper.createPanel(I18n.t(this.name).title)

        let simplifyButton = panel.addButton(
          'A',
          BUTTONS.A.title,
          BUTTONS.A.description,
          () => this.simplifySegmentGeometry(model),
        )

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

        if (model.geometry.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());
        }
      } else {
        // Remove the panel
        element.querySelector('div.form-group.e85')?.remove()
      }
    }

    /**
     * Handler for `segments.wme` event
     * @param {jQuery.Event} event
     * @param {HTMLElement} element
     * @param {Array<Segment>} models
     * @return {void}
     */
    onSegments (event, element, models) {
      // Skip for locked roads
      if (models.filter((model) =>
          this.wmeSDK.DataModel.Segments.isRoadTypeDrivable({ roadType: model.roadType })
          && this.wmeSDK.DataModel.Segments.hasPermissions({ segmentId: model.id })
        ).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)
      )

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

      let modelWithComponents = models.filter(model => model.geometry.coordinates.length > 2)
      if (modelWithComponents.length === 0) {
        simplifyButton.html().disabled = true
      }

      if (models.length === 2) {
        let first = models[0]
        let second = models[1]

        // check connections of the first segment
        // trying to find second one
        let connections = []
        connections = connections.concat(this.wmeSDK.DataModel.Segments.getConnectedSegments({ segmentId: first.id }))
        connections = connections.concat(this.wmeSDK.DataModel.Segments.getConnectedSegments({ segmentId: first.id, reverseDirection: true }))
        connections = connections.map(segment => segment.id)

        if (connections.indexOf(second.id) !== -1) {
          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(first, second, angle),
              ''
            )
          }
        }
      }

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

      /*
      // @todo: need to check how it works with new SDK
      if (W.selectionManager.getSegmentSelection().multipleConnectedComponents) {
        straightenButton.html().disabled = true
      }
      */
    }

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

      if (model.geometry.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.geometry.coordinates.length - 2; i++) {
        let nodeStart = model.geometry.coordinates[i],
          nodeCenter = model.geometry.coordinates[i + 1],
          nodeEnd = model.geometry.coordinates[i + 2]

        nodes[i] = {
          angle: Math.round(GeoUtils.findAngle(nodeStart, nodeCenter, nodeEnd)),
          start: Math.round(GeoUtils.getDistance(nodeStart, nodeCenter)),
          end: Math.round(GeoUtils.getDistance(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; point: ' + (i + 1))
          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; point: ' + (i + 1))
          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) {
        this.log('points to remove: ' + removeNodes.join(', '))
        let geometry = structuredClone(model.geometry)
        geometry.coordinates = []
        // insert coordinates if it shouldn't delete
        for (let i = 0; i < model.geometry.coordinates.length; i++) {
          if (removeNodes.indexOf(i) === -1) {
            this.log(
              'Add point ' + (i + 1) + ' : '
              + model.geometry.coordinates[i][0] + ','
              + model.geometry.coordinates[i][1]
            )
            geometry.coordinates.push(model.geometry.coordinates[i])
          }
        }

        this.wmeSDK.DataModel.Segments.updateSegment({
          segmentId: model.id, geometry
        })
      } else {
        this.log('modification is not needed')
      }
      this.groupEnd()
    }

    /**
     * Remove geometry nodes on all segments on the screen
     * @return {void}
     */
    simplifyAll () {
      this.group('simplify on screen segments')

      let segments = this.getAllSegments()

      segments = segments.filter((model) =>
        this.wmeSDK.DataModel.Segments.isRoadTypeDrivable({ roadType: model.roadType })
        && this.wmeSDK.DataModel.Segments.hasPermissions({ segmentId: model.id })
      )

      this.simplifyStreetGeometry(
        segments
      )
      this.groupEnd()
    }

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

    /**
     * Remove geometry nodes on the target segments
     * @param {Array<Segment>} 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<Segment>} models
     * @return {void}
     */
    straightenStreetGeometry (models) {
      this.group('straighten street geometry')

      /**
       * Find intersection point
       * @param {Number} A
       * @param {Number} B
       * @param {Number} C
       * @param {Number} D
       * @return {Number[]}
       */
      function getIntersectCoordinates (A, B, C, D) {

        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
      }

      this.log('calculating the formula for the straight line')

      /*
      // @todo: need to check how it works with new SDK
      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.id)

        // collect the nodes
        allNodeIds.push(segment.fromNodeId)
        allNodeIds.push(segment.toNodeId)

        let nodes = this.wmeSDK.DataModel.Segments.getVirtualNodes( {segmentId: segment.id} )

        if (nodes.length === 0) {
          // simplify a segment to straight
          this.straightenSegmentGeometry(segment)
        } else {
          // don't do anything if we have virtual nodes
          virtualNodes = virtualNodes.concat(nodes)
        }
      })

      if (virtualNodes.length ) {
        this.log('⚠️ virtual nodes are present, please disconnect all trails and rails from the segments and try again')
        this.groupEnd()
        // 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 = endPointNodeIds.map(id => this.wmeSDK.DataModel.Nodes.getById( {nodeId: id} )),
        endPointNode1Geo = endPointNodes[0].geometry.coordinates,
        endPointNode2Geo = endPointNodes[1].geometry.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 = this.wmeSDK.DataModel.Nodes.getById( {nodeId: nodeId} )
        const nodeCoordinates = node.geometry.coordinates
        const d = nodeCoordinates[1] * a - nodeCoordinates[0] * b
        const newCoordinates = getIntersectCoordinates(a, b, c, d);
        const newGeometry = node.geometry
        newGeometry.coordinates = newCoordinates

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

        this.wmeSDK.DataModel.Nodes.moveNode({
          id: nodeId,
          geometry: newGeometry
        })
      });

      // 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 {Segment} first
     * @param {Segment} second
     * @param {Number} angle
     * @return {void}
     */
    alignStreetGeometry (first, second, angle = 90) {
      this.log('align street geometry ∡' + angle + '°')

      /**
       * Extract coordinates from components
       * @param {Segment} segment
       * @param {'first'|'second'|'next-to-last'|'last'} position
       * @return {*[]}
       */
      function getCoordinatesFromComponent(segment, position) {
        let pos = 0
        switch (position) {
          case 'first':
            pos = 0
            break
          case 'second':
            pos = 1
            break
          case 'next-to-last':
            pos = segment.geometry.coordinates.length - 2
            break
          case 'last':
            pos = segment.geometry.coordinates.length - 1
            break
        }
        return [
          segment.geometry.coordinates[pos][0],
          segment.geometry.coordinates[pos][1],
        ]
      }

      let A, B, C, commonNode

      if (first.toNodeId === second.fromNodeId) {
        // A → B → C
        this.log('A → B → C')
        commonNode = this.wmeSDK.DataModel.Nodes.getById( { nodeId: first.toNodeId } )

        A = getCoordinatesFromComponent(first, 'next-to-last')
        B = getCoordinatesFromComponent(first, 'last')
        C = getCoordinatesFromComponent(second, 'second')
      } else if (first.fromNodeId === second.fromNodeId) {
        // B ← A → C
        this.log('B ← A → C')
        commonNode = this.wmeSDK.DataModel.Nodes.getById( { nodeId: first.fromNodeId } )

        A = getCoordinatesFromComponent(first, 'second')
        B = getCoordinatesFromComponent(first, 'first')
        C = getCoordinatesFromComponent(second, 'second')
      } else if (first.toNodeId === second.toNodeId) {
        // A → B ← C
        this.log('A → B ← C')
        commonNode = this.wmeSDK.DataModel.Nodes.getById( { nodeId: first.toNodeId } )

        A = getCoordinatesFromComponent(first, 'next-to-last')
        B = getCoordinatesFromComponent(first, 'last')
        C = getCoordinatesFromComponent(second, 'next-to-last')
      } else if (first.fromNodeId === second.toNodeId) {
        // B ← A ← C
        this.log('B ← A ← C')
        commonNode = this.wmeSDK.DataModel.Nodes.getById( { nodeId: first.fromNodeId } )

        A = getCoordinatesFromComponent(first, 'second')
        B = getCoordinatesFromComponent(first, 'first')
        C = getCoordinatesFromComponent(second, 'next-to-last')
      }

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

      this.log('common node coords [' + B[0] + ';' + B[1] + ']')
      this.log('current angle is ' + GeoUtils.findAngle(A, B, C) + '°')

      let intersection

      // For 180
      if (parseInt(angle) === 180) {
        // Check the current angle
        let current = GeoUtils.findAngle(A, B, C)

        if (180 === Math.round(current)) {
          this.log('current angle is already ~180°, skipped')
          return
        }

        // Find the distance
        let distAB = GeoUtils.getAngularDistance(A, B)
        let distBC = GeoUtils.getAngularDistance(B, C)
        let distAC = GeoUtils.getAngularDistance(A, C)

        let distance
        if (distAC < distAB && distAC < distBC) {
          distance = distAC / 2
        } else if (distAB < distBC) {
          distance = distAB
        } else {
          distance = distBC
        }

        let bearing = GeoUtils.getBearing(A, C)

        intersection = GeoUtils.getDestination(A, bearing, distance);
      } else {
        // Find the intersection
        intersection = GeoUtils.findIntersection(A, B, C, angle);
      }

      if (!intersection) {
        this.log('intersection not found')
        return
      }

      this.log('point of the intersection is [' + intersection[0] + ', ' + intersection[1] +']')
      this.log('new angle is ' + GeoUtils.findAngle(A, intersection, C) + '°')

      commonNode.geometry.coordinates = intersection

      this.wmeSDK.DataModel.Nodes.moveNode({
        id: commonNode.id,
        geometry: commonNode.geometry
      })
    }

    /**
     * Straighten up segment, remove all geometry nodes except first and last
     * @param {Segment} segment
     */
    straightenSegmentGeometry (segment) {
      this.log('straighten segment geometry')
      if (segment.geometry.coordinates.length > 2) {
        let geometry = structuredClone(segment.geometry)
        // remove all coordinates except first and last
        geometry.coordinates.splice(
          1, geometry.coordinates.length - 2
        )
        this.wmeSDK.DataModel.Segments.updateSegment({
          segmentId: segment.id,
          geometry: geometry
        })
      }
    }
  }
  /**
   * A utility class for spherical geometry (geodesy).
   * Assumes points are [longitude, latitude] in degrees.
   */
  class GeoUtils {
    /**
     * @param {number} degrees
     * @return {number} radians
     * @private
     */
    static _toRadians(degrees) {
      return degrees * (Math.PI / 180);
    }

    /**
     * @param {number} radians
     * @return {number} degrees
     * @private
     */
    static _toDegrees(radians) {
      return radians * (180 / Math.PI);
    }

    /**
     * Normalizes an angle to the range -180 to +180 degrees.
     *
     * @param {number} degrees
     * @return {number} degrees
     */
    static _normalizeAngle(degrees) {
      return (degrees + 540) % 360 - 180;
    }

    /**
     * Calculates the initial bearing from pA to pB.
     *
     * @param {[number,number]} pA - [lon, lat] of start point.
     * @param {[number,number]} pB - [lon, lat] of end point.
     * @returns {number} Initial bearing in degrees (0-360).
     */
    static getBearing(pA, pB) {
      const latA = GeoUtils._toRadians(pA[1]);
      const lonA = GeoUtils._toRadians(pA[0]);
      const latB = GeoUtils._toRadians(pB[1]);
      const lonB = GeoUtils._toRadians(pB[0]);

      const deltaLon = lonB - lonA;

      const y = Math.sin(deltaLon) * Math.cos(latB);
      const x = Math.cos(latA) * Math.sin(latB) -
        Math.sin(latA) * Math.cos(latB) * Math.cos(deltaLon);

      const bearingRad = Math.atan2(y, x);

      // Convert from -180/+180 to 0-360
      return (GeoUtils._toDegrees(bearingRad) + 360) % 360;
    }

    /**
     * Calculates the interior angle at vertex p2.
     *
     * @param {[number,number]} p1
     * @param {[number,number]} p2
     * @param {[number,number]} p3
     */
    static findAngle(p1, p2, p3) {
      const bearing21 = GeoUtils.getBearing(p2, p1);
      const bearing23 = GeoUtils.getBearing(p2, p3);
      let angle = Math.abs(bearing21 - bearing23);

      if (angle > 180) {
        angle = 360 - angle;
      }
      return angle;
    }

    /**
     * Calculate the approximate distance between two coordinates (lat/lon)
     *
     * @param {[number,number]} pA - [lon, lat] of start point.
     * @param {[number,number]} pB - [lon, lat] of end point.
     * @return {number} The distance in meters.
     */
    static getDistance (pA, pB) {
      return GeoUtils.getAngularDistance(pA, pB) * 6371000
    }

    /**
     * Calculates the angular distance between two points using the Haversine formula.
     *
     * @param {[number,number]} pA - [lon, lat] of start point.
     * @param {[number,number]} pB - [lon, lat] of end point.
     * @returns {number} The angular distance in radians.
     */
    static getAngularDistance(pA, pB) {
      const latA = GeoUtils._toRadians(pA[1]);
      const lonA = GeoUtils._toRadians(pA[0]);
      const latB = GeoUtils._toRadians(pB[1]);
      const lonB = GeoUtils._toRadians(pB[0]);

      const deltaLat = latB - latA;
      const deltaLon = lonB - lonA;

      const a = Math.sin(deltaLat / 2) * Math.sin(deltaLat / 2) +
        Math.cos(latA) * Math.cos(latB) *
        Math.sin(deltaLon / 2) * Math.sin(deltaLon / 2);

      // c is the angular distance in radians
      return 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a));
    }

    /**
     * Calculates the destination point given a start point, bearing, and distance.
     * @param {[number,number]} startPoint - [lon, lat] of start point.
     * @param {number} bearing - Bearing in degrees (0-360).
     * @param {number} distanceRad - Angular distance in radians.
     * @returns {number[]} The destination point [lon, lat] in degrees.
     */
    static getDestination(startPoint, bearing, distanceRad) {
      const lat1 = GeoUtils._toRadians(startPoint[1]);
      const lon1 = GeoUtils._toRadians(startPoint[0]);
      const brng = GeoUtils._toRadians(bearing);
      const d = distanceRad;

      const lat2 = Math.asin(
        Math.sin(lat1) * Math.cos(d) +
        Math.cos(lat1) * Math.sin(d) * Math.cos(brng)
      );

      const lon2 = lon1 + Math.atan2(
        Math.sin(brng) * Math.sin(d) * Math.cos(lat1),
        Math.cos(d) - Math.sin(lat1) * Math.sin(lat2)
      );

      // Normalize longitude to -180 to +180
      const lon2Deg = GeoUtils._toDegrees(lon2);
      const lat2Deg = GeoUtils._toDegrees(lat2);

      return [(lon2Deg + 540) % 360 - 180, lat2Deg];
    }

    /**
     * Finds a point D on the great circle path AB such that the angle ADC equals the specified angle.
     *
     * @param {[number,number]} pA - Start of line [lon, lat]
     * @param {[number,number]} pB - End of line [lon, lat]
     * @param {[number,number]} pC - The third point [lon, lat]
     * @param {number} angle - The desired intersection angle at D in degrees (e.g., 90 for perpendicular).
     * @returns {[number,number] | null} The coordinates of D, or null if no such intersection exists.
     */
    static findIntersection(pA, pB, pC, angle) {
      const angleRad = GeoUtils._toRadians(angle);

      // 1. Calculate Angle A (difference in bearings)
      const bearingAB = GeoUtils.getBearing(pA, pB);
      const bearingAC = GeoUtils.getBearing(pA, pC);
      const angleA_rad = GeoUtils._toRadians(bearingAC - bearingAB);

      // 2. Calculate Side b (distance AC)
      const distb_rad = GeoUtils.getAngularDistance(pA, pC);

      // 3. Solve for distance AD (Side c) using the Four-Part Formula (Cotangent Law)
      // The relation for parts (Side b, Angle A, Side c, Angle D) is:
      // sin(c) * cot(b) - cos(c) * cos(A) = sin(A) * cot(D)

      // We solve this linear combination of sin(c) and cos(c) by transforming it into:
      // R * sin(c - phi) = K

      const cot_b = 1.0 / Math.tan(distb_rad);
      const cot_D = 1.0 / Math.tan(angleRad);

      // Coefficients for harmonic addition
      // sin(c)*X - cos(c)*Y = Z
      // X = cot_b, Y = cos(angleA), Z = sin(angleA) * cot_D
      const X = cot_b;
      const Y = Math.cos(angleA_rad);
      const Z = Math.sin(angleA_rad) * cot_D;

      // Calculate auxiliary angle phi and magnitude R
      // We match form: R * sin(c - phi) = Z
      // where R * cos(phi) = X and R * sin(phi) = Y
      const R = Math.hypot(X, Y);
      const phi = Math.atan2(Y, X); // atan2(y, x) -> atan2(cosA, cot_b)

      // Check if solution exists (Z/R must be between -1 and 1)
      const sin_c_minus_phi = Z / R;

      if (Math.abs(sin_c_minus_phi) > 1) {
        return null; // The requested angle is impossible to form (e.g., triangle inequality violation)
      }

      // 4. Calculate final distance c (distAD)
      // c - phi = asin(...)
      const distAD_rad = phi + Math.asin(sin_c_minus_phi);

      // 5. Calculate coordinates of D
      return GeoUtils.getDestination(pA, bearingAB, distAD_rad);
    }

    /**
     * Calculates the coordinates of point D in a right-angled spherical triangle ADC,
     * using Angle A and the hypotenuse AC.
     * Triangle ADC has a right angle at D (angle D = 90 deg),
     * and angle A and side AC are preserved from the original triangle ABC.
     *
     * @param {[number,number]} pA - [lon, lat] of point A.
     * @param {[number,number]} pB - [lon, lat] of point B (used to calculate angle A).
     * @param {[number,number]} pC - [lon, lat] of point C.
     * @returns {[number,number]} The coordinates [lon, lat] of point D.
     */
    static findRightAngleIntersection(pA, pB, pC) {
      // 1. Calculate the required angle at A (angle A_ABC)
      // The angle at A in triangle ABC is the interior angle at pA.
      const angleA_deg = GeoUtils.findAngle(pB, pA, pC);
      const angleA_rad = GeoUtils._toRadians(angleA_deg);

      // 2. Calculate the common side AC (side 'b' in spherical triangle ADC)
      // This is the hypotenuse of the right triangle ADC.
      const distAC_rad = GeoUtils.getAngularDistance(pA, pC);

      // 3. Use Napier's Rules to find side 'c' (distance AD)
      // In right triangle ADC: D = 90 deg, angle A is known, hypotenuse b (AC) is known.
      // We want to find side 'c' (distance AD), which is adjacent to angle A.
      // The correct spherical formula relating adjacent side 'c', hypotenuse 'b', and angle 'A' is:
      // cos(A) = tan(c) / tan(b)

      // Therefore, tan(c) = cos(A) * tan(b)
      // Where:
      // c = distAD_rad (unknown side)
      // b = distAC_rad (hypotenuse)
      // A = angleA_rad (known angle)

      const tan_c = Math.cos(angleA_rad) * Math.tan(distAC_rad);
      const distAD_rad = Math.atan(tan_c);

      // 4. Determine the bearing from A to D
      // The bearing from A to D is the bearing from A to C, adjusted by the angle A.
      const bearingAC_deg = GeoUtils.getBearing(pA, pC);

      // The bearing A->D must be rotated from A->C such that D forms a right angle with CD.
      // This requires D to be along the great circle arc that is perpendicular to C->D.

      // Bearing from A to B
      const bearingAB_deg = GeoUtils.getBearing(pA, pB);

      // Calculate the signed difference: bearingAC - bearingAB
      const angleCAB_raw_diff = GeoUtils._normalizeAngle(bearingAC_deg - bearingAB_deg);

      let bearingAD_deg;

      // The point D is found by rotating the bearing A->C away from B, by the interior angle A.
      if (angleCAB_raw_diff >= 0) {
        // B is counter-clockwise from AC (left side)
        // D needs to be on the other side of AC
        bearingAD_deg = GeoUtils._normalizeAngle(bearingAC_deg - angleA_deg);
      } else {
        // B is clockwise from AC (right side)
        // D needs to be on the other side of AC
        bearingAD_deg = GeoUtils._normalizeAngle(bearingAC_deg + angleA_deg);
      }

      // 5. Calculate the destination point D
      // Start point: pA
      // Bearing: bearingAD_deg
      // Distance: distAD_rad
      return GeoUtils.getDestination(pA, bearingAD_deg, distAD_rad);
    }
  }

  $(document).on('bootstrap.wme', () => {
    new E85(NAME, SETTINGS, BUTTONS)
  })
})()