// ==UserScript==
// @name WME E40 Geometry
// @version 0.4.0
// @description Setup POI geometry properties in one click
// @author Anton Shevchuk
// @license MIT License
// @match https://www.waze.com/editor*
// @match https://www.waze.com/*/editor*
// @match https://beta.waze.com/editor*
// @match https://beta.waze.com/*/editor*
// @exclude https://www.waze.com/user/editor*
// @exclude https://beta.waze.com/user/editor*
// @grant none
// @icon 
// @require https://greasyfork.dpdns.org/scripts/389117-apihelper/code/APIHelper.js?version=1082818
// @require https://greasyfork.dpdns.org/scripts/389577-apihelperui/code/APIHelperUI.js?version=1082967
// @supportURL https://github.com/AntonShevchuk/wme-e40/issues
// @namespace https://greasyfork.dpdns.org/users/227648
// ==/UserScript==
/* jshint esversion: 6 */
/* global require */
/* global $ */
/* global W */
/* global OpenLayers */
/* global I18n */
/* global APIHelper */
/* global APIHelperUI */
(function ($) {
'use strict'
let helper
let panel
let tab
let OL = OpenLayers
// Script name, uses as unique index
const NAME = 'E40'
// Translations
const TRANSLATION = {
'en': {
title: 'Geometry',
description: 'Change geometry in the current view area',
orthogonalize: 'Orthogonalize',
simplify: 'Simplify',
scale: 'Scale',
copy: 'Copy',
},
'uk': {
title: 'Геометрія',
description: 'Змінити геометрію об’єктів у поточному розташуванні',
orthogonalize: 'Вирівняти',
simplify: 'Спростити',
scale: 'Масштабувати',
copy: 'Копіювати',
},
'ru': {
title: 'Геометрия',
description: 'Изменить геометрию объектов в текущем расположении',
orthogonalize: 'Выровнять',
simplify: 'Упростить',
scale: 'Масштабировать',
copy: 'Копировать',
}
}
APIHelper.bootstrap()
APIHelper.addTranslation(NAME, TRANSLATION)
APIHelper.addStyle(
'button.waze-btn.e40 { margin: 0 4px 4px 0; padding: 2px; width: 42px; } ' +
'button.waze-btn.e40:hover { box-shadow: 0 2px 8px 0 rgba(0, 0, 0, 0.1), inset 0 0 100px 100px rgba(255, 255, 255, 0.3); } '
)
const panelButtons = {
A: {
title: '🔲',
description: I18n.t(NAME).orthogonalize,
shortcut: 'S+49',
callback: () => orthogonalize()
},
B: {
title: '<i class="fa fa-magic"></i>',
description: I18n.t(NAME).simplify,
shortcut: 'S+50',
callback: () => simplify()
},
C: {
title: '500m²',
description: I18n.t(NAME).scale,
shortcut: 'S+51',
callback: () => scaleSelected(500)
},
D: {
title: '650m²',
description: I18n.t(NAME).scale,
shortcut: 'S+52',
callback: () => scaleSelected(650)
},
E: {
title: '>650',
description: I18n.t(NAME).scale,
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+53',
callback: () => copyPlace()
}
}
const tabButtons = {
A: {
title: '🔲',
description: I18n.t(NAME).orthogonalize,
shortcut: null,
callback: () => orthogonalizeAll()
},
B: {
title: '<i class="fa fa-magic"></i>',
description: I18n.t(NAME).simplify,
shortcut: null,
callback: () => simplifyAll()
},
C: {
title: '>500',
description: I18n.t(NAME).scale,
shortcut: null,
callback: () => scaleAll(500, true)
}
}
let WazeActionUpdateFeatureGeometry
let WazeActionUpdateFeatureAddress
let WazeFeatureVectorLandmark
let WazeActionAddLandmark
/**
* Get selected Area POI
* @return {Array}
*/
function getSelectedPlaces () {
let selected
selected = APIHelper.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(APIHelper.getVenues(), x, orMore)
return false
}
function scaleArray (elements, x, orMore = false) {
for (let i = 0; i < elements.length; i++) {
let selected = elements[i]
try {
let oldGeometry = selected.geometry.clone()
let newGeometry = selected.geometry.clone()
let scale = Math.sqrt((x + 5) / oldGeometry.getGeodesicArea(W.map.getProjectionObject()))
if (scale < 1 && orMore) {
continue
}
newGeometry.resize(scale, newGeometry.getCentroid())
let action = new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, newGeometry)
W.model.actionManager.add(action)
} catch (e) {
log('skipped')
}
}
}
// 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(APIHelper.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']))
return false
}
function orthogonalizeArray (elements) {
for (let i = 0; i < elements.length; i++) {
let selected = elements[i]
try {
let oldGeometry = selected.geometry.clone()
let newGeometry = orthogonalizeGeometry(selected.geometry.clone().components[0].components)
if (!compare(oldGeometry.components[0].components, newGeometry)) {
selected.geometry.components[0].components = [].concat(newGeometry)
selected.geometry.components[0].clearBounds()
let action = new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, selected.geometry)
W.model.actionManager.add(action)
}
} catch (e) {
log('skipped')
console.log(e)
}
}
return false
}
function orthogonalizeGeometry (geometry, 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 () {
let nodes = geometry,
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
}
var 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
}
function isDisabled (nodes) {
let points = nodes.slice(0, -1).map(function (n) {
let p = n.toLonLat().transform(new OpenLayers.Projection('EPSG:900913'), new OpenLayers.Projection('EPSG:4326'))
return { x: p.lat, y: p.lon }
})
return squareness(points)
}
return Orthogonalize()
}
// 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(APIHelper.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']))
return false
}
function simplifyArray (elements, factor = 8) {
for (let i = 0; i < elements.length; i++) {
let selected = elements[i]
try {
let oldGeometry = selected.geometry.clone()
let ls = new OL.Geometry.LineString(oldGeometry.components[0].components)
ls = ls.simplify(factor)
let newGeometry = new OL.Geometry.Polygon(new OL.Geometry.LinearRing(ls.components))
if (newGeometry.components[0].components.length < oldGeometry.components[0].components.length) {
W.model.actionManager.add(new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, newGeometry))
}
} catch (e) {
log('skipped')
}
}
return false
}
// Compare two polygons point-by-point
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
}
function copyPlace(){
let venues = APIHelper.getSelectedVenues()
if (venues.length > 0) {
let oldPlace = venues[0];
let newPlace = new WazeFeatureVectorLandmark;
newPlace.attributes.name = oldPlace.attributes.name + ' (copy)';
newPlace.attributes.phone = oldPlace.attributes.phone;
newPlace.attributes.url = oldPlace.attributes.url;
newPlace.attributes.categories = [].concat(oldPlace.attributes.categories);
newPlace.attributes.aliases = [].concat(oldPlace.attributes.aliases);
newPlace.attributes.description = oldPlace.attributes.description;
newPlace.attributes.houseNumber = oldPlace.attributes.houseNumber;
newPlace.attributes.lockRank = oldPlace.attributes.lockRank;
newPlace.attributes.geometry = oldPlace.attributes.geometry.clone();
if (oldPlace.attributes.geometry.toString().match(/^POLYGON/)) {
for (let i = 0; i < newPlace.attributes.geometry.components[0].components.length - 1; i++) {
newPlace.attributes.geometry.components[0].components[i].x += 5
newPlace.attributes.geometry.components[0].components[i].y += 5
}
} else {
// Geometry not used for points
// But who knows?
newPlace.attributes.geometry.x += 5
newPlace.attributes.geometry.y += 5
}
newPlace.attributes.services = [].concat(oldPlace.attributes.services);
newPlace.attributes.openingHours = [].concat(oldPlace.attributes.openingHours);
newPlace.attributes.streetID = oldPlace.attributes.streetID;
if (oldPlace.attributes.categories.includes('GAS_STATION')) {
newPlace.attributes.brand = oldPlace.attributes.brand
}
if (oldPlace.attributes.categories.includes('PARKING_LOT')) {
newPlace.attributes.categoryAttributes.PARKING_LOT = {}
let attributes = oldPlace.attributes.categoryAttributes.PARKING_LOT
if ((attributes.lotType != null))
newPlace.attributes.categoryAttributes.PARKING_LOT.lotType = [].concat(oldPlace.attributes.categoryAttributes.PARKING_LOT.lotType)
if ((attributes.canExitWhileClosed != null))
newPlace.attributes.categoryAttributes.PARKING_LOT.canExitWhileClosed = oldPlace.attributes.categoryAttributes.PARKING_LOT.canExitWhileClosed
if ((attributes.costType != null))
newPlace.attributes.categoryAttributes.PARKING_LOT.costType = oldPlace.attributes.categoryAttributes.PARKING_LOT.costType
if ((attributes.estimatedNumberOfSpots != null))
newPlace.attributes.categoryAttributes.PARKING_LOT.estimatedNumberOfSpots = oldPlace.attributes.categoryAttributes.PARKING_LOT.estimatedNumberOfSpots
if ((attributes.hasTBR != null))
newPlace.attributes.categoryAttributes.PARKING_LOT.hasTBR = oldPlace.attributes.categoryAttributes.PARKING_LOT.hasTBR
if ((attributes.lotType != null))
newPlace.attributes.categoryAttributes.PARKING_LOT.lotType = [].concat(oldPlace.attributes.categoryAttributes.PARKING_LOT.lotType)
if ((attributes.parkingType != null))
newPlace.attributes.categoryAttributes.PARKING_LOT.parkingType = oldPlace.attributes.categoryAttributes.PARKING_LOT.parkingType
if ((attributes.paymentType != null))
newPlace.attributes.categoryAttributes.PARKING_LOT.paymentType = [].concat(oldPlace.attributes.categoryAttributes.PARKING_LOT.paymentType)
}
W.model.actionManager.add(new WazeActionAddLandmark(newPlace));
W.selectionManager.setSelectedModels(newPlace);
}
}
// Simple console.log wrapper
function log (message) {
console.log(NAME + ': ' + message)
}
$(document)
.on('init.apihelper', ready)
.on('landmark.apihelper', createPanel)
.on('landmark-collection.apihelper', createPanel)
function ready () {
// Require Waze components
WazeActionUpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry')
WazeActionUpdateFeatureAddress = require('Waze/Action/UpdateFeatureAddress')
WazeFeatureVectorLandmark = require('Waze/Feature/Vector/Landmark')
WazeActionAddLandmark = require('Waze/Action/AddLandmark')
helper = new APIHelperUI(NAME)
panel = helper.createPanel(I18n.t(NAME).title)
panel.addButtons(panelButtons)
if (W.loginManager.user.getRank() > 2) {
tab = helper.createTab(
I18n.t(NAME).title,
I18n.t(NAME).description,
'<i class="w-icon panel-header-component-icon w-icon-polygon"></i>'
)
tab.addButtons(tabButtons)
tab.inject()
}
W.model.actionManager.events.register('afterundoaction', null, updateLabel)
W.model.actionManager.events.register('afterclearactions', null, updateLabel)
W.model.actionManager.events.register('afteraction', null, updateLabel)
}
function createPanel (event, element) {
if (element.querySelector('div.form-group.e40')) {
return
}
let places = getSelectedPlaces()
if (places.length === 0) {
return
}
element.prepend(panel.html())
updateLabel()
}
function 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.geometry.getGeodesicArea(W.map.getProjectionObject())) + 'm²')
}
let label = I18n.t(NAME).title
if (info.length) {
label += ' (' + info.join(', ') + ')'
}
panel.html().querySelector('label').innerText = label
}
// external API
window.E40 = {
scale: function (x) {
scaleSelected(x)
}
}
})(window.jQuery)