CleanURLs and SkipRedirects

Supprime les paramètres de suivi des URLs et passe les redirections

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

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

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

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

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

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @license         MIT
// @name            CleanURLs and SkipRedirects
// @name:en         CleanURLs and SkipRedirects
// @version         8.4
// @description     Supprime les paramètres de suivi des URLs et passe les redirections
// @description:en  Removes tracking parameters from URLs and skips redirects
// @author          LeDimiScript
// @match           *://*/*
// @grant           none
// @run-at          document-start
// @icon            
// @namespace       https://greasyfork.org/users/1291639
// ==/UserScript==
//
// ┌────────────────────────────────────────────┐
// │                  SOMMAIRE                  │
// ├────────────────────────────────────────────┤
// │ [1] Configuration                          │
// │ [2] Cas spéciaux                           │
// │ [3] Outils de décodage                     │
// │ [4] Nettoyage des URLs                     │
// │ [5] Barre d’adresse                        │
// │ [6] Balises <a>                            │
// │ [7] Observateurs DOM & URL                 │
// │ [8] Initialisation                         │
// └────────────────────────────────────────────┘

(function() {
    'use strict';

    // Quitter si le document n'est pas HTML
    if (document.doctype == null) return;



    // ===============[1]===============
    //           CONFIGURATION
    // =================================


    // ---------- Liste des protocoles pris en charge ----------

    const validProtocols = [
      'finger:',
      'ftp://',
      'ftps://',
      'freenet:',
      'gemini:',
      'gopher:',
      'http://',
      'https://',
      'ipfs:',
      'mailto:',
      'magnet:',
      'wap:',
      'xmpp:'
    ];


    // ---------- Liste des paramètres d'URL à nettoyer ----------

    const blacklist = [
      'a',
      'abcId',
      'accountId',
      'acnt',
      'ad',
      'adgroupid',
      'adgrp',
      'adgrpid',
      'ad-location',
      'ad_medium',
      'ad_name',
      'ad_pvid',
      'ad_sub',
      'ad_tags',
      'adt_ei',
      'ad_type',
      'advertising-id',
      'adword',
      'aem_p4p_detail',
      'af',
      'aff',
      'aff_fcid',
      'aff_fsk',
      'affiliate',
      'affiliateCode',
      'affiliationId',
      'aff_platform',
      'affparams',
      'aff_short_key',
      'afSmartRedirect',
      'aff_trace_key',
      'afftrack',
      'af_assettype_id',
      'af_creative_id',
      'af_id',
      'af_placement_id',
      'agid',
      'aid',
      'albag',
      'albch',
      'albcp',
      'algo_exp_id',
      'algo_pvid',
      'APID',
      'ap_id',
      'appver',
      'ar',
      'ascsubtag',
      'asc_contentid',
      'asgtbndr',
      'atc',
      'at_campaign',
      'at_campaign_group',
      'at_creation',
      'at_medium',
      'at_network',
      'at_platform',
      'ats',
      'at_term',
      'at_variant',
      'autostart',
      'bb',
      'bbn',
      'bdmtchtyp ',
      'bdmtchtyp+',
      'bizType',
      'block',
      'bp',
      '_bsa_req',
      'bta',
      'businessType',
      'c',
      'caifrq',
      'campaign',
      'campaignId',
      'campaignid',
      'campaign_id',
      'campid',
      'cd',
      '__cf_chl_rt_tk',
      '__cf_chl_tk',
      '__cf_chl_f_tk',
      '__cf_chl_jschl_tk__',
      '__cf_chl_captcha_tk__',
      '__cf_chl_managed_tk__',
      '__cf_chl_rt_tk__',
      'cha',
      'chb',
      'chbr',
      'chf',
      'child_sku',
      'chm',
      'chmd',
      'chn',
      'chnl',
      'chp',
      'chv',
      'cid',
      'ck',
      'clickid',
      'client_id',
      'cm_mmc',
      'cm_ven',
      'cmd',
      'cmpgn',
      'cnvs',
      'cod',
      'comId',
      'content-id',
      'country',
      'crea',
      'creative',
      'crid',
      'cs',
      'cst',
      'cti',
      'cts',
      'curPageLogUid',
      'custom3',
      'customid',
      'dc',
      'dchild',
      'dclid',
      'deals-widget',
      'device',
      'dgcid',
      'dib',
      'dib_tag',
      'dicbo',
      'dim1',
      'dim2',
      'discounts-widget',
      'docid',
      'ds',
      'dt',
      'dTag',
      'dv',
      'e',
      'e9s',
      'eclog',
      'edd',
      'edm_click_module',
      'ei',
      'email',
      'embed',
      'em_cmp',
      'emid',
      'em_src',
      '_encoding',
      'eventSource',
      'exp_price',
      'fbclid',
      'fdl',
      'feature',
      'febuild',
      'fid',
      'field',
      'field-lbr_brands_browse-bin',
      'fn',
      'forced_click',
      'fr',
      'freq',
      'from',
      'frs',
      '_ga',
      'ga_order',
      'ga_search_query',
      'ga_search_type',
      'ga_view_type',
      'gadid',
      'gatewayAdapt',
      'gbv',
      'gclid',
      'gclsrc',
      'gg_dev',
      'gh_jid',
      'goods_id',
      'googleloc',
      'gps-id',
      'gsAttrs',
      'gs_lcp',
      'gs_lp',
      'gt',
      'gtin',
      'guccounter',
      'hdtime',
      'helpid',
      'hosted_button_id',
      'hvadid',
      'hvbmt',
      'hvdev',
      'hvlocphy',
      'hvnetw',
      'hvqmt',
      'hvtargid',
      'hydadcr',
      'i',
      'ICID',
      'ico',
      'idOffre',
      'idzone',
      'ie',
      'iflsig',
      'ig_rid',
      'im',
      'index',
      'intake',
      'intcmp',
      'irclickid',
      'irgwc',
      'irpid',
      'isdl',
      'is_from_webapp',
      'itemId',
      'itemid',
      'itid',
      'itok',
      'ix',
      'kard',
      'katds_labels',
      'kb',
      'keyno',
      'keywords',
      'KwdID',
      'kwmt',
      'l10n',
      'landed',
      'ld',
      'linkCode',
      'loc_physical_ms',
      'ls',
      'mark',
      'mc',
      'mc_cid',
      'mcid',
      'md',
      'md5',
      'media',
      'merchantid',
      'mid',
      'mkcid',
      '__mk_de_DE',
      'mkevt',
      'mkgroupid',
      'mkrid',
      'mkscid',
      'mortyurl',
      'mp',
      'msadgroup',
      'mscampaign',
      'msclid',
      'msclkid',
      'msdevice',
      'msmatchtype',
      'msnetwork',
      'mstargetid',
      'mtchtyp',
      'mtctp',
      'nats',
      'nbbct',
      'nci',
      'netw',
      'nojs',
      'norover',
      'nsdOptOutParam',
      'ntwrk',
      'obOrigUrl',
      'offerId',
      'offerid',
      'offer_id',
      'opened-from',
      'opt_92',
      'optout',
      'oq',
      'organic_search_click',
      'origin',
      'os',
      'p',
      'p1',
      'p2',
      'p3',
      'pa',
      'Partner',
      'partner',
      'partner_id',
      'partner_ID',
      'pb',
      'pcampaignid',
      'pd_rd_i',
      'pd_rd_r',
      'pd_rd_w',
      'pd_rd_wg',
      'pdp_npi',
      'pf',
      'pf_rd_i',
      'pf_rd_m',
      'pf_rd_p',
      'pf_rd_r',
      'pf_rd_s',
      'pf_rd_t',
      'pg',
      'pgId',
      'PHPSESSID',
      'pid',
      'pk_campaign',
      'pdp_ext_f',
      'pkey',
      'Platform',
      'platform',
      'plkey',
      'pload',
      'plu',
      'pp',
      'pqr',
      'pr',
      'preselect',
      'pro',
      'prod',
      'prodid',
      'product_id',
      'product_partition_id',
      'prom',
      'promo',
      'promocode',
      'promoid',
      'provider',
      'psc',
      'psp',
      'psprogram',
      'psr',
      'pv',
      'pvid',
      'qid',
      'Query',
      'r',
      '_randl_currency',
      '_randl_shipto',
      'rb_css',
      'rb_geo',
      'rb_itemId',
      'rb_pgeo',
      'rb_plang',
      'realDomain',
      'recruiter_id',
      'ref',
      'ref_',
      'ref_src',
      'refcode',
      'referral',
      'referrer',
      'refinements',
      'reftag',
      'related_post_from',
      'retailer',
      'rf',
      'rlp',
      'rlid',
      'rlsatarget',
      'rnid',
      'rowan_id1',
      'rowan_msg_id',
      'rs',
      'ru',
      'rss',
      'sbo',
      'sCh',
      'sca_esv',
      'scene',
      'sclient',
      'scm',
      'scm_id',
      'scm-url',
      'sd',
      'search_id',
      'search_page',
      'search_page_position',
      'searchText',
      'sei',
      'sender_device',
      'service',
      'serviceIdserviceId',
      'sh',
      'shareId',
      'shownav',
      'showVariations',
      'si',
      'sid',
      '___SID',
      '.sig',
      'site',
      'site_id',
      'sk',
      'smid',
      'social_params',
      'soluteclid',
      'source',
      'sourceId',
      'sp',
      'sp_csd',
      'spLa',
      'spm',
      'spreadType',
      'sprefix',
      'sr',
      'src',
      '_src',
      'src_cmp',
      'src_player',
      'src_src',
      'srcSns',
      'start_radio',
      'su',
      'supplier',
      'sxin_0_pb',
      'syslcid',
      '_t',
      'tag',
      't_agid',
      'targetid',
      'tcampaign',
      't_cid',
      't_crid',
      't_crname',
      'td',
      't_device',
      'terminal_id',
      'test',
      'text',
      'tgt',
      'th',
      'tl',
      't_match_type',
      't_network',
      'token',
      'tokenId',
      'toolid',
      'tracelog',
      'trafficChannelId',
      'traffic_id',
      'traffic_source',
      'traffic_type',
      'trgt',
      '.ts',
      't_s',
      'tt',
      'tuid',
      't_validation',
      'tz',
      'uact',
      'ug_edm_item_id',
      'ui',
      'uilcid',
      '_ul',
      'url_from',
      'userId',
      'utm',
      'utm1',
      'utm2',
      'utm3',
      'utm4',
      'utm5',
      'utm6',
      'utm7',
      'utm8',
      'utm9',
      'utm_campaign',
      'utm_channel',
      'utm_content',
      'utm_feed',
      'utm_hash',
      'utm_id',
      'utm_medium',
      'utm_productid',
      'utm_source',
      'utm_source_platform',
      'utm_term',
      'uuid',
      'utype',
      'var',
      'variant',
      'variant_id',
      'variant_sku_code',
      'vcn',
      'vcv',
      've',
      'ved',
      'wait',
      'wcks',
      'wgl',
      'wprov',
      'x',
      '_xiid',
      'xpid',
      'y',
      'zone',
      'zoneid'
    ];


    // ---------- Liste des paramètres de hash à nettoyer ----------

    const hash = [
      '!psicash',
      'back-url',
      'back_url',
      'dealsGridLinkAnchor',
      'ebo',
      'int',
      'intcid',
      'mpos',
      'niche-',
      'searchinput',
      'src',
      'xtor'
    ];


    // ---------- Liste des paramètres de redirection ----------

    const redirectParams = [
      'continue',
      'ds_dest_url',
      'kaRdt',
      'lp',
      'rdr',
      'redirect',
      'redirect_uri',
      'spld',
      'target',
      'tURL',
      'u',
      'url',
      'url64fb'
    ];



    // ===============[2]===============
    //           CAS SPÉCIAUX
    // =================================

    /**
     * Réécrit les URL pour certains sites spécifiques :
     * - Amazon
     * - Wikimedia
     */
    function rewriteSpecialCases(url) {


        // ---------- Cas Amazon ----------

        // Domaines Amazon
        const amazonDomains = [
          'amazon.com',
          'amazon.be',
          'amazon.ae',
          'amazon.ca',
          'amazon.co.uk',
          'amazon.com.au',
          'amazon.com.br',
          'amazon.com.mx',
          'amazon.com.tr',
          'amazon.de',
          'amazon.es',
          'amazon.fr',
          'amazon.in',
          'amazon.it',
          'amazon.nl',
          'amazon.pl',
          'amazon.sa',
          'amazon.se',
          'amazon.sg'
        ];
        const isAmazon = amazonDomains.some(domain => url.hostname.endsWith(domain));
        if (isAmazon && url.pathname.match(/^\/s/) && url.searchParams.has('keywords')) {
            // Unifie le chemin
            url.pathname = '/s';
            // Remplace 'keywords' par 'k'
            const keywords = url.searchParams.get('keywords');
            url.searchParams.delete('keywords');
            url.searchParams.set('k', keywords);
        }


        // ---------- Cas Wikimedia ----------

        // Domaines Wikimedia
        const wikimediaDomains = [
          'mediawiki.org',
          'wikibooks.org',
          'wikidata.org',
          'wikimedia.org',
          'wikinews.org',
          'wikipedia.org',
          'wikiquote.org',
          'wikisource.org',
          'wikiversity.org',
          'wikivoyage.org',
          'wiktionary.org'
        ];
        const isWikimedia = wikimediaDomains.some(domain => url.hostname.endsWith(domain));
        if (isWikimedia && url.pathname.match(/^\/w\/index.php/) && url.searchParams.has('title')) {
            // Récupère la valeur du paramètre 'title'
            const title = url.searchParams.get('title');
            // Supprime le paramètre 'title' de l'URL
            url.searchParams.delete('title');
            // Modifie le chemin en fonction de la valeur de 'title'
            url.pathname = `/wiki/${encodeURIComponent(title)}`;
        }
    }



    // ===============[3]===============
    //        OUTILS DE DÉCODAGE
    // =================================


    // ---------- Décodage en URL ----------

    function tryUrlDecode(value) {
        if (!value) return null;
        try {
            let decoded = decodeURIComponent(value);
            return decoded;
        } catch (e) {
            return null; // Cas de chaîne mal formée
        }
    }


    // ---------- Décodage en base64 ----------

    function tryBase64Decode(value) {
        if (!value) return null;
        try {
            // Corrige le format base64url → base64 standard
            let base64 = value.replace(/-/g, '+').replace(/_/g, '/');
            while (base64.length % 4 !== 0) base64 += '=';
            const decoded = atob(base64);
            return decoded;
        } catch (e) {
            return null;
        }
    }


    // ---------- Décodage JSON ----------

    function tryDecodeJson(value) {
        if (!value) return null;

        // Étape 1 : tentative sans décodage URL
        try {
            return JSON.parse(value);
        } catch (e1) {
          
            // Étape 2 : tentative après décodage URL
            const decoded = tryUrlDecode(value);
            if (decoded) {
                try {
                    return JSON.parse(decoded);
                } catch (e2) {
                    return null;
                }
            }
            return null;
        }
    }


    /**
     * Tente de décoder une URL de redirection encodée :
     * - en URL
     * - en base64
     * - en json
     * Retourne l'url décodée ou null
     */
    function tryDecodeRedirectUrl(value) {
        if (!value) return null;

        // 1. Essayer un décodage URL
        const urlDecoded = tryUrlDecode(value);
        for (const protocol of validProtocols) {
            if (urlDecoded && urlDecoded.startsWith(protocol)) {
                return urlDecoded;
            }
        }

        // 2. Essayer un décodage base64, puis, si besoin, décodage URL
        let temp = value;
        for (let i = 0; i < 4; i++) {
            const base64Decoded = tryBase64Decode(temp);
            if (base64Decoded) {
                // On teste d'abord base64Decoded, puis sa version URL décodée
                const candidates = [base64Decoded, tryUrlDecode(base64Decoded)];
                for (const candidate of candidates) {
                    if (!candidate) continue;
                    for (const protocol of validProtocols) {
                        const index = candidate.indexOf(protocol);
                        if (index !== -1) {
                            return candidate.substring(index);
                        }
                    }
                }
            }
            // Supprime le premier caractère pour la prochaine itération
            temp = temp.substring(1);
        }

        // 3. Essayer un décodage JSON
        const json = tryDecodeJson(value);
        if (json && typeof json === 'object') {
            // On cherche un champ contenant une URL potentielle
            for (const param of redirectParams) {
                const possible = json[param];
                if (typeof possible === 'string') {
                    const candidate = tryDecodeRedirectUrl(possible);
                    if (candidate) {
                        return candidate;
                    }
                }
            }
        }

        // 4. Aucun résultat valide : ce n'était pas une URL de redirection
        return null;
    }



    // ===============[4]===============
    //        NETTOYAGE DES URLS
    // =================================


    /**
     * Suit les redirections et nettoie l'URL en plusieurs étapes :
     * - traitement du chemin
     * - nettoyage des paramètres
     * - nettoyage du hash
     * S'assure qu'il n'y a pas de paramètres encodés en URL.
     */
    function cleanUrl(url) {
        if (typeof url === 'string') url = new URL(url);


       // ---------- Passage des redirections ----------

        const urlString = url.href;
        const originalOrigin = url.origin;
        let lastValidRedirect = url;

        // 1. Traitement des urls mal encodées
        for (const param of redirectParams) {
          let startIndex = 0;
          const pattern = `${param}=`;
          while (true) {
            const index = urlString.indexOf(pattern, startIndex);
            if (index === -1) break;
            const rawValue = urlString.slice(index + pattern.length);
            for (const protocol of validProtocols) {
              if (rawValue.startsWith(protocol)) {
                try {
                  const nextUrl = new URL(rawValue);
                  if (nextUrl.origin !== originalOrigin) {
                    return cleanUrl(nextUrl);
                  } else {
                    lastValidRedirect = cleanUrl(nextUrl);
                  }
                } catch {
                  return null;
                }
                break;
              }
            }
            startIndex = index + pattern.length;
          }
        }

        // 2. Traitement classique
        for (const param of redirectParams) {
          const values = url.searchParams.getAll(param);
          for (const rawValue of values) {
            const decodedRedirect = tryDecodeRedirectUrl(rawValue);
            if (decodedRedirect) {
              const decodedUrl = new URL(decodedRedirect);
              if (decodedUrl.origin !== originalOrigin) {
                return cleanUrl(decodedUrl);
              } else {
                lastValidRedirect = cleanUrl(decodedUrl);
              }
            } else {
              lastValidRedirect.searchParams.delete(param);
            }
          }
        }

        // 3. Retourne l'url de destination
        url = lastValidRedirect;


        // ---------- Nettoyage de l'URL ----------

        // Appliquer les réécriture spécifiques
        rewriteSpecialCases(url);
        let href = url.href;

        // 1. Supprimer les paramètres dans le chemin
        const path = url.pathname;
        const pathParts = path.split('/').filter(part => {
            if (!part.includes('=')) return true;
            const [key, ...rest] = part.split('=');
            const value = rest.join('=');
            // Supprime si la clé est blacklistée ou la valeur est vide
            return !!value && !blacklist.includes(key);
        });
        // Reconstruire le chemin avec ce qu'il reste
        url.pathname = pathParts.join('/');

        // 2. Supprime les paramètres vides et tous les paramètres blacklistés
        if (url.search.includes('=')) {
            const newParams = new URLSearchParams();
            for (const [key, value] of url.searchParams.entries()) {
                  if (!!value && !blacklist.includes(key)) {
                      newParams.append(key, value);
                  }
            }
            url.search = newParams.toString();
        }

        // 3. Supprime les hash vides et tous les paramètres blacklistés dans hash
        if (url.hash) {
          // Enlève le '#'
          let hashContent = url.hash.substring(1);
          if (hashContent.includes('=')) {
              // Parser comme une liste de paramètres
              const hashParams = new URLSearchParams(hashContent);
              // Nettoyage
              const newHashParams = new URLSearchParams();
              for (const [key, value] of hashParams.entries()) {
                  if (!!value && !hash.includes(key)) {
                      newHashParams.append(key, value);
                  }
              }
              // Reconstruire le hash avec ce qu'il reste
              url.hash = newHashParams.toString() ? '#' + newHashParams.toString() : '';
            }
        }


        // ---------- Décodage des paramètres restants ----------

        let finalURL = url.href;
        const newParams = new URLSearchParams();

        // 1. Parcourir tous les paramètres de l'URL
        for (const [key, value] of url.searchParams.entries()) {
            // Décoder la valeur du paramètre
            let decodedValue = tryUrlDecode(value);
            // On cherche si la valeur est une suite de paramètres encodés
            if (decodedValue && decodedValue.includes('=')) {
                const innerParams = new URLSearchParams(decodedValue);

                // Si la valeur décodée ressemble à une série de paramètres, les ajouter comme nouveaux paramètres en remplaçant la clef par sa valeur décodée
                for (const [innerKey, innerValue] of innerParams.entries()) {
                    newParams.append(innerKey, innerValue);  // Ajouter chaque paramètre comme s'il était indépendant
                }
                // On ne garde pas le param original, car il est remplacé par ses composants décodés
            } else {
                // Si ce n'est pas une suite de paramètres, ajouter le paramètre normalement
                newParams.append(key, decodedValue);
            }
        }

        // 2. Remplacer les paramètres dans l'URL nettoyée avec les nouveaux paramètres décodés
        url.search = newParams.toString();

        // 3. Si l'URL a changé suite à ce décodage des paramètres, rappeler cleanUrl
        if (url.href !== finalURL) {
            url = cleanUrl(url);
        }

        return url;  // Retourner l'URL nettoyée avec les paramètres réécrits
    }



    // ===============[5]===============
    //         BARRE D'ADRESSE
    // =================================

    // Modifie l'URL actuelle dans la barre d'adresse pour supprimer les paramètres de suivi
    function modifyURL() {
        const url = new URL(window.location.href);
        const cleanedUrl = cleanUrl(url);
        const newURL = cleanedUrl.href;
        if (window.location.href !== newURL) {
            window.history.replaceState(null, '', newURL);
        }
    }



    // ===============[6]===============
    //           BALISES <a>
    // =================================


    // ---------- Redéfinir la propriété `href` de HTMLAnchorElement ----------

    (function redefineHrefSetter() {
        const descriptor = Object.getOwnPropertyDescriptor(HTMLAnchorElement.prototype, 'href');
        Object.defineProperty(HTMLAnchorElement.prototype, 'href', {
            get: descriptor.get,
            set(value) {
                try {
                    const cleaned = validateAndCleanUrl(value);
                    descriptor.set.call(this, cleaned ?? value);
                } catch (e) {
                    descriptor.set.call(this, value);
                }
            }
        });
    })();


    // ---------- Nettoyer une URL et éventuellement suivre la redirection ----------

    /**
     * Valide et nettoie une URL.
     * Retourne une URL absolue nettoyée ou null si invalide.
     */
    function validateAndCleanUrl(href) {
        if (!href) return null; // Ignore les valeurs vides
        try {
            // Transforme les href en URL absolue
            const url = new URL(href, window.location.href);
            // Nettoie l'URL
            let cleaned = cleanUrl(url);
            // Retourne l'URL nettoyée (toujours absolue)
            return cleaned.href;
        } catch (e) {
            return null;
        }
    }


    // ---------- Intercepter les modifications de href via setAttribute() ----------

    (function redefineSetAttribute() {
        const original = Element.prototype.setAttribute;
        Element.prototype.setAttribute = function(name, value) {
            if (this.tagName === 'A' && name.toLowerCase() === 'href') {
                try {
                    const cleaned = validateAndCleanUrl(value);
                    if (cleaned) {
                        return original.call(this, name, cleaned);
                    }
                } catch (e) {}
            }
            return original.call(this, name, value);
        };
    })();


    // ---------- Traiter tous les liens dans un conteneur ----------

    /**
     * Parcourir tous les liens du conteneur,
     * nettoyer leur attribut href si nécessaire,
     * et remplacer par l’URL nettoyée.
     */
    function handleLinks(container = document) {
        const links = container.getElementsByTagName('a');
        for (const link of links) {
            try {
                // On récupère l'attribut brut, sans résolution automatique en URL absolue
                if (!link.hasAttribute('href')) continue;
                const originalHref = link.getAttribute('href');
                const cleanedHref = validateAndCleanUrl(originalHref);
                // Si une URL nettoyée valide existe et qu'elle est différente, on remplace
                if (cleanedHref && cleanedHref !== originalHref) {
                    link.setAttribute('href', cleanedHref);
                }
            } catch (e) {}
        }
    }



    // ===============[7]===============
    //      OBSERVATEURS DOM & URL
    // =================================


    // ---------- Configure un MutationObserver pour nettoyer dynamiquement les URLs ----------

    function setupMutationObserver() {
        const observer = new MutationObserver(function(mutations) {
            mutations.forEach(function(mutation) {
                for (const node of mutation.addedNodes) {
                    if (node.nodeType === Node.ELEMENT_NODE) {
                        handleLinks(node);
                    }
                }
            });
        });
        function tryObserve() {
            if (document.body) {
                observer.observe(document.body, { childList: true, subtree: true });
            } else {
                requestAnimationFrame(tryObserve);
            }
        }
        tryObserve();
    }


    // ---------- Observation des changement dynamique de l'URL ----------

    function observeURLChanges() {
        let lastUrl = window.location.href;
        const observer = new MutationObserver(() => {
            const currentUrl = window.location.href;
            if (currentUrl !== lastUrl) {
                modifyURL();
                lastUrl = currentUrl;
            }
        });
        observer.observe(document, { subtree: true, childList: true });
        // Observe les ajouts de hash
        window.addEventListener('hashchange', () => {
            const currentUrl = window.location.href;
            if (currentUrl !== lastUrl) {
                modifyURL();
                lastUrl = currentUrl;
            }
        });
    }



    // ===============[8]===============
    //          INITIALISATION
    // =================================


    // ---------- Nettoyage initial ----------

    modifyURL();


    // ---------- Nettoyage une fois la page chargée ----------

    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', function() {
            handleLinks();
            setupMutationObserver();
            observeURLChanges();
        });
    } else {
        handleLinks();
        setupMutationObserver();
        observeURLChanges();
    }
})();