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://gf.qytechs.cn/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://gf.qytechs.cn/scripts/24851-wazewrap/code/WazeWrap.js
  16. // eslint-disable-next-line max-len
  17. // @icon 
  18. // @author seb-d59, WazeDev (2023-?)
  19. // @version 2025-02-19-000
  20. // @license GPLv3
  21. // @grant GM_xmlhttpRequest
  22. // @connect gf.qytechs.cn
  23. // ==/UserScript==
  24.  
  25. /* global W */
  26. /* global OpenLayers */
  27. /* global WazeWrap */
  28.  
  29. (function main() {
  30. 'use strict';
  31.  
  32. const DEBUG = false;
  33. const SCRIPT_VERSION = GM_info.script.version;
  34. const SCRIPT_NAME = GM_info.script.name;
  35. const DOWNLOAD_URL = 'https://gf.qytechs.cn/scripts/13008-wme-split-poi/code/WME%20Split%20POI.user.js';
  36. const MINIMUM_AREA = 100.0;
  37.  
  38. let LandmarkVectorFeature;
  39. let DeleteObjectAction;
  40. let DeleteSegmentAction;
  41. let UpdateFeatureAddressAction;
  42. let MultiAction;
  43.  
  44. function bootstrap() {
  45. if (WazeWrap.Ready) {
  46. initialize();
  47. } else {
  48. setTimeout(bootstrap, 100);
  49. }
  50. }
  51.  
  52. function getId(node) {
  53. return document.getElementById(node);
  54. }
  55.  
  56. function log(msg, obj) {
  57. if (obj == null) {
  58. console.log(`WME Split POI v${SCRIPT_VERSION} - ${msg}`);
  59. } else if (DEBUG) {
  60. console.debug(`WME Split POI v${SCRIPT_VERSION} - ${msg} `, obj);
  61. }
  62. }
  63.  
  64. function initialize() {
  65. log('init');
  66. startScriptUpdateMonitor();
  67. initializeWazeObjects();
  68. }
  69.  
  70. function startScriptUpdateMonitor() {
  71. let updateMonitor;
  72. try {
  73. updateMonitor = new WazeWrap.Alerts.ScriptUpdateMonitor(SCRIPT_NAME, SCRIPT_VERSION, DOWNLOAD_URL, GM_xmlhttpRequest);
  74. updateMonitor.start();
  75. } catch (ex) {
  76. // Report, but don't stop if ScriptUpdateMonitor fails.
  77. console.error(`${SCRIPT_NAME}:`, ex);
  78. }
  79. }
  80.  
  81. function initializeWazeObjects() {
  82. DeleteObjectAction = require('Waze/Action/DeleteObject');
  83. DeleteSegmentAction = require('Waze/Action/DeleteSegment');
  84. LandmarkVectorFeature = require('Waze/Feature/Vector/Landmark');
  85. UpdateFeatureAddressAction = require('Waze/Action/UpdateFeatureAddress');
  86. MultiAction = require('Waze/Action/MultiAction');
  87. W.selectionManager.events.register('selectionchanged', null, onSelectionChanged);
  88. }
  89.  
  90. function onSelectionChanged() {
  91. try {
  92. if (W.selectionManager.getSelectedDataModelObjects().length !== 1) return;
  93.  
  94. const selectedObject = W.selectionManager.getSelectedDataModelObjects()[0];
  95. if (selectedObject.type !== 'venue' || selectedObject.isPoint()) return;
  96.  
  97. // const landmarkPoi = '(NATURAL_FEATURES|ISLAND|SEA_LAKE_POOL|RIVER_STREAM|FOREST_GROVE|FARM|CANAL|SWAMP_MARSH|DAM|PARK)';
  98. // if (new RegExp(landmarkPoi).test(attributes.categories) === false) return;
  99.  
  100. log('selectionManager', W.selectionManager);
  101.  
  102. const editPanel = getId('edit-panel');
  103. if (editPanel.firstElementChild.style.display === 'none') {
  104. window.setTimeout(onSelectionChanged, 100);
  105. }
  106.  
  107. // ok: 1 selected item and pannel is shown
  108.  
  109. // On verifie que le segment est éditable
  110. if (!objIsEditable(selectedObject)) return;
  111.  
  112. // Exclude gas station and EVCS categories (don't ever want to delete those by splitting):
  113. if (selectedObject.attributes.categories.some(cat => ['GAS_STATION', 'CHARGING_STATION'].includes(cat))) return;
  114.  
  115. if (selectedObject.type === 'venue' && !$('#split-poi-button').length) {
  116. let $btnHandle = $('.geometry-type-control-area')[0];
  117. if (!$btnHandle) {
  118. $btnHandle = $('.external-providers-control')[0];
  119. }
  120. const WMESP_Controle = document.createElement('wz-button');
  121. WMESP_Controle.color = 'secondary';
  122. WMESP_Controle.size = 'sm';
  123. WMESP_Controle.id = 'split-poi-button';
  124. WMESP_Controle.className = 'geometry-type-control-button geometry-type-control-point';
  125. WMESP_Controle.innerHTML = '<i class="fa fa-cut" style="font-size:24px;" title="Split POI"></i>';
  126.  
  127. $btnHandle.after(WMESP_Controle);
  128.  
  129. WMESP_Controle.onclick = onSplitPoiButtonClick;
  130. }
  131. } catch (ex) {
  132. console.error('Split POI:', ex);
  133. }
  134. }
  135.  
  136. function onScreen(obj) {
  137. if (obj.geometry) {
  138. return (W.map.getExtent().intersectsBounds(obj.geometry.getBounds()));
  139. }
  140. return false;
  141. }
  142.  
  143. function objIsEditable(obj) {
  144. if (obj == null) return false;
  145. if (W.loginManager.user.isCountryManager()) return true;
  146. if (obj.attributes.permissions === 0) return false;
  147.  
  148. return true;
  149. }
  150.  
  151. // This will return null if more than one object is selected
  152. function getSelectedAreaPlace() {
  153. const selectedObjects = W.selectionManager.getSelectedDataModelObjects();
  154. if (selectedObjects.length > 1) return null;
  155. const object = selectedObjects[0];
  156. if (object.type !== 'venue' || object.isPoint()) return null;
  157. return object;
  158. }
  159.  
  160. function getNewestUnconnectedOnScreenSegment() {
  161. const newSegs = W.model.segments.getObjectArray(seg => seg.isNew());
  162. let newestSeg;
  163. let newestId = 0;
  164. newSegs.forEach(seg => {
  165. const hasConnections = seg.getToNode().getSegmentIds().length > 1 || seg.getFromNode().getSegmentIds().length > 1;
  166. if (seg.getID() < newestId && onScreen(seg) && !hasConnections) {
  167. newestSeg = seg;
  168. newestId = seg.getID();
  169. }
  170. });
  171. return newestSeg;
  172. }
  173.  
  174. function getPoiAndSegIntersectionPoints(poi, seg) {
  175. function clearComponent(geometry) {
  176. geometry.removeComponent(0);
  177. geometry.removeComponent(1);
  178. }
  179.  
  180. function copyComponent(sourceGeometry, sourceIndex, targetGeometry) {
  181. targetGeometry.components[0] = sourceGeometry.components[sourceIndex].clone();
  182. targetGeometry.components[1] = sourceGeometry.components[sourceIndex + 1].clone();
  183. }
  184.  
  185. const poiAttr = poi.attributes;
  186. const poiGeo = poiAttr.geometry.clone();
  187. const poiLineString = poiGeo.components[0].clone();
  188. const segLineString = seg.attributes.geometry.clone();
  189.  
  190. const intersectPoint = [];
  191. const poiLine = new OpenLayers.Geometry.LinearRing();
  192. const segLine = new OpenLayers.Geometry.LinearRing();
  193.  
  194. // Calcul des point d'intersection seg // poi
  195. for (let n = 0; n < poiLineString.components.length - 1; n++) {
  196. copyComponent(poiLineString, n, poiLine);
  197. for (let m = 0; m < segLineString.components.length - 1; m++) {
  198. copyComponent(segLineString, m, segLine);
  199. if (poiLine.intersects(segLine)) {
  200. intersectPoint.push({ index: n, intersect: intersection(poiLine, segLine) });
  201. }
  202. clearComponent(segLine);
  203. }
  204. clearComponent(poiLine);
  205. }
  206.  
  207. return intersectPoint;
  208. }
  209.  
  210. function createTwoPolygonsFromIntersectPoints(poi, intersectPoints) {
  211. const poiLineString = poi.attributes.geometry.components[0].clone();
  212. // intégration des points au contour du POI avec memo du nouvel index
  213. let i = 1;
  214. for (let n = 0; n < intersectPoints.length; n++) {
  215. const point = intersectPoints[n].intersect;
  216. const index = intersectPoints[n].index + i;
  217. poiLineString.addComponent(point, index);
  218. intersectPoints[n].newIndex = index;
  219. i++;
  220. }
  221.  
  222. // création des deux nouvelles géométries
  223. const lineString1 = [];
  224. const lineString2 = [];
  225.  
  226. const index1 = intersectPoints[0].newIndex;
  227. const index2 = intersectPoints[1].newIndex;
  228.  
  229. for (let n = 0; n < poiLineString.components.length; n++) {
  230. const x = poiLineString.components[n].x;
  231. const y = poiLineString.components[n].y;
  232. const point = new OpenLayers.Geometry.Point(x, y);
  233.  
  234. if (n < index1) {
  235. lineString1.push(point);
  236. } else if (n === index1) {
  237. lineString1.push(point);
  238. lineString2.push(point.clone());
  239. } else if ((index1 < n) && (n < index2)) {
  240. lineString2.push(point);
  241. } else if (n === index2) {
  242. lineString1.push(point);
  243. lineString2.push(point.clone());
  244. } else if (index2 < n) {
  245. lineString1.push(point);
  246. }
  247. }
  248.  
  249. return [
  250. new OpenLayers.Geometry.Polygon(new OpenLayers.Geometry.LinearRing(lineString1)),
  251. new OpenLayers.Geometry.Polygon(new OpenLayers.Geometry.LinearRing(lineString2))
  252. ];
  253. }
  254.  
  255. function cloneAttribute(poi, attrName, newAttributesObject) {
  256. if (poi.attributes.hasOwnProperty(attrName)) {
  257. let value = poi.attributes[attrName];
  258.  
  259. if (Array.isArray(value)) {
  260. value = value.slice(0); // copy array
  261. }
  262. newAttributesObject[attrName] = poi.attributes[attrName];
  263. }
  264. }
  265.  
  266. function addClonePoiAction(poi, newGeometry, nameSuffixIndex, actions) {
  267. const clonePoi = new LandmarkVectorFeature({ geoJSONGeometry: W.userscripts.toGeoJSONGeometry(newGeometry) });
  268. [
  269. 'aliases',
  270. 'categories',
  271. 'description',
  272. 'entryExitPoints',
  273. 'externalProviderIDs',
  274. 'houseNumber',
  275. 'lockRank',
  276. 'name',
  277. 'openingHours',
  278. 'phone',
  279. 'services',
  280. 'streetID',
  281. 'url'
  282. ].forEach(attrName => cloneAttribute(poi, attrName, clonePoi.attributes));
  283. if (clonePoi.attributes.name) clonePoi.attributes.name += ` (copy ${nameSuffixIndex})`; // IMPORTANT! Won't save for some reason without changing the names (at least for PLAs).
  284. if (poi.attributes.categoryAttributes.PARKING_LOT) {
  285. clonePoi.attributes.categoryAttributes.PARKING_LOT = JSON.parse(JSON.stringify(poi.attributes.categoryAttributes.PARKING_LOT));
  286. }
  287.  
  288. const WazeActionAddLandmark = require('Waze/Action/AddLandmark');
  289. actions.push(new WazeActionAddLandmark(clonePoi));
  290.  
  291. const street = W.model.streets.getObjectById(poi.attributes.streetID);
  292. const streetName = street.attributes.name;
  293. const cityID = street.attributes.cityID;
  294. const city = W.model.cities.getObjectById(cityID);
  295. const stateID = city.attributes.stateID;
  296. const countryID = city.attributes.countryID;
  297. const houseNumber = poi.attributes.houseNumber;
  298. if (!street.attributes.isEmpty || !city.attributes.isEmpty) { // nok
  299. const newAtts = {
  300. emptyStreet: street.attributes.isEmpty, // TODO: fix this
  301. stateID,
  302. countryID,
  303. cityName: city.attributes.name,
  304. houseNumber,
  305. streetName,
  306. emptyCity: city.attributes.isEmpty // TODO: fix this
  307. };
  308. const updateAddressAction = new UpdateFeatureAddressAction(clonePoi, newAtts);
  309. updateAddressAction.options.updateHouseNumber = true;
  310. actions.push(updateAddressAction);
  311. }
  312. }
  313.  
  314. function confirmBeforeSplitting(poi) {
  315. const entryExitPointsLen = poi.attributes.entryExitPoints?.length;
  316. const imagesLen = poi.attributes.images?.length;
  317. const extProvidersLen = poi.attributes.externalProviderIDs?.length;
  318. let warningText = 'WARNING: The original place will be deleted!';
  319.  
  320. if (imagesLen) {
  321. warningText += '\n\nThe following property(s) will be lost:';
  322. if (imagesLen) warningText += `\n ${imagesLen} photo${imagesLen === 1 ? '' : 's'} (permanently deleted after saving)`;
  323. }
  324. warningText += '\n\nThe following properties likely need to be changed after splitting:';
  325. warningText += '\n • name ("copy #" will be appended)';
  326. if (entryExitPointsLen) warningText += `\n ${entryExitPointsLen} entry/exit point${entryExitPointsLen === 1 ? '' : 's'}`;
  327. if (extProvidersLen) warningText += `\n ${extProvidersLen} linked Google place${extProvidersLen === 1 ? '' : 's'}`;
  328. warningText += '\n\nReview <i>all</i> properties of both new places before saving.';
  329. warningText += '\n';
  330. return new Promise(resolve => {
  331. WazeWrap.Alerts.confirm(
  332. SCRIPT_NAME,
  333. warningText,
  334. () => resolve(true),
  335. () => resolve(false)
  336. );
  337. });
  338. }
  339.  
  340. async function onSplitPoiButtonClick() {
  341. const poi = getSelectedAreaPlace();
  342. if (!poi) return;
  343.  
  344. // This is needed in case the category is changed to GS or EVCS and the split button is still there.
  345. if (poi.attributes.categories.some(cat => ['GAS_STATION', 'CHARGING_STATION'].includes(cat))) {
  346. WazeWrap.Alerts.error(SCRIPT_NAME, 'Cannot split gas stations or EV charging stations');
  347. return;
  348. }
  349.  
  350. const seg = getNewestUnconnectedOnScreenSegment();
  351. if (!seg) {
  352. WazeWrap.Alerts.error(SCRIPT_NAME, 'Create a temporary unconnected road segment through the area place first.');
  353. return;
  354. }
  355.  
  356. if (seg.geometry.components.some(pt => poi.geometry.containsPoint(pt))) {
  357. WazeWrap.Alerts.error(SCRIPT_NAME, 'The splitting road segment must be straight (no geometry handles within the POI).');
  358. return;
  359. }
  360.  
  361. const intersectPoints = getPoiAndSegIntersectionPoints(poi, seg);
  362. if (intersectPoints.length !== 2) {
  363. WazeWrap.Alerts.error(SCRIPT_NAME, 'The temporary road segment must intersect the area place boundary at two points.');
  364. return;
  365. }
  366.  
  367. const newPolygons = createTwoPolygonsFromIntersectPoints(poi, intersectPoints);
  368. if (newPolygons[0].getArea() < MINIMUM_AREA || newPolygons[1].getArea() < MINIMUM_AREA) {
  369. WazeWrap.Alerts.error(SCRIPT_NAME, 'New area place would be too small. Move the temporary road segment.');
  370. return;
  371. }
  372.  
  373. const confirm = await confirmBeforeSplitting(poi);
  374. if (confirm) {
  375. const actions = [];
  376. addClonePoiAction(poi, newPolygons[0], 1, actions);
  377. addClonePoiAction(poi, newPolygons[1], 2, actions);
  378. actions.push(new DeleteObjectAction(poi, null));
  379. actions.push(new DeleteSegmentAction(seg));
  380. const multiaction = new MultiAction(actions, { description: 'Split POI' });
  381. W.model.actionManager.add(multiaction);
  382. }
  383. }
  384.  
  385. function intersection(D1, D2) {
  386. // let a, b, c, d, x, y;
  387. // const seg = {}; // {x1, y1, x2, y2};
  388. const seg1 = {}; // {x1, y1, x2, y2};
  389. const seg2 = {}; // {x1, y1, x2, y2};
  390. const options = {};
  391. options.point = true;
  392.  
  393. if (D1.components[0].x <= D1.components[1].x) {
  394. seg1.x1 = D1.components[0].x;
  395. seg1.y1 = D1.components[0].y;
  396. seg1.x2 = D1.components[1].x;
  397. seg1.y2 = D1.components[1].y;
  398. } else if (D1.components[0].x > D1.components[1].x) {
  399. seg1.x1 = D1.components[1].x;
  400. seg1.y1 = D1.components[1].y;
  401. seg1.x2 = D1.components[0].x;
  402. seg1.y2 = D1.components[0].y;
  403. }
  404.  
  405. if (D2.components[0].x <= D2.components[1].x) {
  406. seg2.x1 = D2.components[0].x;
  407. seg2.y1 = D2.components[0].y;
  408. seg2.x2 = D2.components[1].x;
  409. seg2.y2 = D2.components[1].y;
  410. } else if (D2.components[0].x > D2.components[1].x) {
  411. seg2.x1 = D2.components[1].x;
  412. seg2.y1 = D2.components[1].y;
  413. seg2.x2 = D2.components[0].x;
  414. seg2.y2 = D2.components[0].y;
  415. }
  416. return OpenLayers.Geometry.segmentsIntersect(seg1, seg2, options);
  417. }
  418.  
  419. bootstrap();
  420. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址