WME Split POI

Split POI with a new seg

目前為 2018-03-01 提交的版本,檢視 最新版本

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==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*
// @icon            
// @author          seb-d59
// @version	        2.1
// @grant           none
// ==/UserScript==

var debug=false;
var WMESP_Version = GM_info.script.version;
var WMESP_OldVersion = WMESP_Version;
var WMESP_API = {};
WMESP_API.require = {};

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

/* bootstrap, will call initialize() */
function WMESP_bootstrap(){
	log('Init');
	if (typeof(unsafeWindow) === "undefined"){
		unsafeWindow = ( function () {
			var dummyElem = document.createElement('p');
			dummyElem.setAttribute('onclick', 'return window;');
			return dummyElem.onclick();
		}) ();
	}
	/* 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");
    initializeWazeObjects();   
    
}

function waitForObject(object)
{
    var obj=null;
    if (object.r==true)
    {
        if (debug) log ("eval: " + (object.s!=null?object.s:'dummy') + "="+ ((typeof(window.require)=="function")?"require":"WMESP_API.require") +"(\"" + object.o + "\")");
        eval ((object.s!=null?object.s:'dummy') + '=' + ((typeof(window.require)=="function")?'require':'WMESP_API.require') +'("' + object.o + '")');
        eval ("obj=" + (object.s!=null?object.s:'dummy'));
    		log("obj", obj);
    }else{
    
        if (debug) log ("eval: " + "typeof(unsafeWindow." + object.o.replace(/\//g, '.') + ")");
        obj=eval("typeof(unsafeWindow." + object.o.replace(/\//g, '.') + ")");
    }
    if(obj === "undefined")
    {
        log(object.o + ' KO');
        window.setTimeout(waitForObject.caller, 500);
        return false;
    }
    if (debug) log(object.s + ' OK');


    if (object.s!=null && object.r==false)
        eval (object.s + "=" + object.o.replace(/\//g, '.'));

    return true;



}


function initializeWazeObjects()
{
  var objectToCheck = [
      {o: "W",																s: "waze",														r: false},
      {o: "W.map",											      s: "wazeMap",                         r: false},
		  {o: "W.model",													s: "wazeModel",												r: false},
		  {o: "OL",													s: "wazeOL",													r: false},
		  {o: "W.loginManager",									s: "loginManager",										r: false},
		  {o: "W.selectionManager",							s: "selectionManager",								r: false},
		  {o: "W.loginManager.user",							s: "me",															r: false},
		  {o: "Waze/Action/UpdateObject",						s: "WazeActionUpdateObject",					r: true},
			{o: "Waze/Action/UpdateSegmentGeometry",  s: "WazeUpdateSegmentGeometry",       r: true},
      {o: "Waze/Action/UpdateFeatureAddress",		s: "WazeActionUpdateFeatureAddress",	r: true},
		  {o: "Waze/Action/DeleteObject",						s: "WazeActionDeleteObject", 					r: true},
		  {o: "Waze/Action/UpdateFeatureGeometry",	s: "WazeActionUpdateFeatureGeometry", r: true},
      {o: "Waze/Feature/Vector/Landmark",			  s: "WazefeatureVectorLandmark",       r: true},
      {o: "Waze/Action/CreateObject",						s: "WazeActionCreateObject",           r: true},
      {o: "Waze/Action/AddLandmark",						s: "WazeActionAddLandmark",           r: true},
      {o: "localStorage",										    s: null,                              r: false}
		  ];
 
	for (var i=0; i<objectToCheck.length; i++)
	{
			if (!waitForObject(objectToCheck[i])) return;    
	}
  
  
  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();
    
		selectionManager.events.register("selectionchanged", null, WMESP_newSelectionAvailable);
    
    log("init done.");
}

function WMESP_newSelectionAvailable()
{
    if (selectionManager.selectedItems.length!=1)
        return;
    var selectedObject = selectionManager.selectedItems[0].model;
    if (selectedObject.type!="venue")
        return;
    
    var attributes = selectedObject.attributes;
		
		if (!attributes.geometry.hasOwnProperty("components"))
		  return;
		
		if (!((attributes.categories == "NATURAL_FEATURES" ) || (attributes.categories == "ISLAND" ) || (attributes.categories == "SEA_LAKE_POOL") || (attributes.categories == "RIVER_STREAM" ) || (attributes.categories == "FOREST_GROVE") || (attributes.categories == "FARM" ) || (attributes.categories == "CANAL" ) || (attributes.categories == "SWAMP_MARSH" ) || (attributes.categories == "DAM" )))
		  return;
		log("selectionManager",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")
    {
        item=getId("landmark-edit-general");
        
        var attSection = getElementsByClassName("attributes-form side-panel-section", item);
        /*var btnResidential = getElementsByClassName("btn-link toggle-residential",item);
        
        var parent1 = btnResidential[0].parentNode;
        var parent = parent1.parentNode;
        var WMESP_Controle=document.createElement('Div');
    
        WMESP_Controle.id="WMESP-Controle";
        WMESP_Controle.innerHTML+='<br><input type="button" id="_btnSplitPOI" class="action-button waze-btn waze-btn-white" value="Split POI"><br>';
		    attSection[0].insertBefore(WMESP_Controle, parent);
        getId("_btnSplitPOI").onclick=SplitPOI;
        */
        var btnGoogle = getElementsByClassName("external-providers-view",item);
        var parent1 = btnGoogle[0].parentNode;
        var parent = parent1.parentNode;
        var WMESP_Controle=document.createElement('Div');
    
        WMESP_Controle.id="WMESP-Controle";
        WMESP_Controle.innerHTML+='<br><input type="button" id="_btnSplitPOI" class="action-button waze-btn waze-btn-white" value="Split POI"><br>';
		    attSection[0].insertBefore(WMESP_Controle, parent);
        getId("_btnSplitPOI").onclick=SplitPOI;
        
    }
    
}


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

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

    return true;
}

