您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Setup POI geometry properties in one click
当前为
- // ==UserScript==
- // @name WME E40 Geometry
- // @version 0.5.7
- // @description Setup POI geometry properties in one click
- // @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://greasyfork.org/scripts/450160-wme-bootstrap/code/WME-Bootstrap.js?version=1126584
- // @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=1101617
- // @require https://greasyfork.org/scripts/450320-wme-ui/code/WME-UI.js?version=1127621
- // ==/UserScript==
- /* jshint esversion: 8 */
- /* global require */
- /* global $ */
- /* global W */
- /* global I18n */
- /* global OpenLayers */
- /* global WME, WMEBase, WMEUI, WMEUIHelper */
- /* global Container, Settings, SimpleCache, Tools */
- (function () {
- 'use strict'
- // 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',
- about: '<a href="https://greasyfork.org/uk/scripts/388271-wme-e40-geometry">WME E40 Geometry</a>',
- },
- 'uk': {
- title: 'Геометрія',
- description: 'Змінити геометрію об’єктів у поточному розташуванні',
- orthogonalize: 'Вирівняти',
- simplify: 'Спростити',
- scale: 'Масштабувати',
- copy: 'Копіювати',
- about: '<a href="https://greasyfork.org/uk/scripts/388271-wme-e40-geometry">WME E40 Geometry</a>',
- },
- 'ru': {
- title: 'Геометрия',
- description: 'Изменить геометрию объектов в текущем расположении',
- 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: 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); } ' +
- 'p.e40-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)
- // 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,
- 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+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,
- 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)
- if (W.loginManager.user.getRank() > 2) {
- this.tab = this.helper.createTab(
- I18n.t(name).title,
- I18n.t(name).description,
- {
- 'icon': '<i class="w-icon panel-header-component-icon w-icon-polygon"></i>'
- }
- )
- this.tab.addButtons(tabButtons)
- this.tab.addText(
- 'info',
- '<a href="' + GM_info.scriptUpdateURL + '">' + GM_info.script.name + '</a> ' + GM_info.script.version
- )
- this.tab.inject()
- }
- }
- /**
- * Handler for `place.wme` event
- * @param {jQuery.Event} event
- * @param {HTMLElement} element
- * @param {W.model} model
- * @return {Null}
- */
- onPlace (event, element, model) {
- 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())
- 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.geometry.getGeodesicArea(W.map.getProjectionObject())) + 'm²')
- }
- let label = I18n.t(NAME).title
- if (info.length) {
- label += ' (' + info.join(', ') + ')'
- }
- document.querySelector('div.form-group.e40 label').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.group(
- '%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 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)
- 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.group(
- '%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.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)
- total++
- }
- } catch (e) {
- console.log('skipped')
- console.log(e)
- }
- }
- console.log(total + ' element(s) was orthogonalized')
- console.groupEnd()
- }
- 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
- }
- 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
- }
- 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(WME.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']).filter(el => !el.isPoint()))
- return false
- }
- function simplifyArray (elements, factor = 8) {
- console.group(
- '%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 oldGeometry = selected.geometry.clone()
- let ls = new OpenLayers.Geometry.LineString(oldGeometry.components[0].components)
- ls = ls.simplify(factor)
- let newGeometry = new OpenLayers.Geometry.Polygon(new OpenLayers.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))
- 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) {
- 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 as is
- // But you can use select multiple venues, and then click "copy"
- 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)
- }
- })()