WME Split POI

Split POI with a new seg

当前为 2019-02-28 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

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

您需要先安装一款用户脚本管理器扩展,例如 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.4
// @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",
	en: "Update WME Split POI: v" + WMESP_Version + "\nCompatibility New WME"
	};

/* 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.getSelectedFeatures().length !=1 )
        return;
    
    var selectedObject = selectionManager.getSelectedFeatures()[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")
    {
        var item=getId("landmark-edit-general");
        //var attSection = getElementsByClassName("attributes-form side-panel-section", item);
   	    var btnGoogleGroup = getElementsByClassName("external-providers form-group",item);
        var parent = btnGoogleGroup[0].parentNode;
 
        var WMESP_Controle=document.createElement('Div');
        WMESP_Controle.id="WMESP-Controle";
        WMESP_Controle.className="form-group";
        WMESP_Controle.innerHTML+='<br><input type="button" id="_btnSplitPOI" class="action-button waze-btn waze-btn-white" value="Split POI"><br>';
		    parent.insertBefore(WMESP_Controle, btnGoogleGroup[0]);
        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 (loginManager.user.isCountryManager()) return true;
    
    if (obj.attributes.permissions == 0)
      return false;

    return true;
}

function SplitPOI()
{
    
    if (selectionManager.getSelectedFeatures().length !=1 )
        return;
    
    var poi = selectionManager.getSelectedFeatures()[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 wazeModel.segments.objects)
    {
      var segment = typeof(wazeModel.segments.getObjectById) == "function" ? wazeModel.segments.getObjectById(seg) : wazeModel.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);
}