WME Split POI

Split POI with a new seg

目前為 2025-05-08 提交的版本,檢視 最新版本

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