WME E40

Setup POI geometry properties in one click

目前为 2019-08-30 提交的版本。查看 最新版本

  1. // ==UserScript==
  2. // @name WME E40
  3. // @version 0.1.2
  4. // @description Setup POI geometry properties in one click
  5. // @author Anton Shevchuk
  6. // @license MIT License
  7. // @include https://www.waze.com/editor*
  8. // @include https://www.waze.com/*/editor*
  9. // @include https://beta.waze.com/editor*
  10. // @include https://beta.waze.com/*/editor*
  11. // @exclude https://www.waze.com/user/editor*
  12. // @exclude https://beta.waze.com/user/editor*
  13. // @grant none
  14. // @icon 
  15. // @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
  16. // @require https://greasyfork.org/scripts/389117-apihelper/code/APIHelper.js?version=729372
  17. // @require https://greasyfork.org/scripts/389577-apihelperui/code/APIHelperUI.js?version=729353
  18. // @supportURL https://github.com/AntonShevchuk/wme-e40/issues
  19. // @namespace https://greasyfork.org/users/227648
  20. // ==/UserScript==
  21.  
  22. /* jshint esversion: 6 */
  23. /* global require, APIHelper, WazeWrap, W, I18n, OL */
  24.  
  25. (function ($) {
  26. 'use strict';
  27.  
  28. let helper;
  29. let panel;
  30. let tab;
  31.  
  32. // Script name, uses as unique index
  33. const NAME = 'E40';
  34.  
  35. // Translations
  36. const TRANSLATION = {
  37. 'en': {
  38. title: 'Geometry',
  39. orthogonalize: 'Orthogonalize',
  40. simplify: 'Simplify',
  41. scale: 'Scale',
  42. },
  43. 'uk': {
  44. title: 'Геометрія',
  45. orthogonalize: 'Вирівняти',
  46. simplify: 'Спростити',
  47. scale: 'Масштабувати',
  48. },
  49. 'ru': {
  50. title: 'Геометрия',
  51. orthogonalize: 'Выровнять',
  52. simplify: 'Упростить',
  53. scale: 'Масштабировать',
  54. }
  55. };
  56.  
  57. APIHelper.bootstrap();
  58. APIHelper.addTranslation(NAME, TRANSLATION);
  59. APIHelper.appendStyle(
  60. 'button.waze-btn.e40 { margin: 0 4px 4px 0; padding: 2px; width: 42px; } ' +
  61. '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); } '
  62. );
  63.  
  64. const panelButtons = {
  65. A: {
  66. title: '🔲',
  67. description: I18n.t(NAME).orthogonalize,
  68. shortcut: 'S+49',
  69. callback: () => orthogonalize()
  70. },
  71. B: {
  72. title: '〽️',
  73. description: I18n.t(NAME).simplify,
  74. shortcut: 'S+50',
  75. callback: () => simplify()
  76. },
  77. C: {
  78. title: '500m²',
  79. description: I18n.t(NAME).scale,
  80. shortcut: 'S+51',
  81. callback: () => scaleSelected(500)
  82. },
  83. D: {
  84. title: '650m²',
  85. description: I18n.t(NAME).scale,
  86. shortcut: 'S+52',
  87. callback: () => scaleSelected(650)
  88. },
  89. E: {
  90. title: '>650',
  91. description: I18n.t(NAME).scale,
  92. shortcut: 'S+53',
  93. callback: () => scaleSelected(650, true)
  94. }
  95. };
  96.  
  97. const tabButtons = {
  98. A: {
  99. title: '🔲',
  100. description: I18n.t(NAME).orthogonalize,
  101. shortcut: null,
  102. callback: () => orthogonalizeAll()
  103. },
  104. B: {
  105. title: '〽️',
  106. description: I18n.t(NAME).simplify,
  107. shortcut: null,
  108. callback: () => simplifyAll()
  109. },
  110. C: {
  111. title: '>650',
  112. description: I18n.t(NAME).scale,
  113. shortcut: null,
  114. callback: () => scaleAll(650, true)
  115. }
  116. };
  117.  
  118. let WazeActionUpdateFeatureGeometry = require('Waze/Action/UpdateFeatureGeometry');
  119.  
  120. /**
  121. * Get selected Area POI
  122. * @return {Array}
  123. */
  124. function getSelectedPlaces() {
  125. let selected;
  126. selected = APIHelper.getSelectedVenues();
  127. selected = selected.filter((el) => !el.isPoint());
  128. return selected;
  129. }
  130. // Scale selected place(s) to X m²
  131. function scaleSelected(x, orMore = false) {
  132. scaleArray(getSelectedPlaces(), x, orMore);
  133. return false;
  134. }
  135. // Scale all places in the editor area to X m²
  136. function scaleAll(x = 650, orMore = true) {
  137. scaleArray(APIHelper.getVenues(), x, orMore);
  138. return false;
  139. }
  140. function scaleArray(elements, x, orMore = false) {
  141. for (let i = 0; i < elements.length; i++) {
  142. let selected = elements[i];
  143. try {
  144. let oldGeometry = selected.geometry.clone();
  145. let newGeometry = selected.geometry.clone();
  146.  
  147. let scale = Math.sqrt((x + 5) / oldGeometry.getGeodesicArea(W.map.getProjectionObject()));
  148. if (scale < 1 && orMore) {
  149. continue;
  150. }
  151. newGeometry.resize(scale, newGeometry.getCentroid());
  152.  
  153. let action = new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, newGeometry);
  154. W.model.actionManager.add(action);
  155. } catch (e) {
  156. log('skipped');
  157. }
  158. }
  159. }
  160. // Orthogonalize selected place(s)
  161. function orthogonalize() {
  162. orthogonalizeArray(getSelectedPlaces());
  163. return false;
  164. }
  165. // Orthogonalize all places in the editor area
  166. function orthogonalizeAll() {
  167. // skip parking, natural and outdoors
  168. // TODO: make options for filters
  169. orthogonalizeArray(APIHelper.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']));
  170. return false;
  171. }
  172. function orthogonalizeArray(elements) {
  173. for (let i = 0; i < elements.length; i++) {
  174. let selected = elements[i];
  175. try {
  176. let oldGeometry = selected.geometry.clone();
  177. let newGeometry = WazeWrap.Util.OrthogonalizeGeometry(selected.geometry.clone().components[0].components);
  178.  
  179. if (!compare(oldGeometry.components[0].components, newGeometry)) {
  180. selected.geometry.components[0].components = [].concat(newGeometry);
  181. selected.geometry.components[0].clearBounds();
  182.  
  183. let action = new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, selected.geometry);
  184. W.model.actionManager.add(action);
  185. }
  186. } catch (e) {
  187. log('skipped');
  188. }
  189. }
  190. return false;
  191. }
  192. // Simplify selected place(s)
  193. function simplify(factor = 8) {
  194. simplifyArray(getSelectedPlaces(), factor);
  195. return false;
  196. }
  197. // Simplify all places in the editor area
  198. function simplifyAll() {
  199. // skip parking, natural and outdoors
  200. // TODO: make options for filters
  201. simplifyArray(APIHelper.getVenues(['OUTDOORS', 'PARKING_LOT', 'NATURAL_FEATURES']));
  202. return false;
  203. }
  204. function simplifyArray(elements, factor = 8) {
  205. for (let i = 0; i < elements.length; i++) {
  206. let selected = elements[i];
  207. try {
  208. let oldGeometry = selected.geometry.clone();
  209. let ls = new OL.Geometry.LineString(oldGeometry.components[0].components);
  210. ls = ls.simplify(factor);
  211. let newGeometry = new OL.Geometry.Polygon(new OL.Geometry.LinearRing(ls.components));
  212.  
  213. if (newGeometry.components[0].components.length < oldGeometry.components[0].components.length) {
  214. W.model.actionManager.add(new WazeActionUpdateFeatureGeometry(selected, W.model.venues, oldGeometry, newGeometry));
  215. }
  216. } catch (e) {
  217. log('skipped');
  218. }
  219. }
  220. return false;
  221. }
  222. // Compare two polygons point-by-point
  223. function compare(geo1, geo2) {
  224. if (geo1.length !== geo2.length) {
  225. return false;
  226. }
  227. for (let i = 0; i < geo1.length; i++) {
  228. if (Math.abs(geo1[i].x - geo2[i].x) > .1
  229. || Math.abs(geo1[i].y - geo2[i].y) > .1) {
  230. return false;
  231. }
  232. }
  233. return true;
  234. }
  235.  
  236. // Simple console.log wrapper
  237. function log(message) {
  238. console.log(NAME + ': ' + message);
  239. }
  240.  
  241. $(document)
  242. .on('ready.apihelper', ready)
  243. .on('landmark.apihelper', '#edit-panel', createPanel)
  244. .on('landmark-collection.apihelper', '#edit-panel', createPanel)
  245. ;
  246.  
  247. function ready() {
  248. helper = new APIHelperUI(NAME);
  249.  
  250. panel = helper.createPanel(I18n.t(NAME).title);
  251. panel.addButtons(panelButtons);
  252.  
  253. if (W.loginManager.user.getRank() > 2) {
  254. tab = helper.createTab(I18n.t(NAME).title);
  255. tab.addButtons(tabButtons);
  256. tab.init();
  257. }
  258.  
  259. WazeWrap.Events.register('selectionchanged', null, updateLabel);
  260. WazeWrap.Events.register('afterundoaction', null, updateLabel);
  261. WazeWrap.Events.register('afterclearactions', null, updateLabel);
  262. WazeWrap.Events.register('afteraction', null, updateLabel);
  263. }
  264. function createPanel(event, element) {
  265. if (element.querySelector('div.form-group.e40')) {
  266. return;
  267. }
  268. let places = getSelectedPlaces();
  269. if (places.length === 0) {
  270. return;
  271. }
  272.  
  273. element.prepend(panel.toHTML());
  274. updateLabel();
  275. }
  276.  
  277. function updateLabel() {
  278. let places = getSelectedPlaces();
  279. if (places.length === 0) {
  280. return;
  281. }
  282. let info = [];
  283. for (let i = 0; i < places.length; i++) {
  284. let selected = places[i];
  285. info.push(Math.round(selected.geometry.getGeodesicArea(W.map.getProjectionObject())) + 'm²');
  286. }
  287. let label = I18n.t(NAME).title;
  288. if (info.length) {
  289. label += ' (' + info.join(', ') + ')';
  290. }
  291. $('div.form-group.e40 label.control-label').text(label);
  292. }
  293.  
  294. // external API
  295. window.E40 = {
  296. scale: function (x) {
  297. scaleSelected(x);
  298. }
  299. };
  300. })(window.jQuery);