您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Setup POI geometry properties in one click
当前为
// ==UserScript== // @name WME E40 Geometry // @version 0.3.2 // @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.org/scripts/389117-apihelper/code/APIHelper.js?version=1082818 // @require https://greasyfork.org/scripts/389577-apihelperui/code/APIHelperUI.js?version=1082967 // @supportURL https://github.com/AntonShevchuk/wme-e40/issues // @namespace https://greasyfork.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', }, 'uk': { title: 'Геометрія', description: 'Змінити геометрію об’єктів у поточному розташуванні', orthogonalize: 'Вирівняти', simplify: 'Спростити', scale: 'Масштабувати', }, 'ru': { title: 'Геометрия', description: 'Изменить геометрию объектов в текущем расположении', orthogonalize: 'Выровнять', simplify: 'Упростить', scale: 'Масштабировать', } } 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) } } 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 /** * 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 () { var 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) var 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 } var 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 { var 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 } // 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 component WazeActionUpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry') 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)