function SplitPOI()
{
    
    if (selectionManager.selectedItems.length!=1)
        return;
    var poi = selectionManager.selectedItems[0].model;
    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 Waze.model.segments.objects)
    {
      var segment = Waze.model.segments.get(seg);
      var segAttr = segment.attributes;
      if (segAttr.primaryStreetID==null)
      {
        if (onScreen(segment))
	      {
          var segLineString = segAttr.geometry.clone();
        }
      }
		}
		
		var poiGeo = poiAttr.geometry.clone();
		var oldPoiGeo = poiAttr.geometry.clone(); 
		var poiLineString = poiGeo.components[0].clone();
    
    var poiLine = new OpenLayers.Geometry.LinearRing();
    var segLine = new OpenLayers.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 OpenLayers.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 OpenLayers.Geometry.LinearRing(TabLine1);
    var LineString2 = new OpenLayers.Geometry.LinearRing(TabLine2);
    log('LineString1= ',LineString1);
    log('LineString2= ',LineString2);
    
    poiGeo = new OpenLayers.Geometry.Polygon(LineString1);
    log('poiGeo = ',poiGeo);
            
    wazeModel.actionManager.add(new WazeActionUpdateFeatureGeometry(poi, Waze.model.venues,oldPoiGeo,poiGeo));
   
    
    // Création du nouveau poi
    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 OpenLayers.Geometry.Polygon(LineString2);

    log('clonePoi',clonePoi);
    
    wazeModel.actionManager.add(new WazeActionAddLandmark(clonePoi));
    
    // copie du nom et mise à jour du nouveau poi
    
    var street = wazeModel.streets.objects[poi.attributes.streetID];
    streetName = street.name;
    var cityID = street.cityID;
    var city = wazeModel.cities.objects[cityID];
    var stateID = wazeModel.cities.objects[cityID].attributes.stateID;
    var state = wazeModel.states.objects[stateID];
    var countryID = wazeModel.cities.objects[cityID].attributes.countryID;
    var country = wazeModel.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');
      wazeModel.actionManager.add(new WazeActionUpdateFeatureAddress(poi, newAtts));
		}
		var street = wazeModel.streets.objects[clonePoi.attributes.streetID];
    streetName = street.name;
    var cityID = street.cityID;
    var city = wazeModel.cities.objects[cityID];
    var stateID = wazeModel.cities.objects[cityID].attributes.stateID;
    var state = wazeModel.states.objects[stateID];
    var countryID = wazeModel.cities.objects[cityID].attributes.countryID;
    var country = wazeModel.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');
      wazeModel.actionManager.add(new WazeActionUpdateFeatureAddress(clonePoi, newAtts));
		}
		
    //log('wazeModel.actionManager = ',wazeModel.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 OpenLayers.Geometry.segmentsIntersect(seg1,seg2,options);
          
   
}

