// ==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://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/1619452/WME-Bootstrap.js
// @require https://update.greasyfork.org/scripts/452563/1218878/WME.js
// @require https://update.greasyfork.org/scripts/450221/1137043/WME-Base.js
// @require https://update.greasyfork.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)
})
})()