WME Split POI

Split POI with a new seg

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

/* 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();
})();