if (typeof(window.require) == "function"){
  log("bootstrap classique");
  WMESP_bootstrap();
}else{
  ///////////////////////////////////////////////
  ///  Big thanks to dummyd2 for this Patch    //
  ///////////////////////////////////////////////
  log("load patch by dummyd2");

  // setup one global var and put all in
  var WMESPAPI = {};


  // detect URL of WME source code
  WMESPAPI.scripts = document.getElementsByTagName('script');
  WMESPAPI.url=null;
  for (i=0;i<WMESPAPI.scripts.length;i++)
  {
      if (WMESPAPI.scripts[i].src.indexOf('/assets-editor/js/app')!=-1)
      {
          WMESPAPI.url=WMESPAPI.scripts[i].src;
          break;
      }
  }
  if (WMESPAPI.url==null)
  {
      throw new Error("WME Hack: can't detect WME main JS");
  }



  // setup a fake require and require.define
  WMESPAPI.require=function (e) {
      return WMESPAPI.require.define.modules[e];
  };

  WMESPAPI.require.define=function (m) {
      if (WMESPAPI.require.define.hasOwnProperty('modules')==false)
          WMESPAPI.require.define.modules={};
      for (var p in m)
      {
          WMESPAPI.require.define.modules[p]=m[p];
      }
  };


  // save the original webpackJsonp function
  WMESPAPI.tmp = window.webpackJsonp;

  // taken from WME code: this function is a wrapper that setup the API and may call recursively other functions
  WMESPAPI.t = function (n) {
      if (WMESPAPI.s[n]) return WMESPAPI.s[n].exports;
      var r = WMESPAPI.s[n] = {
          exports: {},
          id: n,
          loaded: !1
      };
      return WMESPAPI.e[n].call(r.exports, r, r.exports, WMESPAPI.t), r.loaded = !0, r.exports;
  };

  // e is a copy of all WME funcs because function t need to access to this list
  WMESPAPI.e=[];

  // the patch
  window.webpackJsonp = function(a, i) {
      // our API but we will use it only to build the require stuffs
      var api={};
      // taken from WME code. a is [1], so...
      for (var o, d, u = 0, l = []; u < a.length; u++) d = a[u], WMESPAPI.r[d] && l.push.apply(l, WMESPAPI.r[d]), WMESPAPI.r[d] = 0;
      
      var unknownCount=0;
      var classname, funcStr;
      
      // copy i in e and keep a link from classname to index in e
      for (o in i)
      {
          WMESPAPI.e[o] = i[o];
          funcStr = i[o].toString();
          classname = funcStr.match(/CLASS_NAME:\"([^\"]*)\"/);
          if (classname)
          {
              // keep the link.
              api[classname[1].replace(/\./g,'/').replace(/^W\//, 'Waze/')]={index: o, func: WMESPAPI.e[o]};
          }
          else
          {
              api['Waze/Unknown/' + unknownCount]={index: o, func: WMESPAPI.e[o]};
              unknownCount++;
          }
          
      }
      
      // taken from WME code: it calls the original webpackJsonp and do something else, but I don't really know what.
      // removed the call to the original webpackJsonp: still works...
      //for (tmp && tmp(a, i); l.length;) l.shift().call(null, t);
      for (; l.length;) l.shift().call(null, WMESPAPI.t);
      WMESPAPI.s[0] = 0;
      
      // run the first func of WME. This first func will call recusrsively all funcs needed to setup the API.
      // After this call, s will contain all instanciables classes.
      //var ret = WMESPAPI.t(0);
      
      // now, build the requires thanks to the link we've built in var api.
      var module={};
      var apiFuncName;
      unknownCount=0;
      
      for (o in i)
      {
          funcStr = i[o].toString();
          classname = funcStr.match(/CLASS_NAME:\"([^\"]*)\"/);
          if (classname)
          {
              module={};
              apiFuncName = classname[1].replace(/\./g,'/').replace(/^W\//, 'Waze/');
              module[apiFuncName]=WMESPAPI.t(api[apiFuncName].index);
              WMESPAPI.require.define(module);
          }
          else
          {
              var matches = funcStr.match(/SEGMENT:"segment",/);
              if (matches)
              {
                  module={};
                  apiFuncName='Waze/Model/ObjectType';
                  module[apiFuncName]=WMESPAPI.t(api['Waze/Unknown/' + unknownCount].index);
                  WMESPAPI.require.define(module);
              }
              unknownCount++;
          }
      }
       

      // restore the original func
      window.webpackJsonp=WMESPAPI.tmp;

      // set the require public if needed
      // if so: others scripts must wait for the window.require to be available before using it.
      //window.require=WMESPAPI.require;
      WMESP_API.require=WMESPAPI.require;
      // all available functions are in WMESPAPI.require.define.modules
      // console.debug this variable to read it:
      //console.debug('WMESPAPI Modules: ', WMESPAPI.require.define.modules);
      
      // run your script here:
      setTimeout(WMESP_bootstrap, 1000);
      
      // again taken from WME code. Not sure about what it does.
      //if (i[0]) return ret;
  };

  // some kind of global vars and init
  WMESPAPI.s = {};
  WMESPAPI.r = {
      0: 0
  };

  // hacking finished

// load again WME through our patched func
WMESPAPI.WMEHACK_Injected_script = document.createElement("script");
WMESPAPI.WMEHACK_Injected_script.setAttribute("type", "application/javascript");
WMESPAPI.WMEHACK_Injected_script.src = WMESPAPI.url;
document.body.appendChild(WMESPAPI.WMEHACK_Injected_script);
}