// ==UserScript==
// @name WME E40
// @version 0.1.2
// @description Setup POI geometry properties in one click
// @author Anton Shevchuk
// @license MIT License
// @include https://www.waze.com/editor*
// @include https://www.waze.com/*/editor*
// @include https://beta.waze.com/editor*
// @include 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.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require https://greasyfork.org/scripts/389117-apihelper/code/APIHelper.js?version=729372
// @require https://greasyfork.org/scripts/389577-apihelperui/code/APIHelperUI.js?version=729353
// @supportURL https://github.com/AntonShevchuk/wme-e40/issues
// @namespace https://greasyfork.org/users/227648
// ==/UserScript==
/* jshint esversion: 6 */
/* global require, APIHelper, WazeWrap, W, I18n, OL */
(function ($) {
'use strict';
let helper;
let panel;
let tab;
// Script name, uses as unique index
const NAME = 'E40';
// Translations
const TRANSLATION = {
'en': {
title: 'Geometry',
orthogonalize: 'Orthogonalize',
simplify: 'Simplify',
scale: 'Scale',
},
'uk': {
title: 'Геометрія',
orthogonalize: 'Вирівняти',
simplify: 'Спростити',
scale: 'Масштабувати',
},
'ru': {
title: 'Геометрия',
orthogonalize: 'Выровнять',
simplify: 'Упростить',
scale: 'Масштабировать',
}
};
APIHelper.bootstrap();
APIHelper.addTranslation(NAME, TRANSLATION);
APIHelper.appendStyle(
'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: '〽️',
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)
}
};
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: '>650',
description: I18n.t(NAME).scale,
shortcut: null,
callback: () => scaleAll(650, true)
}
};
let WazeActionUpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry');
/**
* 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 = WazeWrap.Util.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');
}
}
return false;
}
// 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;
}
// Simple console.log wrapper
function log(message) {
console.log(NAME + ': ' + message);
}
$(document)
.on('ready.apihelper', ready)
.on('landmark.apihelper', '#edit-panel', createPanel)
.on('landmark-collection.apihelper', '#edit-panel', createPanel)
;
function ready() {
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);
tab.addButtons(tabButtons);
tab.init();
}
WazeWrap.Events.register('selectionchanged', null, updateLabel);
WazeWrap.Events.register('afterundoaction', null, updateLabel);
WazeWrap.Events.register('afterclearactions', null, updateLabel);
WazeWrap.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.toHTML());
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(', ') + ')';
}
$('div.form-group.e40 label.control-label').text(label);
}
// external API
window.E40 = {
scale: function (x) {
scaleSelected(x);
}
};
})(window.jQuery);