- // ==UserScript==
- // @name WME E40 Geometry
- // @name:uk WME 🇺🇦 E40 Geometry
- // @version 0.7.4
- // @description A script that allows aligning, scaling, and copying POI geometry
- // @description:uk За допомогою цього скрипта ви можете легко змінювати площу та вирівнювати POI
- // @license MIT License
- // @author Anton Shevchuk
- // @namespace https://greatest.deepsurf.us/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.greatest.deepsurf.us/scripts/450160/1218867/WME-Bootstrap.js
- // @require https://update.greatest.deepsurf.us/scripts/452563/1218878/WME.js
- // @require https://update.greatest.deepsurf.us/scripts/450221/1137043/WME-Base.js
- // @require https://update.greatest.deepsurf.us/scripts/450320/1555446/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://greatest.deepsurf.us/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://greatest.deepsurf.us/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://greatest.deepsurf.us/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)')
-
- let geometry = { ... oldPlace.getGeometry()}
-
- // little move for new POI, uses geoJSON
- for (let i = 0; i < geometry.coordinates[0].length; i++) {
- geometry.coordinates[0][i][0] += 0.0001
- geometry.coordinates[0][i][1] += 0.00005
- }
-
- newPlace.setGeometry(geometry)
-
- // 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)
- }
-
- })()