WME Split POI (Mod)

Split POI with a new seg. Temporary patch to fix recent incompatibility issues with WME, and added the Other category.

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name            WME Split POI (Mod)
// @namespace	      https://greasyfork.org/en/scripts/13008-wme-split-poi/
// @description	    Split POI with a new seg. Temporary patch to fix recent incompatibility issues with WME, and added the Other category.
// @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*
// @icon            
// @version	    2023-07-14-001
// @license         GPLv3
// @grant           none
// ==/UserScript==

// Based on https://greasyfork.org/en/scripts/13008-wme-split-poi/
// Updated by Frantzisko

var debug=false;
var WMESP_Version = GM_info.script.version;
var WMESP_OldVersion = WMESP_Version;

var WMESP_Maj = {
    fr: "Mise à jour WME Split POI: v" + WMESP_Version + "\nCompatibilité New WME",
    en: "Update WME Split POI: v" + WMESP_Version + "\nCompatibility New WME"
};

/* bootstrap, will call initialize() */
function WMESP_bootstrap(){
    log('Init');
    /* begin running the code! */
    setTimeout(initialize, 1000);
}

//==========  Helper ==============================//
function getElementsByClassName(classname, node) {
    if(!node) node = document.getElementsByTagName("body")[0];
    var a = [];
    var re = new RegExp('\\b' + classname + '\\b');
    var els = node.getElementsByTagName("*");
    for (var i=0,j=els.length; i<j; i++)
        if (re.test(els[i].className)) a.push(els[i]);
    return a;
}

function getId(node) {
    return document.getElementById(node);
}

function log(msg, obj)
{
    if (obj==null)
        console.log ("WME Split POI v" + WMESP_Version + " - " + msg);
    else if (debug)
        console.debug("WME Split POI v" + WMESP_Version + " - " + msg + " " ,obj);

}
function IsJsonString(str) {
    try {
        JSON.parse(str);
    } catch (e) {
        return false;
    }
    return true;
}

function cloneObj(obj){
    var copy = JSON.parse(JSON.stringify(obj));
    return copy;
}

//==========  /Helper ==============================//

function WMESP_TestVersion() {

    if (typeof(localStorage.WMESPVersion) !== "undefined" && IsJsonString(localStorage.getItem('WMESPVersion'))) {
        WMESP_OldVersion=JSON.parse(localStorage.WMESPVersion);
    }else WMESP_OldVersion = "1.1";

    var locale = navigator.language.match(/fr|en/);
    var WMESPMaj = "";

    if(locale != null){
        switch(locale[0]) {
            case "fr":
                WMESPMaj=WMESP_Maj.fr;
                break;
            case "en":
                WMESPMaj=WMESP_Maj.en;
                break;
        }
    }else if(locale == null){
        WMESPMaj=WMESP_Maj.en;
    }
    log('WMESP_OldVersion ='+WMESP_OldVersion+'; WMESP_Version ='+WMESP_Version);
    if (WMESP_OldVersion != WMESP_Version) {
        alert(WMESPMaj);
        WMESP_OldVersion = WMESP_Version;
    }
    localStorage.setItem('WMESPVersion', JSON.stringify(WMESP_Version));


}

function initialize()
{
    log("init");
        if (typeof (W) === 'undefined') { setTimeout(initialize, 500); return; }
        if (typeof (W.model) == 'undefined') { setTimeout(initialize, 500);  return; }
        if (typeof (W.map) == 'undefined') { setTimeout(initialize, 500); return; }
        if (typeof (OL) === 'undefined') { setTimeout(initialize, 500); return; }
        if (W.loginManager.user == null) { setTimeout(initialize, 500); return; }
    initializeWazeObjects();

}


function initializeWazeObjects()
{
    initializeWazeUI();
}

function initializeWazeUI()
{

    var userInfo = getId('user-info');
    if (userInfo==null)
    {
        window.setTimeout(initializeWazeUI, 500);
        return;
    }

    var navTabs=userInfo.getElementsByTagName('ul');
    if (navTabs.length==0)
    {
        window.setTimeout(initializeWazeUI, 500);
        return;
    }
    if (typeof(navTabs[0])==='undefined')
    {
        window.setTimeout(initializeWazeUI, 500);
        return;
    }

    var tabContents=userInfo.getElementsByTagName('div');
    if (tabContents.length==0)
    {
        window.setTimeout(initializeWazeUI, 500);
        return;
    }
    if (typeof(tabContents[0])==='undefined')
    {
        window.setTimeout(initializeWazeUI, 500);
        return;
    }

    WMESP_TestVersion();

    W.selectionManager.events.register("selectionchanged", null, WMESP_newSelectionAvailable);

    log("init done.");
}

