// ==UserScript== // @name [HFR] Black List // @version 0.7.2.1 // @namespace http://nykal.fr // @description Mask messages from black listed users on forum.hardware.fr // @include https://forum.hardware.fr/* // @exclude https://forum.hardware.fr/message.php* // @exclude https://forum.hardware.fr/forum1.php* // @exclude https://forum.hardware.fr/forum1f.php* // @grant GM_info // @grant GM_deleteValue // @grant GM_getValue // @grant GM_listValues // @grant GM_setValue // @grant GM_getResourceText // @grant GM_getResourceURL // @grant GM_addStyle // @grant GM_log // @grant GM_openInTab // @grant GM_registerMenuCommand // @grant GM_setClipboard // @grant GM_xmlhttpRequest // ==/UserScript== // historique modifs r21 : // 0.7.2.1 (03/12/2017) : // - passage au https // file: hfr_black_list.user.js // author: Nykal // author: anonymous contributors // license: This file is in the public domain and comes with no warranty. // history: 0.1 2008/12/21 first revision // history: 0.2 2008/12/27 ajout de la fenetre d'édition. Correction du réaffichage des messages masqués // history: 0.3 2009/02/17 petits correctifs // history: 0.4 2009/03/02 blocage des quotes // history: 0.5 2009/03/03 refont présentation // history: 0.6 2009/05/11 Ajout masquage total des messages et quotes. Ajout actualisation du masquage de quotes à l'ajout. // history: 0.7 2010/11/16 Gestion des pseudos longs (caractère zwsp). // version: $$ var cle_masquage_total = "hfr_black_list_masquage_total"; masquerQuotesBloques(); (function () { injectCSS(".rightListeNoire{float:right;text-align:right}"); var root = document.getElementById('mesdiscussions'); var mp = getElementByXpath('table//tr//td//div[@class="left"]//div[@class="left"]', root)[0]; var imgListenoire = document.createElement('img'); imgListenoire.src = ''; imgListenoire.setAttribute("style", "vertical-align: bottom"); var divListeNoire = document.createElement('div'); divListeNoire.setAttribute("class", "left"); var textListeNoire=document.createTextNode("\u00a0"); //divListeNoire.innerHTML="Liste noire"; var aListeNoire = this.document.createElement('span'); aListeNoire.setAttribute("class", "s1Ext linkListeNoire"); //aListeNoire.setAttribute("style", "cursor: pointer;"); injectCSS("span.linkListeNoire:hover {text-decoration:underline;} span.linkListeNoire{cursor: pointer;margin-left:0.5em;}"); //aListeNoire.setAttribute('href', '#'); divListeNoire.addEventListener('click', hfr_edition_liste_noire, false); aListeNoire.appendChild(this.document.createTextNode("Liste\u00a0noire")); divListeNoire.appendChild(textListeNoire); divListeNoire.appendChild(imgListenoire); divListeNoire.appendChild(aListeNoire); if (mp != null) { mp.parentNode.appendChild(divListeNoire); } // liste des auteurs de message var pseudos = getElementByXpath('//table//td[@class="messCase1"]//b[@class="s2"]', root); var linksProfil = getElementByXpath('//table//tr[@class="message"]//a[starts-with(@href, "/hfr/profil" ) and contains(@href, "#im") = false]', root); var cancel = false; // Supression de profil supprime de la liste des pseudos pseudos = pseudos.filter(function(pseudo){ if (pseudo.innerHTML.match(/Profil su.*prim.*/) == null) { return pseudo; } }); var blacklist = hfr_GM_getValue("ignore_list", new Array()); // parcours de la liste des pseudos pour ajouter les boutons d'edition de la blacklist for(var i = 0; i < pseudos.length; i++) { var pseudo = pseudos[i]; var imgListenoire = document.createElement('img'); imgListenoire.src = ''; imgListenoire.setAttribute("style", "vertical-align: bottom"); var divListeNoire = document.createElement('div'); divListeNoire.setAttribute("class", "right"); divListeNoire.appendChild(imgListenoire); //pseudo.parentNode.previousSibling.appendChild(divListeNoire); pseudo.parentNode.parentNode.insertBefore(divListeNoire, pseudo.parentNode); // masque le message si l'utilisateur est sur la liste noire if (hfr_isPseudoBlacklist(pseudo.innerHTML)) { hfr_masquer_message(pseudo.parentNode.parentNode.parentNode, pseudo.innerHTML); } // associe la fenetre de gestion de la liste noire au clic sur un pseudo divListeNoire.addEventListener('click', function(event) { var theEvent = event; var pseudoClique = event.target.parentNode.nextSibling.textContent; var width = 450; var newDivBL; // Suppression d'une fenetre d'edition de liste noire déjà ouverte if (document.getElementById('edit_black_list')) { newDivBL = document.getElementById('edit_black_list'); newDivBL.parentNode.removeChild(newDivBL); } // pop up de gestion de la liste noire newDivBL = document.createElement('div'); newDivBL.setAttribute('id', 'edit_black_list'); newDivBL.style.position = 'absolute'; newDivBL.style.border = '1px solid black'; newDivBL.style.background = "white"; newDivBL.style.zIndex = '1001'; newDivBL.style.width = 'auto'; newDivBL.style.paddingBottom = '5px'; newDivBL.style.textAlign = 'justify'; // Message de la popup var divQuestion = document.createElement('div'); //divQuestion.style.display = 'block'; divQuestion.style.margin = '5px'; divQuestion.style.fontSize = '8pt'; divQuestion.style.width = 'auto'; var divBas = document.createElement('div'); divBas.style.clear = 'none'; divBas.style.width = '100%'; divBas.style.display = 'inline'; var divView = document.createElement('div'); //divView.style.textAlign ='left'; //divView.style.float ='left'; //divView.style.width ='70px'; var viewListe = document.createElement('input'); viewListe.type = 'image'; viewListe.src = ''; viewListe.style.marginLeft = '6px'; viewListe.style.textAlign = 'left'; viewListe.setAttribute("pseudo", pseudoClique); viewListe.addEventListener('click', hfr_edition_liste_noire_bis, false); var divValidation = document.createElement('div'); divValidation.setAttribute("class", "rightListeNoire"); // bouton de validation de la popup var inputOk = document.createElement('input'); inputOk.type = 'image'; inputOk.src = ''; inputOk.style.marginRight = '6px'; // si le pseudo cliqué est sur la liste noire le message et le bouton de validation seront différents if (hfr_isPseudoBlacklist(pseudoClique)) { divQuestion.innerHTML = "Supprimer " + pseudoClique + " de la liste des utilisateurs bloqués ?"; inputOk.addEventListener('click', function(event) { hfr_removeFromBlacklist(pseudoClique); hfr_show_blacklisted(pseudoClique); afficherQuotesBloques(pseudoClique); newDivBL.style.display = 'none'; } , false); } else { divQuestion.innerHTML = "Ajouter " + pseudoClique + " à la liste des utilisateurs bloqués ?"; inputOk.addEventListener('click', function(event) { hfr_addToBlacklist(pseudoClique); hfr_hide_new_blacklisted(pseudoClique); masquerQuotesBloques(); newDivBL.style.display = 'none'; } , false); }; // bouton d'annnulation de la popup var inputCancel = document.createElement('input'); inputCancel.type = 'image'; inputCancel.src = ''; inputCancel.style.marginRight = '5px'; inputCancel.addEventListener('click', function(event) { newDivBL.style.display = 'none'; } , false); divValidation.appendChild(inputOk); divValidation.appendChild(inputCancel); divView.appendChild(viewListe); divBas.appendChild(divValidation); divBas.appendChild(divView); newDivBL.appendChild(divQuestion); newDivBL.appendChild(divBas); //positionnement de la popup if (theEvent.clientX + width + 25 > document.documentElement.clientWidth) { newDivBL.style.left = (document.documentElement.clientWidth - width - 25) + 'px'; } else { newDivBL.style.left = (theEvent.clientX + 8) + 'px'; } newDivBL.style.display = 'block'; newDivBL.style.top = (window.pageYOffset + theEvent.clientY + 8) + 'px'; root.appendChild(newDivBL); } , false); }; })(); function masquerQuotesBloques() { injectCSS(".citationListeNoire {display:none}"); var classNameCitation = "citation"; //var xpathResultCitation = document.evaluate("//*[@class = '"+classNameCitation+"']", document, null, 0, null); var quoteBloques = hfr_getElementsByClassName(document, classNameCitation); //var quoteBloques = new Array(); //var elt = xpathResultCitation.iterateNext(); //while (elt) { var elt; for( var i = 0; i < quoteBloques.length; i++ ) { elt = quoteBloques[i]; var titreQuote = elt.firstChild.firstChild.firstChild.firstChild.textContent; var pseudoQuote = titreQuote.substring(0, titreQuote.length - " a ecrit :".length).toLowerCase(); if (hfr_isPseudoBlacklist(pseudoQuote)){ elt.setAttribute("class", "citation citationListeNoire"); if (!hfr_GM_getValue(cle_masquage_total, false)) { var quoteListeNoire = this.document.createElement('table'); quoteListeNoire.setAttribute("class", classNameCitation); quoteListeNoire.setAttribute("quoteBloque", pseudoQuote); var aListeNoire = document.createElement('a'); aListeNoire.appendChild(document.createTextNode(pseudoQuote + " a été bloqué")); aListeNoire.setAttribute("class", "Topic"); aListeNoire.addEventListener('click', function(event) { event.target.parentNode.parentNode.parentNode.parentNode.style.display = 'none'; event.target.parentNode.parentNode.parentNode.parentNode.parentNode.firstChild.setAttribute("class", "citation"); } , false); var trListeNoire = document.createElement('tr'); trListeNoire.setAttribute("class", "none") var bListeNoire = document.createElement('b'); bListeNoire.setAttribute("class", "s1") quoteListeNoire.appendChild(trListeNoire).appendChild(document.createElement('td')).appendChild(bListeNoire).appendChild(aListeNoire); elt.parentNode.appendChild(quoteListeNoire); } } } } function afficherQuotesBloques(pseudoDebloque) { var xpathResult = document.evaluate("//*[@quoteBloque = '"+pseudoDebloque.toLowerCase()+"']", document, null, 0, null); var outArray = new Array(); var elt = xpathResult.iterateNext(); while (elt) { outArray[outArray.length] = elt; elt.style.display = 'none'; elt.parentNode.firstChild.setAttribute("class", "citation"); elt = xpathResult.iterateNext(); } } /** récupère les éléments d'une classe donnée */ function hfr_getElementsByClassName(doc, className) { var xpathResult = doc.evaluate("//*[@class = '"+className+"']", doc, null, 0, null); var outArray = new Array(); var elt = xpathResult.iterateNext(); while (elt) { outArray[outArray.length] = elt; elt = xpathResult.iterateNext(); } return outArray; } /** Masque les messages d'un utilisateur black listé */ function hfr_hide_new_blacklisted(blackListed) { var blacklist = hfr_GM_getValue("ignore_list", new Array()); if (blacklist.length == 0) { return; } var nicks = hfr_getElementsByClassName(this.document, 's2'); for (var i=0, j=nicks.length; i document.documentElement.clientWidth) { newDivLN.style.left = (document.documentElement.clientWidth - width - 25) + 'px'; } else { newDivLN.style.left = (theEvent.clientX + 8) + 'px'; } newDivLN.style.display = 'block'; newDivLN.style.top = (window.pageYOffset + theEvent.clientY + 8) + 'px'; var root = document.getElementById('mesdiscussions'); root.appendChild(newDivLN); } /** Affiche la pop up d'édition de la liste des utilisateurs bloqués */ function hfr_edition_liste_noire_bis(event) { var width = 450; var theEvent = event; var pseudoClique = event.target.getAttribute("pseudo").toLowerCase(); // Suppression d'une fenetre d'edition de liste noire déjà ouverte if (document.getElementById('edit_complete_black_list_bis')) { var newDivLN = document.getElementById('edit_complete_black_list_bis'); newDivLN.parentNode.removeChild(newDivLN); event.target.src = ''; } else { event.target.src = ''; // pop up de gestion de la liste noire var newDivLN = document.createElement('div'); newDivLN.setAttribute('id', 'edit_complete_black_list_bis'); newDivLN.style.border = '1px solid black'; newDivLN.style.backgroundColor = "rgb(250,250,250)"; newDivLN.style.margin = '5px'; event.target.parentNode.parentNode.parentNode.appendChild(newDivLN); // Message de la popup var divQuestion = document.createElement('div'); divQuestion.style.display = 'block'; divQuestion.style.margin = '5px'; divQuestion.style.fontSize = '8pt'; divQuestion.style.width = 'auto'; divQuestion.style.textAlign= 'left'; divQuestion.innerHTML="Editer la liste noire"; var blacklist = hfr_GM_getValue("ignore_list", new Array()); var tableLN = document.createElement('table'); tableLN.setAttribute("class", "s1Ext"); var newLN = document.createElement('input'); newLN.type = 'text'; newLN.style.display = 'block'; newLN.style.margin = '5px'; var trLN = document.createElement('tr'); var tdLNpseudo = document.createElement('td'); tdLNpseudo.setAttribute("style", 'border-width:0px'); tdLNpseudo.appendChild(newLN); var tdLNbtn = document.createElement('td'); var inputAdd = document.createElement('input'); inputAdd.type = 'image'; inputAdd.src = ''; inputAdd.style.marginRight = '6px'; inputAdd.addEventListener('click', function() { var newBlackListed = this.parentNode.parentNode.firstChild.firstChild.value.toLowerCase(); if (pseudoClique != newBlackListed) { if (!hfr_isPseudoBlacklist(newBlackListed)) { hfr_addToBlacklist(newBlackListed); } hfr_hide_new_blacklisted(newBlackListed); masquerQuotesBloques(); hfr_edition_liste_noire_bis(theEvent); } } , false); inputAdd.style.textAlign ='right'; inputAdd.style.float ='right'; tdLNbtn.appendChild(inputAdd); tdLNbtn.setAttribute("style", 'border-width:0px'); trLN.appendChild(tdLNpseudo); trLN.appendChild(tdLNbtn); tableLN.appendChild(trLN); for (var i=0, j=blacklist.length; i document.documentElement.clientWidth) { newDivLN.style.left = (document.documentElement.clientWidth - width - 25) + 'px'; } else { newDivLN.style.left = (theEvent.clientX + 8) + 'px'; } newDivLN.style.display = 'block'; newDivLN.style.top = (window.pageYOffset + theEvent.clientY + 8) + 'px'; } //var root = document.getElementById('mesdiscussions'); //root.appendChild(newDivLN); } /** Affiche les messages d'un utilisateur rétiré de la liste noire */ function hfr_show_blacklisted(unBlackListed) { var nicks = hfr_getElementsByClassName(this.document, 's2'); for (var i=0, j=nicks.length; i= 0); } function getRealPseudo(pseudoValue) { // Suppression du caractère spécial dans les pseudos longs return pseudoValue.split(String.fromCharCode(8203)).join(''); } /** Supprime un utilisateur de la liste noire */ function hfr_removeFromBlacklist(pseudoASupprimer) { var blacklist = hfr_GM_getValue('ignore_list', new Array()); var i = searchArray(pseudoASupprimer.toLowerCase(), blacklist); if (i >=0) { blacklist.splice(i, 1); hfr_GM_setValue('ignore_list', blacklist); } } /** Récupère un élément via son path */ function getElementByXpath(path, element) { var arr = Array(), xpr = document.evaluate(path, element, null, XPathResult.UNORDERED_NODE_ITERATOR_TYPE, null); for (;item = xpr.iterateNext();) arr.push(item); return arr; } function hfr_GM_setValue( cookieName, cookieValue, lifeTime ) { if( !cookieName ) { return; } if( lifeTime == "delete" ) { lifeTime = -10; } else { lifeTime = 31536000; } document.cookie = escape( cookieName ) + "=" + escape( hfr_getRecoverableString( cookieValue ) ) + ";expires=" + ( new Date( ( new Date() ).getTime() + ( 1000 * lifeTime ) ) ).toGMTString() + ";path=/"; } function hfr_GM_getValue( cookieName, oDefault ) { var cookieJar = document.cookie.split( "; " ); for( var x = 0; x < cookieJar.length; x++ ) { var oneCookie = cookieJar[x].split( "=" ); if( oneCookie[0] == escape( cookieName ) ) { try { eval('var footm = '+unescape( oneCookie[1] )); } catch(e) { return oDefault; } return footm; } } return oDefault; } function hfr_getRecoverableString(oVar,notFirst) { var oType = typeof(oVar); if( ( oType == 'null' ) || ( oType == 'object' && !oVar ) ) { //most browsers say that the typeof for null is 'object', but unlike a real //object, it will not have any overall value return 'null'; } if( oType == 'undefined' ) { return 'window.uDfXZ0_d'; } if( oType == 'object' ) { //Safari throws errors when comparing non-objects with window/document/etc if( oVar == window ) { return 'window'; } if( oVar == document ) { return 'document'; } if( oVar == document.body ) { return 'document.body'; } if( oVar == document.documentElement ) { return 'document.documentElement'; } } if( oVar.nodeType && ( oVar.childNodes || oVar.ownerElement ) ) { return '{error:\'DOM node\'}'; } if( !notFirst ) { Object.prototype.toRecoverableString = function (oBn) { if( this.tempLockIgnoreMe ) { return '{\'LoopBack\'}'; } this.tempLockIgnoreMe = true; var retVal = '{', sepChar = '', j; for( var i in this ) { if( i == 'toRecoverableString' || i == 'tempLockIgnoreMe' || i == 'prototype' || i == 'constructor' ) { continue; } if( oBn && ( i == 'index' || i == 'input' || i == 'length' || i == 'toRecoverableObString' ) ) { continue; } j = this[i]; if( !i.match(hfr_basicObPropNameValStr) ) { //for some reason, you cannot use unescape when defining peoperty names inline for( var x = 0; x < cleanStrFromAr.length; x++ ) { i = i.replace(cleanStrFromAr[x],cleanStrToAr[x]); } i = '\''+i+'\''; } else if( window.ActiveXObject && navigator.userAgent.indexOf('Mac') + 1 && !navigator.__ice_version && window.ScriptEngine && ScriptEngine() == 'JScript' && i.match(/^\d+$/) ) { //IE mac does not allow numerical property names to be used unless they are quoted i = '\''+i+'\''; } retVal += sepChar+i+':'+hfr_getRecoverableString(j,true); sepChar = ','; } retVal += '}'; this.tempLockIgnoreMe = false; return retVal; }; Array.prototype.toRecoverableObString = Object.prototype.toRecoverableString; Array.prototype.toRecoverableString = function () { if( this.tempLock ) { return '[\'LoopBack\']'; } if( !this.length ) { var oCountProp = 0; for( var i in this ) { if( i != 'toRecoverableString' && i != 'toRecoverableObString' && i != 'tempLockIgnoreMe' && i != 'prototype' && i != 'constructor' && i != 'index' && i != 'input' && i != 'length' ) { oCountProp++; } } if( oCountProp ) { return this.toRecoverableObString(true); } } this.tempLock = true; var retVal = '['; for( var i = 0; i < this.length; i++ ) { retVal += (i?',':'')+hfr_getRecoverableString(this[i],true); } retVal += ']'; delete this.tempLock; return retVal; }; Boolean.prototype.toRecoverableString = function () { return ''+this+''; }; Date.prototype.toRecoverableString = function () { return 'new Date('+this.getTime()+')'; }; Function.prototype.toRecoverableString = function () { return this.toString().replace(/^\s+|\s+$/g,'').replace(/^function\s*\w*\([^\)]*\)\s*\{\s*\[native\s+code\]\s*\}$/i,'function () {[\'native code\'];}'); }; Number.prototype.toRecoverableString = function () { if( isNaN(this) ) { return 'Number.NaN'; } if( this == Number.POSITIVE_INFINITY ) { return 'Number.POSITIVE_INFINITY'; } if( this == Number.NEGATIVE_INFINITY ) { return 'Number.NEGATIVE_INFINITY'; } return ''+this+''; }; RegExp.prototype.toRecoverableString = function () { return '\/'+this.source+'\/'+(this.global?'g':'')+(this.ignoreCase?'i':''); }; String.prototype.toRecoverableString = function () { var oTmp = escape(this); if( oTmp == this ) { return '\''+this+'\''; } return 'unescape(\''+oTmp+'\')'; }; } if( !oVar.toRecoverableString ) { return '{error:\'internal object\'}'; } var oTmp = oVar.toRecoverableString(); if( !notFirst ) { //prevent it from changing for...in loops that the page may be using delete Object.prototype.toRecoverableString; delete Array.prototype.toRecoverableObString; delete Array.prototype.toRecoverableString; delete Boolean.prototype.toRecoverableString; delete Date.prototype.toRecoverableString; delete Function.prototype.toRecoverableString; delete Number.prototype.toRecoverableString; delete RegExp.prototype.toRecoverableString; delete String.prototype.toRecoverableString; } return oTmp; } var hfr_basicObPropNameValStr = /^\w+$/, cleanStrFromAr = new Array(/\\/g,/'/g,/"/g,/\r/g,/\n/g,/\f/g,/\t/g,new RegExp('-'+'->','g'),new RegExp('',' aChercher) { high = mid - 1; } else if (element < aChercher) { low = mid + 1; } else { return mid; } } return -1; }; // Inject your own CSS in the page. // Example: Do not underline link: // injectCSS("a{text-decoration: none;}") function injectCSS(cssdata){ head = document.getElementsByTagName("head")[0]; style = document.createElement("style"); style.setAttribute("type", 'text/css'); style.innerHTML = cssdata; head.appendChild(style); } // ============ Module d'auto update du script ============ ({ check4Update : function() { var autoUpdate = this; var mirrorUrl = GM_getValue('mirrorUrl', 'null'); if (mirrorUrl == 'null') autoUpdate.retrieveMirrorUrl(); var currentVersion = GM_getValue('currentVersion', '0.7.2'); // On met éventuellement la version stockée à jour avec la version courante, si la version courante est plus récente if (autoUpdate.isLater('0.7.2', currentVersion)) { GM_setValue('currentVersion', '0.7.2'); currentVersion = '0.7.2'; } // Par contre, si la version stockée est plus récente que la version courante -> création un menu d'update pour la dernière version else if (autoUpdate.isLater(currentVersion, '0.7.2')) { GM_registerMenuCommand("[HFR] Black List -> Installer la version " + currentVersion, function() { GM_openInTab(mirrorUrl + 'others/hfr_black_list.user.js'); } ); } // Si la version courante et la version stockée sont identiques, on ne fait rien if (GM_getValue('lastVersionCheck') == undefined || GM_getValue('lastVersionCheck') == '') GM_setValue('lastVersionCheck', new Date().getTime() + ''); // Pas eu de check depuis 24h, on vérifie... if ((new Date().getTime() - GM_getValue('lastVersionCheck')) > 86400000 && mirrorUrl != 'null') { var checkUrl = mirrorUrl + 'getLastVersion.php5?name=' + encodeURIComponent('[HFR] Black List'); if (isNaN(currentVersion.substring(currentVersion.length - 1))) checkUrl += '&sversion=' + currentVersion.substring(currentVersion.length - 1); GM_xmlhttpRequest({ method: "GET", url: checkUrl, onload: function(response) { var regExpVersion = new RegExp('^[0-9]{1,2}\.[0-9]{1,2}\.[0-9]{1,2}[a-zA-Z]?$'); var lastVersion = response.responseText; // Pas d'erreur et nouvelle version plus récente if (lastVersion != '-1' && regExpVersion.test(lastVersion) && autoUpdate.isLater(lastVersion, currentVersion)) { if (confirm('Une nouvelle version de [HFR] Black List est disponible : ' + lastVersion + '\nVoulez-vous l\'installer ?')) { GM_openInTab(mirrorUrl + 'others/hfr_black_list.user.js'); } else { // Mémorisation de la version refusée : elle servira de version de référence GM_setValue('currentVersion', lastVersion); } } GM_setValue('lastVersionCheck', new Date().getTime() + ''); } }); } }, max : function(v1, v2) { var tabV1 = v1.split('.'); var tabV2 = v2.split('.'); if (isNaN(tabV1[2].substring(tabV1[2].length - 1))) tabV1[2] = tabV1[2].substring(0, tabV1[2].length - 1); if (isNaN(tabV2[2].substring(tabV2[2].length - 1))) tabV2[2] = tabV2[2].substring(0, tabV2[2].length - 1); if ((tabV1[0] > tabV2[0]) || (tabV1[0] == tabV2[0] && tabV1[1] > tabV2[1]) || (tabV1[0] == tabV2[0] && tabV1[1] == tabV2[1] && tabV1[2] > tabV2[2])) { return v1; } else { return v2; } }, isLater : function(v1, v2) { return v1 != v2 && this.max(v1, v2) == v1; }, retrieveMirrorUrl : function() { var mirrors = 'http://hfr.toyonos.info/gm/;http://hfr-mirror.toyonos.info/gm/'.split(';'); var checkMirror = function (i) { var mirror = mirrors[i]; GM_xmlhttpRequest({ url: mirror + 'getLastVersion.php5', method: "HEAD", onload: function(response) { // Dès qu'un miroir répond, on le mémorise. if (response.status == 200) { GM_setValue('mirrorUrl', mirror); } else { // Sinon on test le prochain if ((i + 1) < mirrors.length) { checkMirror(i + 1); } else { GM_setValue('mirrorUrl', 'null'); } } } }); }; checkMirror(0); }, }).check4Update();