WME E85 Simplify Street Geometry

Simplify Street Geometry, looks like fork

安装此脚本?
作者推荐脚本

你可能也喜欢 WME E50 Fetch POI Data

安装此脚本
  1. // ==UserScript==
  2. // @name WME E85 Simplify Street Geometry
  3. // @name:uk WME 🇺🇦 E85 Simplify Street Geometry
  4. // @version 0.2.4
  5. // @description Simplify Street Geometry, looks like fork
  6. // @description:uk Спрощуємо та вирівнюємо геометрію вулиць
  7. // @license MIT License
  8. // @author Anton Shevchuk
  9. // @namespace https://greasyfork.org/users/227648-anton-shevchuk
  10. // @supportURL https://github.com/AntonShevchuk/wme-e85/issues
  11. // @match https://*.waze.com/editor*
  12. // @match https://*.waze.com/*/editor*
  13. // @exclude https://*.waze.com/user/editor*
  14. // @icon 
  15. // @grant none
  16. // @require https://update.greasyfork.org/scripts/389765/1090053/CommonUtils.js
  17. // @require https://update.greasyfork.org/scripts/450160/1218867/WME-Bootstrap.js
  18. // @require https://update.greasyfork.org/scripts/452563/1218878/WME.js
  19. // @require https://update.greasyfork.org/scripts/450221/1137043/WME-Base.js
  20. // @require https://update.greasyfork.org/scripts/450320/1555446/WME-UI.js
  21. // ==/UserScript==
  22.  
  23. /* jshint esversion: 8 */
  24.  
  25. /* global require */
  26. /* global $, jQuery */
  27. /* global W */
  28. /* global I18n */
  29. /* global OpenLayers */
  30. /* global WME, WMEBase */
  31. /* global WMEUI, WMEUIHelper, WMEUIHelperPanel, WMEUIHelperModal, WMEUIHelperTab, WMEUIShortcut */
  32. /* global Container, Settings, SimpleCache, Tools */
  33.  
  34. (function () {
  35. 'use strict'
  36.  
  37. // Script name, uses as unique index
  38. const NAME = 'E85'
  39.  
  40. // Translations
  41. const TRANSLATION = {
  42. 'en': {
  43. title: 'Street Geometry',
  44. description: 'Simplify and straighten up streets',
  45. buttons: {
  46. A: 'Simplify',
  47. B: 'Straighten',
  48. C: '∡90°',
  49. },
  50. settings: {
  51. title: 'Settings',
  52. description: 'Settings for simplifying segments',
  53. simplifyShort: 'Remove a fragment shorter than',
  54. simplifyAngle: 'If the angle is bigger than',
  55. simplifyTwoShort: 'and fragments shorter than',
  56. },
  57. },
  58. 'uk': {
  59. title: 'Геометрія вулиць',
  60. description: 'Спрощуйте та вирівнюйте вулиці',
  61. buttons: {
  62. A: 'Спростити',
  63. B: 'Вирівняти',
  64. C: '∡90°',
  65. },
  66. settings: {
  67. title: 'Налаштування',
  68. description: 'Для спрощення сегментів будуть враховані наступні параметри',
  69. simplifyShort: 'Видаляти фрагменти менші ніж',
  70. simplifyAngle: 'Або якщо кут більше ніж',
  71. simplifyTwoShort: 'та фрагменти меньші ніж',
  72. },
  73. },
  74. 'ru': {
  75. title: 'Геометрия улиц',
  76. description: 'Упрощайте и выравнивайте геометрию улиц',
  77. buttons: {
  78. A: 'Упростить',
  79. B: 'Выровнять',
  80. C: '∡90°',
  81. },
  82. settings: {
  83. title: 'Настройки',
  84. description: 'Параметры для упрощения геометрии сегмента',
  85. simplifyShort: 'Если фрагмент короче, чем',
  86. simplifyAngle: 'Или угол больше чем',
  87. simplifyTwoShort: 'и фрагменты меньше, чем',
  88. },
  89. }
  90. }
  91.  
  92. const STYLE =
  93. 'button.e85.e85-A { background-color: #0f9; margin-right: 2px }' +
  94. 'button.e85.e85-B { background-color: #09f; color: #fff }' +
  95. 'button.e85.e85-C { background-color: #f99; margin-left: 2px }' +
  96. 'button.e85.e85-A:disabled, button.e85.e85-B:disabled { background-color: #ccc }' +
  97. '.e85 legend { cursor:pointer; font-size: 12px; font-weight: bold; width: auto; text-align: right; border: 0; margin: 0; padding: 0 8px; }' +
  98. '.e85 fieldset { border: 1px solid #ddd; padding: 8px; }' +
  99. '.e85 fieldset.e85 div.controls label { white-space: normal; font-weight: normal; line-height: 32px; font-size: 13px; }' +
  100. '.e85 fieldset.e85 div.controls input[type="number"] { float:right; wight: 32px }' +
  101. 'p.e85-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }'
  102.  
  103. WMEUI.addTranslation(NAME, TRANSLATION)
  104. WMEUI.addStyle(STYLE)
  105.  
  106. const BUTTONS = {
  107. A: {
  108. title: I18n.t(NAME).buttons.A,
  109. description: I18n.t(NAME).buttons.A,
  110. shortcut: '',
  111. },
  112. B: {
  113. title: I18n.t(NAME).buttons.B,
  114. description: I18n.t(NAME).buttons.B,
  115. shortcut: '',
  116. },
  117. C: {
  118. title: I18n.t(NAME).buttons.C,
  119. description: I18n.t(NAME).buttons.C,
  120. shortcut: '',
  121. },
  122. }
  123.  
  124. // Default settings
  125. const SETTINGS = {
  126. simplifyShort: 5,
  127. simplifyAngle: 176,
  128. simplifyTwoShort: 50,
  129. }
  130.  
  131. let WazeActionAddNode
  132. let WazeActionMoveNode
  133. let WazeActionMultiAction
  134. let WazeActionUpdateSegmentGeometry
  135.  
  136. class E85 extends WMEBase {
  137. /**
  138. * Initial UI elements
  139. * @param {Object} buttons
  140. */
  141. init (buttons) {
  142. /** @type {WMEUIHelper} */
  143. this.helper = new WMEUIHelper(this.name)
  144.  
  145. /** @type {WMEUIHelperTab} */
  146. this.tab = this.helper.createTab(
  147. I18n.t(this.name).title,
  148. {
  149. image: GM_info.script.icon
  150. }
  151. )
  152.  
  153. // Setup options for script
  154. let fieldset = this.helper.createFieldset(I18n.t(NAME).settings.title)
  155. fieldset.addText('description', I18n.t(NAME).settings.description)
  156. let settings = this.settings.get()
  157. for (let item in settings) {
  158. if (settings.hasOwnProperty(item)) {
  159. fieldset.addNumber(
  160. 'settings-' + item,
  161. I18n.t(NAME).settings[item],
  162. event => this.settings.set([item], event.target.value),
  163. this.settings.get(item),
  164. (item === 'simplifyAngle') ? 150 : 0,
  165. (item === 'simplifyAngle') ? 180 : 200,
  166. 1
  167. )
  168. }
  169. }
  170. this.tab.addElement(fieldset)
  171. this.tab.addText(
  172. 'info',
  173. '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
  174. )
  175.  
  176. // Inject custom HTML to container in the WME interface
  177. this.tab.inject()
  178. }
  179.  
  180. /**
  181. * Handler for `segment.wme` event
  182. * @param {jQuery.Event} event
  183. * @param {HTMLElement} element
  184. * @param {W.model} model
  185. * @return {void}
  186. */
  187. onSegment (event, element, model) {
  188. // Skip for blocked roads
  189. if (model.isLockedByHigherRank() || !model.isGeometryEditable()) {
  190. return
  191. }
  192.  
  193. let panel = this.helper.createPanel(I18n.t(this.name).title)
  194. let simplifyButton = panel.addButton(
  195. 'A',
  196. BUTTONS.A.title,
  197. BUTTONS.A.description,
  198. () => this.simplifySegmentGeometry(model),
  199. BUTTONS.A.shortcut
  200. )
  201.  
  202. let straightenButton = panel.addButton(
  203. 'B',
  204. BUTTONS.B.title,
  205. BUTTONS.B.description,
  206. () => this.straightenSegmentGeometry(model),
  207. BUTTONS.B.shortcut
  208. )
  209. if (model.getGeometry().coordinates.length < 3) {
  210. simplifyButton.html().disabled = true
  211. straightenButton.html().disabled = true
  212. }
  213.  
  214. const existingFormGroup = element.querySelector('div.form-group.e85');
  215. if (existingFormGroup) {
  216. existingFormGroup.replaceWith(panel.html());
  217. } else {
  218. element.prepend(panel.html());
  219. }
  220. }
  221.  
  222. /**
  223. * Handler for `segments.wme` event
  224. * @param {jQuery.Event} event
  225. * @param {HTMLElement} element
  226. * @param {Array} models
  227. * @return {void}
  228. */
  229. onSegments (event, element, models) {
  230. // Skip for locked roads
  231. if (models.filter((model) => model.isLockedByHigherRank() || !model.isGeometryEditable()).length > 0) {
  232. element.querySelector('div.form-group.e85')?.remove()
  233. return
  234. }
  235.  
  236. let panel = this.helper.createPanel(I18n.t(this.name).title)
  237. let simplifyButton = panel.addButton(
  238. 'A',
  239. BUTTONS.A.title,
  240. BUTTONS.A.description,
  241. () => this.simplifyStreetGeometry(models),
  242. BUTTONS.A.shortcut
  243. )
  244.  
  245. // Don't straighten multiple components
  246. let straightenButton = panel.addButton(
  247. 'B',
  248. BUTTONS.B.title,
  249. BUTTONS.B.description,
  250. () => this.straightenStreetGeometry(models),
  251. BUTTONS.B.shortcut
  252. )
  253.  
  254. let modelWithComponents = models.filter(model => model.getGeometry().coordinates.length > 2)
  255.  
  256. if (modelWithComponents.length === 0) {
  257. simplifyButton.html().disabled = true
  258. }
  259.  
  260. if (W.selectionManager.getSegmentSelection().multipleConnectedComponents) {
  261. straightenButton.html().disabled = true
  262. }
  263.  
  264. if (!W.selectionManager.getSegmentSelection().multipleConnectedComponents
  265. && models.length === 2) {
  266. panel.addButton(
  267. 'C',
  268. BUTTONS.C.title,
  269. BUTTONS.C.description,
  270. () => this.orthogonalizeStreetGeometry(models[0], models[1]),
  271. BUTTONS.C.shortcut
  272. )
  273. }
  274.  
  275. const existingFormGroup = element.querySelector('div.form-group.e85');
  276. if (existingFormGroup) {
  277. existingFormGroup.replaceWith(panel.html());
  278. } else {
  279. element.prepend(panel.html());
  280. }
  281. }
  282.  
  283. /**
  284. * Remove geometry nodes on the target segment
  285. * @param {Object} model
  286. * @return {void}
  287. */
  288. simplifySegmentGeometry (model) {
  289. if (model.getGeometry().coordinates.length < 3) {
  290. return
  291. }
  292.  
  293. this.group('simplify segment geometry')
  294. this.log('check geometry of the segment with ID ' + model.getID())
  295. let nodes = []
  296.  
  297. // calculate angles for every inside point
  298. for (let i = 0; i < model.getGeometry().coordinates.length - 2; i++) {
  299. let nodeStart = model.getGeometry().coordinates[i],
  300. nodeCenter = model.getGeometry().coordinates[i + 1],
  301. nodeEnd = model.getGeometry().coordinates[i + 2]
  302.  
  303. nodes[i] = {
  304. angle: Math.round(this.findAngle(nodeStart, nodeCenter, nodeEnd)),
  305. start: Math.round(this.findLength(nodeStart, nodeCenter)),
  306. end: Math.round(this.findLength(nodeCenter, nodeEnd)),
  307. }
  308. this.log('point ' + (i+1) + ' : ' + nodes[i].angle + '°, ' + nodes[i].start + 'm, ' + nodes[i].end + 'm')
  309. }
  310.  
  311. let removeNodes = []
  312.  
  313. for (let i = 0; i < nodes.length; i++) {
  314. let node = nodes[i]
  315.  
  316. // mark to remove a node with short START segment
  317. if (node.start < this.settings.get('simplifyShort')) {
  318. this.log('found too short segment: ' + node.start + 'm')
  319. removeNodes.push(i+1)
  320. continue // skip next rule
  321. }
  322. // mark to remove a node with short END segment and big ANGLE
  323. if (node.angle >= this.settings.get('simplifyAngle')
  324. && node.end < this.settings.get('simplifyShort')) {
  325. this.log('found too short fragment: ' + node.end + 'm')
  326. removeNodes.push(i+1)
  327. i++ // skip next node
  328. continue // skip next rule
  329. }
  330. // mark to remove a node with big angle and short segments
  331. if (node.angle >= this.settings.get('simplifyAngle')
  332. && node.start + node.end < this.settings.get('simplifyTwoShort')) {
  333. this.log(
  334. 'found point with short fragment: ' + node.start + ' + ' + node.end + ' = ' +
  335. (node.start + node.end) + 'm and angle equal to ' + node.angle + '°'
  336. )
  337. removeNodes.push(i+1)
  338. // continue // skip next rule
  339. }
  340. }
  341.  
  342. // remove nodes from geometry
  343. if (removeNodes.length) {
  344. let newGeometry = { ... model.getGeometry() }
  345. let coordinates = []
  346. for (let i = 0; i < newGeometry.coordinates.length; i++) {
  347. if (removeNodes.indexOf(i) === -1) {
  348. coordinates.push(newGeometry.coordinates[i])
  349. }
  350. }
  351. newGeometry.coordinates = coordinates
  352. W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(model, model.getGeometry(), newGeometry))
  353. }
  354. this.groupEnd()
  355. }
  356.  
  357. /**
  358. * Calculates the angle (in radians) between two vectors pointing outward from one center
  359. *
  360. * @param {Object} start first point
  361. * @param {Object} center second point
  362. * @param {Object} end third point
  363. */
  364. findAngle (start, center, end) {
  365. let b = Math.pow(center[0] - start[0], 2) + Math.pow(center[1] - start[1], 2),
  366. a = Math.pow(center[0] - end[0], 2) + Math.pow(center[1] - end[1], 2),
  367. c = Math.pow(end[0] - start[0], 2) + Math.pow(end[1] - start[1], 2)
  368. return Math.acos((a + b - c) / Math.sqrt(4 * a * b)) * (180 / Math.PI)
  369. }
  370.  
  371. /**
  372. * Get the length of the line by point coordinates
  373. * @param {Array<number,number>} start point
  374. * @param {Array<number,number>} end point
  375. * @return {Number} length in meters
  376. */
  377. findLength (start, end) {
  378. return distance(start[0], start[1], end[0], end[1])
  379. }
  380.  
  381. /**
  382. * Remove geometry nodes on the target segments
  383. * @param {Array} models
  384. * @return {void}
  385. */
  386. simplifyStreetGeometry (models) {
  387. this.group('simplify street geometry')
  388. for (let i = 0; i < models.length; i++) {
  389. this.simplifySegmentGeometry(models[i])
  390. }
  391. this.groupEnd()
  392. }
  393.  
  394. /**
  395. * Aligns the segments into a straight line by moving the intermediate
  396. * nodes to the intersection points of the perpendiculars with
  397. * the calculated line passing through the start and end nodes of the selection.
  398. *
  399. * A, B, and C are the parameters of the calculated line equation:
  400. * Ax + By + C = 0
  401. *
  402. * @param {Array} models
  403. * @return {void}
  404. */
  405. straightenStreetGeometry (models) {
  406. this.group('straighten street geometry')
  407. this.log('calculating the formula for the straight line')
  408.  
  409. let segmentSelection = W.selectionManager.getSegmentSelection()
  410.  
  411. if (segmentSelection.multipleConnectedComponents) {
  412. this.log('don\'t try to straighten multiple segments without connection')
  413. }
  414.  
  415. let
  416. allNodeIds = [], // all nodes for selected segments
  417. dupNodeIds = [], // only nodes inside connections
  418. virtualNodes = [] // virtual nodes of segments
  419.  
  420. models.forEach(segment => {
  421. this.log('straighten segment #' + segment.getID())
  422.  
  423. // simplify segment to straight
  424. this.straightenSegmentGeometry(segment)
  425.  
  426. // collect the nodes
  427. allNodeIds.push(segment.getFromNode().getID())
  428. allNodeIds.push(segment.getToNode().getID())
  429. virtualNodes = virtualNodes.concat(segment.getVirtualNodes())
  430. })
  431.  
  432. if (virtualNodes.length ) {
  433. this.log('⚠️ virtual nodes are present, please disconnect all trails and rails from the segments and try again')
  434.  
  435. // doesn't work, but why? what is wrong with this code?
  436. // virtualNodes.forEach(node => {
  437. // let element = document.getElementById(node.getOLGeometry.id)
  438. // element.setAttribute("fill","#dd7700")
  439. //
  440. // element.addEventListener("click", () => {
  441. // element.setAttribute("fill","#00ece3")
  442. // });
  443. // })
  444.  
  445. return
  446. }
  447.  
  448. allNodeIds.forEach((nodeId, idx) => {
  449. if (allNodeIds.indexOf(nodeId, idx + 1) > -1) {
  450. if (!dupNodeIds.includes(nodeId))
  451. dupNodeIds.push(nodeId);
  452. }
  453. });
  454.  
  455. let distinctNodeIds = [...new Set(allNodeIds)];
  456. let endPointNodeIds = distinctNodeIds.filter((nodeId) => !dupNodeIds.includes(nodeId));
  457. let endPointNodes = W.model.nodes.getByIds(endPointNodeIds),
  458. endPointNode1Geo = endPointNodes[0].getGeometry().coordinates,
  459. endPointNode2Geo = endPointNodes[1].getGeometry().coordinates
  460.  
  461. const a = endPointNode2Geo[1] - endPointNode1Geo[1],
  462. b = endPointNode1Geo[0] - endPointNode2Geo[0],
  463. c = endPointNode2Geo[0] * endPointNode1Geo[1] - endPointNode1Geo[0] * endPointNode2Geo[1];
  464.  
  465. dupNodeIds.forEach((nodeId) => {
  466. const node = W.model.nodes.getObjectById(nodeId),
  467. nodeCoordinates = node.getGeometry().coordinates;
  468. const d = nodeCoordinates[1] * a - nodeCoordinates[0] * b,
  469. newCoordinates = getIntersectCoordinates(a, b, c, d);
  470.  
  471. this.log('move node #' + nodeId + ' to [' + newCoordinates[0] + ';' + newCoordinates[1] + ']')
  472. this.moveNode(node, newCoordinates)
  473. });
  474.  
  475. // I don't understand why doesn't it work, in the WME all looks good, but it fails when try to save changes
  476. // virtualNodes.forEach((node) => {
  477. // const nodeCoordinates = node.getGeometry().coordinates;
  478. // const d = nodeCoordinates[1] * a - nodeCoordinates[0] * b,
  479. // newCoordinates = getIntersectCoordinates(a, b, c, d);
  480. //
  481. // this.log('move node #' + node.getID() + ' to [' + newCoordinates[0] + ';' + newCoordinates[1] + ']')
  482. // this.moveNode(node, newCoordinates)
  483. // });
  484.  
  485. this.groupEnd()
  486. }
  487.  
  488. /**
  489. * Orthogonalize two segments
  490. * This method move the node to new point
  491. *
  492. * @param {Object} segment1
  493. * @param {Object} segment2
  494. * @return {void}
  495. */
  496. orthogonalizeStreetGeometry (segment1, segment2) {
  497. this.log('orthogonalize street geometry')
  498.  
  499. if (segment1.getType() !== 'segment'
  500. || segment2.getType() !== 'segment') {
  501. this.log('only segments must be selected')
  502. return
  503. }
  504.  
  505. /**
  506. * Extract coordinates from components
  507. * @param {Object} segment
  508. * @param {'first'|'second'|'last-but-one'|'last'} position
  509. * @return {*[]}
  510. */
  511. function getCoordinatesFromComponent(segment, position) {
  512. let pos = 0
  513. switch (position) {
  514. case 'first':
  515. pos = 0
  516. break
  517. case 'second':
  518. pos = 1
  519. break
  520. case 'last-but-one':
  521. pos = segment.getOLGeometry().components.length - 2
  522. break
  523. case 'last':
  524. pos = segment.getOLGeometry().components.length - 1
  525. break
  526. }
  527. return [
  528. segment.getOLGeometry().components[pos].x,
  529. segment.getOLGeometry().components[pos].y,
  530. ]
  531. }
  532.  
  533. let A, B, C, commonNode
  534.  
  535. if (segment1.getToNode().getID() === segment2.getFromNode().getID()) {
  536. // A → B → C
  537. commonNode = segment1.getToNode()
  538. A = getCoordinatesFromComponent(segment1, 'last-but-one')
  539. B = getCoordinatesFromComponent(segment1, 'last')
  540. C = getCoordinatesFromComponent(segment2, 'second')
  541. } else if (segment1.getFromNode().getID() === segment2.getFromNode().getID()) {
  542. // B ← A → C
  543. commonNode = segment1.getFromNode()
  544. A = getCoordinatesFromComponent(segment1, 'second')
  545. B = getCoordinatesFromComponent(segment1, 'first')
  546. C = getCoordinatesFromComponent(segment2, 'second')
  547. } else if (segment1.getToNode().getID() === segment2.getToNode().getID()) {
  548. // A → B ← C
  549. commonNode = segment1.getToNode()
  550. A = getCoordinatesFromComponent(segment1, 'last-but-one')
  551. B = getCoordinatesFromComponent(segment1, 'last')
  552. C = getCoordinatesFromComponent(segment2, 'last-but-one')
  553. } else if (segment1.getFromNode().getID() === segment2.getToNode().getID()) {
  554. // B ← A ← C
  555. commonNode = segment1.getFromNode()
  556. A = getCoordinatesFromComponent(segment1, 'second')
  557. B = getCoordinatesFromComponent(segment1, 'first')
  558. C = getCoordinatesFromComponent(segment2, 'last-but-one')
  559. }
  560.  
  561. if (!commonNode) {
  562. this.log('segments does not have common node')
  563. return
  564. }
  565.  
  566. this.log('common node coords [' + B[0] + ';' + B[1] + ']')
  567.  
  568. // Coordinates of points A, B and C
  569. // First selected segment uses it as line for calculation
  570. let intersection = findIntersection(A, B, C)
  571.  
  572. // Uses OpenLayers with convertation, because
  573. intersection = W.userscripts.toGeoJSONGeometry(new OpenLayers.Geometry.Point( ...intersection ))
  574.  
  575. this.log('point of the intersection is [' + intersection[0] + ', ' + intersection[1] +']')
  576.  
  577. this.moveNode(commonNode, intersection.coordinates)
  578. }
  579.  
  580. /**
  581. * Straighten up segment, remove all geometry nodes except first and last
  582. * @param {Object} segment
  583. */
  584. straightenSegmentGeometry (segment) {
  585. this.group('straighten segment geometry')
  586. if (segment.getGeometry().coordinates.length > 2) {
  587. let multiAction = new WazeActionMultiAction()
  588. let newGeometry = structuredClone(segment.attributes.geoJSONGeometry)
  589. // just left the first and last elements
  590. newGeometry.coordinates.splice(1, newGeometry.coordinates.length - 2)
  591. // W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(segment, segment.getGeometry(), newGeometry))
  592. let updateSegmentGeometry = new WazeActionUpdateSegmentGeometry(segment, segment.attributes.geoJSONGeometry, newGeometry)
  593. updateSegmentGeometry.generateDescription();
  594. multiAction.doSubAction(W.model, updateSegmentGeometry);
  595. W.model.actionManager.add(multiAction);
  596. }
  597. }
  598.  
  599. /**
  600. * Move node to new position
  601. * @param {Object} node target
  602. * @param {Array<2>} coords of the new position, array of the wo elements
  603. */
  604. moveNode (node, coords) {
  605. let nodeGeometry = node.getGeometry()
  606. nodeGeometry.coordinates = coords
  607.  
  608. let connectedSegObjs = {}
  609. let emptyObj = {}
  610.  
  611. node.getSegmentIds().forEach((id) => {
  612. connectedSegObjs[id] = { ...W.model.segments.getObjectById(id).getGeometry() }
  613. })
  614.  
  615. W.model.actionManager.add(
  616. new WazeActionMoveNode(
  617. node,
  618. node.getGeometry(),
  619. nodeGeometry,
  620. connectedSegObjs,
  621. emptyObj
  622. )
  623. )
  624. }
  625. }
  626.  
  627. /**
  628. * @param {Array<number,number>} A point of line
  629. * @param {Array<number,number>} B point of line
  630. * @param {Array<number,number>} C point
  631. * @return {(number|*)[]}
  632. */
  633. function findIntersection(A, B, C) {
  634. // Функция для вычисления углового коэффициента прямой
  635. function slope(point1, point2) {
  636. return (point2[1] - point1[1]) / (point2[0] - point1[0]);
  637. }
  638.  
  639. // Функция для вычисления c (точка пересечения с осью Y) для уравнения прямой
  640. function intercept(point, slope) {
  641. return point[1] - slope * point[0];
  642. }
  643.  
  644. // Вычисляем угловой коэффициент для прямой AB
  645. let mAB = slope(A, B);
  646. // Вычисляем c для прямой AB
  647. let cAB = intercept(A, mAB);
  648.  
  649. // Для перпендикуляра угловой коэффициент будет обратным и противоположным
  650. let mPerpendicular = -1 / mAB;
  651. // Вычисляем c для перпендикулярной прямой, проходящей через C
  652. let cPerpendicular = intercept(C, mPerpendicular);
  653.  
  654. // Находим точку пересечения прямых
  655. let x = (cPerpendicular - cAB) / (mAB - mPerpendicular);
  656. let y = mAB * x + cAB;
  657.  
  658. return [x, y];
  659. }
  660.  
  661.  
  662. /**
  663. * Find intersection point
  664. * @param {Number} A
  665. * @param {Number} B
  666. * @param {Number} C
  667. * @param {Number} D
  668. * @return {Number[]}
  669. */
  670. function getIntersectCoordinates (A, B, C, D) {
  671. // http://rsdn.ru/forum/alg/2589531.hot
  672. let r = [2]
  673. r[1] = -1.0 * (C * B - A * D) / (A * A + B * B)
  674. r[0] = (-r[1] * (B + A) - C + D) / (A - B)
  675.  
  676. return r
  677. }
  678.  
  679. /**
  680. * Detect direction
  681. * @param {Number} A
  682. * @param {Number} B
  683. * @return {Number}
  684. */
  685. function getDeltaDirect (A, B) {
  686. if (A < B) {
  687. return 1.0
  688. } else if (A > B) {
  689. return -1.0
  690. }
  691.  
  692. return 0.0
  693. }
  694.  
  695. /**
  696. * Calculate the approximate distance between two coordinates (lat/lon)
  697. *
  698. * © Chris Veness, MIT-licensed,
  699. * http://www.movable-type.co.uk/scripts/latlong.html#equirectangular
  700. *
  701. * @param {number} λ1 first point latitude
  702. * @param {number} φ1 first point longitude
  703. * @param {number} λ2 second point latitude
  704. * @param {number} φ2 second point longitude
  705. */
  706. function distance 1, φ1, λ2, φ2) {
  707. let R = 6371000;
  708. let Δλ = 2 - λ1) * Math.PI / 180;
  709. φ1 = φ1 * Math.PI / 180;
  710. φ2 = φ2 * Math.PI / 180;
  711. let x = Δλ * Math.cos((φ12)/2);
  712. let y = 21);
  713. let d = Math.sqrt(x*x + y*y);
  714. return R * d;
  715. };
  716.  
  717. $(document).on('bootstrap.wme', () => {
  718.  
  719. WazeActionAddNode = require('Waze/Action/AddNode')
  720. WazeActionMoveNode = require('Waze/Action/MoveNode')
  721. WazeActionMultiAction = require('Waze/Action/MultiAction');
  722. WazeActionUpdateSegmentGeometry = require('Waze/Action/UpdateSegmentGeometry')
  723.  
  724. let Instance = new E85(NAME, SETTINGS)
  725. Instance.init(BUTTONS)
  726.  
  727. // setup name for a shortcut section
  728. WMEUIShortcut.setGroupTitle(NAME, I18n.t(NAME).title)
  729. })
  730. })()