function WMESP_newSelectionAvailable()
{
    if (W.selectionManager.getSelectedFeatures().length !=1 )
        return;

    var selectedObject = W.selectionManager.getSelectedFeatures()[0].attributes;
    selectedObject = (selectedObject.repositoryObject) ? selectedObject.repositoryObject : selectedObject.wazeFeature._wmeObject;

    if (selectedObject.type!="venue")
        return;

    var attributes = selectedObject.attributes;

    if (!attributes.geometry.hasOwnProperty("components"))
        return;

    var landmarkPoi="(NATURAL_FEATURES|ISLAND|SEA_LAKE_POOL|RIVER_STREAM|FOREST_GROVE|FARM|CANAL|SWAMP_MARSH|DAM|PARK|OTHER)"
    if (new RegExp(landmarkPoi).test(attributes.categories) === false)
        return;

    log("selectionManager",W.selectionManager);

    var editPanel=getId('edit-panel');
    if (editPanel.firstElementChild.style.display=='none')
        window.setTimeout(WMESP_newSelectionAvailable, 100);

    // ok: 1 selected item and pannel is shown

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

    if (selectedObject.type=="venue")
    {
        var btnHandle = getElementsByClassName("geometry-type-control-area")[0].parentNode;
        var WMESP_Controle=document.createElement('wz-button');
        WMESP_Controle.color='secondary';
        WMESP_Controle.size='sm';
        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.insertBefore(WMESP_Controle, btnHandle.nextSibling);
        WMESP_Controle.onclick=SplitPOI;
    }
}


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;
}

