« MediaWiki:Vaulttec.js » : différence entre les versions

Kims (discussion | contributions)
Aucun résumé des modifications
Kims (discussion | contributions)
Aucun résumé des modifications
 
(11 versions intermédiaires par le même utilisateur non affichées)
Ligne 1 : Ligne 1 :
/* ── Bandeau Discord ───────────────────────────────────────── */
/**
* Bandeau Discord — MediaWiki:Vaulttec.js
* v2 : affiché uniquement en lecture d'article, rappel après 60 jours,
* compteur de présence mis en cache 30 min par session.
*/
( function () {
( function () {
  'use strict';
'use strict';


  var STORAGE_KEY = 'vtDiscordBannerDismissed';
var DISCORD_URL  = 'https://discord.gg/DxFnyyR';
  var DISCORD_URL  = 'https://discord.gg/DxFnyyR';
var STORAGE_KEY  = 'vt-discord-banner-dismissed';
  var GUILD_ID    = '181800470082355200'; // ← Remplacer par l'ID numérique du serveur
var COUNT_KEY    = 'vt-discord-count-cache';
var GUILD_ID    = '181800470082355200';
var REMIND_AFTER = 60 * 24 * 3600 * 1000; // 60 jours
var COUNT_TTL    = 30 * 60 * 1000;        // 30 min


  // Ne pas afficher si déjà fermé
// Uniquement en lecture, dans les espaces de contenu
  if ( localStorage.getItem( STORAGE_KEY ) === '1' ) { return; }
if ( mw.config.get( 'wgNamespaceNumber' ) < 0 ||
mw.config.get( 'wgAction' ) !== 'view' ) {
return;
}


  // Icône Discord SVG (officielle, monochrome)
try {
  var iconSVG = '<svg class="vt-banner-icon" viewBox="0 0 24 24" fill="#7dff9c" xmlns="http://www.w3.org/2000/svg">'
var dismissed = localStorage.getItem( STORAGE_KEY );
    + '<path d="M20.317 4.492c-1.53-.69-3.17-1.2-4.885-1.49a.075.075 0 0 0-.079.036c-.21.369-.444.85-.608 1.23a18.566 18.566 0 0 0-5.487 0 12.36 12.36 0 0 0-.617-1.23A.077.077 0 0 0 8.562 3c-1.714.29-3.354.8-4.885 1.491a.07.07 0 0 0-.032.027C.533 9.093-.32 13.555.099 17.961a.08.08 0 0 0 .031.055 20.03 20.03 0 0 0 5.993 2.98.078.078 0 0 0 .084-.026 13.83 13.83 0 0 0 1.226-1.963.074.074 0 0 0-.041-.104 13.175 13.175 0 0 1-1.872-.878.075.075 0 0 1-.008-.125c.126-.093.252-.19.372-.287a.075.075 0 0 1 .078-.01c3.927 1.764 8.18 1.764 12.061 0a.075.075 0 0 1 .079.009c.12.098.245.195.372.288a.075.075 0 0 1-.006.125c-.598.344-1.22.635-1.873.877a.075.075 0 0 0-.041.105c.36.687.772 1.341 1.225 1.962a.077.077 0 0 0 .084.028 19.963 19.963 0 0 0 6.002-2.981.076.076 0 0 0 .032-.054c.5-5.094-.838-9.52-3.549-13.442a.06.06 0 0 0-.031-.028zM8.02 15.278c-1.182 0-2.157-1.069-2.157-2.38 0-1.312.956-2.38 2.157-2.38 1.21 0 2.176 1.077 2.157 2.38 0 1.312-.956 2.38-2.157 2.38zm7.975 0c-1.183 0-2.157-1.069-2.157-2.38 0-1.312.955-2.38 2.157-2.38 1.21 0 2.176 1.077 2.157 2.38 0 1.312-.946 2.38-2.157 2.38z"/>'
// Compat ancienne valeur '1' (fermeture définitive) : on la garde 60 j
    + '</svg>';
// à partir de maintenant en la convertissant en timestamp.
if ( dismissed === '1' ) {
localStorage.setItem( STORAGE_KEY, String( Date.now() ) );
return;
}
if ( dismissed && Date.now() - Number( dismissed ) < REMIND_AFTER ) {
return;
}
} catch ( e ) {}


  // Construction du bandeau
var banner = document.createElement( 'div' );
  var banner = document.createElement( 'div' );
banner.className = 'vt-discord-banner';
  banner.className = 'vt-discord-banner';
banner.setAttribute( 'role', 'complementary' );
  banner.setAttribute( 'role', 'complementary' );
banner.setAttribute( 'aria-label', 'Invitation Discord' );
  banner.setAttribute( 'aria-label', 'Invitation Discord' );


  banner.innerHTML = iconSVG
banner.innerHTML =
    + '<span class="vt-banner-text">'
'<span class="vt-banner-text">' +
    +  '<strong>Communauté française</strong> — Rejoignez-nous sur Discord pour discuter, jouer, poser vos questions et contribuer au wiki !'
'Rejoignez la communauté des <strong>Archives de Vault-Tec</strong> sur Discord pour discuter, contribuer et partager.' +
    + '</span>'
'</span>' +
    + '<span class="vt-banner-members" aria-live="polite" style="display:none">'
'<span class="vt-banner-info">' +
    +  '<span id="vt-discord-count"></span> en ligne'
'<span class="vt-banner-icon" aria-hidden="true">' +
    + '</span>'
'<svg viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">' +
    + '<a class="vt-banner-btn" href="' + DISCORD_URL + '" target="_blank" rel="noopener noreferrer">'
'<path d="M20.317 4.37a19.79 19.79 0 0 0-4.885-1.515.074.074 0 0 0-.079.037c-.21.375-.444.864-.608 1.25a18.27 18.27 0 0 0-5.487 0 12.64 12.64 0 0 0-.617-1.25.077.077 0 0 0-.079-.037A19.736 19.736 0 0 0 3.677 4.37a.07.07 0 0 0-.032.027C.533 9.046-.32 13.58.099 18.057a.082.082 0 0 0 .031.057 19.9 19.9 0 0 0 5.993 3.03.078.078 0 0 0 .084-.028c.462-.63.874-1.295 1.226-1.994a.076.076 0 0 0-.041-.106 13.107 13.107 0 0 1-1.872-.892.077.077 0 0 1-.008-.128 10.2 10.2 0 0 0 .372-.292.074.074 0 0 1 .077-.01c3.928 1.793 8.18 1.793 12.062 0a.074.074 0 0 1 .078.01c.12.098.246.198.373.292a.077.077 0 0 1-.006.127 12.299 12.299 0 0 1-1.873.892.077.077 0 0 0-.041.107c.36.698.772 1.362 1.225 1.993a.076.076 0 0 0 .084.028 19.839 19.839 0 0 0 6.002-3.03.077.077 0 0 0 .032-.054c.5-5.177-.838-9.674-3.549-13.66a.061.061 0 0 0-.031-.03zM8.02 15.33c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.956-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.956 2.418-2.157 2.418zm7.975 0c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.955-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.946 2.418-2.157 2.418z"/>' +
    +  '↗ Rejoindre'
'</svg>' +
    + '</a>'
'</span>' +
    + '<button class="vt-banner-close" aria-label="Fermer">×</button>';
'<span class="vt-banner-members" style="display:none" aria-live="polite">' +
'<span id="vt-discord-count"></span> en ligne' +
'</span>' +
'</span>' +
'<a class="vt-banner-btn" href="' + DISCORD_URL + '" target="_blank" rel="noopener noreferrer">' +
'↗ Rejoindre' +
'</a>' +
'<button class="vt-banner-close" aria-label="Fermer">×</button>';


  document.body.appendChild( banner );
document.body.appendChild( banner );


  // Animation d'entrée
requestAnimationFrame( function () {
  requestAnimationFrame( function () {
banner.style.opacity = '0';
    banner.style.opacity = '0';
banner.style.transform = 'translateY(100%)';
    banner.style.transform = 'translateY(100%)';
requestAnimationFrame( function () {
    requestAnimationFrame( function () {
banner.style.opacity = '';
      banner.style.transition = 'transform 0.35s ease, opacity 0.35s ease';
banner.style.transform = '';
      banner.style.opacity = '';
} );
      banner.style.transform = '';
} );
    } );
  } );


  // Fermeture
banner.querySelector( '.vt-banner-close' ).addEventListener( 'click', function () {
  banner.querySelector( '.vt-banner-close' ).addEventListener( 'click', function () {
banner.classList.add( 'vt-banner-hidden' );
    banner.classList.add( 'vt-banner-hidden' );
try { localStorage.setItem( STORAGE_KEY, String( Date.now() ) ); } catch ( e ) {}
    localStorage.setItem( STORAGE_KEY, '1' );
setTimeout( function () { banner.remove(); }, 350 );
    setTimeout( function () { banner.remove(); }, 350 );
} );
  } );


   // Compteur de membres en ligne Discord Widget API (public, sans clé)
function showCount( n ) {
  // Prérequis : activer dans Paramètres du serveur → Engagement → Widget du serveur
var countEl   = document.getElementById( 'vt-discord-count' );
  if ( GUILD_ID !== 'VOTRE_GUILD_ID' ) {
var membersEl = banner.querySelector( '.vt-banner-members' );
    fetch( 'https://discord.com/api/guilds/' + GUILD_ID + '/widget.json' )
if ( countEl && membersEl ) {
      .then( function ( r ) { return r.ok ? r.json() : null; } )
countEl.textContent    = n;
      .then( function ( data ) {
membersEl.style.display = '';
        if ( !data || !data.presence_count ) { return; }
}
        var countEl  = document.getElementById( 'vt-discord-count' );
}
        var membersEl = banner.querySelector( '.vt-banner-members' );
 
        if ( countEl && membersEl ) {
// Compteur de présence cache de session pour éviter une requête
          countEl.textContent    = data.presence_count;
// vers discord.com à chaque page vue.
          membersEl.style.display = '';
var cached = null;
        }
try { cached = JSON.parse( sessionStorage.getItem( COUNT_KEY ) ); } catch ( e ) {}
      } )
if ( cached && Date.now() - cached.t < COUNT_TTL ) {
      .catch( function () { /* silencieux si widget désactivé */ } );
showCount( cached.n );
  }
return;
}
 
fetch( 'https://discord.com/api/guilds/' + GUILD_ID + '/widget.json' )
.then( function ( r ) { return r.ok ? r.json() : null; } )
.then( function ( data ) {
if ( !data || !data.presence_count ) { return; }
showCount( data.presence_count );
try {
sessionStorage.setItem( COUNT_KEY,
JSON.stringify( { n: data.presence_count, t: Date.now() } ) );
} catch ( e ) {}
} )
.catch( function () {} );


}() );
}() );
/* LCP — fetchpriority=high sur la première image de l'article */
( function () {
  var img = document.querySelector( '#mw-content-text .mw-file-element' );
  if ( img && !img.complete ) {
    img.setAttribute( 'fetchpriority', 'high' );
  }
} )();

