WME Split POI

Split POI with a new seg

  1. /* eslint-disable max-len */
  2. /* eslint-disable prefer-destructuring */
  3. /* eslint-disable camelcase */
  4. // ==UserScript==
  5. // @name WME Split POI
  6. // @namespace https://greasyfork.org/fr/scripts/13008-wme-split-poi
  7. // @description Split POI with a new seg
  8. // @description:fr Découpage d'un POI en deux en utisant un nouveau segment
  9. // @include https://www.waze.com/editor*
  10. // @include https://www.waze.com/*/editor*
  11. // @include https://beta.waze.com/editor*
  12. // @include https://beta.waze.com/*/editor*
  13. // @exclude https://www.waze.com/user*
  14. // @exclude https://www.waze.com/*/user*
  15. // @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
  16. // @require https://cdn.jsdelivr.net/npm/@turf/turf@7/turf.min.js
  17. // eslint-disable-next-line max-len
  18. // @icon 
  19. // @author seb-d59, WazeDev (2023-?)
  20. // @version 2025.05.08.000
  21. // @license GPLv3
  22. // @grant GM_xmlhttpRequest
  23. // @connect greasyfork.org
  24. // ==/UserScript==
  25.  
  26. /* global WazeWrap */
  27. /* global getWmeSdk */
  28. /* global turf */
  29.  
  30. (function main() {
  31. 'use strict';
  32.  
  33. const DEBUG = false;
  34. const SCRIPT_VERSION = GM_info.script.version;
  35. const SCRIPT_NAME = GM_info.script.name;
  36. const DOWNLOAD_URL = 'https://greasyfork.org/scripts/13008-wme-split-poi/code/WME%20Split%20POI.user.js';
  37. const MINIMUM_AREA = 500.0;
  38.  
  39. let sdk;
  40.  
  41. function bootstrap() {
  42. if (unsafeWindow.getWmeSdk && WazeWrap.Ready) {
  43. initialize();
  44. } else {
  45. setTimeout(bootstrap, 100);
  46. }
  47. }
  48.  
  49. function getId(node) {
  50. return document.getElementById(node);
  51. }
  52.  
  53. function log(msg, obj) {
  54. if (obj == null) {
  55. console.log(`WME Split POI v${SCRIPT_VERSION} - ${msg}`);
  56. } else if (DEBUG) {
  57. console.debug(`WME Split POI v${SCRIPT_VERSION} - ${msg} `, obj);
  58. }
  59. }
  60.  
  61. function initialize() {
  62. log('init');
  63. sdk = getWmeSdk({ scriptId: 'wmeSplitPOI', scriptName: 'WME Split POI' });
  64. startScriptUpdateMonitor();
  65. initializeWazeObjects();
  66. }
  67.  
  68. function startScriptUpdateMonitor() {
  69. let updateMonitor;
  70. try {
  71. updateMonitor = new WazeWrap.Alerts.ScriptUpdateMonitor(SCRIPT_NAME, SCRIPT_VERSION, DOWNLOAD_URL, GM_xmlhttpRequest);
  72. updateMonitor.start();
  73. } catch (ex) {
  74. // Report, but don't stop if ScriptUpdateMonitor fails.
  75. console.error(`${SCRIPT_NAME}:`, ex);
  76. }
  77. }
  78.  
  79. function onSelectionChanged(tries = 0) {
  80. if (tries === 30) return;
  81. try {
  82. const venue = getSelectedAreaVenue();
  83. if (!venue) return;
  84.  
  85. // const landmarkPoi = '(NATURAL_FEATURES|ISLAND|SEA_LAKE_POOL|RIVER_STREAM|FOREST_GROVE|FARM|CANAL|SWAMP_MARSH|DAM|PARK)';
  86. // if (new RegExp(landmarkPoi).test(attributes.categories) === false) return;
  87.  
  88. const editPanel = getId('edit-panel');
  89. if (editPanel.firstElementChild.style.display === 'none') {
  90. window.setTimeout(onSelectionChanged, 100, ++tries);
  91. return;
  92. }
  93.  
  94. // ok: 1 selected item and panel is shown
  95.  
  96. // On verifie que le segment est éditable
  97. if (!sdk.DataModel.Venues.hasPermissions({ venueId: venue.id })) return;
  98.  
  99. // Exclude gas station and EVCS categories (don't ever want to delete those by splitting):
  100. if (venue.categories.some(cat => ['GAS_STATION', 'CHARGING_STATION'].includes(cat))) return;
  101.  
  102. if (!$('#split-poi-button').length) {
  103. let addAfter = true;
  104. let insertAtElement = document.querySelector('.geometry-type-control-area');
  105. if (!insertAtElement) {
  106. insertAtElement = document.querySelector('.external-providers-control');
  107. if (!insertAtElement) {
  108. setTimeout(onSelectionChanged, 100, ++tries);
  109. return;
  110. }
  111. addAfter = false;
  112. }
  113. const WMESP_Controle = document.createElement('wz-button');
  114. WMESP_Controle.color = 'secondary';
  115. WMESP_Controle.size = 'sm';
  116. WMESP_Controle.id = 'split-poi-button';
  117. WMESP_Controle.className = 'geometry-type-control-button geometry-type-control-point';
  118. WMESP_Controle.innerHTML = '<i class="fa fa-cut" style="font-size:24px;" title="Split POI"></i>';
  119. if (addAfter) {
  120. insertAtElement.after(WMESP_Controle);
  121. } else {
  122. insertAtElement.before(WMESP_Controle);
  123. }
  124. WMESP_Controle.onclick = onSplitPoiButtonClick;
  125. }
  126. } catch (ex) {
  127. console.error('Split POI:', ex);
  128. }
  129. }
  130.  
  131. function initializeWazeObjects() {
  132. sdk.Events.on({
  133. eventName: 'wme-selection-changed',
  134. eventHandler: () => setTimeout(onSelectionChanged, 0)
  135. });
  136. // call OnSelectionChanged once to catch selected venue in PL
  137. onSelectionChanged();
  138. }
  139.  
  140. // This will return null if more than one object is selected
  141. function getSelectedAreaVenue() {
  142. const selection = sdk.Editing.getSelection();
  143. if (selection?.ids.length !== 1 || selection.objectType !== 'venue') return null;
  144. const venue = sdk.DataModel.Venues.getById({ venueId: selection.ids[0] });
  145. if (venue.geometry.type !== 'Polygon') return null;
  146. return venue;
  147. }
  148.  
  149. // function cloneAttribute(venue, attrName, newAttributesObject) {
  150. // if (venue.attributes.hasOwnProperty(attrName)) {
  151. // let value = venue.attributes[attrName];
  152.  
  153. // if (Array.isArray(value)) {
  154. // value = value.slice(0); // copy array
  155. // }
  156. // newAttributesObject[attrName] = venue.attributes[attrName];
  157. // }
  158. // }
  159.  
  160. function cloneVenue(venue, newGeometry) {
  161. const cloneId = sdk.DataModel.Venues.addVenue({
  162. // SDK: Update this if/when more attributes are available.
  163. category: venue.categories[0],
  164. geometry: newGeometry
  165. }).toString(); // toString is needed because a string is expected later
  166. const address = sdk.DataModel.Venues.getAddress({ venueId: venue.id });
  167. sdk.DataModel.Venues.updateAddress({ venueId: cloneId, houseNumber: address.houseNumber, streetId: address.street?.id });
  168. sdk.DataModel.Venues.updateVenue({
  169. venueId: cloneId,
  170. aliases: venue.aliases,
  171. openingHours: venue.openingHours,
  172. phone: venue.phone,
  173. services: venue.services,
  174. url: venue.url
  175. });
  176. // const clonePoi = new LandmarkVectorFeature({ geoJSONGeometry: W.userscripts.toGeoJSONGeometry(newGeometry) });
  177. // [
  178. // 'aliases',
  179. // 'categories',
  180. // 'description',
  181. // 'entryExitPoints',
  182. // 'externalProviderIDs',
  183. // 'houseNumber',
  184. // 'lockRank',
  185. // 'name',
  186. // 'openingHours',
  187. // 'phone',
  188. // 'services',
  189. // 'streetID',
  190. // 'url'
  191. // ].forEach(attrName => cloneAttribute(poi, attrName, clonePoi.attributes));
  192. // if (clonePoi.attributes.name) clonePoi.attributes.name += ` (copy ${nameSuffixIndex})`; // IMPORTANT! Won't save for some reason without changing the names (at least for PLAs).
  193. // if (poi.attributes.categoryAttributes.PARKING_LOT) {
  194. // clonePoi.attributes.categoryAttributes.PARKING_LOT = JSON.parse(JSON.stringify(poi.attributes.categoryAttributes.PARKING_LOT));
  195. // }
  196.  
  197. // const WazeActionAddLandmark = require('Waze/Action/AddLandmark');
  198. // actions.push(new WazeActionAddLandmark(clonePoi));
  199.  
  200. // const street = W.model.streets.getObjectById(poi.attributes.streetID);
  201. // const streetName = street.attributes.name;
  202. // const cityID = street.attributes.cityID;
  203. // const city = W.model.cities.getObjectById(cityID);
  204. // const stateID = city.attributes.stateID;
  205. // const countryID = city.attributes.countryID;
  206. // const houseNumber = poi.attributes.houseNumber;
  207. // if (!street.attributes.isEmpty || !city.attributes.isEmpty) { // nok
  208. // const newAtts = {
  209. // emptyStreet: street.attributes.isEmpty, // TODO: fix this
  210. // stateID,
  211. // countryID,
  212. // cityName: city.attributes.name,
  213. // houseNumber,
  214. // streetName,
  215. // emptyCity: city.attributes.isEmpty // TODO: fix this
  216. // };
  217. // const updateAddressAction = new UpdateFeatureAddressAction(clonePoi, newAtts);
  218. // updateAddressAction.options.updateHouseNumber = true;
  219. // actions.push(updateAddressAction);
  220. // }
  221. }
  222.  
  223. // function confirmBeforeSplitting(venue) {
  224. // // SDK: FR submitted to add venue attribues
  225. // const entryExitPointsLen = venue.attributes.entryExitPoints?.length;
  226. // const imagesLen = venue.attributes.images?.length;
  227. // const extProvidersLen = venue.attributes.externalProviderIDs?.length;
  228. // let warningText = 'WARNING: The original place will be deleted!';
  229.  
  230. // if (imagesLen) {
  231. // warningText += '\n\nThe following property(s) will be lost:';
  232. // if (imagesLen) warningText += `\n • ${imagesLen} photo${imagesLen === 1 ? '' : 's'} (permanently deleted after saving)`;
  233. // }
  234. // warningText += '\n\nThe following properties likely need to be changed after splitting:';
  235. // warningText += '\n • name ("copy #" will be appended)';
  236. // if (entryExitPointsLen) warningText += `\n • ${entryExitPointsLen} entry/exit point${entryExitPointsLen === 1 ? '' : 's'}`;
  237. // if (extProvidersLen) warningText += `\n • ${extProvidersLen} linked Google place${extProvidersLen === 1 ? '' : 's'}`;
  238. // warningText += '\n\nReview <i>all</i> properties of both new places before saving.';
  239. // warningText += '\n';
  240. // return new Promise(resolve => {
  241. // WazeWrap.Alerts.confirm(
  242. // SCRIPT_NAME,
  243. // warningText,
  244. // () => resolve(true),
  245. // () => resolve(false)
  246. // );
  247. // });
  248. // }
  249.  
  250. async function onDrawLineFinished(line, venue) {
  251. // if (!await confirmBeforeSplitting(venue)) return;
  252.  
  253. const intersections = turf.lineIntersect(venue.geometry, line);
  254. if (intersections.features.length === 0) {
  255. WazeWrap.Alerts.error(SCRIPT_NAME, 'The cut line must intersect the place\'s geometry.');
  256. return;
  257. }
  258.  
  259. if (intersections.features.length % 2) {
  260. WazeWrap.Alerts.error(SCRIPT_NAME, 'The cut line cannot begin or end inside the place\'s geometry and it cannot cross itself.');
  261. return;
  262. }
  263.  
  264. // const newPolygons = createTwoPolygonsFromIntersectPoints(venue, intersectPoints);
  265. // if (newPolygons[0].getArea() < MINIMUM_AREA || newPolygons[1].getArea() < MINIMUM_AREA) {
  266. // WazeWrap.Alerts.error(SCRIPT_NAME, 'New area place would be too small. Move the temporary road segment.');
  267. // return;
  268. // }
  269. unsafeWindow.intersections = intersections;
  270. unsafeWindow.line = line;
  271. unsafeWindow.poly = venue.geometry;
  272. console.log(intersections.features.length === 2);
  273. const venuePolygon = venue.geometry;
  274. const newPolygons = [];
  275. const cutResults1 = cutPolygon(venuePolygon, line, 1);
  276. if (cutResults1) {
  277. newPolygons.push(...cutResults1);
  278. }
  279. const cutResults2 = cutPolygon(venuePolygon, line, -1);
  280. if (cutResults2) {
  281. newPolygons.push(...cutResults2);
  282. }
  283.  
  284. if (newPolygons.some(poly => { console.log(turf.area(poly)); return turf.area(poly) < MINIMUM_AREA; })) {
  285. WazeWrap.Alerts.error(SCRIPT_NAME, 'At least one of the new polygons would be too small to appear in the app.');
  286. return;
  287. }
  288.  
  289. let largest;
  290. newPolygons.forEach(polygon => {
  291. const area = turf.area(polygon);
  292. if (!largest || area > largest.area) {
  293. largest = { polygon, area };
  294. }
  295. });
  296.  
  297. newPolygons.forEach(polygon => {
  298. if (polygon === largest.polygon) {
  299. sdk.DataModel.Venues.updateVenue({ venueId: venue.id, geometry: polygon.geometry });
  300. } else {
  301. cloneVenue(venue, polygon.geometry);
  302. }
  303. });
  304. }
  305.  
  306. function onSplitPoiButtonClick() {
  307. const venue = getSelectedAreaVenue();
  308. if (!venue) return;
  309.  
  310. // This is needed in case the category is changed to GS or EVCS and the split button is still there.
  311. if (venue.categories.some(cat => ['GAS_STATION', 'CHARGING_STATION'].includes(cat))) {
  312. WazeWrap.Alerts.error(SCRIPT_NAME, 'Cannot split gas stations or EV charging stations');
  313. return;
  314. }
  315.  
  316. sdk.Map.drawLine().then(line => {
  317. onDrawLineFinished(line, venue);
  318.  
  319. // const confirm = await confirmBeforeSplitting(venue);
  320. // if (confirm) {
  321. // const actions = [];
  322. // addClonePoiAction(venue, newPolygons[0], 1, actions);
  323. // addClonePoiAction(venue, newPolygons[1], 2, actions);
  324.  
  325. // actions.push(new DeleteSegmentAction(seg));
  326. // const multiaction = new MultiAction(actions, { description: 'Split POI' });
  327. // W.model.actionManager.add(multiaction);
  328. // }
  329. }).catch(ex => {
  330. if (ex instanceof sdk.Errors.InvalidStateError) {
  331. // log, but ignore it
  332. console.log(ex);
  333. } else {
  334. console.error(ex);
  335. }
  336. });
  337. }
  338.  
  339. function cutPolygon(polygon, cutLine, direction) {
  340. let j;
  341. const polyCoords = [];
  342. const cutPolyGeoms = [];
  343.  
  344. if ((polygon.type !== 'Polygon') || (cutLine.type !== 'LineString')) return null;
  345.  
  346. const intersectPoints = turf.lineIntersect(polygon, cutLine);
  347. const nPoints = intersectPoints.features.length;
  348. if ((nPoints === 0) || ((nPoints % 2) !== 0)) return null;
  349.  
  350. const offsetLine = turf.lineOffset(cutLine, (0.01 * direction), { units: 'kilometers' });
  351.  
  352. for (j = 0; j < cutLine.coordinates.length; j++) {
  353. polyCoords.push(cutLine.coordinates[j]);
  354. }
  355. for (j = (offsetLine.geometry.coordinates.length - 1); j >= 0; j--) {
  356. polyCoords.push(offsetLine.geometry.coordinates[j]);
  357. }
  358. polyCoords.push(cutLine.coordinates[0]);
  359. const thickLineString = turf.lineString(polyCoords);
  360. const thickLinePolygon = turf.lineToPolygon(thickLineString);
  361.  
  362. polygon = turf.feature(polygon);
  363. const clipped = turf.difference(turf.featureCollection([polygon, thickLinePolygon]));
  364. for (j = 0; j < clipped.geometry.coordinates.length; j++) {
  365. const polyg = turf.polygon(clipped.geometry.coordinates[j]);
  366. const overlap = turf.lineOverlap(polyg, cutLine, { tolerance: 0.005 });
  367. if (overlap.features.length > 0) {
  368. cutPolyGeoms.push(polyg.geometry.coordinates);
  369. }
  370. }
  371.  
  372. let result = null;
  373. if (cutPolyGeoms.length === 1) {
  374. result = [turf.polygon(cutPolyGeoms[0])];
  375. } else if (cutPolyGeoms.length > 1) {
  376. result = cutPolyGeoms.map(geometry => turf.polygon(geometry));
  377. }
  378. return result;
  379. }
  380.  
  381. bootstrap();
  382. })();