WME E95

Setup road properties in one click

当前为 2019-05-04 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name WME E95
  3. // @namespace http://tampermonkey.net/
  4. // @version 0.1
  5. // @description Setup road properties in one click
  6. // @author Anton Shevchuk
  7. // @include https://www.waze.com/editor*
  8. // @include https://www.waze.com/*/editor*
  9. // @exclude https://www.waze.com/user/editor*
  10. // @icon 
  11. // @grant none
  12. // ==/UserScript==
  13. /* jshint esversion: 6 */
  14. /* global require */
  15.  
  16. (function($, WazeApi) {
  17. 'use strict';
  18.  
  19. let WazeActionUpdateObject = require("Waze/Action/UpdateObject");
  20. let WazeActionUpdateFeatureAddress = require("Waze/Action/UpdateFeatureAddress");
  21.  
  22. // Road Types
  23. const types = {
  24. street : 1,
  25. primary : 2,
  26. private : 17,
  27. };
  28.  
  29. // Road Flags
  30. // for setup flags use binary operators
  31. // e.g. flags.tunnel | flags.headlights
  32. const flags = {
  33. tunnel : 0b00000001,
  34. // ??? : 0b00000010,
  35. // ??? : 0b00000100,
  36. // ??? : 0b00001000,
  37. unpaved : 0b00010000,
  38. headlights : 0b00100000,
  39. };
  40.  
  41. // Buttons:
  42. // title - for buttons
  43. // keyCode - key for shortcuts (Alt+...)
  44. // detectCity - try to detect city name by closures segments
  45. // clearCity - clear city name
  46. // attributes - native settings for object
  47. const buttons = {
  48. A: {
  49. title: 'PR',
  50. keyCode: 49,
  51. detectCity: true,
  52. attributes: {
  53. fwdMaxSpeed: 20,
  54. revMaxSpeed: 20,
  55. roadType: types.private,
  56. }
  57. },
  58. B: {
  59. title: '50',
  60. keyCode: 50,
  61. detectCity: true,
  62. attributes: {
  63. fwdMaxSpeed: 50,
  64. revMaxSpeed: 50,
  65. roadType: types.street,
  66. }
  67. },
  68. C: {
  69. title: '90',
  70. keyCode: 51,
  71. clearCity: true,
  72. attributes: {
  73. fwdMaxSpeed: 90,
  74. revMaxSpeed: 90,
  75. roadType: types.street,
  76. flags: flags.headlights,
  77. }
  78. }
  79. };
  80.  
  81. // Update segment attributes
  82. function setupRoad(segment, settings) {
  83. let addr = segment.getAddress().attributes;
  84. // Change address
  85. let address = {
  86. countryID: addr.country ? addr.country.id : WazeApi.model.countries.top.id,
  87. stateID: addr.state ? addr.state.id : WazeApi.model.states.top.id,
  88. cityName: addr.city ? addr.city.attributes.name : null,
  89. streetName: addr.street ? addr.street.name : null,
  90. };
  91.  
  92. // Settings: Clear city
  93. if (settings.clearCity) {
  94. address.cityName = null;
  95. }
  96.  
  97. // Settings: Detect city
  98. if (settings.detectCity) {
  99. address.cityName = getCity(segment);
  100. }
  101.  
  102. // Check city
  103. address.emptyCity = (address.cityName === null);
  104.  
  105. // Check street
  106. address.emptyStreet = (address.streetName === null);
  107.  
  108. // Update segment properties
  109. WazeApi.model.actionManager.add(
  110. new WazeActionUpdateObject(
  111. segment,
  112. settings.attributes
  113. )
  114. );
  115. // Update segment address
  116. WazeApi.model.actionManager.add(
  117. new WazeActionUpdateFeatureAddress(
  118. segment,
  119. address,
  120. {
  121. streetIDField: 'primaryStreetID'
  122. }
  123. )
  124. );
  125. }
  126.  
  127. // Update street handler
  128. function processHandler() {
  129. process(this.dataset.e95);
  130. }
  131. function process(index) {
  132. // Get all selected segments
  133. let selected = WazeApi.selectionManager.getSelectedFeatures();
  134. for (let i = 0, total = selected.length; i < total; i++) {
  135. let segment = WazeApi.model.segments.getObjectById(selected[i].model.attributes.id);
  136. if (!segment) continue;
  137. if (!segment.getPermissions()) {
  138. console.log('E95: you don\'t have permissions');
  139. continue;
  140. }
  141. setupRoad(segment, buttons[index]);
  142. }
  143. }
  144.  
  145. // Detect city name by connected segments
  146. function getCity(segment) {
  147. // get cities
  148. // W.model.cities.getValidCities();
  149. // distance to city center
  150. // W.model.cities.getObjectById(644304).getAttributes().geometry.distanceTo(W.model.segments.getObjectById(374688209).getAttributes().geometry);
  151. let cityName = null;
  152. // TODO: replace follow magic with segment.getConnectedSegments() and segment.getConnectedSegmentsByDirection() when it will work
  153. let connected = WazeApi.model.nodes.getObjectById(segment.getAttributes().fromNodeID).getSegmentIds(); // segments from point A
  154. connected = connected.concat(WazeApi.model.nodes.getObjectById(segment.getAttributes().toNodeID).getSegmentIds()); // segments from point B
  155. connected.filter(id => id !== segment.getID());
  156.  
  157. for (let i = 0, total = connected.length; i < total; i++) {
  158. let city = WazeApi.model.segments.getObjectById(connected[i]).getAddress().getCity();
  159. // skip segments with empty cities
  160. if (city && !city.isEmpty()) {
  161. cityName = city.getName();
  162. break;
  163. }
  164. }
  165. console.log('E-95: detected city ' + cityName);
  166. return cityName;
  167. }
  168.  
  169. // Create UI controls everytime
  170. // Uses native JS function for better performance
  171. function createUI() {
  172. // container for buttons
  173. let controls = document.createElement('div');
  174. controls.className = 'controls';
  175. // create all buttons
  176. for (let btn in buttons) {
  177. let button = document.createElement('button');
  178. button.className = 'waze-btn waze-btn-small waze-btn-white road-e95 road-' + btn;
  179. button.style.marginRight = '4px';
  180. button.innerHTML = buttons[btn].title;
  181. button.dataset.e95 = btn;
  182. controls.appendChild(button);
  183. }
  184.  
  185. let label = document.createElement('label');
  186. label.className = 'control-label';
  187. label.innerHTML = 'Quick properties';
  188.  
  189. let group = document.createElement('div');
  190. group.className = 'form-group e95';
  191. group.appendChild(label);
  192. group.appendChild(controls);
  193.  
  194. document.getElementById('segment-edit-general').prepend(group);
  195. }
  196.  
  197. function init() {
  198. // Check for changes in the edit-panel
  199. var speedlimitsObserver = new MutationObserver(function(mutations) {
  200. mutations.forEach(function(mutation) {
  201. for (let i = 0, total = mutation.addedNodes.length; i < total; i++) {
  202. let node = mutation.addedNodes[i];
  203. // Only fire up if it's a node
  204. if (node.nodeType === Node.ELEMENT_NODE &&
  205. node.querySelector('div.selection') &&
  206. !node.querySelector('div.form-group.e95')) {
  207. createUI();
  208. }
  209. }
  210. });
  211. });
  212.  
  213. speedlimitsObserver.observe(document.getElementById('edit-panel'), { childList: true, subtree: true });
  214. console.log('E95: observer was run');
  215.  
  216. // Handler for all buttons
  217. $('#edit-panel').on('click', 'button.road-e95', processHandler);
  218.  
  219. // Handler for button shortcuts
  220. $(document).on('keyup', function(e) {
  221. if (e.altKey && !e.ctrlKey && !e.shiftKey) {
  222. for (let btn in buttons) {
  223. if (buttons[btn].keyCode === e.which) {
  224. process(btn);
  225. break;
  226. }
  227. }
  228. }
  229. });
  230. console.log('E95: handler was initialized');
  231. }
  232.  
  233. // Bootstrap plugin
  234. function bootstrap(tries = 1) {
  235. console.log('E95: attempt ' + tries);
  236. if (WazeApi &&
  237. WazeApi.map &&
  238. WazeApi.model &&
  239. WazeApi.loginManager.user) {
  240. console.log('E95: was initialized');
  241. init();
  242. } else if (tries < 100) {
  243. tries++;
  244. setTimeout(() => bootstrap(tries), 800);
  245. } else {
  246. console.error('E95: initialization failed');
  247. }
  248. }
  249.  
  250. console.log('E95: initialization');
  251. bootstrap();
  252. })(window.jQuery, window.W);