/* eslint-disable max-len */
/* eslint-disable prefer-destructuring */
/* eslint-disable camelcase */
// ==UserScript==
// @name WME Split POI
// @namespace https://greasyfork.org/fr/scripts/13008-wme-split-poi
// @description Split POI with a new seg
// @description:fr Découpage d'un POI en deux en utisant un nouveau segment
// @include https://www.waze.com/editor*
// @include https://www.waze.com/*/editor*
// @include https://beta.waze.com/editor*
// @include https://beta.waze.com/*/editor*
// @exclude https://www.waze.com/user*
// @exclude https://www.waze.com/*/user*
// @require https://greasyfork.org/scripts/24851-wazewrap/code/WazeWrap.js
// @require https://cdn.jsdelivr.net/npm/@turf/turf@7/turf.min.js
// eslint-disable-next-line max-len
// @icon 
// @author seb-d59, WazeDev (2023-?)
// @version 2025.05.08.000
// @license GPLv3
// @grant GM_xmlhttpRequest
// @connect greasyfork.org
// ==/UserScript==
/* global WazeWrap */
/* global getWmeSdk */
/* global turf */
(function main() {
'use strict';
const DEBUG = false;
const SCRIPT_VERSION = GM_info.script.version;
const SCRIPT_NAME = GM_info.script.name;
const DOWNLOAD_URL = 'https://greasyfork.org/scripts/13008-wme-split-poi/code/WME%20Split%20POI.user.js';
const MINIMUM_AREA = 500.0;
let sdk;
function bootstrap() {
if (unsafeWindow.getWmeSdk && WazeWrap.Ready) {
initialize();
} else {
setTimeout(bootstrap, 100);
}
}
function getId(node) {
return document.getElementById(node);
}
function log(msg, obj) {
if (obj == null) {
console.log(`WME Split POI v${SCRIPT_VERSION} - ${msg}`);
} else if (DEBUG) {
console.debug(`WME Split POI v${SCRIPT_VERSION} - ${msg} `, obj);
}
}
function initialize() {
log('init');
sdk = getWmeSdk({ scriptId: 'wmeSplitPOI', scriptName: 'WME Split POI' });
startScriptUpdateMonitor();
initializeWazeObjects();
}
function startScriptUpdateMonitor() {
let updateMonitor;
try {
updateMonitor = new WazeWrap.Alerts.ScriptUpdateMonitor(SCRIPT_NAME, SCRIPT_VERSION, DOWNLOAD_URL, GM_xmlhttpRequest);
updateMonitor.start();
} catch (ex) {
// Report, but don't stop if ScriptUpdateMonitor fails.
console.error(`${SCRIPT_NAME}:`, ex);
}
}
function onSelectionChanged(tries = 0) {
if (tries === 30) return;
try {
const venue = getSelectedAreaVenue();
if (!venue) return;
// const landmarkPoi = '(NATURAL_FEATURES|ISLAND|SEA_LAKE_POOL|RIVER_STREAM|FOREST_GROVE|FARM|CANAL|SWAMP_MARSH|DAM|PARK)';
// if (new RegExp(landmarkPoi).test(attributes.categories) === false) return;
const editPanel = getId('edit-panel');
if (editPanel.firstElementChild.style.display === 'none') {
window.setTimeout(onSelectionChanged, 100, ++tries);
return;
}
// ok: 1 selected item and panel is shown
// On verifie que le segment est éditable
if (!sdk.DataModel.Venues.hasPermissions({ venueId: venue.id })) return;
// Exclude gas station and EVCS categories (don't ever want to delete those by splitting):
if (venue.categories.some(cat => ['GAS_STATION', 'CHARGING_STATION'].includes(cat))) return;
if (!$('#split-poi-button').length) {
let addAfter = true;
let insertAtElement = document.querySelector('.geometry-type-control-area');
if (!insertAtElement) {
insertAtElement = document.querySelector('.external-providers-control');
if (!insertAtElement) {
setTimeout(onSelectionChanged, 100, ++tries);
return;
}
addAfter = false;
}
const WMESP_Controle = document.createElement('wz-button');
WMESP_Controle.color = 'secondary';
WMESP_Controle.size = 'sm';
WMESP_Controle.id = 'split-poi-button';
WMESP_Controle.className = 'geometry-type-control-button geometry-type-control-point';
WMESP_Controle.innerHTML = '<i class="fa fa-cut" style="font-size:24px;" title="Split POI"></i>';
if (addAfter) {
insertAtElement.after(WMESP_Controle);
} else {
insertAtElement.before(WMESP_Controle);
}
WMESP_Controle.onclick = onSplitPoiButtonClick;
}
} catch (ex) {
console.error('Split POI:', ex);
}
}
function initializeWazeObjects() {
sdk.Events.on({
eventName: 'wme-selection-changed',
eventHandler: () => setTimeout(onSelectionChanged, 0)
});
// call OnSelectionChanged once to catch selected venue in PL
onSelectionChanged();
}
// This will return null if more than one object is selected
function getSelectedAreaVenue() {
const selection = sdk.Editing.getSelection();
if (selection?.ids.length !== 1 || selection.objectType !== 'venue') return null;
const venue = sdk.DataModel.Venues.getById({ venueId: selection.ids[0] });
if (venue.geometry.type !== 'Polygon') return null;
return venue;
}
// function cloneAttribute(venue, attrName, newAttributesObject) {
// if (venue.attributes.hasOwnProperty(attrName)) {
// let value = venue.attributes[attrName];
// if (Array.isArray(value)) {
// value = value.slice(0); // copy array
// }
// newAttributesObject[attrName] = venue.attributes[attrName];
// }
// }
function cloneVenue(venue, newGeometry) {
const cloneId = sdk.DataModel.Venues.addVenue({
// SDK: Update this if/when more attributes are available.
category: venue.categories[0],
geometry: newGeometry
}).toString(); // toString is needed because a string is expected later
const address = sdk.DataModel.Venues.getAddress({ venueId: venue.id });
sdk.DataModel.Venues.updateAddress({ venueId: cloneId, houseNumber: address.houseNumber, streetId: address.street?.id });
sdk.DataModel.Venues.updateVenue({
venueId: cloneId,
aliases: venue.aliases,
openingHours: venue.openingHours,
phone: venue.phone,
services: venue.services,
url: venue.url
});
// const clonePoi = new LandmarkVectorFeature({ geoJSONGeometry: W.userscripts.toGeoJSONGeometry(newGeometry) });
// [
// 'aliases',
// 'categories',
// 'description',
// 'entryExitPoints',
// 'externalProviderIDs',
// 'houseNumber',
// 'lockRank',
// 'name',
// 'openingHours',
// 'phone',
// 'services',
// 'streetID',
// 'url'
// ].forEach(attrName => cloneAttribute(poi, attrName, clonePoi.attributes));
// if (clonePoi.attributes.name) clonePoi.attributes.name += ` (copy ${nameSuffixIndex})`; // IMPORTANT! Won't save for some reason without changing the names (at least for PLAs).
// if (poi.attributes.categoryAttributes.PARKING_LOT) {
// clonePoi.attributes.categoryAttributes.PARKING_LOT = JSON.parse(JSON.stringify(poi.attributes.categoryAttributes.PARKING_LOT));
// }
// const WazeActionAddLandmark = require('Waze/Action/AddLandmark');
// actions.push(new WazeActionAddLandmark(clonePoi));
// const street = W.model.streets.getObjectById(poi.attributes.streetID);
// const streetName = street.attributes.name;
// const cityID = street.attributes.cityID;
// const city = W.model.cities.getObjectById(cityID);
// const stateID = city.attributes.stateID;
// const countryID = city.attributes.countryID;
// const houseNumber = poi.attributes.houseNumber;
// if (!street.attributes.isEmpty || !city.attributes.isEmpty) { // nok
// const newAtts = {
// emptyStreet: street.attributes.isEmpty, // TODO: fix this
// stateID,
// countryID,
// cityName: city.attributes.name,
// houseNumber,
// streetName,
// emptyCity: city.attributes.isEmpty // TODO: fix this
// };
// const updateAddressAction = new UpdateFeatureAddressAction(clonePoi, newAtts);
// updateAddressAction.options.updateHouseNumber = true;
// actions.push(updateAddressAction);
// }
}
// function confirmBeforeSplitting(venue) {
// // SDK: FR submitted to add venue attribues
// const entryExitPointsLen = venue.attributes.entryExitPoints?.length;
// const imagesLen = venue.attributes.images?.length;
// const extProvidersLen = venue.attributes.externalProviderIDs?.length;
// let warningText = 'WARNING: The original place will be deleted!';
// if (imagesLen) {
// warningText += '\n\nThe following property(s) will be lost:';
// if (imagesLen) warningText += `\n • ${imagesLen} photo${imagesLen === 1 ? '' : 's'} (permanently deleted after saving)`;
// }
// warningText += '\n\nThe following properties likely need to be changed after splitting:';
// warningText += '\n • name ("copy #" will be appended)';
// if (entryExitPointsLen) warningText += `\n • ${entryExitPointsLen} entry/exit point${entryExitPointsLen === 1 ? '' : 's'}`;
// if (extProvidersLen) warningText += `\n • ${extProvidersLen} linked Google place${extProvidersLen === 1 ? '' : 's'}`;
// warningText += '\n\nReview <i>all</i> properties of both new places before saving.';
// warningText += '\n';
// return new Promise(resolve => {
// WazeWrap.Alerts.confirm(
// SCRIPT_NAME,
// warningText,
// () => resolve(true),
// () => resolve(false)
// );
// });
// }
async function onDrawLineFinished(line, venue) {
// if (!await confirmBeforeSplitting(venue)) return;
const intersections = turf.lineIntersect(venue.geometry, line);
if (intersections.features.length === 0) {
WazeWrap.Alerts.error(SCRIPT_NAME, 'The cut line must intersect the place\'s geometry.');
return;
}
if (intersections.features.length % 2) {
WazeWrap.Alerts.error(SCRIPT_NAME, 'The cut line cannot begin or end inside the place\'s geometry and it cannot cross itself.');
return;
}
// const newPolygons = createTwoPolygonsFromIntersectPoints(venue, intersectPoints);
// if (newPolygons[0].getArea() < MINIMUM_AREA || newPolygons[1].getArea() < MINIMUM_AREA) {
// WazeWrap.Alerts.error(SCRIPT_NAME, 'New area place would be too small. Move the temporary road segment.');
// return;
// }
unsafeWindow.intersections = intersections;
unsafeWindow.line = line;
unsafeWindow.poly = venue.geometry;
console.log(intersections.features.length === 2);
const venuePolygon = venue.geometry;
const newPolygons = [];
const cutResults1 = cutPolygon(venuePolygon, line, 1);
if (cutResults1) {
newPolygons.push(...cutResults1);
}
const cutResults2 = cutPolygon(venuePolygon, line, -1);
if (cutResults2) {
newPolygons.push(...cutResults2);
}
if (newPolygons.some(poly => { console.log(turf.area(poly)); return turf.area(poly) < MINIMUM_AREA; })) {
WazeWrap.Alerts.error(SCRIPT_NAME, 'At least one of the new polygons would be too small to appear in the app.');
return;
}
let largest;
newPolygons.forEach(polygon => {
const area = turf.area(polygon);
if (!largest || area > largest.area) {
largest = { polygon, area };
}
});
newPolygons.forEach(polygon => {
if (polygon === largest.polygon) {
sdk.DataModel.Venues.updateVenue({ venueId: venue.id, geometry: polygon.geometry });
} else {
cloneVenue(venue, polygon.geometry);
}
});
}
function onSplitPoiButtonClick() {
const venue = getSelectedAreaVenue();
if (!venue) return;
// This is needed in case the category is changed to GS or EVCS and the split button is still there.
if (venue.categories.some(cat => ['GAS_STATION', 'CHARGING_STATION'].includes(cat))) {
WazeWrap.Alerts.error(SCRIPT_NAME, 'Cannot split gas stations or EV charging stations');
return;
}
sdk.Map.drawLine().then(line => {
onDrawLineFinished(line, venue);
// const confirm = await confirmBeforeSplitting(venue);
// if (confirm) {
// const actions = [];
// addClonePoiAction(venue, newPolygons[0], 1, actions);
// addClonePoiAction(venue, newPolygons[1], 2, actions);
// actions.push(new DeleteSegmentAction(seg));
// const multiaction = new MultiAction(actions, { description: 'Split POI' });
// W.model.actionManager.add(multiaction);
// }
}).catch(ex => {
if (ex instanceof sdk.Errors.InvalidStateError) {
// log, but ignore it
console.log(ex);
} else {
console.error(ex);
}
});
}
function cutPolygon(polygon, cutLine, direction) {
let j;
const polyCoords = [];
const cutPolyGeoms = [];
if ((polygon.type !== 'Polygon') || (cutLine.type !== 'LineString')) return null;
const intersectPoints = turf.lineIntersect(polygon, cutLine);
const nPoints = intersectPoints.features.length;
if ((nPoints === 0) || ((nPoints % 2) !== 0)) return null;
const offsetLine = turf.lineOffset(cutLine, (0.01 * direction), { units: 'kilometers' });
for (j = 0; j < cutLine.coordinates.length; j++) {
polyCoords.push(cutLine.coordinates[j]);
}
for (j = (offsetLine.geometry.coordinates.length - 1); j >= 0; j--) {
polyCoords.push(offsetLine.geometry.coordinates[j]);
}
polyCoords.push(cutLine.coordinates[0]);
const thickLineString = turf.lineString(polyCoords);
const thickLinePolygon = turf.lineToPolygon(thickLineString);
polygon = turf.feature(polygon);
const clipped = turf.difference(turf.featureCollection([polygon, thickLinePolygon]));
for (j = 0; j < clipped.geometry.coordinates.length; j++) {
const polyg = turf.polygon(clipped.geometry.coordinates[j]);
const overlap = turf.lineOverlap(polyg, cutLine, { tolerance: 0.005 });
if (overlap.features.length > 0) {
cutPolyGeoms.push(polyg.geometry.coordinates);
}
}
let result = null;
if (cutPolyGeoms.length === 1) {
result = [turf.polygon(cutPolyGeoms[0])];
} else if (cutPolyGeoms.length > 1) {
result = cutPolyGeoms.map(geometry => turf.polygon(geometry));
}
return result;
}
bootstrap();
})();