Dernière version du 10 juin 2026 à 23:08

/**
 * Bandeau Discord — MediaWiki:Vaulttec.js
 * v2 : affiché uniquement en lecture d'article, rappel après 60 jours,
 * compteur de présence mis en cache 30 min par session.
 */
( function () {
	'use strict';

	var DISCORD_URL  = 'https://discord.gg/DxFnyyR';
	var STORAGE_KEY  = 'vt-discord-banner-dismissed';
	var COUNT_KEY    = 'vt-discord-count-cache';
	var GUILD_ID     = '181800470082355200';
	var REMIND_AFTER = 60 * 24 * 3600 * 1000; // 60 jours
	var COUNT_TTL    = 30 * 60 * 1000;        // 30 min

	// Uniquement en lecture, dans les espaces de contenu
	if ( mw.config.get( 'wgNamespaceNumber' ) < 0 ||
		mw.config.get( 'wgAction' ) !== 'view' ) {
		return;
	}

	try {
		var dismissed = localStorage.getItem( STORAGE_KEY );
		// Compat ancienne valeur '1' (fermeture définitive) : on la garde 60 j
		// à partir de maintenant en la convertissant en timestamp.
		if ( dismissed === '1' ) {
			localStorage.setItem( STORAGE_KEY, String( Date.now() ) );
			return;
		}
		if ( dismissed && Date.now() - Number( dismissed ) < REMIND_AFTER ) {
			return;
		}
	} catch ( e ) {}

	var banner = document.createElement( 'div' );
	banner.className = 'vt-discord-banner';
	banner.setAttribute( 'role', 'complementary' );
	banner.setAttribute( 'aria-label', 'Invitation Discord' );

	banner.innerHTML =
		'<span class="vt-banner-text">' +
			'Rejoignez la communauté des <strong>Archives de Vault-Tec</strong> sur Discord pour discuter, contribuer et partager.' +
		'</span>' +
		'<span class="vt-banner-info">' +
			'<span class="vt-banner-icon" aria-hidden="true">' +
				'<svg viewBox="0 0 24 24" fill="currentColor" xmlns="http://www.w3.org/2000/svg">' +
					'<path d="M20.317 4.37a19.79 19.79 0 0 0-4.885-1.515.074.074 0 0 0-.079.037c-.21.375-.444.864-.608 1.25a18.27 18.27 0 0 0-5.487 0 12.64 12.64 0 0 0-.617-1.25.077.077 0 0 0-.079-.037A19.736 19.736 0 0 0 3.677 4.37a.07.07 0 0 0-.032.027C.533 9.046-.32 13.58.099 18.057a.082.082 0 0 0 .031.057 19.9 19.9 0 0 0 5.993 3.03.078.078 0 0 0 .084-.028c.462-.63.874-1.295 1.226-1.994a.076.076 0 0 0-.041-.106 13.107 13.107 0 0 1-1.872-.892.077.077 0 0 1-.008-.128 10.2 10.2 0 0 0 .372-.292.074.074 0 0 1 .077-.01c3.928 1.793 8.18 1.793 12.062 0a.074.074 0 0 1 .078.01c.12.098.246.198.373.292a.077.077 0 0 1-.006.127 12.299 12.299 0 0 1-1.873.892.077.077 0 0 0-.041.107c.36.698.772 1.362 1.225 1.993a.076.076 0 0 0 .084.028 19.839 19.839 0 0 0 6.002-3.03.077.077 0 0 0 .032-.054c.5-5.177-.838-9.674-3.549-13.66a.061.061 0 0 0-.031-.03zM8.02 15.33c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.956-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.956 2.418-2.157 2.418zm7.975 0c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.955-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.946 2.418-2.157 2.418z"/>' +
				'</svg>' +
			'</span>' +
			'<span class="vt-banner-members" style="display:none" aria-live="polite">' +
				'<span id="vt-discord-count"></span> en ligne' +
			'</span>' +
		'</span>' +
		'<a class="vt-banner-btn" href="' + DISCORD_URL + '" target="_blank" rel="noopener noreferrer">' +
			'↗ Rejoindre' +
		'</a>' +
		'<button class="vt-banner-close" aria-label="Fermer">×</button>';

	document.body.appendChild( banner );

	requestAnimationFrame( function () {
		banner.style.opacity = '0';
		banner.style.transform = 'translateY(100%)';
		requestAnimationFrame( function () {
			banner.style.opacity = '';
			banner.style.transform = '';
		} );
	} );

	banner.querySelector( '.vt-banner-close' ).addEventListener( 'click', function () {
		banner.classList.add( 'vt-banner-hidden' );
		try { localStorage.setItem( STORAGE_KEY, String( Date.now() ) ); } catch ( e ) {}
		setTimeout( function () { banner.remove(); }, 350 );
	} );

	function showCount( n ) {
		var countEl   = document.getElementById( 'vt-discord-count' );
		var membersEl = banner.querySelector( '.vt-banner-members' );
		if ( countEl && membersEl ) {
			countEl.textContent     = n;
			membersEl.style.display = '';
		}
	}

	// Compteur de présence — cache de session pour éviter une requête
	// vers discord.com à chaque page vue.
	var cached = null;
	try { cached = JSON.parse( sessionStorage.getItem( COUNT_KEY ) ); } catch ( e ) {}
	if ( cached && Date.now() - cached.t < COUNT_TTL ) {
		showCount( cached.n );
		return;
	}

	fetch( 'https://discord.com/api/guilds/' + GUILD_ID + '/widget.json' )
		.then( function ( r ) { return r.ok ? r.json() : null; } )
		.then( function ( data ) {
			if ( !data || !data.presence_count ) { return; }
			showCount( data.presence_count );
			try {
				sessionStorage.setItem( COUNT_KEY,
					JSON.stringify( { n: data.presence_count, t: Date.now() } ) );
			} catch ( e ) {}
		} )
		.catch( function () {} );

}() );