WME Split POI

Split POI with a new seg

当前为 2025-02-19 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

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

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

/* 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
// eslint-disable-next-line max-len
// @icon            
// @author          seb-d59, WazeDev (2023-?)
// @version         2025-02-19-000
// @license         GPLv3
// @grant           GM_xmlhttpRequest
// @connect         greasyfork.org
// ==/UserScript==

/* global W */
/* global OpenLayers */
/* global WazeWrap */

(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 = 100.0;

    let LandmarkVectorFeature;
    let DeleteObjectAction;
    let DeleteSegmentAction;
    let UpdateFeatureAddressAction;
    let MultiAction;

    function bootstrap() {
        if (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');
        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 initializeWazeObjects() {
        DeleteObjectAction = require('Waze/Action/DeleteObject');
        DeleteSegmentAction = require('Waze/Action/DeleteSegment');
        LandmarkVectorFeature = require('Waze/Feature/Vector/Landmark');
        UpdateFeatureAddressAction = require('Waze/Action/UpdateFeatureAddress');
        MultiAction = require('Waze/Action/MultiAction');
        W.selectionManager.events.register('selectionchanged', null, onSelectionChanged);
    }

    function onSelectionChanged() {
        try {
            if (W.selectionManager.getSelectedDataModelObjects().length !== 1) return;

            const selectedObject = W.selectionManager.getSelectedDataModelObjects()[0];
            if (selectedObject.type !== 'venue' || selectedObject.isPoint()) 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;

            log('selectionManager', W.selectionManager);

            const editPanel = getId('edit-panel');
            if (editPanel.firstElementChild.style.display === 'none') {
                window.setTimeout(onSelectionChanged, 100);
            }

            // ok: 1 selected item and pannel is shown

            // On verifie que le segment est éditable
            if (!objIsEditable(selectedObject)) return;

            // Exclude gas station and EVCS categories (don't ever want to delete those by splitting):
            if (selectedObject.attributes.categories.some(cat => ['GAS_STATION', 'CHARGING_STATION'].includes(cat))) return;

            if (selectedObject.type === 'venue' && !$('#split-poi-button').length) {
                let $btnHandle = $('.geometry-type-control-area')[0];
                if (!$btnHandle) {
                    $btnHandle = $('.external-providers-control')[0];
                }
                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>';

                $btnHandle.after(WMESP_Controle);

                WMESP_Controle.onclick = onSplitPoiButtonClick;
            }
        } catch (ex) {
            console.error('Split POI:', ex);
        }
    }

    function onScreen(obj) {
        if (obj.geometry) {
            return (W.map.getExtent().intersectsBounds(obj.geometry.getBounds()));
        }
        return false;
    }

    function objIsEditable(obj) {
        if (obj == null) return false;
        if (W.loginManager.user.isCountryManager()) return true;
        if (obj.attributes.permissions === 0) return false;

        return true;
    }

    // This will return null if more than one object is selected
    function getSelectedAreaPlace() {
        const selectedObjects = W.selectionManager.getSelectedDataModelObjects();
        if (selectedObjects.length > 1) return null;
        const object = selectedObjects[0];
        if (object.type !== 'venue' || object.isPoint()) return null;
        return object;
    }

    function getNewestUnconnectedOnScreenSegment() {
        const newSegs = W.model.segments.getObjectArray(seg => seg.isNew());
        let newestSeg;
        let newestId = 0;
        newSegs.forEach(seg => {
            const hasConnections = seg.getToNode().getSegmentIds().length > 1 || seg.getFromNode().getSegmentIds().length > 1;
            if (seg.getID() < newestId && onScreen(seg) && !hasConnections) {
                newestSeg = seg;
                newestId = seg.getID();
            }
        });
        return newestSeg;
    }

    function getPoiAndSegIntersectionPoints(poi, seg) {
        function clearComponent(geometry) {
            geometry.removeComponent(0);
            geometry.removeComponent(1);
        }

        function copyComponent(sourceGeometry, sourceIndex, targetGeometry) {
            targetGeometry.components[0] = sourceGeometry.components[sourceIndex].clone();
            targetGeometry.components[1] = sourceGeometry.components[sourceIndex + 1].clone();
        }

        const poiAttr = poi.attributes;
        const poiGeo = poiAttr.geometry.clone();
        const poiLineString = poiGeo.components[0].clone();
        const segLineString = seg.attributes.geometry.clone();

        const intersectPoint = [];
        const poiLine = new OpenLayers.Geometry.LinearRing();
        const segLine = new OpenLayers.Geometry.LinearRing();

        // Calcul des point d'intersection seg // poi
        for (let n = 0; n < poiLineString.components.length - 1; n++) {
            copyComponent(poiLineString, n, poiLine);
            for (let m = 0; m < segLineString.components.length - 1; m++) {
                copyComponent(segLineString, m, segLine);
                if (poiLine.intersects(segLine)) {
                    intersectPoint.push({ index: n, intersect: intersection(poiLine, segLine) });
                }
                clearComponent(segLine);
            }
            clearComponent(poiLine);
        }

        return intersectPoint;
    }

    function createTwoPolygonsFromIntersectPoints(poi, intersectPoints) {
        const poiLineString = poi.attributes.geometry.components[0].clone();
        // intégration des points au contour du POI avec memo du nouvel index
        let i = 1;
        for (let n = 0; n < intersectPoints.length; n++) {
            const point = intersectPoints[n].intersect;
            const index = intersectPoints[n].index + i;
            poiLineString.addComponent(point, index);
            intersectPoints[n].newIndex = index;
            i++;
        }

        // création des deux nouvelles géométries
        const lineString1 = [];
        const lineString2 = [];

        const index1 = intersectPoints[0].newIndex;
        const index2 = intersectPoints[1].newIndex;

        for (let n = 0; n < poiLineString.components.length; n++) {
            const x = poiLineString.components[n].x;
            const y = poiLineString.components[n].y;
            const point = new OpenLayers.Geometry.Point(x, y);

            if (n < index1) {
                lineString1.push(point);
            } else if (n === index1) {
                lineString1.push(point);
                lineString2.push(point.clone());
            } else if ((index1 < n) && (n < index2)) {
                lineString2.push(point);
            } else if (n === index2) {
                lineString1.push(point);
                lineString2.push(point.clone());
            } else if (index2 < n) {
                lineString1.push(point);
            }
        }

        return [
            new OpenLayers.Geometry.Polygon(new OpenLayers.Geometry.LinearRing(lineString1)),
            new OpenLayers.Geometry.Polygon(new OpenLayers.Geometry.LinearRing(lineString2))
        ];
    }

    function cloneAttribute(poi, attrName, newAttributesObject) {
        if (poi.attributes.hasOwnProperty(attrName)) {
            let value = poi.attributes[attrName];

            if (Array.isArray(value)) {
                value = value.slice(0); // copy array
            }
            newAttributesObject[attrName] = poi.attributes[attrName];
        }
    }

    function addClonePoiAction(poi, newGeometry, nameSuffixIndex, actions) {
        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(poi) {
        const entryExitPointsLen = poi.attributes.entryExitPoints?.length;
        const imagesLen = poi.attributes.images?.length;
        const extProvidersLen = poi.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 onSplitPoiButtonClick() {
        const poi = getSelectedAreaPlace();
        if (!poi) return;

        // This is needed in case the category is changed to GS or EVCS and the split button is still there.
        if (poi.attributes.categories.some(cat => ['GAS_STATION', 'CHARGING_STATION'].includes(cat))) {
            WazeWrap.Alerts.error(SCRIPT_NAME, 'Cannot split gas stations or EV charging stations');
            return;
        }

        const seg = getNewestUnconnectedOnScreenSegment();
        if (!seg) {
            WazeWrap.Alerts.error(SCRIPT_NAME, 'Create a temporary unconnected road segment through the area place first.');
            return;
        }

        if (seg.geometry.components.some(pt => poi.geometry.containsPoint(pt))) {
            WazeWrap.Alerts.error(SCRIPT_NAME, 'The splitting road segment must be straight (no geometry handles within the POI).');
            return;
        }

        const intersectPoints = getPoiAndSegIntersectionPoints(poi, seg);
        if (intersectPoints.length !== 2) {
            WazeWrap.Alerts.error(SCRIPT_NAME, 'The temporary road segment must intersect the area place boundary at two points.');
            return;
        }

        const newPolygons = createTwoPolygonsFromIntersectPoints(poi, 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;
        }

        const confirm = await confirmBeforeSplitting(poi);
        if (confirm) {
            const actions = [];
            addClonePoiAction(poi, newPolygons[0], 1, actions);
            addClonePoiAction(poi, newPolygons[1], 2, actions);
            actions.push(new DeleteObjectAction(poi, null));
            actions.push(new DeleteSegmentAction(seg));
            const multiaction = new MultiAction(actions, { description: 'Split POI' });
            W.model.actionManager.add(multiaction);
        }
    }

    function intersection(D1, D2) {
        // let a, b, c, d, x, y;
        // const seg = {}; // {x1, y1, x2, y2};
        const seg1 = {}; // {x1, y1, x2, y2};
        const seg2 = {}; // {x1, y1, x2, y2};
        const options = {};
        options.point = true;

        if (D1.components[0].x <= D1.components[1].x) {
            seg1.x1 = D1.components[0].x;
            seg1.y1 = D1.components[0].y;
            seg1.x2 = D1.components[1].x;
            seg1.y2 = D1.components[1].y;
        } else if (D1.components[0].x > D1.components[1].x) {
            seg1.x1 = D1.components[1].x;
            seg1.y1 = D1.components[1].y;
            seg1.x2 = D1.components[0].x;
            seg1.y2 = D1.components[0].y;
        }

        if (D2.components[0].x <= D2.components[1].x) {
            seg2.x1 = D2.components[0].x;
            seg2.y1 = D2.components[0].y;
            seg2.x2 = D2.components[1].x;
            seg2.y2 = D2.components[1].y;
        } else if (D2.components[0].x > D2.components[1].x) {
            seg2.x1 = D2.components[1].x;
            seg2.y1 = D2.components[1].y;
            seg2.x2 = D2.components[0].x;
            seg2.y2 = D2.components[0].y;
        }
        return OpenLayers.Geometry.segmentsIntersect(seg1, seg2, options);
    }

    bootstrap();
})();