// ==UserScript==
// @name WME E40 Geometry
// @name:uk WME 🇺🇦 E40 Geometry
// @version 0.7.1
// @description A script that allows aligning, scaling, and copying POI geometry
// @description:uk За допомогою цього скрипта ви можете легко змінювати площу та вирівнювати POI
// @license MIT License
// @author Anton Shevchuk
// @namespace https://greasyfork.org/users/227648-anton-shevchuk
// @supportURL https://github.com/AntonShevchuk/wme-e40/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/450160/1218867/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/1281847/WME-UI.js
// ==/UserScript==
/* jshint esversion: 8 */
/* global require */
/* global $, jQuery */
/* global W */
/* global I18n */
/* global OpenLayers */
/* global WME, WMEBase, WMEUI, WMEUIHelper, WMEUIShortcut */
/* global Container, Settings, SimpleCache, Tools */
(function () {
'use strict'
// Script name, uses as unique index
const NAME = 'E40'
// User level required for apply geometry for all entities in the view area
const REQUIRED_LEVEL = 2
// Translations
const TRANSLATION = {
'en': {
title: 'POI Geometry',
description: 'Change geometry in the current view area',
warning: '⚠️ This option is available for editors with a rank higher than ' + REQUIRED_LEVEL,
orthogonalize: 'Orthogonalize',
simplify: 'Simplify',
scale: 'Scale',
copy: 'Copy',
about: '<a href="https://greasyfork.org/uk/scripts/388271-wme-e40-geometry">WME E40 Geometry</a>',
},
'uk': {
title: 'Геометрія POI',
description: 'Змінити геометрію об’єктів у поточному розташуванні',
warning: '⚠️ Ця опція доступна лише для редакторів з рангом вищім ніж ' + REQUIRED_LEVEL,
orthogonalize: 'Вирівняти',
simplify: 'Спростити',
scale: 'Масштабувати',
copy: 'Копіювати',
about: '<a href="https://greasyfork.org/uk/scripts/388271-wme-e40-geometry">WME E40 Geometry</a>',
},
'ru': {
title: 'Геометрия POI',
description: 'Изменить геометрию объектов в текущем расположении',
warning: '⚠️ Эта опция доступна для редакторов с рангов выше ' + REQUIRED_LEVEL,
orthogonalize: 'Выровнять',
simplify: 'Упростить',
scale: 'Масштабировать',
copy: 'Копировать',
about: '<a href="https://greasyfork.org/uk/scripts/388271-wme-e40-geometry">WME E40 Geometry</a>',
}
}
const STYLE =
'button.waze-btn.e40 { margin: 0 4px 4px 0; padding: 2px; width: 45px; border: 1px solid #ddd; } ' +
'p.e40-info { border-top: 1px solid #ccc; color: #777; font-size: x-small; margin-top: 15px; padding-top: 10px; text-align: center; }' +
'p.e40-warning { color: #f77 }'
WMEUI.addTranslation(NAME, TRANSLATION)
WMEUI.addStyle(STYLE)
// Set shortcuts title
WMEUIShortcut.setGroupTitle(NAME, I18n.t(NAME).title)
const panelButtons = {
A: {
title: '🔲',
description: I18n.t(NAME).orthogonalize,
shortcut: 'S+49',
callback: () => orthogonalize()
},
B: {
title: '〽️',
description: I18n.t(NAME).simplify,
shortcut: 'S+50',
callback: () => simplify()
},
C: {
title: '500m²',
description: I18n.t(NAME).scale + ' 500m²',
shortcut: 'S+51',
callback: () => scaleSelected(500)
},
D: {
title: '650m²',
description: I18n.t(NAME).scale + ' 650m²',
shortcut: 'S+52',
callback: () => scaleSelected(650)
},
E: {
title: '650+',
description: I18n.t(NAME).scale + ' 650+',
shortcut: 'S+53',
callback: () => scaleSelected(650, true)
},
F: {
title: '<i class="fa fa-clone" aria-hidden="true"></i>',
description: I18n.t(NAME).copy,
shortcut: 'S+54',
callback: () => copyPlaces()
}
}
const tabButtons = {
A: {
title: '🔲',
description: I18n.t(NAME).orthogonalize,
shortcut: null,
callback: () => orthogonalizeAll()
},
B: {
title: '〽️',
description: I18n.t(NAME).simplify,
shortcut: null,
callback: () => simplifyAll()
},
C: {
title: '500+',
description: I18n.t(NAME).scale + ' 500m²+',
shortcut: null,
callback: () => scaleAll(500, true)
}
}
let WazeActionUpdateFeatureGeometry
let WazeActionUpdateFeatureAddress
let WazeFeatureVectorLandmark
let WazeActionAddLandmark
class E40 extends WMEBase {
constructor (name) {
super(name)
this.helper = new WMEUIHelper(name)
this.panel = this.helper.createPanel(I18n.t(name).title)
this.panel.addButtons(panelButtons)
let tab = this.helper.createTab(
I18n.t(name).title,
{
image: GM_info.script.icon
}
)
tab.addText('description', I18n.t(name).description)
if (W.loginManager.user.getRank() > REQUIRED_LEVEL) {
tab.addButtons(tabButtons)
} else {
tab.addText('warning', I18n.t(name).warning)
}
tab.addText(
'info',
'<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
)
tab.inject()
}
/**
* Handler for `place.wme` event
* @param {jQuery.Event} event
* @param {HTMLElement} element
* @param {W.model} model
*/
onPlace (event, element, model) {
if (!model.isGeometryEditable()) {
return
}
this.createPanel(event, element)
}
/**
* Handler for `venues.wme` event
* @param {jQuery.Event} event
* @param {HTMLElement} element
* @param {Array} models
* @return {Null}
*/
onVenues (event, element, models) {
models = models.filter(el => !el.isPoint() && el.isGeometryEditable())
if (models.length > 0) {
this.createPanel(event, element)
}
}
/**
* Create panel with buttons
* @param event
* @param element
*/
createPanel (event, element) {
if (element.querySelector('div.form-group.e40')) {
return
}
element.prepend(this.panel.html())
this.updateLabel()
}
/**
* Updated label
*/
updateLabel () {
let places = getSelectedPlaces()
if (places.length === 0) {
return
}
let info = []
for (let i = 0; i < places.length; i++) {
let selected = places[i]
info.push(Math.round(selected.getOLGeometry().getGeodesicArea(W.map.getProjectionObject())) + 'm²')
}
let label = I18n.t(NAME).title
if (info.length) {
label += ' (' + info.join(', ') + ')'
}
let elm = document.querySelector('div.form-group.e40 label')
if (elm) elm.innerText = label
}
}
$(document).on('bootstrap.wme', () => {
// Require Waze components
WazeActionUpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry')
WazeActionUpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress')
WazeFeatureVectorLandmark = require('Waze/Feature/Vector/Landmark')
WazeActionAddLandmark = require('Waze/Action/AddLandmark')
let E40Instance = new E40(NAME)
W.model.actionManager.events.register('afterundoaction', null, E40Instance.updateLabel)
W.model.actionManager.events.register('afterclearactions', null, E40Instance.updateLabel)
W.model.actionManager.events.register('afteraction', null, E40Instance.updateLabel)
})
/**
* Get selected Area POI
* @return {Array}
*/
function getSelectedPlaces () {
let selected
selected = WME.getSelectedVenues()
selected = selected.filter(el => !el.isPoint())
return selected
}
// Scale selected place(s) to X m²
function scaleSelected (x, orMore = false) {
scaleArray(getSelectedPlaces(), x, orMore)
return false
}
// Scale all places in the editor area to X m²
function scaleAll (x = 650, orMore = true) {
scaleArray(WME.getVenues().filter(el => !el.isPoint()), x, orMore)
return false
}
function scaleArray (elements, x, orMore = false) {
console.groupCollapsed(
'%c' + NAME + ': 📏 %c try to scale ' + (elements.length) + ' element(s) to ' + x + 'm²',
'color: #0DAD8D; font-weight: bold',
'color: dimgray; font-weight: normal'
)
let total = 0
for (let i = 0; i < elements.length; i++) {
let selected = elements[i]
try {
let oldOLGeometry = selected.getOLGeometry().clone()
let newOLGeometry = selected.getOLGeometry().clone()
let scale = Math.sqrt((x + 5) / oldOLGeometry.getGeodesicArea(W.map.getProjectionObject()))
if (scale < 1 && orMore) {
continue
}
newOLGeometry.resize(scale, newOLGeometry.getCentroid())
let action = new WazeActionUpdateFeatureGeometry(
selected,
W.model.venues,
W.userscripts.toGeoJSONGeometry(oldOLGeometry),
W.userscripts.toGeoJSONGeometry(newOLGeometry)
)
W.model.actionManager.add(action)
total++
} catch (e) {
console.log('skipped', e)
}
}
console.log(total + ' element(s) was scaled')
console.groupEnd()
}
// Orthogonalize selected place(s)
function orthogonalize () {
orthogonalizeArray(getSelectedPlaces())
return false
}
// Orthogonalize all places in the editor area
function orthogonalizeAll () {
// skip parking, natural and outdoors
// TODO: make options for filters
orthogonalizeArray(WME.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']).filter(el => !el.isPoint()))
return false
}
function orthogonalizeArray (elements) {
console.groupCollapsed(
'%c' + NAME + ': 🔲 %c try to orthogonalize ' + (elements.length) + ' element(s)',
'color: #0DAD8D; font-weight: bold',
'color: dimgray; font-weight: normal'
)
let total = 0
// skip points
for (let i = 0; i < elements.length; i++) {
let selected = elements[i]
try {
let oldGeometry = { ...selected.getGeometry() }
let currentOLGeometry = selected.getOLGeometry()
let oldNodes = currentOLGeometry.clone().components[0].components
let newNodes = orthogonalizeGeometry(selected.getOLGeometry().clone().components[0].components)
if (!compare(oldNodes, newNodes)) {
currentOLGeometry.components[0].components = [].concat(newNodes)
currentOLGeometry.components[0].clearBounds()
selected.setOLGeometry(currentOLGeometry)
let action = new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, selected.getGeometry())
W.model.actionManager.add(action)
total++
}
} catch (e) {
console.log('skipped', e)
}
}
console.log(total + ' element(s) was orthogonalized')
console.groupEnd()
}
/**
* Clone OL Geometry and orthogonalize it
* @param nodes
* @param threshold
* @return {*}
*/
function orthogonalizeGeometry (nodes, threshold = 12) {
let nomthreshold = threshold, // degrees within right or straight to alter
lowerThreshold = Math.cos((90 - nomthreshold) * Math.PI / 180),
upperThreshold = Math.cos(nomthreshold * Math.PI / 180)
function Orthogonalize (nodes) {
let points = nodes.slice(0, -1).map(function (n) {
let p = n.clone().transform(new OpenLayers.Projection('EPSG:900913'), new OpenLayers.Projection('EPSG:4326'))
p.y = lat2latp(p.y)
return p
}),
corner = { i: 0, dotp: 1 },
epsilon = 1e-4,
i, j, score, motions
// Triangle
if (nodes.length === 4) {
for (i = 0; i < 1000; i++) {
motions = points.map(calcMotion)
let tmp = addPoints(points[corner.i], motions[corner.i])
points[corner.i].x = tmp.x
points[corner.i].y = tmp.y
score = corner.dotp
if (score < epsilon) {
break
}
}
let n = points[corner.i]
n.y = latp2lat(n.y)
let pp = n.transform(new OpenLayers.Projection('EPSG:4326'), new OpenLayers.Projection('EPSG:900913'))
let id = nodes[corner.i].id
for (i = 0; i < nodes.length; i++) {
if (nodes[i].id !== id) {
continue
}
nodes[i].x = pp.x
nodes[i].y = pp.y
}
return nodes
} else {
let best,
originalPoints = nodes.slice(0, -1).map(function (n) {
let p = n.clone().transform(new OpenLayers.Projection('EPSG:900913'), new OpenLayers.Projection('EPSG:4326'))
p.y = lat2latp(p.y)
return p
})
score = Infinity
for (i = 0; i < 1000; i++) {
motions = points.map(calcMotion)
for (j = 0; j < motions.length; j++) {
let tmp = addPoints(points[j], motions[j])
points[j].x = tmp.x
points[j].y = tmp.y
}
let newScore = squareness(points)
if (newScore < score) {
best = [].concat(points)
score = newScore
}
if (score < epsilon) {
break
}
}
points = best
for (i = 0; i < points.length; i++) {
// only move the points that actually moved
if (originalPoints[i].x !== points[i].x || originalPoints[i].y !== points[i].y) {
let n = points[i]
n.y = latp2lat(n.y)
let pp = n.transform(new OpenLayers.Projection('EPSG:4326'), new OpenLayers.Projection('EPSG:900913'))
let id = nodes[i].id
for (j = 0; j < nodes.length; j++) {
if (nodes[j].id !== id) {
continue
}
nodes[j].x = pp.x
nodes[j].y = pp.y
}
}
}
// remove empty nodes on straight sections
for (i = 0; i < points.length; i++) {
let dotp = normalizedDotProduct(i, points)
if (dotp < -1 + epsilon) {
let id = nodes[i].id
for (j = 0; j < nodes.length; j++) {
if (nodes[j].id !== id) {
continue
}
nodes[j] = false
}
}
}
return nodes.filter(item => item !== false)
}
function calcMotion (b, i, array) {
let a = array[(i - 1 + array.length) % array.length],
c = array[(i + 1) % array.length],
p = subtractPoints(a, b),
q = subtractPoints(c, b),
scale, dotp
scale = 2 * Math.min(euclideanDistance(p, { x: 0, y: 0 }), euclideanDistance(q, { x: 0, y: 0 }))
p = normalizePoint(p, 1.0)
q = normalizePoint(q, 1.0)
dotp = filterDotProduct(p.x * q.x + p.y * q.y)
// nasty hack to deal with almost-straight segments (angle is closer to 180 than to 90/270).
if (array.length > 3) {
if (dotp < -0.707106781186547) {
dotp += 1.0
}
} else if (dotp && Math.abs(dotp) < corner.dotp) {
corner.i = i
corner.dotp = Math.abs(dotp)
}
return normalizePoint(addPoints(p, q), 0.1 * dotp * scale)
}
}
function lat2latp (lat) {
return 180 / Math.PI * Math.log(Math.tan(Math.PI / 4 + lat * (Math.PI / 180) / 2))
}
function latp2lat (a) {
return 180 / Math.PI * (2 * Math.atan(Math.exp(a * Math.PI / 180)) - Math.PI / 2)
}
function squareness (points) {
return points.reduce(function (sum, val, i, array) {
let dotp = normalizedDotProduct(i, array)
dotp = filterDotProduct(dotp)
return sum + 2.0 * Math.min(Math.abs(dotp - 1.0), Math.min(Math.abs(dotp), Math.abs(dotp + 1)))
}, 0)
}
function normalizedDotProduct (i, points) {
let a = points[(i - 1 + points.length) % points.length],
b = points[i],
c = points[(i + 1) % points.length],
p = subtractPoints(a, b),
q = subtractPoints(c, b)
p = normalizePoint(p, 1.0)
q = normalizePoint(q, 1.0)
return p.x * q.x + p.y * q.y
}
function subtractPoints (a, b) {
return { x: a.x - b.x, y: a.y - b.y }
}
function addPoints (a, b) {
return { x: a.x + b.x, y: a.y + b.y }
}
function euclideanDistance (a, b) {
let x = a.x - b.x, y = a.y - b.y
return Math.sqrt((x * x) + (y * y))
}
function normalizePoint (point, scale) {
let vector = { x: 0, y: 0 }
let length = Math.sqrt(point.x * point.x + point.y * point.y)
if (length !== 0) {
vector.x = point.x / length
vector.y = point.y / length
}
vector.x *= scale
vector.y *= scale
return vector
}
function filterDotProduct (dotp) {
if (lowerThreshold > Math.abs(dotp) || Math.abs(dotp) > upperThreshold) {
return dotp
}
return 0
}
return Orthogonalize(nodes)
}
// Simplify selected place(s)
function simplify (factor = 8) {
simplifyArray(getSelectedPlaces(), factor)
return false
}
// Simplify all places in the editor area
function simplifyAll () {
// skip parking, natural and outdoors
// TODO: make options for filters
simplifyArray(WME.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']).filter(el => !el.isPoint()))
return false
}
function simplifyArray (elements, factor = 8) {
console.groupCollapsed(
'%c' + NAME + ': 〽️ %c try to simplify ' + (elements.length) + ' element(s)',
'color: #0DAD8D; font-weight: bold',
'color: dimgray; font-weight: normal'
)
let total = 0
for (let i = 0; i < elements.length; i++) {
let selected = elements[i]
try {
let oldOLGeometry = selected.getOLGeometry().clone()
let ls = new OpenLayers.Geometry.LineString(oldOLGeometry.components[0].components)
ls = ls.simplify(factor)
let newOLGeometry = new OpenLayers.Geometry.Polygon(new OpenLayers.Geometry.LinearRing(ls.components))
if (newOLGeometry.components[0].components.length < oldOLGeometry.components[0].components.length) {
W.model.actionManager.add(
new WazeActionUpdateFeatureGeometry(
selected,
W.model.venues,
W.userscripts.toGeoJSONGeometry(oldOLGeometry),
W.userscripts.toGeoJSONGeometry(newOLGeometry)
)
)
total++
}
} catch (e) {
console.log('skipped', e)
}
}
console.log(total + ' element(s) was simplified')
console.groupEnd()
}
/**
* Compare two polygons point-by-point
*
* @return boolean
*/
function compare (geo1, geo2) {
if (geo1.length !== geo2.length) {
return false
}
for (let i = 0; i < geo1.length; i++) {
if (Math.abs(geo1[i].x - geo2[i].x) > .1
|| Math.abs(geo1[i].y - geo2[i].y) > .1) {
return false
}
}
return true
}
/**
* Copy selected places
* Last of them will be chosen
*/
function copyPlaces () {
let venues = getSelectedPlaces()
for (let i = 0; i < venues.length; i++) {
copyPlace(venues[i])
}
}
/**
* Create copy for place
* @param oldPlace
*/
function copyPlace (oldPlace) {
console.log(
'%c' + NAME + ': %c created a copy of the POI ' + oldPlace.attributes.name,
'color: #0DAD8D; font-weight: bold',
'color: dimgray; font-weight: normal'
)
// copy all attributes of the old place
// maybe we should except something in the feature
let newPlace = new WazeFeatureVectorLandmark({ ...oldPlace.attributes})
newPlace.setAttribute('name', oldPlace.getAttribute('name') + ' (copy)')
// little move for new POI, uses geoJSON
for (let i = 0; i < newPlace.getGeometry().coordinates[0].length; i++) {
newPlace.getGeometry().coordinates[0][i][0] += 0.0001
newPlace.getGeometry().coordinates[0][i][1] += 0.00005
}
// add new POI
W.model.actionManager.add(new WazeActionAddLandmark(newPlace))
// update address of new POI
// set the same Country/State/Street and skip the house number
let address = {
countryID: oldPlace.getAddress().getCountry().getID(),
stateID: oldPlace.getAddress().getState().getID(),
cityName: oldPlace.getAddress().getCityName(),
streetName: oldPlace.getAddress().getStreetName()
}
W.model.actionManager.add(new WazeActionUpdateFeatureAddress(newPlace, address))
W.selectionManager.setSelectedModels(newPlace)
}
})()