// ==UserScript==
// @name WME E85 Simplify Street Geometry
// @name:uk WME 🇺🇦 E85 Simplify Street Geometry
// @version 0.1.6
// @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://greasyfork.org/scripts/389765-common-utils/code/CommonUtils.js?version=1090053
// @require https://greasyfork.org/scripts/450160-wme-bootstrap/code/WME-Bootstrap.js?version=1153357
// @require https://greasyfork.org/scripts/452563-wme/code/WME.js?version=1101598
// @require https://greasyfork.org/scripts/450221-wme-base/code/WME-Base.js?version=1137043
// @require https://greasyfork.org/scripts/450320-wme-ui/code/WME-UI.js?version=1137289
// ==/UserScript==
/* jshint esversion: 8 */
/* global require */
/* global $, jQuery */
/* global W */
/* global I18n */
/* global OpenLayers */
/* global WME, WMEBase */
/* global WMEUI, WMEUIHelper, WMEUIHelperPanel, WMEUIHelperModal, WMEUIHelperTab, WMEUIShortcut */
/* global Container, Settings, SimpleCache, Tools */
(function () {
'use strict'
// Script name, uses as unique index
const NAME = 'E85'
// Translations
const TRANSLATION = {
'en': {
title: 'Street Geometry',
description: 'Simplify and straighten up streets',
buttons: {
A: 'Simplify',
B: 'Straighten',
C: '∡90°',
},
settings: {
title: 'Settings',
description: 'Settings for simplifying segments',
simplifyShort: 'Remove a fragment shorter than',
simplifyAngle: 'If the angle is bigger than',
simplifyTwoShort: 'and fragments shorter than',
},
},
'uk': {
title: 'Геометрія вулиць',
description: 'Спрощуйте та вирівнюйте вулиці',
buttons: {
A: 'Спростити',
B: 'Вирівняти',
C: '∡90°',
},
settings: {
title: 'Налаштування',
description: 'Для спрощення сегментів будуть враховані наступні параметри',
simplifyShort: 'Видаляти фрагменти менші ніж',
simplifyAngle: 'Або якщо кут більше ніж',
simplifyTwoShort: 'та фрагменти меньші ніж',
},
},
'ru': {
title: 'Геометрия улиц',
description: 'Упрощайте и выравнивайте геометрию улиц',
buttons: {
A: 'Упростить',
B: 'Выровнять',
C: '∡90°',
},
settings: {
title: 'Настройки',
description: 'Параметры для упрощения геометрии сегмента',
simplifyShort: 'Если фрагмент короче, чем',
simplifyAngle: 'Или угол больше чем',
simplifyTwoShort: 'и фрагменты меньше, чем',
},
}
}
const STYLE =
'button.e85.e85-A { background-color: #0f9; margin-right: 2px }' +
'button.e85.e85-B { background-color: #09f; color: #fff }' +
'button.e85.e85-C { background-color: #f99; margin-left: 2px }' +
'button.e85.e85-A:disabled, button.e85.e85-B:disabled { background-color: #ccc }' +
'.e85 legend { cursor:pointer; font-size: 12px; font-weight: bold; width: auto; text-align: right; border: 0; margin: 0; padding: 0 8px; }' +
'.e85 fieldset { border: 1px solid #ddd; padding: 8px; }' +
'.e85 fieldset.e85 div.controls label { white-space: normal; font-weight: normal; line-height: 32px; font-size: 13px; }' +
'.e85 fieldset.e85 div.controls input[type="number"] { float:right; wight: 32px }' +
'p.e85-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }'
WMEUI.addTranslation(NAME, TRANSLATION)
WMEUI.addStyle(STYLE)
const BUTTONS = {
A: {
title: I18n.t(NAME).buttons.A,
description: I18n.t(NAME).buttons.A,
shortcut: '',
},
B: {
title: I18n.t(NAME).buttons.B,
description: I18n.t(NAME).buttons.B,
shortcut: '',
},
C: {
title: I18n.t(NAME).buttons.C,
description: I18n.t(NAME).buttons.C,
shortcut: '',
},
}
// Default settings
const SETTINGS = {
simplifyShort: 5,
simplifyAngle: 176,
simplifyTwoShort: 50,
}
let WazeActionUpdateSegmentGeometry
let WazeActionMoveNode
let WazeActionAddNode
class E85 extends WMEBase {
/**
* Initial UI elements
* @param {Object} buttons
*/
init (buttons) {
/** @type {WMEUIHelper} */
this.helper = new WMEUIHelper(this.name)
/** @type {WMEUIHelperTab} */
this.tab = this.helper.createTab(
I18n.t(this.name).title,
{
'icon': 'route'
}
)
// Setup options for script
let fieldset = this.helper.createFieldset(I18n.t(NAME).settings.title)
fieldset.addText('description', I18n.t(NAME).settings.description)
let settings = this.settings.get()
for (let item in settings) {
if (settings.hasOwnProperty(item)) {
fieldset.addNumber(
'settings-' + item,
I18n.t(NAME).settings[item],
event => this.settings.set([item], event.target.value),
this.settings.get(item),
(item === 'simplifyAngle') ? 150 : 0,
(item === 'simplifyAngle') ? 180 : 200,
1
)
}
}
this.tab.addElement(fieldset)
this.tab.addText(
'info',
'<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
)
// Inject custom HTML to container in the WME interface
this.tab.inject()
}
/**
* Handler for `segment.wme` event
* @param {jQuery.Event} event
* @param {HTMLElement} element
* @param {W.model} model
* @return {void}
*/
onSegment (event, element, model) {
// Skip for blocked roads
if (model.isLockedByHigherRank()) {
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.geometry.components.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()).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
)
panel.addButton(
'B',
BUTTONS.B.title,
BUTTONS.B.description,
() => this.straightenStreetGeometry(models),
BUTTONS.B.shortcut
)
let modelWithComponents = models.filter(model => model.geometry.components.length > 2)
if (modelWithComponents.length === 0) {
simplifyButton.html().disabled = true
}
if (models.length === 2) {
panel.addButton(
'C',
BUTTONS.C.title,
BUTTONS.C.description,
() => this.orthogonalizeStreetGeometry(models[0], models[1]),
BUTTONS.C.shortcut
)
}
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) {
if (model.geometry.components.length < 3) {
return
}
this.group('simplify segment geometry')
this.log('check geometry of the segment with ID ' + model.getID())
let nodes = []
// calculate angles for every inside point
for (let i = 0; i < model.geometry.components.length - 2; i++) {
let nodeStart = model.geometry.components[i],
nodeCenter = model.geometry.components[i + 1],
nodeEnd = model.geometry.components[i + 2]
nodes[i] = {
angle: Math.round(this.findAngle(nodeStart, nodeCenter, nodeEnd)),
start: Math.round(this.findLength(nodeStart, nodeCenter)),
end: Math.round(this.findLength(nodeCenter, nodeEnd)),
}
this.log('point ' + (i+1) + ' : ' + nodes[i].angle + '°, ' + nodes[i].start + 'm, ' + nodes[i].end + 'm')
}
let removeNodes = []
for (let i = 0; i < nodes.length; i++) {
let node = nodes[i]
// mark to remove a node with short START segment
if (node.start < this.settings.get('simplifyShort')) {
this.log('found too short segment: ' + node.start + 'm')
removeNodes.push(i+1)
continue // skip next rule
}
// mark to remove a node with short END segment and big ANGLE
if (node.angle >= this.settings.get('simplifyAngle')
&& node.end < this.settings.get('simplifyShort')) {
this.log('found too short fragment: ' + node.end + 'm')
removeNodes.push(i+1)
i++ // skip next node
continue // skip next rule
}
// mark to remove a node with big angle and short segments
if (node.angle >= this.settings.get('simplifyAngle')
&& node.start + node.end < this.settings.get('simplifyTwoShort')) {
this.log(
'found point with short fragment: ' + node.start + ' + ' + node.end + ' = ' +
(node.start + node.end) + 'm and angle equal to ' + node.angle + '°'
)
removeNodes.push(i+1)
// continue // skip next rule
}
}
// remove nodes from geometry
if (removeNodes.length) {
let newGeometry = model.geometry.clone()
let components = []
for (let i = 0; i < newGeometry.components.length; i++) {
if (removeNodes.indexOf(i) === -1) {
components.push(newGeometry.components[i])
}
}
newGeometry.components = components
W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(model, model.geometry, newGeometry))
}
this.groupEnd()
}
/**
* Calculates the angle (in radians) between two vectors pointing outward from one center
*
* @param {Object} start first point
* @param {Object} center second point
* @param {Object} end third point
*/
findAngle (start, center, end) {
let b = Math.pow(center.x - start.x, 2) + Math.pow(center.y - start.y, 2),
a = Math.pow(center.x - end.x, 2) + Math.pow(center.y - end.y, 2),
c = Math.pow(end.x - start.x, 2) + Math.pow(end.y - start.y, 2)
return Math.acos((a + b - c) / Math.sqrt(4 * a * b)) * (180 / Math.PI)
}
/**
* Get the length of the line by point coordinates
* @param {Object} start
* @param {Object} end
* @return {Number} length in meters
*/
findLength (start, end) {
let line = new OpenLayers.Geometry.LineString([start, end])
return line.getGeodesicLength('EPSG:900913')
}
/**
* Remove geometry nodes on the target segments
* @param {Array} models
* @return {void}
*/
simplifyStreetGeometry (models) {
this.group('simplify street geometry')
for (let i = 0; i < models.length; i++) {
this.simplifySegmentGeometry(models[i])
}
this.groupEnd()
}
/**
* Выравнивает сегменты в прямую линию, перемещая промежуточные узлы
* в точки пересечения перпендикуляров к вычисленной прямой, проходящей через
* начальный и конечный узлы выделения
* A,B,C - параметры вычисленной прямой уравнения Аx + By + C = 0
*
* @param {Array} models
* @return {void}
*/
straightenStreetGeometry (models) {
this.group('straighten street geometry')
this.log('calculating the formula for the straight line')
let T1, T2,
t,
A = 0.0,
B = 0.0,
C = 0.0
for (let i = 0; i < models.length; i++) {
let segment = models[i]
let geometry = segment.geometry
if (geometry.components.length < 2) {
continue
}
// определяем формулу наклонной прямой
let A1 = geometry.components[0].clone(),
A2 = geometry.components[geometry.components.length - 1].clone()
let dX = getDeltaDirect(A1.x, A2.x)
let dY = getDeltaDirect(A1.y, A2.y)
// looks very strange
let tX = i > 0 ? getDeltaDirect(T1.x, T2.x) : 0
let tY = i > 0 ? getDeltaDirect(T1.y, T2.y) : 0
this.log('vector of the line - tX=' + tX + ', tY=' + tY)
this.log('segment #' + (i + 1) + ' (' + A1.x + '; ' + A1.y + ') - (' + A2.x + '; ' + A2.y + '), dX=' + dX + ', dY=' + dY)
if (dX < 0) {
t = A1.x
A1.x = A2.x
A2.x = t
t = A1.y
A1.y = A2.y
A2.y = t
dX = getDeltaDirect(A1.x, A2.x)
dY = getDeltaDirect(A1.y, A2.y)
this.log('rotate segment #' + (i + 1) + ' (' + A1.x + '; ' + A1.y + ') - (' + A2.x + '; ' + A2.y + '), dX=' + dX + ', dY=' + dY)
}
// looks very strange
if (i === 0) {
T1 = A1.clone()
T2 = A2.clone()
} else {
if (A1.x < T1.x) {
T1.x = A1.x
T1.y = A1.y
}
if (A2.x > T2.x) {
T2.x = A2.x
T2.y = A2.y
}
this.log('calculated straight line (' + T1.x + '; ' + T1.y + ') - (' + T2.x + '; ' + T2.y + ')')
}
}
A = T2.y - T1.y
B = T1.x - T2.x
C = T2.x * T1.y - T1.x * T2.y
this.log('line coords: (' + T1.x + ';' + T1.y + ') - (' + T2.x + ';' + T2.y + ')')
this.log('line formula: ' + A + 'x + ' + B + 'y + ' + C)
for (let i = 0; i < models.length; i++) {
let segment = models[i]
this.group('straighten segment #' + i)
// упрощаем сегмент, если нужно
this.straightenSegmentGeometry(segment)
// работа с узлом
let node = W.model.nodes.getObjectById(segment.attributes.fromNodeID)
let D = node.attributes.geometry.y * A - node.attributes.geometry.x * B
let r1 = getIntersectCoordinates(A, B, C, D)
this.log('move node A')
this.moveNode(node, r1)
let node2 = W.model.nodes.getObjectById(segment.attributes.toNodeID)
let D2 = node2.attributes.geometry.y * A - node2.attributes.geometry.x * B
let r2 = getIntersectCoordinates(A, B, C, D2)
this.log('move node B')
this.moveNode(node2, r2)
this.log('segment #' + (i + 1) + ' (' + r1[0] + ';' + r1[1] + ') - (' + r2[0] + ';' + r2[1] + ')')
this.groupEnd()
}
}
/**
* Orthogonalize two segments
* This method move the node to new point
*
* @param {Object} segment1
* @param {Object} segment2
* @return {void}
*/
orthogonalizeStreetGeometry (segment1, segment2) {
this.log('orthogonalize street geometry')
if (segment1.type !== 'segment' || segment2.type !== 'segment') {
this.log('only segments must be selected')
return
}
let seg1Attrs = segment1.attributes,
seg2Attrs = segment2.attributes
let commonNodeID
// find ID of the common node
let node = {}
if (seg1Attrs.fromNodeID === seg2Attrs.fromNodeID) commonNodeID = seg1Attrs.fromNodeID
else if (seg1Attrs.fromNodeID === seg2Attrs.toNodeID) commonNodeID = seg1Attrs.fromNodeID
else if (seg1Attrs.toNodeID === seg2Attrs.fromNodeID) commonNodeID = seg1Attrs.toNodeID
else if (seg1Attrs.toNodeID === seg2Attrs.toNodeID) commonNodeID = seg1Attrs.toNodeID
if (!commonNodeID) {
this.log('segments does not have common node')
return
}
this.log('common node ID: ' + commonNodeID)
node = W.model.nodes.getObjectById(commonNodeID)
// ID другого узла второго сегмента. От него будем строить перпендикуляр
let otherNodeID = commonNodeID === seg2Attrs.fromNodeID ? seg2Attrs.toNodeID : seg2Attrs.fromNodeID
let otherNode = W.model.nodes.getObjectById(otherNodeID)
// упростим оба сегмента
// TODO: подумать, можно ли использовать координаты промежуточных узлов и не упрощать сегменты
this.straightenSegmentGeometry(segment1)
this.straightenSegmentGeometry(segment2)
// вычислим новое положение общего узла
// координаты концов первого сегмента
let x1 = segment1.getFromNode().attributes.geometry.x,
y1 = segment1.getFromNode().attributes.geometry.y,
x2 = segment1.getToNode().attributes.geometry.x,
y2 = segment1.getToNode().attributes.geometry.y
// коэффициенты в формуле прямой, проходящей через концы первого сегмента
let A = y1 - y2,
B = x2 - x1,
C = x1 * y2 - x2 * y1,
// что такое D ???
D = otherNode.attributes.geometry.y * A - otherNode.attributes.geometry.x * B
// move node and its segments to calculated position
this.moveNode(node, getIntersectCoordinates(A, B, C, D))
}
/**
* Straighten up segment, remove all geometry nodes except first and last
* @param {Object} segment
*/
straightenSegmentGeometry (segment) {
if (segment.geometry.components.length > 2) {
let newGeometry = segment.geometry.clone()
newGeometry.components.splice(1, newGeometry.components.length - 2)
W.model.actionManager.add(new WazeActionUpdateSegmentGeometry(segment, segment.geometry, newGeometry))
}
}
/**
* Move node to new position
* @param {Object} node target
* @param {Array<2>} coords of the new position, array of the wo elements
*/
moveNode (node, coords) {
let nodeGeo = node.geometry.clone()
nodeGeo.x = coords[0]
nodeGeo.y = coords[1]
nodeGeo.calculateBounds()
let connectedSegObjs = {}
let emptyObj = {}
for (let j = 0; j < node.attributes.segIDs.length; j++) {
let segId = node.attributes.segIDs[j]
connectedSegObjs[segId] = W.model.segments.getObjectById(segId).geometry.clone()
}
W.model.actionManager.add(new WazeActionMoveNode(node, node.geometry, nodeGeo, connectedSegObjs, emptyObj))
}
}
/**
* Find intersection point
* @param {Number} A
* @param {Number} B
* @param {Number} C
* @param {Number} D
* @return {Number[]}
*/
function getIntersectCoordinates (A, B, C, D) {
// http://rsdn.ru/forum/alg/2589531.hot
let r = [2]
r[1] = -1.0 * (C * B - A * D) / (A * A + B * B)
r[0] = (-r[1] * (B + A) - C + D) / (A - B)
return r
}
/**
* Detect direction
* @param {Number} A
* @param {Number} B
* @return {Number}
*/
function getDeltaDirect (A, B) {
let d = 0.0
if (A < B) {
d = 1.0
} else if (A > B) {
d = -1.0
}
return d
}
$(document).on('bootstrap.wme', () => {
WazeActionUpdateSegmentGeometry = require('Waze/Action/UpdateSegmentGeometry')
WazeActionMoveNode = require('Waze/Action/MoveNode')
WazeActionAddNode = require('Waze/Action/AddNode')
let Instance = new E85(NAME, SETTINGS)
Instance.init(BUTTONS)
// setup name for shortcut section
WMEUIShortcut.setGroupTitle(NAME, I18n.t(NAME).title)
})
})()