function SplitPOI()
{

    if (W.selectionManager.getSelectedFeatures().length !=1 )
        return;

    var poi = W.selectionManager.getSelectedFeatures()[0].attributes.wazeFeature._wmeObject;
    if (poi.type!="venue")
        return;

    var poiAttr = poi.attributes;

    if (!poiAttr.geometry.components[0].hasOwnProperty("components"))
        return;


    var poiPoints = [];
    var segPoints = [];

    log("poi",poi);
    log("poiAttr",poiAttr);
    for (var seg in W.model.segments.objects)
    {
        var segment = typeof(W.model.segments.getObjectById) == "function" ? W.model.segments.getObjectById(seg) : W.model.segments.get(seg);
        var segAttr = segment.attributes;
        if (segAttr.primaryStreetID==null)
        {
            if (onScreen(segment))
            {
                var segLineString = segAttr.geometry.clone();
            }
        }
    }
    if (typeof(segLineString) === "undefined") {
        alert("Need a new segment through");
        return;
    }

    var poiGeo = poiAttr.geometry.clone();
    var oldPoiGeo = poiAttr.geometry.clone();
    var poiLineString = poiGeo.components[0].clone();

    var poiLine = new OL.Geometry.LinearRing();
    var segLine = new OL.Geometry.LinearRing();

    var intersectPoint = [];
    var intersectLine  = [];

    // Calcul des point d'intersection seg // poi
    for (var n=0; n < parseInt(poiLineString.components.length-1); n++)
    {
        poiLine.components["0"] = poiLineString.components[n].clone();
        poiLine.components["1"] = poiLineString.components[n+1].clone();

        for (var m=0; m < parseInt(segLineString.components.length-1); m++)
        {
            segLine.components["0"] = segLineString.components[m].clone();
            segLine.components["1"] = segLineString.components[m+1].clone();
            if (poiLine.intersects(segLine))
            {
                intersectPoint.push({index: n, intersect: intersection(poiLine, segLine)});
            }
            segLine.removeComponent("0");
            segLine.removeComponent("1");
        }
        poiLine.removeComponent("0");
        poiLine.removeComponent("1");
    }
    log('intersectPoint= ',intersectPoint);
    // intégration des points au contour du POI avec memo du nouvel index
    var i=1;
    for (var n=0; n < intersectPoint.length; n++)
    {
        var point = intersectPoint[n].intersect;
        var index = parseInt(intersectPoint[n].index)+i;
        poiLineString.addComponent(point, index);
        intersectPoint[n].newIndex = index;
        i++;
    }

    // création des deux nouvelles géométries
    var TabLine1 = [];
    var TabLine2 = [];

    var index1 = parseInt(intersectPoint[0].newIndex);
    var index2 = parseInt(intersectPoint[1].newIndex);

    for (var n=0; n < parseInt(poiLineString.components.length); n++)
    {
        var x = poiLineString.components[n].x;
        var y = poiLineString.components[n].y;
        var point = new OL.Geometry.Point(x ,y);


        if (n < index1){
            TabLine1.push(point);

        }
        if (n == index1){
            TabLine1.push(point);
            TabLine2.push(point);
        }
        if ((index1 < n) && (n < index2)){
            TabLine2.push(point);
        }
        if (n == index2){
            TabLine1.push(point);
            TabLine2.push(point);
        }
        if (index2 < n){
            TabLine1.push(point);
        }
    }

    /*
    log('TabLine1['+0+']= ',TabLine1[0]);
    log('TabLine1['+(TabLine1.length-1)+']= ',TabLine1[(TabLine1.length-1)]);
    log('TabLine2['+0+']= ',TabLine2[0]);
    log('TabLine2['+(TabLine2.length-1)+']= ',TabLine2[(TabLine2.length-1)]);
    */
    //log('TabLine1= ',TabLine1);
    //log('TabLine2= ',TabLine2);

    var LineString1 = new OL.Geometry.LinearRing(TabLine1);
    var LineString2 = new OL.Geometry.LinearRing(TabLine2);
    log('LineString1= ',LineString1);
    log('LineString2= ',LineString2);

    poiGeo = new OL.Geometry.Polygon(LineString1);
    log('poiGeo = ',poiGeo);

    var WazeActionUpdateFeatureGeometry = require("Waze/Action/UpdateFeatureGeometry");
    W.model.actionManager.add(new WazeActionUpdateFeatureGeometry(poi, Waze.model.venues,oldPoiGeo,poiGeo));


    // Création du nouveau poi
    var WazefeatureVectorLandmark = require("Waze/Feature/Vector/Landmark");
    clonePoi = new WazefeatureVectorLandmark();
    var clonePoiAttr = clonePoi.attributes;

    clonePoiAttr.adLocked = poi.attributes.adLocked;
    clonePoiAttr.aliases = poi.attributes.aliases;
    clonePoiAttr.approved = poi.attributes.approved;
    clonePoiAttr.categories = poi.attributes.categories;
    clonePoiAttr.description = poi.attributes.description;
    clonePoiAttr.externalProviderIDs = poi.attributes.externalProviderIDs;
    clonePoiAttr.houseNumber = poi.attributes.houseNumber;
    clonePoiAttr.openingHours = poi.attributes.openingHours;
    clonePoiAttr.lockRank = poi.attributes.lockRank;
    clonePoiAttr.name = poi.attributes.name;
    clonePoiAttr.residential = poi.attributes.residential;
    clonePoiAttr.phone = poi.attributes.phone;
    clonePoiAttr.services = poi.attributes.services;
    clonePoiAttr.url = poi.attributes.url;
    //clonePoiAttr.entryExitPoints = poi.attributes.entryExitPoints;
    //clonePoiAttr.images = poi.attributes.images;

    clonePoi.geometry = new OL.Geometry.Polygon(LineString2);

    log('clonePoi',clonePoi);

    var WazeActionAddLandmark = require("Waze/Action/AddLandmark");
    W.model.actionManager.add(new WazeActionAddLandmark(clonePoi));

    // copie du nom et mise à jour du nouveau poi

    var street = W.model.streets.objects[poi.attributes.streetID];
    streetName = street.name;
    var cityID = street.cityID;
    var city = W.model.cities.objects[cityID];
    var stateID = W.model.cities.objects[cityID].attributes.stateID;
    var state = W.model.states.objects[stateID];
    var countryID = W.model.cities.objects[cityID].attributes.countryID;
    var country = W.model.countries.objects[countryID];

    if (!street.isEmpty || !city.attributes.isEmpty){ // nok
        var newAtts = { emptyStreet: true, stateID: stateID, countryID: countryID, cityName: city.attributes.name, streetName: streetName, emptyCity: true };
        log ('Natural feature POI: no street name and city');
        var WazeActionUpdateFeatureAddress = require("Waze/Action/UpdateFeatureAddress");
        W.model.actionManager.add(new WazeActionUpdateFeatureAddress(poi, newAtts));
    }
    var street = W.model.streets.objects[clonePoi.attributes.streetID];
    streetName = street.name;
    var cityID = street.cityID;
    var city = W.model.cities.objects[cityID];
    var stateID = W.model.cities.objects[cityID].attributes.stateID;
    var state = W.model.states.objects[stateID];
    var countryID = W.model.cities.objects[cityID].attributes.countryID;
    var country = W.model.countries.objects[countryID];

    if (!street.isEmpty || !city.attributes.isEmpty){ // nok
        var newAtts = { emptyStreet: true, stateID: stateID, countryID: countryID, cityName: city.attributes.name, streetName: streetName, emptyCity: true };
        log ('Natural feature POI: no street name and city');
        var WazeActionUpdateFeatureAddress = require("Waze/Action/UpdateFeatureAddress");
        W.model.actionManager.add(new WazeActionUpdateFeatureAddress(clonePoi, newAtts));
    }

    //log('model.actionManager = ',model.actionManager);

}


function intersection(D1, D2)
{
    var a,b,c,d,x,y;
    var seg = {}; //{x1, y1, x2, y2};
    var seg1 = {}; //{x1, y1, x2, y2};
    var seg2 = {}; //{x1, y1, x2, y2};
    var 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 OL.Geometry.segmentsIntersect(seg1,seg2,options);


}


WMESP_bootstrap();