您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Split POI with a new seg
- /* 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();
- })();