/* jshint esversion: 6, scripturl: true */
/* globals window, document, URL, setTimeout */
/* globals mw, $, importScript, MonobookToolbar, CadreOnglets_Init */
/* exported obtenir, addSpecialCharset, addSpecialCharsetHTML, addCustomButton */

/**
 * N'importe quel JavaScript ici sera chargé pour n'importe quel utilisateur et pour chaque page accédée.
 *
 * ATTENTION : Avant de modifier cette page, veuillez tester vos changements avec votre propre
 * vector.js. Une erreur sur cette page peut faire bugger le site entier (et gêner l'ensemble des
 * visiteurs), même plusieurs heures après la modification !
 *
 * Prière de ranger les nouvelles fonctions dans les sections adaptées :
 * - Fonctions JavaScript
 * - Fonctions spécifiques pour MediaWiki
 * - Applications spécifiques à la fenêtre d'édition
 * - Applications qui peuvent être utilisées sur toute page
 * - Applications spécifiques à un espace de nom ou une page
 *
 * <nowiki> /!\ Ne pas retirer cette balise
 */

// Encapsulation de tout le code dans une IIFE globale
// (le code doit être exécuté immédiatement, donc ne pas mettre ici de mw.loader.using(), etc.)
( function () {
	'use strict';

	/*
	 * Les fonctions appliquées au contenu de page sont ajoutées à cet array,
	 * et à la fin du script ces fonctions sont exécutées dans un hook "wikipage.content",
	 * mais avec l'élément approprié pour le contenu de page.
	 */
	const pageContentHooks = ;


	/***********************/
	/* Fonctions générales */
	/***********************/

	/**
	 * Projet JavaScript
	 */
	window.obtenir = function ( name ) {
		if ( mw.loader.getState( 'ext.gadget.' + name ) !== null ) {
			mw.loader.load( 'ext.gadget.' + name );
		} else {
			importScript( 'MediaWiki:Gadget-' + name + '.js' );
		}
	};


	/****************************************/
	/* Applications pour l'ensemble du site */
	/****************************************/

	/**
	 * Transformer les pages du Bistro, du BA et les pages spécifiées en page de discussion
	 */
	if ( mw.config.get( 'wgNamespaceNumber' ) >= 2 && mw.config.get( 'wgNamespaceNumber' ) % 2 === 0 ) {
		$( function ( $ ) { // eslint-disable-line no-unused-vars
			var alwaysTransform = ( function () {
				if ( !.includes( mw.config.get( 'wgNamespaceNumber' ) ) ) {
					return false;
				}
				// voir ] et par ailleurs ]
				var pageName = mw.config.get( 'wgPageName' ).replace( /_/g, ' ' );
				var basePages = [
					"Wikipédia:Le Bistro",
					"Wikipédia:Bulletin des administrateurs",
					"Wikipédia:Bulletin des bureaucrates",
					"Wikipédia:Bulletin des patrouilleurs",
					"Wikipédia:Demande d'intervention sur une page protégée",
					"Wikipédia:Demande d'intervention sur un message système",
					"Wikipédia:Demande de protection de page",
					"Wikipédia:Demande de fusion d'historiques",
					"Wikipédia:Demande de purge d'historique",
					"Wikipédia:Demande de renommage",
					"Wikipédia:Demande de suppression immédiate",
					"Wikipédia:Demande de restauration de page",
					"Wikipédia:Bot/Requêtes",
					"Wikipédia:Forum de relecture",
					"Wikipédia:Le salon de médiation",
					"Wikipédia:Legifer",
					"Wikipédia:Pages à fusionner",
					"Wikipédia:Pages à scinder",
					"Wikipédia:Questions techniques",
					"Wikipédia:Requête aux administrateurs",
					"Wikipédia:Sondage",
					"Wikipédia:Vandalisme en cours",
					"Wikipédia:Bulletin du filtrage",
					"Wikipédia:AbuseFilter/Requêtes",
					"Wikipédia:AbuseFilter/Faux positifs",
					"Wikipédia:Oracle",
					"Projet:Modèle/Demandes",
					"Projet:Infobox/Demandes",
				];
				return basePages.some( function ( basePage ) {
					return pageName === basePage || pageName.startsWith( basePage + '/' );
				} );
			} )();
			if ( alwaysTransform ) {
				document.body.classList.replace( 'ns-subject', 'ns-talk' );
			} else {
				pageContentHooks.push( function ( $pageContent ) { // eslint-disable-line no-unused-vars
					if ( $pageContent.find( '#transformeEnPageDeDiscussion' ).length ) {
						document.body.classList.replace( 'ns-subject', 'ns-talk' );
					} else {
						// on repose sur le fait que le code est exécuté uniquement dans les namespaces pairs (subject),
						// autrement la ligne suivante irait transformer à tort des pages "talk" en "subject"
						document.body.classList.replace( 'ns-talk', 'ns-subject' );
					}
				} );
			}
		} );
	}


	/**
	 * Tout ce qui concerne la page d'édition
	 */
	if ( .includes( mw.config.get( 'wgAction' ) ) ) {

		// chargement de ]
		mw.loader.load( 'ext.gadget.CommonEdit' );

		// pour que les fonctions soient définies dès maintenant,
		// mais l'exécution réelle ne se fait qu'une fois le module chargé
		window.addSpecialCharset = function ( title, chars ) {
			mw.hook( 'CommonEdit.charsetFunctions.ready' ).add( function ( charsetFunctions ) {
				charsetFunctions.addSpecialCharset( title, chars );
			} );
		};
		window.addSpecialCharsetHTML = function ( title, charsHTML ) {
			mw.hook( 'CommonEdit.charsetFunctions.ready' ).add( function ( charsetFunctions ) {
				charsetFunctions.addSpecialCharsetHTML( title, charsHTML );
			} );
		};

		// fonction pour ajouter un bouton à la fin de la barre d'outils
		// permet d'utiliser ] sans se préoccuper de son chargement
		window.addCustomButton = ( function () {
			var promise;

			return function () {
				var buttonArguments = arguments;

				if ( !promise ) {
					promise = mw.loader.using( 'ext.gadget.MonobookToolbar' );
				}

				promise.done( function () {
					MonobookToolbar.addButton.apply( MonobookToolbar, buttonArguments );
				} );
			};
		} )();

	} else {
		// pour que les fonctions soient toujours définies,
		// afin d'éviter aux scripts utilisateur de planter
		window.addSpecialCharset = function () {};
		window.addSpecialCharsetHTML = function () {};
		window.addCustomButton = function () {};
	}


	/**
	 * Réécriture des titres
	 *
	 * Fonction utilisée par ]
	 *
	 * La fonction cherche un bandeau de la forme
	 * <div id="RealTitleBanner">
	 *   <span id="RealTitle">titre</span>
	 * </div>
	 *
	 * Un élément comportant id="DisableRealTitle" désactive la fonction
	 */
	function rewritePageTitle( $ ) {
		var $realTitle, titleHtml, $h1, $header,
			$realTitleBanner = $( '#RealTitleBanner' );
		if ( $realTitleBanner.length && !$( '#DisableRealTitle' ).length ) {
			$realTitle = $( '#RealTitle' );
			$h1 = $( '.mw-first-heading' );
			if ( mw.config.get( 'skin' ) === 'vector-2022' ) {
				$header = $( '.mw-body-header' );
			} else {
				$header = $h1;
			}
			if ( $realTitle.length && $h1.length && $header.length ) {
				titleHtml = $realTitle.html();
				if ( titleHtml === '' ) {
					$h1.hide();
				} else {
					$h1.html( titleHtml );
					if ( mw.config.get( 'wgAction' ) === 'view' && !mw.config.get( 'wgDiffNewId' ) ) { // page views, exclude diff pages
						// using a callback for replacement, to prevent interpreting "$" characters that realTitle might contain
						document.title = document.title.replace( /^.+(  Wikipédia)$/, function ( match, p1 ) {
							return $realTitle.text() + p1;
						} );
					}
				}
				$realTitleBanner.hide();
				// voir ] et par ailleurs ]
				$( '<p>' ).css( 'font-size', '80%' )
					.append( 'Titre à utiliser pour créer un lien interne : ', $( '<b>' ).text( mw.config.get( 'wgPageName' ).replace( /_/g, ' ' ) ) )
					.insertAfter( $header );
			}
		}
	}
	$( rewritePageTitle ); // TODO: éventuellement, compatibilité avec l'aperçu rapide

	/**
	 * Ajout d'un sous-titre
	 *
	 * Fonction utilisée par ] et quelques modules de taxobox
	 *
	 * La fonction cherche un élément de la forme
	 * <span id="sous_titre_h1">Sous-titre</span>
	 */
	function sousTitreH1( $pageContent ) {
		$( '#sous_titre_h1_moved' ).remove();
		var $span = $pageContent.find( '#sous_titre_h1' );
		if ( $span.length ) {
			$span.attr( 'id', 'sous_titre_h1_moved' );
			$span.prepend( ' ' );
			$( '.mw-first-heading' ).append( $span );
		}
	}
	pageContentHooks.push( sousTitreH1 );

	/**
	 * permet d'ajouter un petit lien (par exemple d'aide) à la fin du titre d'une page.
	 * utilisé par ]
	 * known bug : conflit avec le changement de titre classique.
	 * Pour les commentaires, merci de contacter ].
	 */
	function rewritePageH1bis( $ ) { // eslint-disable-line no-unused-vars
		var helpPage = document.getElementById( 'helpPage' );
		if ( helpPage ) {
			var h1 = document.getElementsByClassName( 'mw-first-heading' );
			if ( h1 ) {
				h1.innerHTML += '<span id="h1-helpPage">' + helpPage.innerHTML + '</span>';
			}
		}
	}
	$( rewritePageH1bis ); // TODO: éventuellement, compatibilité avec l'aperçu rapide


	/**
	 * Boîtes déroulantes
	 *
	 * Pour ]
	 */

	var Palette_Derouler = '';
	var Palette_Enrouler = '';

	var Palette_max = 1;

	function Palette_toggle( $table ) {
		/*
		direct children, car il ne faut pas prendre les lignes des éventuelles tables imbriquées
		table > tbody (peut-être aussi thead à l'avenir) > tr

		ensuite, on applique à toutes les lignes sauf la première
		*/
		$table.children().children( 'tr' ).slice( 1 ).toggleClass( 'navboxHidden' );
	}

	function Palette( $pageContent ) {
		var tableToGroup = new WeakMap();
		var groupLengths = new WeakMap();

		var $tables = $pageContent.find( '.collapsible' );

		$tables.each( function ( _, table ) {
			var group = table.parentNode.closest( '.navbox-container, .collapsible' );
			if ( group ) {
				tableToGroup.set( table, group );
				groupLengths.set( group, ( groupLengths.get( group ) || 0 ) + 1 );
			}
		} );

		$tables.each( function ( _, table ) {
			var $table = $( table );

			var collapsed = false;
			if ( table.classList.contains( 'autocollapse' ) ) {
				var group = tableToGroup.get( table );
				if ( group && groupLengths.get( group ) > Palette_max ) {
					collapsed = true;
				}
			} else if ( table.classList.contains( 'collapsed' ) ) {
				collapsed = true;
			}

			// le modèle dispose d'une classe "navbox-title",
			// sauf que les palettes "inlinées" (e.g. « class="navbox collapsible collapsed" ») n'ont pas cette classe
			$table.find( 'tr' ).eq( 0 ).find( 'th' ).eq( 0 ).prepend(
				$( '<span class="navboxToggle">\xA0</span>' ).append(
					$( '<a href="javascript:">' + ( collapsed ? Palette_Derouler : Palette_Enrouler ) + '</a>' ).click( function ( e ) {
						e.preventDefault();
						if ( this.textContent === Palette_Enrouler ) {
							this.textContent = Palette_Derouler;
						} else {
							this.textContent = Palette_Enrouler;
						}
						Palette_toggle( $table );
					} )
				)
			);
			if ( collapsed ) {
				Palette_toggle( $table );
			}
		} );

		// for garbage collection
		tableToGroup = null;
		groupLengths = null;
		$tables = null;

		// permet de dérouler/enrouler les palettes en cliquant n'importe où sur l'entête
		// (utilisation de la classe "navbox-title", comme ça seules les vraies palettes utilisant le modèle sont ciblées)
		$pageContent.find( '.navbox-title' )
			.click( function ( e ) {
				if ( e.target.closest( 'a' ) ) {
					return;
				}
				$( this ).find( '.navboxToggle a' ).click();
			} )
			.css( 'cursor', 'pointer' );
	}

	pageContentHooks.push( Palette );


	/**
	 * Pour ]
	 */

	var BoiteDeroulante_Derouler = '';
	var BoiteDeroulante_Enrouler = '';

	function BoiteDeroulante_toggle( NavToggle ) {
		var NavFrame = NavToggle.parentNode;

		var caption = ;
		caption = NavFrame.dataset.boiteDeroulanteDerouler;
		caption = NavFrame.dataset.boiteDeroulanteEnrouler;

		var $NavContent = $( NavFrame ).find( '.NavContent' ).eq( 0 );

		if ( NavToggle.textContent === caption ) {
			NavToggle.textContent = caption;
			$NavContent.hide();
		} else {
			NavToggle.textContent = caption;
			$NavContent.show();
		}
	}

	function BoiteDeroulante( $pageContent ) {

		$pageContent.find( '.NavFrame' ).each( function ( _, NavFrame ) {
			var CustomTexts, Derouler, Enrouler, NavToggle;

			if ( NavFrame.title && NavFrame.title.includes( '/' ) ) {
				CustomTexts = NavFrame.title.split( '/' );
				Derouler = CustomTexts;
				Enrouler = CustomTexts;
			} else {
				Derouler = BoiteDeroulante_Derouler;
				Enrouler = BoiteDeroulante_Enrouler;
			}
			NavFrame.title = '';
			NavFrame.dataset.boiteDeroulanteDerouler = Derouler;
			NavFrame.dataset.boiteDeroulanteEnrouler = Enrouler;

			NavToggle = document.createElement( 'a' );
			NavToggle.className = 'NavToggle';
			NavToggle.href = 'javascript:';
			NavToggle.onclick = function ( e ) {
				e.preventDefault();
				BoiteDeroulante_toggle( e.target );
			};
			NavToggle.textContent = Enrouler;

			NavFrame.prepend( NavToggle );

			BoiteDeroulante_toggle( NavToggle );
		} );

		// permet de dérouler/enrouler les boîtes en cliquant n'importe où sur l'entête
		$pageContent.find( '.NavHead' )
			.click( function ( e ) {
				if ( e.target.closest( 'a' ) ) {
					return;
				}
				var toggle = $( this ).siblings( 'a.NavToggle' );
				if ( toggle ) {
					toggle.click(); // pas du jquery, mais du vanilla js
				}
			} )
			.css( 'cursor', 'pointer' );
	}

	pageContentHooks.push( BoiteDeroulante );


	/**
	 * Script pour alterner entre plusieurs cartes de géolocalisation
	 */

	function GeoBox_Init( $pageContent ) {
		// noter qu'une classe "imgtoggle" (sans l'underscore) est aussi présente sur le wiki, sans rapport avec celle-ci
		$pageContent.find( '.img_toggle' ).each( function ( i, Container ) {
			Container.id = 'img_toggle_' + i;
			var Boxes = $( Container ).find( '.geobox' );
			if ( Boxes.length < 2 ) {
				return;
			}
			var ToggleLinksDiv = document.createElement( 'ul' );
			ToggleLinksDiv.id = 'geoboxToggleLinks_' + i;
			Boxes.each( function ( a, ThisBox ) {
				ThisBox.id = 'geobox_' + i + '_' + a;
				var ThisAlt;
				var ThisImg = ThisBox.getElementsByTagName( 'img' );
				if ( ThisImg ) {
					ThisAlt = ThisImg.alt;
				}
				if ( !ThisAlt ) {
					ThisAlt = 'erreur : description non trouvée';
				}
				var toggle = document.createElement( 'a' );
				toggle.id = 'geoboxToggle_' + i + '_' + a;
				toggle.textContent = ThisAlt;
				toggle.href = 'javascript:';
				toggle.onclick = function ( e ) {
					e.preventDefault();
					GeoBox_Toggle( this );
				};
				var Li = document.createElement( 'li' );
				Li.append( toggle );
				ToggleLinksDiv.append( Li );
				if ( a === 0 ) {
					toggle.style.color = '#888';
					toggle.style.pointerEvents = 'none';
				} else {
					ThisBox.style.display = 'none';
				}
			} );
			Container.append( ToggleLinksDiv );
		} );
	}

	function GeoBox_Toggle( link ) {
		var matches = link.id.match( /^geoboxToggle_(\d+)_(\d+)$/ );
		if ( !matches ) {
			return;
		}
		var ImgToggleIndex = matches;
		var GeoBoxIndex = matches;
		var ImageToggle = document.getElementById( 'img_toggle_' + ImgToggleIndex );
		var Links = document.getElementById( 'geoboxToggleLinks_' + ImgToggleIndex );
		var Geobox = document.getElementById( 'geobox_' + ImgToggleIndex + '_' + GeoBoxIndex );
		var Link = document.getElementById( 'geoboxToggle_' + ImgToggleIndex + '_' + GeoBoxIndex );
		if ( !ImageToggle || !Links || !Geobox || !Link ) {
			return;
		}
		$( ImageToggle ).find( '.geobox' ).each( function ( _, ThisgeoBox ) {
			if ( ThisgeoBox === Geobox ) {
				ThisgeoBox.style.display = '';
			} else {
				ThisgeoBox.style.display = 'none';
			}
		} );
		$( Links ).find( 'a' ).each( function ( _, thisToggleLink ) {
			if ( thisToggleLink === Link ) {
				thisToggleLink.style.color = '#888';
				thisToggleLink.style.pointerEvents = 'none';
			} else {
				thisToggleLink.style.color = '';
				thisToggleLink.style.pointerEvents = '';
			}
		} );
	}

	pageContentHooks.push( GeoBox_Init );


	/**
	 * Configuration du tri des diacritique dans les tables de class "sortable"
	 */
	mw.config.set( 'tableSorterCollation', {'à':'a', 'â':'a', 'æ':'ae', 'é':'e', 'è':'e', 'ê':'e', 'î':'i', 'ï':'i', 'ô':'o', 'œ':'oe', 'û':'u', 'ç':'c',  } );


	/**
	 * Direct imagelinks to Commons
	 *
	 * Required modules: mediawiki.util, user.options
	 *
	 * @source www.mediawiki.orghttps://wiki95.com/fr/Snippets/Direct_imagelinks_to_Commons
	 * @author Krinkle
	 * @version 2015-06-23
	 * Ajouté le 'uselang' ce 18 janvier 2016 — Ltrlg
	 */
	if ( mw.config.get( 'wgNamespaceNumber' ) >= 0 ) {
		mw.loader.using( , function () {
			pageContentHooks.push( function ( $pageContent ) {
				var
					uploadBase = '//upload.wikimedia.org/wikipedia/commons/',

					fileNamespace = mw.config.get( 'wgFormattedNamespaces' ),
					localBasePath = mw.util.getUrl( fileNamespace + ':' ),
					localBaseScript = mw.util.wikiScript() + '?title=' + mw.util.wikiUrlencode( fileNamespace + ':' ),

					commonsBasePath = '//commons.wikimedia.orgview_image.php?q=MediaWiki:Common.js&sq=MediaWiki:Common.js&lang=fr&file=File:',
					commonsBaseScript = '//commons.wikimedia.org/w/index.php?title=File:',

					lang = mw.user.options.get( 'language' );

				// see ]
				$pageContent.find( '.mw-file-description' ).each( function ( i, link ) {
					if ( link.tagName !== 'A' ) {
						return;
					}

					var img = link.querySelector( 'img' );

					// attention : on lit l'attribut, et non la propriété (elle contient en plus le protocole)
					// (en prime, il est plus performant dans ce cas de lire l'attribut)
					if ( img && img.getAttribute( 'src' ).startsWith( uploadBase ) ) {
						var currVal = link.getAttribute( 'href' );
						if ( currVal.startsWith( localBasePath ) ) {
							link.setAttribute( 'href', currVal.replace( localBasePath, commonsBasePath ) + '?uselang=' + lang );
						} else if ( currVal.startsWith( localBaseScript ) ) {
							link.setAttribute( 'href', currVal.replace( localBaseScript, commonsBaseScript ) + '&uselang=' + lang );
						}
					}
				} );
			} );
		} );
	}


	/**
	 * Ajout d'un lien « Ajouter un sujet » en bas de page
	 *
	 * TODO : éventuellement, ne pas ajouter ce lien lorsqu'il y a déjà l'élément « Démarrer une discussion »
	 * ajouté par DiscussionTools (sur les pdd n'ayant encore aucune discussion)
	 *
	 * solutions pour tester la présence de cet élément (pas étudié ce qui serait le plus adéquat) :
	 * - classe sur le body : body.ext-discussiontools-emptystate-shown
	 * - élément ".ext-discussiontools-emptystate" dans le contenu
	 */
	if ( mw.config.get( 'wgAction' ) === 'view' ) {
		$( function ( $ ) {
			var addSectionLink = $( '#ca-addsection' ).find( 'a' );
			if ( !addSectionLink ) { // pas d'onglet « Ajouter un sujet »
				return;
			}

			var $container = $( '<div style="clear:both; text-align:right; font-size:0.9em; margin:1em 0 -0.5em;" class="noprint">' );

			var link = document.createElement( 'a' );
			link.href = addSectionLink.href; // ce href sert encore, pour les middle-click, Ctrl+click... (ouverture dans un nouvel onglet)
			link.title = addSectionLink.title;
			link.textContent = addSectionLink.textContent;

			// compatibilité avec la fonctionnalité beta "New Discussion Tool", voir ]
			link.addEventListener( 'click', function ( e ) {
				if ( !e.ctrlKey ) {
					e.preventDefault();
					addSectionLink.click(); // .click() JS natif, pour information le .click() jQuery ne fonctionne pas dans le cas présent
				}
			} );

			$container.append( link );

			$( '#mw-content-text' ).append( $container );
		} );
	}


	/**
	 * Permet d'afficher les catégories cachées pour les contributeurs enregistrés, en ajoutant un (+) à la manière des boîtes déroulantes
	 */
	function addHiddenCatsToggler( $catlinks ) {
		if ( new URL( document.URL ).searchParams.get( 'printable' ) === 'yes' ) {
			return;
		}
		var cl = $catlinks;
		if ( !cl || cl.id !== 'catlinks' ) {
			mw.log.warn( 'wikipage.categories hook did not receive the #catlinks element' );
			return;
		}
		var hc = cl.querySelector( '#mw-hidden-catlinks' );
		if ( !hc ) {
			return;
		}
		if ( hc.classList.contains( 'mw-hidden-cats-user-shown' ) ) {
			return;
		}
		if ( hc.classList.contains( 'mw-hidden-cats-ns-shown' ) ) {
			hc.classList.add( 'mw-hidden-cats-hidden' );
		}
		var nc = cl.querySelector( '#mw-normal-catlinks' );
		if ( !nc ) {
			nc = document.createElement( 'div' );
			nc.id = 'mw-normal-catlinks';
			nc.className = 'mw-normal-catlinks';
			var a = document.createElement( 'a' );
			a.href = 'https://wiki95.com/fr/Catégorie:Accueil';
			a.title = 'Catégorie:Accueil';
			a.textContent = 'Catégories';
			nc.append( a, ' : ' );
			cl.prepend( nc );
			cl.classList.remove( 'catlinks-allhidden' );
		}
		var lnk = document.createElement( 'a' );
		lnk.id = 'mw-hidden-cats-link';
		lnk.title = 'Cet article contient des catégories cachées';
		lnk.style.cursor = 'pointer';
		lnk.style.color = 'var(--color-emphasized, black)';
		lnk.style.marginLeft = '0.3em';
		lnk.addEventListener( 'click', toggleHiddenCats );
		lnk.textContent = '';
		nc.append( lnk );
	}

	function toggleHiddenCats( e ) {
		var hc = document.getElementById( 'mw-hidden-catlinks' );
		if ( hc.classList.contains( 'mw-hidden-cats-hidden' ) ) {
			hc.classList.remove( 'mw-hidden-cats-hidden' );
			hc.classList.add( 'mw-hidden-cat-user-shown' );
			e.target.textContent = '';
		} else {
			hc.classList.remove( 'mw-hidden-cat-user-shown' );
			hc.classList.add( 'mw-hidden-cats-hidden' );
			e.target.textContent = '';
		}
	}

	mw.hook( 'wikipage.categories' ).add( addHiddenCatsToggler );


	/**
	 * Repositionnement de la page sur l'ancre avec laquelle elle a été appelée
	 * après le repli des boîtes déroulantes, entre autres.
	 */
	if ( window.location.hash ) {
		$( function ( $ ) { // eslint-disable-line no-unused-vars
			setTimeout( function () {
				var currentTarget = document.getElementById( decodeURIComponent( window.location.hash.substring( 1 ) ) );
				if ( currentTarget ) {
					currentTarget.scrollIntoView();
				}
			}, 1 );
		} );
	}


	/*********************************************************************/
	/* Fonctions strictement spécifiques à un espace de nom ou à une page */
	/*********************************************************************/

	// ESPACE DE NOM 'SPECIAL'
	if ( mw.config.get( 'wgNamespaceNumber' ) === -1 ) {

		/**
		 * Ajoute le namespace et l'éventuelle limite de pagination en cours aux filtres personnalisés sur ]
		 * Voir aussi ]
		 */
		if ( mw.config.get( 'wgCanonicalSpecialPageName' ) === 'Whatlinkshere' ) {
			const query = new URL( document.URL ).searchParams;

			let append = '';
			for ( const name of  ) {
				const value = query.getAll( name ).pop();
				if ( value ) {
					append += '&' + name + '=' + encodeURIComponent( value );
				}
			}

			if ( append !== '' ) {
				$( function ( $ ) {
					$( '#whatlinkshere-customfilters' ).find( 'a' ).each( function () {
						this.href += append;
					} );
				} );
			}
		}

		/**
		 * Affiche un modèle Information sur la page de téléchargement de fichiers ]
		 * Voir aussi ]
		 */
		if ( mw.config.get( 'wgCanonicalSpecialPageName' ) === 'Upload' ) {
			importScript( 'MediaWiki:Onlyifuploading.js' );
		}

	} // Fin du code concernant l'espace de nom 'Special'


	// ESPACE DE NOM 'UTILISATEUR'
	if ( mw.config.get( 'wgNamespaceNumber' ) === 2 ) {

		/*
		 * Fonctionnement du ]
		 * Le JavaScript principal se situe dans ]
		 */
		pageContentHooks.push( function ( $pageContent ) {
			if ( $pageContent.find( '.cadre_a_onglets' ).length ) {
				mw.loader.using( 'ext.gadget.CadreOnglets', function () {
					CadreOnglets_Init( $pageContent );
				} );
			}
		} );

	} // Fin du code concernant l'espace de nom 'Utilisateur'

	/*********************************************************************/
	/* Modification du comportement d'un bouton du menu */
	/*********************************************************************/

	/**
	 * Ajout d'écouteurs d'événement au bouton Développer / Réduire pour qu'il affecte également les éléments déroulants définis sur cette page :
	 * - les palettes
	 * - les boîtes déroulantes
	 * - les catégories cachées (le bouton ne sera pas ajouté s'il n'y a que des catégories cachées à dérouler)
	 */

	mw.loader.using( , function () {
		let expand_text = 'Tout développer';
		let collapse_text = 'Tout réduire';
		let expand_tooltip = 'Développer tous les éléments réductibles de la page actuelle';
		let collapse_tooltip = 'Réduire tous les éléments réductibles de la page actuelle';
	
		let allExpanded = false;
		let toggleAll;
		toggleAll = document.querySelector('#t-collapsible-toggle-all a');
		
		if ( !toggleAll && $('.mw-body-content .collapsible, .mw-body-content .NavToggle').length ) {
			// Si présence d'éléments déroulants et bouton absent, on ajoute le bouton nous-même.
			const portletLink = mw.util.addPortletLink(
				'p-tb',
				'#',
				expand_text,
				't-collapsible-toggle-all',
				expand_tooltip
			);
			if ( portletLink ) {
				// set up the toggle link
				toggleAll = portletLink.querySelector( 'a' );
				toggleAll.setAttribute( 'role', 'button' );
				toggleAll.setAttribute( 'aria-expanded', 'false' );
			}
		}
		if ( toggleAll ) {
			toggleAll.addEventListener( 'click', ( e ) => {
				e.preventDefault();
	
				if ( !allExpanded ) { /* Tout est réduit ou dans son état initial -> on déroule */
					/* Palettes */
					$( '.navboxToggle a' ).each(function() {
						if ($(this).text() === Palette_Derouler) {
							$(this).click();
						}
					});
					
					/* Boîtes déroulantes */
					$( 'a.NavToggle' ).each(function() {
						if ($(this).text() === BoiteDeroulante_Derouler) {
							$(this).click();
						}
					});
					
					/* Catégories cachées */
					let cl = $('#mw-hidden-catlinks');
					if ( cl.hasClass( 'mw-hidden-cats-hidden' ) )  {
						cl.removeClass( 'mw-hidden-cats-hidden' );
						cl.addClass( 'mw-hidden-cat-user-shown' );
						$('#mw-hidden-cats-link').text('');
					}
					
					 /* Mise à jour des propriétés du bouton pour le cas où on l'a ajouté nous-mêmes */
					toggleAll.textContent = collapse_text;
					toggleAll.title = collapse_tooltip;
					toggleAll.setAttribute( 'aria-expanded', 'true' );
					allExpanded = true;
	
				} else { /* Tout est déplié -> on réduit */
					/* Palettes */
					$( '.navboxToggle a' ).each(function() {
						if ($(this).text() === Palette_Enrouler) {
							$(this).click();
						}
					});
					
					/* Boîtes déroulantes */
					$( 'a.NavToggle' ).each(function() {
						if ($(this).text() === BoiteDeroulante_Enrouler) {
							$(this).click();
						}
					});
					
					/* Catégories cachées */
					let cl = $('#mw-hidden-catlinks');
					if ( !cl.hasClass( 'mw-hidden-cats-hidden' ) )  {
						cl.removeClass( 'mw-hidden-cat-user-shown' );
						cl.addClass( 'mw-hidden-cats-hidden' );
						$('#mw-hidden-cats-link').text('');
					}
					
					/* Mise à jour des propriétés du bouton pour le cas où on l'a ajouté nous-mêmes */
					toggleAll.textContent = expand_text;
					toggleAll.title = expand_tooltip;
					toggleAll.setAttribute( 'aria-expanded', 'false' );
					allExpanded = false;
				}
			} );
		}
	});

	/*
	 * Exécution des fonctions ajoutées à l'array "pageContentHooks"
	 *
	 * Pour info : nous sommes sûrs que $content est déjà inséré dans la page, cf. ]
	 * Edit : pas confiance. Par exemple, avec l'éditeur visuel c'est détaché malgré le warning, cf. ]
	 * Par conséquent, considérer que l'élément peut aussi être détaché.
	 */
	mw.hook( 'wikipage.content' ).add( function ( $content ) {
		/*
		 * Désactivation de toutes les fonctions en "pageContentHooks" de cette page
		 * avec les (nombreuses) prévisualisations de DiscussionTools (voir ]) :
		 * - soit ça bugue parce que ça agit sur du contenu en dehors de $content (c'était le cas de "addHiddenCatsToggler")
		 * - soit ça fonctionne quand même, mais ça serait gênant (exemple : "transformeEnPageDeDiscussion")
		 * - soit il est peu probable que cela soit utilisé dans un message (exemples : palettes, boîtes déroulantes, etc.)
		 */
		if ( $content.is( '.ext-discussiontools-ui-replyWidget-preview' ) ) {
			return;
		}

		/*
		 * Sélection plus précise de l'élément avec le contenu de page,
		 * pour éviter l'interface de diff, l'interface de modification, le print footer...
		 * ainsi que les pages qui n'ont pas de contenu (historiques, pages spéciales, etc.)
		 */
		let $pageContent;

		const $parserOutput = $content.find( '.mw-parser-output' );
		if ( $parserOutput.length ) { // élément avec le contenu de page
			$pageContent = $parserOutput;
		} else if ( $content.hasClass( 'mw-parser-output' ) ) { // ]
			$pageContent = $content;
		} else { // pas de contenu de page (historiques, pages spéciales, etc.)
			return;
		}

		for ( const hook of pageContentHooks ) {
			hook( $pageContent );
		}
	} );

} )(); // Fermeture de la IIFE globale

// </nowiki> /!\ Ne pas retirer cette balise