Zum Inhalt springen

MediaWiki:Common.js: Unterschied zwischen den Versionen

Aus ADOS Wiki
Keine Bearbeitungszusammenfassung
Markierung: Zurückgesetzt
Keine Bearbeitungszusammenfassung
 
(11 dazwischenliegende Versionen desselben Benutzers werden nicht angezeigt)
Zeile 396: Zeile 396:


});
});




Zeile 458: Zeile 454:


   /* ========== Kategorien rekursiv einsammeln (inkl. Subkats, Namespaces) ========== */
   /* ========== Kategorien rekursiv einsammeln (inkl. Subkats, Namespaces) ========== */
// NIMMT nsStr (z. B. "*", "0|102|14"). Bei "*" wird cmnamespace NICHT gesetzt.
function fetchCategoryMembersRecursiveSingleResolved(api, catTitle, limit, outSet, pages, nsStr){
function fetchCategoryMembersRecursiveSingleResolved(api, catTitle, limit, outSet, pages, nsStr){
   var visited = {}, queue = [catTitle];
   var visited = {}, queue = [catTitle];
Zeile 468: Zeile 463:
       list: 'categorymembers',
       list: 'categorymembers',
       cmtitle: title,
       cmtitle: title,
      // cmnamespace nur setzen, wenn NICHT "*"
       cmtype: 'page|subcat',
       cmtype: 'page|subcat',
       cmlimit: Math.min(200, limit),
       cmlimit: Math.min(200, limit),
Zeile 577: Zeile 571:
   /* ========== Render ========== */
   /* ========== Render ========== */
   function renderTopN(container, rows, N, minVotes) {
   function renderTopN(container, rows, N, minVotes) {
    // Statusbox parken, falls keep aktiv
     var keep = (container.getAttribute && container.getAttribute('data-keep-status') === 'true');
     var keep = (container.getAttribute && container.getAttribute('data-keep-status') === 'true');
     var statusBox = keep ? container.querySelector('.whisky-top5__status') : null;
     var statusBox = keep ? container.querySelector('.whisky-top5__status') : null;
Zeile 637: Zeile 630:
       container.setAttribute('data-top5-init','1');
       container.setAttribute('data-top5-init','1');


      // Kategorien (Zeilenumbruch ODER Semikolon getrennt)
       var rawCats = container.getAttribute('data-categories') || container.getAttribute('data-category') || '';
       var rawCats = container.getAttribute('data-categories') || container.getAttribute('data-category') || '';
       var parts = rawCats.split(/\n|;/), rootCats = [], i;
       var parts = rawCats.split(/\n|;/), rootCats = [], i;
Zeile 647: Zeile 639:
       var minVotes = parseInt(container.getAttribute('data-min-votes') || '1', 10);
       var minVotes = parseInt(container.getAttribute('data-min-votes') || '1', 10);
       var includeHidden = (container.getAttribute('data-include-hidden') === 'true');
       var includeHidden = (container.getAttribute('data-include-hidden') === 'true');
       var nsStr = container.getAttribute('data-namespaces') || '0|14'; // z. B. "*", "0|102|14"
       var nsStr = container.getAttribute('data-namespaces') || '0|14';


       var rawC = container.getAttribute('data-contests') || 'NASE,GESCHMACK,ABGANG,GESAMTEINDRUCK';
       var rawC = container.getAttribute('data-contests') || 'NASE,GESCHMACK,ABGANG,GESAMTEINDRUCK';
Zeile 658: Zeile 650:
       status('Sammle Seiten …');
       status('Sammle Seiten …');


      // 1) Artikel einsammeln
       fetchCategoryMembersRecursiveMulti(rootCats, lim, status, nsStr).then(function(members){
       fetchCategoryMembersRecursiveMulti(rootCats, lim, status, nsStr).then(function(members){
         status('Gefundene Seiten gesamt: ' + (members ? members.length : 0) + ' – lade Bewertungen …', true);
         status('Gefundene Seiten gesamt: ' + (members ? members.length : 0) + ' – lade Bewertungen …', true);
Zeile 703: Zeile 693:


});
});




Zeile 712: Zeile 700:
     var el = this, val = parseFloat(el.getAttribute('data-rating') || '0');
     var el = this, val = parseFloat(el.getAttribute('data-rating') || '0');
     if (isNaN(val)) val = 0;
     if (isNaN(val)) val = 0;
    // clamp 0..5
     val = Math.max(0, Math.min(5, val));
     val = Math.max(0, Math.min(5, val));
    // set CSS variable for width percentage (0..5 -> 0..5 stars)
     el.style.setProperty('--stars', (val).toString());
     el.style.setProperty('--stars', (val).toString());
     el.setAttribute('aria-label', val + ' von 5 Sternen');
     el.setAttribute('aria-label', val + ' von 5 Sternen');
Zeile 721: Zeile 707:
});
});


// --------------------------------


// Force light color-scheme at document level (helps Mobile Safari)
// Force light color-scheme at document level (helps Mobile Safari)
Zeile 736: Zeile 720:
   }
   }
});
});
// -------------------------------------------------




Zeile 751: Zeile 733:
     if (!chart || !block) return;
     if (!chart || !block) return;


    // Summe aller Werte berechnen
     const total = chart.data.datasets.reduce((sum, ds) =>
     const total = chart.data.datasets.reduce((sum, ds) =>
       sum + (ds.data || []).reduce((a, b) => a + (parseFloat(b) || 0), 0)
       sum + (ds.data || []).reduce((a, b) => a + (parseFloat(b) || 0), 0)
     , 0);
     , 0);


    // Bestehende Anzeige entfernen (falls Neurender)
     const oldInfo = block.querySelector(':scope > .chart-total-info');
     const oldInfo = block.querySelector(':scope > .chart-total-info');
     if (oldInfo) oldInfo.remove();
     if (oldInfo) oldInfo.remove();


    // Neue Anzeige einfügen
     const info = document.createElement('div');
     const info = document.createElement('div');
     info.className = 'chart-total-info';
     info.className = 'chart-total-info';
Zeile 775: Zeile 754:
}
}


// -------------------------------------


/* === ADOS Multi-Serien-Chart (Chart.js) ============================= *
/* === ADOS Multi-Serien-Chart (Chart.js) ============================= *
Zeile 784: Zeile 761:


(function () {
(function () {
  // 1) Chart.js nur 1x laden und erst dann rendern
   var _chartReady = null;
   var _chartReady = null;
   function ensureChartJS() {
   function ensureChartJS() {
Zeile 791: Zeile 767:
       if (window.Chart) return resolve();
       if (window.Chart) return resolve();
       var s = document.createElement('script');
       var s = document.createElement('script');
       s.src = 'https://cdn.jsdelivr.net/npm/chart.js'; // UMD-Bundle
       s.src = 'https://cdn.jsdelivr.net/npm/chart.js';
       s.async = true;
       s.async = true;
       s.onload = function(){ resolve(); };
       s.onload = function(){ resolve(); };
Zeile 800: Zeile 776:
   }
   }


  // 2) Farben je Serie (gut unterscheidbar)
   var ADOS_COLORS = {
   var ADOS_COLORS = {
     'A Dream of Scotland':                '#C2410C', // Kupferbraun
     'A Dream of Scotland':                '#C2410C',
     'A Dream of Ireland':                  '#15803D', // Flaschengrün
     'A Dream of Ireland':                  '#15803D',
     'A Dream of... – Der Rest der Welt':  '#1D4ED8', // Mittelblau
     'A Dream of... – Der Rest der Welt':  '#1D4ED8',
     'Friendly Mr. Z':                      '#9333EA', // Violett
     'Friendly Mr. Z':                      '#9333EA',
     'Die Whisky Elfen':                    '#0891B2', // Türkis
     'Die Whisky Elfen':                    '#0891B2',
     'The Fine Art of Whisky':              '#CA8A04' // Goldgelb
     'The Fine Art of Whisky':              '#CA8A04'
   };
   };
   var COLOR_CYCLE = ['#2563eb','#16a34a','#f97316','#dc2626','#a855f7','#0ea5e9','#f59e0b','#10b981'];
   var COLOR_CYCLE = ['#2563eb','#16a34a','#f97316','#dc2626','#a855f7','#0ea5e9','#f59e0b','#10b981'];
Zeile 822: Zeile 797:
   }
   }


  // 3) Tabelle (Jahr | Serie | Anzahl) -> {labels, datasets}
   function buildDatasetsFromTable(tbl){
   function buildDatasetsFromTable(tbl){
     var rows = Array.from(tbl.querySelectorAll('tr'));
     var rows = Array.from(tbl.querySelectorAll('tr'));
Zeile 828: Zeile 802:


     var yearsSet = new Set();
     var yearsSet = new Set();
     var bySeries  = new Map(); // serie -> Map(year -> count)
     var bySeries  = new Map();


     rows.slice(1).forEach(function(tr){
     rows.slice(1).forEach(function(tr){
Zeile 863: Zeile 837:
   }
   }


   // 4) Einen Chart-Container rendern: nimmt die NÄCHSTE Tabelle als Datenquelle
   function renderOne(block){
function renderOne(block){
    if (block.dataset.rendered === '1') return;
  // schon gerendert? (z. B. durch AJAX/Minerva-Reloads)
  if (block.dataset.rendered === '1') return;


  // nächste Tabelle (auch wenn sie in einem Wrapper steckt) finden
    var el = block.nextElementSibling, tbl = null, wrapToHide = null;
  var el = block.nextElementSibling, tbl = null, wrapToHide = null;
    while (el) {
  while (el) {
      if (/^H[1-6]$/.test(el.tagName) || (el.classList && el.classList.contains('ados-chart-multi'))) break;
    if (/^H[1-6]$/.test(el.tagName) || (el.classList && el.classList.contains('ados-chart-multi'))) break;
      if (el.tagName === 'TABLE') {
    if (el.tagName === 'TABLE') {
        tbl = el;
      tbl = el;
      } else if (el.querySelector) {
    } else if (el.querySelector) {
        var t = el.querySelector('table');
      var t = el.querySelector('table');
        if (t) tbl = t;
      if (t) tbl = t;
      }
      if (tbl) {
        wrapToHide = tbl.parentElement;
        break;
      }
      el = el.nextElementSibling;
     }
     }
     if (tbl) {
     if (!tbl) return;
      // Wrapper merken – aber später nur verstecken, wenn er "quasi nur" die Tabelle enthält
      wrapToHide = tbl.parentElement;
      break;
    }
    el = el.nextElementSibling;
  }
  if (!tbl) return;


  var out = buildDatasetsFromTable(tbl);
    var out = buildDatasetsFromTable(tbl);
  if (!out.labels.length || !out.datasets.length) return;
    if (!out.labels.length || !out.datasets.length) return;


  // Tabelle verstecken, wenn gewünscht (nur die Tabelle – oder den Wrapper, falls er sonst leer ist)
    var hide = (block.dataset.hideTable || '').toLowerCase() === 'true';
  var hide = (block.dataset.hideTable || '').toLowerCase() === 'true';
    if (hide) {
  if (hide) {
      var onlyTable = wrapToHide && wrapToHide.children.length === 1 && wrapToHide.firstElementChild === tbl;
    // Hat der Wrapper außer der Tabelle noch sichtbaren Inhalt?
      if (onlyTable) {
    var onlyTable = wrapToHide && wrapToHide.children.length === 1 && wrapToHide.firstElementChild === tbl;
        wrapToHide.setAttribute('aria-hidden','true');
    if (onlyTable) {
        wrapToHide.style.display = 'none';
      wrapToHide.setAttribute('aria-hidden','true');
      } else {
      wrapToHide.style.display = 'none';
        tbl.setAttribute('aria-hidden','true');
    } else {
        tbl.style.display = 'none';
      tbl.setAttribute('aria-hidden','true');
      }
      tbl.style.display = 'none';
     }
     }
  }


  // Zeichenfläche einsetzen (mobil/desktop)
    var wrap = document.createElement('div');
  var wrap = document.createElement('div');
    wrap.style.position = 'relative';
  wrap.style.position = 'relative';
    wrap.style.width = '100%';
  wrap.style.width = '100%';
    wrap.style.height = block.dataset.height || (window.matchMedia('(min-width:768px)').matches ? '450px' : '300px');
  wrap.style.height = block.dataset.height || (window.matchMedia('(min-width:768px)').matches ? '450px' : '300px');
    var canvas = document.createElement('canvas');
  var canvas = document.createElement('canvas');
    wrap.appendChild(canvas);
  wrap.appendChild(canvas);
    block.innerHTML = '';
  block.innerHTML = '';
    block.appendChild(wrap);
  block.appendChild(wrap);


  var type  = (block.dataset.type || 'line').toLowerCase();
    var type  = (block.dataset.type || 'line').toLowerCase();
  var title = block.dataset.title || '';
    var title = block.dataset.title || '';
  var cumulative = (block.dataset.cumulative || '').toLowerCase() === 'true';
    var cumulative = (block.dataset.cumulative || '').toLowerCase() === 'true';


  // optional: kumulative Werte pro Serie bauen
    if (cumulative) {
  if (cumulative) {
      out.datasets = out.datasets.map(function(ds){
    out.datasets = out.datasets.map(function(ds){
        var acc = 0;
      var acc = 0;
        return Object.assign({}, ds, {
      return Object.assign({}, ds, {
          data: ds.data.map(function(v){ acc += v; return acc; })
        data: ds.data.map(function(v){ acc += v; return acc; })
        });
       });
       });
     });
     }
  }


ensureChartJS().then(function(){
    ensureChartJS().then(function(){
  const chart = new Chart(canvas.getContext('2d'), {
      const chart = new Chart(canvas.getContext('2d'), {
    type: type,
        type: type,
    data: { labels: out.labels, datasets: out.datasets },
        data: { labels: out.labels, datasets: out.datasets },
    options: {
        options: {
      responsive: true,
          responsive: true,
      maintainAspectRatio: false,
          maintainAspectRatio: false,
      interaction: { mode: 'nearest', intersect: false },
          interaction: { mode: 'nearest', intersect: false },
      plugins: {
          plugins: {
        legend: { position: 'bottom', labels: { font: { size: 13 }, boxWidth: 20 } },
            legend: { position: 'bottom', labels: { font: { size: 13 }, boxWidth: 20 } },
        title:  { display: !!title, text: title, font: { size: 16 } },
            title:  { display: !!title, text: title, font: { size: 16 } },
        tooltip:{ backgroundColor: 'rgba(0,0,0,0.8)', titleFont: {size:14}, bodyFont: {size:13} }
            tooltip:{ backgroundColor: 'rgba(0,0,0,0.8)', titleFont: {size:14}, bodyFont: {size:13} }
      },
          },
      scales: {
          scales: {
        x: { ticks: { font: { size: 12 } } },
            x: { ticks: { font: { size: 12 } } },
        y: { beginAtZero: true, ticks: { precision: 0, font: { size: 12 } } }
            y: { beginAtZero: true, ticks: { precision: 0, font: { size: 12 } } }
      }
          }
    }
        }
  });
      });


  // --- NEU ---
      const hideTotal = (block.dataset.hideTotal || '').toLowerCase() === 'true';
  const hideTotal = (block.dataset.hideTotal || '').toLowerCase() === 'true';
      const oldInfo = block.querySelector(':scope > .chart-total-info');
      if (oldInfo) oldInfo.remove();


  // existierende Anzeige entfernen (falls Seite neu gerendert wird)
      if (!hideTotal) {
  const oldInfo = block.querySelector(':scope > .chart-total-info');
        addTotalBelowLegend(chart, block);
  if (oldInfo) oldInfo.remove();


  if (!hideTotal) {
        if (window.ResizeObserver) {
    // Gesamtzahl einfügen
          const obs = new ResizeObserver(() => addTotalBelowLegend(chart, block));
    addTotalBelowLegend(chart, block);
          obs.observe(chart.canvas);
          chart.$adosTotalObserver = obs;
        }
      }


    // bei Größenänderung (mobil <-> desktop) neu berechnen
       block.dataset.rendered = '1';
    if (window.ResizeObserver) {
     });
       const obs = new ResizeObserver(() => addTotalBelowLegend(chart, block));
      obs.observe(chart.canvas);
      chart.$adosTotalObserver = obs;
     }
   }
   }


  block.dataset.rendered = '1';
});
}
  // 5) Start: erst wenn DOM fertig, dann Chart.js laden, dann rendern
   function boot($scope){
   function boot($scope){
     var blocks = ($scope && $scope[0] ? $scope[0] : document).querySelectorAll('.ados-chart-multi');
     var blocks = ($scope && $scope[0] ? $scope[0] : document).querySelectorAll('.ados-chart-multi');
Zeile 982: Zeile 941:
     mw.hook('wikipage.content').add(boot);
     mw.hook('wikipage.content').add(boot);
   } else {
   } else {
    // Fallback
     (document.readyState === 'loading')
     (document.readyState === 'loading')
       ? document.addEventListener('DOMContentLoaded', function(){ boot(); })
       ? document.addEventListener('DOMContentLoaded', function(){ boot(); })
Zeile 988: Zeile 946:
   }
   }
})();
})();




// ==========================Scan==================================
// ==========================Scan==================================
mw.loader.using('mediawiki.util').then(function () {
mw.loader.using('mediawiki.util').then(function () {
   if (mw.config.get('wgPageName') !== 'LabelScan') return;
   if (mw.config.get('wgPageName') !== 'LabelScan') return;
Zeile 998: Zeile 954:
   mw.loader.load('/index.php?title=MediaWiki:Gadget-LabelScan.css&action=raw&ctype=text/css', 'text/css');
   mw.loader.load('/index.php?title=MediaWiki:Gadget-LabelScan.css&action=raw&ctype=text/css', 'text/css');
});
});




// ==========================ScanApp==================================
// ==========================ScanApp==================================
/* ==== PWA: Manifest + Service Worker + Install-Button (ES5) ==== */
/* ==== PWA: Manifest + Service Worker + Install-Button (ES5) ==== */


Zeile 1.032: Zeile 985:
   });
   });


   // Sicherstellen, dass DOM existiert, bevor wir Handler setzen
   function onReady(fn){ if (document.readyState === "loading") document.addEventListener("DOMContentLoaded", fn); else fn(); }
  function onReady(fn){ if (document.readyState === "loading") document.addEventListener("DOMContentLoaded", fn); else fn(); }
   onReady(function () {
   onReady(function () {
     var btn = document.getElementById("ados-install");
     var btn = document.getElementById("ados-install");
Zeile 1.045: Zeile 997:
   });
   });
})();
})();


if ('serviceWorker' in navigator) {
if ('serviceWorker' in navigator) {
Zeile 1.052: Zeile 1.003:




// ============================================================


mw.loader.using('mediawiki.util').then(function () {
  function checkNeuBadges() {
    var badges = document.querySelectorAll('.ados-neu-badge');
    var now = new Date();


    badges.forEach(function (badge) {
      var expiry = badge.getAttribute('data-expiry');
      if (!expiry) return;


/* =========================================================
      var expiryDate = new Date(expiry + "T23:59:59");
  ADOS XMAS: Timer-Bar & Snow (nur im Dezember)
       if (now > expiryDate) {
  ========================================================= */
        badge.style.display = "none";
(function () {
       }
  'use strict';
     });
 
  if (typeof window === 'undefined' || typeof document === 'undefined') return;
 
  var now = new Date();
 
  // Nur im Dezember aktiv (Monat 11 = Dezember)
  if (now.getMonth() !== 11) {
    return;
  }
 
  // Wenn der Nutzer reduzierte Animationen bevorzugt → Schneefall aus
  var prefersReducedMotion = false;
  try {
    prefersReducedMotion = window.matchMedia &&
      window.matchMedia('(prefers-reduced-motion: reduce)').matches;
  } catch (e) {}
 
  /* -----------------------------
    1) Nur bestehenden Timer „weihnachtlich anziehen“
    ----------------------------- */
  function decorateXmasTimer() {
    var bar = document.getElementById('ados-timer-bar');
    if (!bar) return; // Dein eigener Timer erzeugt dieses Element
 
    // Weihnachts-Design via CSS-Klasse aktivieren
    bar.classList.add('ados-xmas');
 
    // Optional: Nachricht dezent mit 🎄 ergänzen (ohne alles zu überschreiben)
    var msgEl = document.getElementById('ados-timer-message');
    if (msgEl && !msgEl.dataset.xmasDecorated) {
      msgEl.textContent = '🎄 ' + msgEl.textContent;
       msgEl.dataset.xmasDecorated = '1';
    }
  }
 
  /* -----------------------------
    2) Schneefall-Effekt
    ----------------------------- */
  function initSnow() {
    if (prefersReducedMotion) return;
 
    // Optional: auf sehr kleinen Geräten deaktivieren
    if (window.innerWidth < 600) return;
 
    // Container anlegen
    var container = document.createElement('div');
    container.id = 'ados-snow';
    document.body.appendChild(container);
 
    var flakeChars = ['❄', '✻', '✼', '✥', '✶'];
    var flakeCount = 60; // dezent halten
 
    for (var i = 0; i < flakeCount; i++) {
      var flake = document.createElement('span');
      flake.className = 'ados-snowflake';
      flake.textContent = flakeChars[Math.floor(Math.random() * flakeChars.length)];
 
      // Zufällige Start-Parameter
      var startX = Math.random() * 100;          // vw
      var drift  = (Math.random() * 40) - 20;    // -20 bis +20 vw
      var dur    = 12 + Math.random() * 10;      // 12–22 Sekunden
      var delay  = Math.random() * 20;          // bis zu 20 Sekunden
 
      flake.style.left = startX + 'vw';
      flake.style.setProperty('--x-start', '0vw');
      flake.style.setProperty('--x-end', drift.toFixed(1) + 'vw');
      flake.style.animationDuration = dur.toFixed(1) + 's';
      flake.style.animationDelay = delay.toFixed(1) + 's';
 
      container.appendChild(flake);
    }
  }
 
  /* -----------------------------
    3) Init nach DOM-Ready
    ----------------------------- */
  function onReady(fn) {
    if (document.readyState === 'loading') {
       document.addEventListener('DOMContentLoaded', fn, { once: true });
     } else {
      fn();
    }
   }
   }


  onReady(function () {
   if (document.readyState === "loading") {
    // nur dekorieren, NICHT neu bauen
     document.addEventListener("DOMContentLoaded", checkNeuBadges);
    decorateXmasTimer();
    initSnow();
  });
 
})();
 
 
/* === ADOS – Winter-Schneefall (Desktop dezent, mobil deutlich sichtbar) === */
(function() {
 
  // Rücksicht auf Nutzer mit "Bewegung reduzieren"
  if (window.matchMedia("(prefers-reduced-motion: reduce)").matches) return;
 
  function initSnow() {
    const isMobile = window.innerWidth < 700;
 
    const container = document.createElement('div');
    container.id = 'ados-snow';
    container.style.position = 'fixed';
    container.style.top = '0';
    container.style.left = '0';
    container.style.width = '100%';
    container.style.height = '100%';
    container.style.pointerEvents = 'none';
    container.style.zIndex = '9999';
    container.style.overflow = 'hidden';
    document.body.appendChild(container);
 
    const flakeChars = ['•', '·', '∙'];
 
    // Desktop dezent, mobil kräftig
    const flakeCount = isMobile ? 60 : 30;
 
    for (let i = 0; i < flakeCount; i++) {
      const flake = document.createElement('div');
      flake.textContent = flakeChars[Math.floor(Math.random() * flakeChars.length)];
      flake.style.position = 'absolute';
 
      // Größe: mobil deutlich größer
      const size = isMobile
        ? (12 + Math.random() * 10)      // Mobil: 12–22 px
        : (5 + Math.random() * 4);    // Desktop: 5–9 px
 
      flake.style.fontSize = size + 'px';
 
      // Sichtbarkeit (Opacity): mobil deutlich höher
      const opacity = isMobile
        ? (0.45 + Math.random() * 0.4)  // Mobil: 0.45–0.85
        : (0.15 + Math.random() * 0.18); // Desktop: 0.15–0.33
 
      flake.style.opacity = opacity.toFixed(2);
      flake.style.color = '#ffffff';
 
      // Startposition
      flake.style.left = Math.random() * 100 + 'vw';
      flake.style.top  = -(Math.random() * 20) + 'vh';
 
      // Geschwindigkeit: mobil etwas schneller, aber nicht hektisch
      const duration = isMobile
        ? (10 + Math.random() * 12)    // Mobil: 10–22 s
        : (18 + Math.random() * 22);    // Desktop: 18–40 s
 
      const drift = isMobile ? 18 : 10; // seitliche Drift in px
 
      flake.style.animation = `adosSnowSoft ${duration}s linear infinite`;
      flake.style.setProperty('--ados-snow-drift', drift + 'px');
      flake.style.animationDelay = (-Math.random() * duration) + 's';
 
      container.appendChild(flake);
    }
  }
 
  // Keyframes: benutzen CSS-Variable für Drift
  const style = document.createElement('style');
  style.textContent = `
    @keyframes adosSnowSoft {
      0%  { transform: translateY(-12vh) translateX(0); }
      100% { transform: translateY(110vh) translateX(var(--ados-snow-drift, 10px)); }
    }
  `;
  document.head.appendChild(style);
 
   if (document.readyState === 'loading') {
     document.addEventListener('DOMContentLoaded', initSnow);
   } else {
   } else {
     initSnow();
     checkNeuBadges();
   }
   }
});


})();
/* ============================================================
  ADOS – Feuerwerk mit Raketen + Explosion (sichtbar, ES5)
  - dauerhaft
  - kein Abdunkeln (kein schwarzer Schleier)
  - Raketen mit Trail, Explosion deutlich
  ============================================================ */
(function () {
  'use strict';
  // true = nur 31.12/01.01, false = immer
  var onlyOnNewYears = false;
  // Debug in Konsole
  var DEBUG = false;
  function isNewYears() {
    var d = new Date();
    var m = d.getMonth() + 1;
    var day = d.getDate();
    return (m === 12 && day === 31) || (m === 1 && day === 1);
  }
  if (onlyOnNewYears && !isNewYears()) return;
  function log() {
    if (!DEBUG || !window.console) return;
    try { console.log.apply(console, arguments); } catch (e) {}
  }
  // Canvas anlegen
  function createCanvas() {
    // falls schon existiert (z.B. nach Reload), entfernen
    var old = document.getElementById('ados-fireworks-canvas');
    if (old && old.parentNode) old.parentNode.removeChild(old);
    var c = document.createElement('canvas');
    c.id = 'ados-fireworks-canvas';
    c.style.position = 'fixed';
    c.style.left = '0';
    c.style.top = '0';
    c.style.width = '100%';
    c.style.height = '100%';
    c.style.pointerEvents = 'none';
    c.style.zIndex = '9999';
    c.style.opacity = '1';
    document.body.appendChild(c);
    return c;
  }
  function fitCanvas(c) {
    var dpr = window.devicePixelRatio || 1;
    c.width = Math.floor(window.innerWidth * dpr);
    c.height = Math.floor(window.innerHeight * dpr);
    var ctx = c.getContext('2d');
    // Koordinaten in CSS-Pixeln nutzen
    ctx.setTransform(dpr, 0, 0, dpr, 0, 0);
    // Optional: additiver Look für „Glow“
    ctx.globalCompositeOperation = 'lighter';
    return ctx;
  }
  function rand(min, max) { return min + Math.random() * (max - min); }
  function clamp(v, a, b) { return Math.max(a, Math.min(b, v)); }
  var canvas, ctx;
  var rockets = [];
  var particles = [];
  var last = 0;
  var running = true;
  // ---- Rocket Spawn ----
  function spawnRocket() {
    var w = window.innerWidth;
    var h = window.innerHeight;
    var x = rand(60, w - 60);
    var y = h + rand(20, 120);
    // Explosionshöhe
    var targetY = rand(h * 0.12, h * 0.50);
    // Start-Geschwindigkeit nach oben (negativ)
    var vy = rand(-14.0, -18.0);
    var vx = rand(-1.8, 1.8);
    // „Glow“-Farbe (gold/weiß)
    var r = Math.floor(rand(220, 255));
    var g = Math.floor(rand(170, 240));
    var b = Math.floor(rand(80, 170));
    rockets.push({
      x: x,
      y: y,
      vx: vx,
      vy: vy,
      targetY: targetY,
      age: 0,
      life: rand(2200, 3200),
      r: r, g: g, b: b,
      trail: [] // Liste von Punkten für Spur
    });
  }
  // ---- Explosion ----
  function explode(x, y) {
    var count = Math.floor(rand(70, 140));
    var i;
    log('[Fireworks] explode @', x, y, 'count', count);
    for (i = 0; i < count; i++) {
      var angle = rand(0, Math.PI * 2);
      var speed = rand(2.5, 7.5);
      // knalligere Farben
      var rr = Math.floor(rand(120, 255));
      var gg = Math.floor(rand(80, 240));
      var bb = Math.floor(rand(120, 255));
      // etwas „Gold“-Bias
      if (Math.random() < 0.40) {
        rr = Math.floor(rand(220, 255));
        gg = Math.floor(rand(150, 230));
        bb = Math.floor(rand(30, 120));
      }
      particles.push({
        x: x,
        y: y,
        vx: Math.cos(angle) * speed,
        vy: Math.sin(angle) * speed,
        age: 0,
        life: rand(900, 1600),
        size: rand(1.8, 3.8),
        r: rr, g: gg, b: bb
      });
    }
  }
  // ---- Frame ----
  function tick(ts) {
    if (!running) return;
    if (!last) last = ts;
    var dt = ts - last;
    last = ts;
    var w = window.innerWidth;
    var h = window.innerHeight;
    // KEIN Abdunkeln: wir „faden“ mit transparenter WEISS-Schicht minimal.
    // Dadurch gibt es einen soften Trail-Effekt, aber keinen dunklen Schleier.
    // (Wenn du absolut gar keinen Trail willst: ctx.clearRect(...) statt fillRect)
    ctx.globalCompositeOperation = 'source-over';
    ctx.fillStyle = 'rgba(255,255,255,0.10)'; // kleiner Wert = weniger „Aufhellen“
    ctx.fillRect(0, 0, w, h);
    // Glow wieder aktivieren
    ctx.globalCompositeOperation = 'lighter';
    // --- Rockets ---
    var i, r;
    for (i = rockets.length - 1; i >= 0; i--) {
      r = rockets[i];
      r.age += dt;
      // Trail-Punkt speichern
      r.trail.push({ x: r.x, y: r.y });
      if (r.trail.length > 18) r.trail.shift();
      // Physik: Gravity zieht runter (vy wird weniger negativ)
      r.vy += 0.010 * (dt / 16); // stärker, damit es natürlicher wirkt
      r.x += r.vx * (dt / 16);
      r.y += r.vy * (dt / 16);
      // Explodieren, wenn Zielhöhe erreicht
      if (r.y <= r.targetY) {
        explode(r.x, r.y);
        rockets.splice(i, 1);
        continue;
      }
      // Sicherheit
      if (r.age >= r.life || r.y < -200 || r.x < -200 || r.x > w + 200) {
        rockets.splice(i, 1);
        continue;
      }
      // Trail zeichnen
      ctx.beginPath();
      ctx.lineWidth = 3.2;
      ctx.strokeStyle = 'rgba(' + r.r + ',' + r.g + ',' + r.b + ',0.35)';
      var t;
      for (t = 0; t < r.trail.length; t++) {
        var pt = r.trail[t];
        if (t === 0) ctx.moveTo(pt.x, pt.y);
        else ctx.lineTo(pt.x, pt.y);
      }
      ctx.stroke();
      // Rocket-Kopf (hell)
      ctx.beginPath();
      ctx.fillStyle = 'rgba(' + r.r + ',' + r.g + ',' + r.b + ',1)';
      ctx.arc(r.x, r.y, 3.0, 0, Math.PI * 2, false);
      ctx.fill();
    }
    // --- Particles ---
    var p;
    for (i = particles.length - 1; i >= 0; i--) {
      p = particles[i];
      p.age += dt;
      if (p.age >= p.life) {
        particles.splice(i, 1);
        continue;
      }
      // Bewegung
      p.vy += 0.020 * (dt / 16);
      p.vx *= Math.pow(0.985, dt / 16);
      p.vy *= Math.pow(0.985, dt / 16);
      p.x += p.vx * (dt / 16);
      p.y += p.vy * (dt / 16);


      var alpha = clamp(1 - (p.age / p.life), 0, 1);
// ============================================================


      ctx.beginPath();
      ctx.fillStyle = 'rgba(' + p.r + ',' + p.g + ',' + p.b + ',' + alpha + ')';
      ctx.arc(p.x, p.y, p.size, 0, Math.PI * 2, false);
      ctx.fill();
    }


    requestAnimationFrame(tick);
mw.loader.load('/wiki/MediaWiki:WhiskybaseBatch.js?action=raw&ctype=text/javascript');
  }
 
  // ---- Spawner ----
  function scheduleRockets() {
    function loop() {
      if (!running) return;
 
      // 1–3 Raketen pro Tick
      spawnRocket();
      if (Math.random() < 0.55) spawnRocket();
      if (Math.random() < 0.20) spawnRocket();
 
      // Frequenz (kleiner = mehr Feuerwerk)
      setTimeout(loop, Math.floor(rand(650, 1200)));
    }
    loop();
  }
 
  function init() {
    canvas = createCanvas();
    ctx = fitCanvas(canvas);
 
    window.addEventListener('resize', function () {
      if (!canvas) return;
      ctx = fitCanvas(canvas);
    });
 
    log('[Fireworks] init');
    scheduleRockets();
    requestAnimationFrame(tick);
  }
 
  if (document.readyState === 'loading') {
    document.addEventListener('DOMContentLoaded', init, false);
  } else {
    init();
  }
})();

Aktuelle Version vom 6. April 2026, 15:42 Uhr

/* Das folgende JavaScript wird für alle Benutzer geladen. */

/* ADOS Whisky-Ratings – RatePage Frontend (ES5, Widgets + Stats + Summary, Doppel-Init-Schutz; ANON VOTING ERLAUBT) */
mw.loader.using(['mediawiki.api', 'mediawiki.user']).then(function () {

  // ---------- kleine Hilfsfunktion ----------
  function get(obj, path) {
    var cur = obj, i;
    for (i = 0; i < path.length; i++) {
      if (!cur || typeof cur !== 'object') return undefined;
      cur = cur[path[i]];
    }
    return cur;
  }

  // ---------- Bootstrapping ----------
  function boot(root) {
    var scope = root || document;
    var i, nodes;

    nodes = scope.querySelectorAll('.whisky-rating__item');
    for (i = 0; i < nodes.length; i++) setupWidget(nodes[i]);

    initMetaOnly(scope);

    nodes = scope.querySelectorAll('[data-ratepage-summary="true"]');
    for (i = 0; i < nodes.length; i++) renderSummary(nodes[i]);
  }

  if (document.readyState === 'loading') {
    document.addEventListener('DOMContentLoaded', function(){ boot(document); });
  } else {
    boot(document);
  }
  mw.hook('wikipage.content').add(function($content){
    if ($content && $content[0]) boot($content[0]);
  });

  // ---------- Interaktives Widget ----------
  function setupWidget(box) {
    if (box.getAttribute('data-rating-init') === '1') return;
    box.setAttribute('data-rating-init', '1');

    var pageId  = mw.config.get('wgArticleId');
    var contest = box.dataset.ratepageContest || undefined;
    var scale   = parseInt(box.dataset.ratepageScale || '10', 10);

    var widget  = box.querySelector('.whisky-rating__widget');
    var meta    = box.querySelector('.whisky-rating__meta');

    while (widget.firstChild) widget.removeChild(widget.firstChild);

    // Anonyme NICHT blockieren – nur optionaler Hinweis
    var isAnon  = mw.user.isAnon();
    if (isAnon && meta && !meta.textContent) {
      meta.textContent = 'Bewerte diesen Whisky!';
    }

    var buttons = [];
    var i;
    for (i = 1; i <= scale; i++) {
      (function(iVal){
        var btn = document.createElement('button');
        btn.type = 'button';
        btn.className = 'whisky-glass';
        btn.setAttribute('aria-label', iVal + ' von ' + scale);
        btn.setAttribute('aria-pressed', 'false');

        // Immer klickbar – egal ob anonym oder eingeloggt
        btn.title = iVal + ' / ' + scale;
        btn.addEventListener('mouseenter', function(){ highlight(iVal); });
        btn.addEventListener('mouseleave', function(){ highlight(current); });
        btn.addEventListener('click',      function(){ vote(iVal); });
        btn.addEventListener('keydown',    function(e){
          if (e.key === 'Enter' || e.key === ' ') { e.preventDefault(); vote(iVal); }
          if (e.key === 'ArrowRight' && iVal < scale) buttons[iVal].focus();
          if (e.key === 'ArrowLeft'  && iVal > 1)     buttons[iVal-2].focus();
        });

        widget.appendChild(btn);
        buttons.push(btn);
      })(i);
    }

    var current = 0;
    highlight(current);

    function highlight(n) {
      var j;
      for (j = 0; j < buttons.length; j++) {
        var active = (j < n);
        buttons[j].classList.toggle('is-active', active);
        buttons[j].setAttribute('aria-pressed', active ? 'true' : 'false');
      }
    }

    function updateStats() {
      var api = new mw.Api();
      api.get({
        action: 'query',
        prop: 'pagerating',
        pageids: pageId,
        prcontest: contest || undefined,
        format: 'json',
        errorformat: 'plaintext'
      }).done(function (data) {
        try {
          var pages = get(data, ['query','pages']) || {};
          var keys = []; for (var k in pages) if (Object.prototype.hasOwnProperty.call(pages, k)) keys.push(k);
          var pid = keys.length ? keys[0] : String(pageId);
          var page = pages[pid] || {};
          var pr = page.pagerating;

          if (!meta) return;

          if (!pr) {
            if (!meta.textContent) meta.textContent = 'Noch keine Bewertungen';
            return;
          }

          if (typeof pr.canSee !== 'undefined' && pr.canSee === 0) {
            meta.textContent = 'Bewertungen sind verborgen.';
          } else {
            var hist = pr.pageRating || {};
            var total = 0, sum = 0;
            for (var key in hist) {
              if (Object.prototype.hasOwnProperty.call(hist, key)) {
                var s = parseInt(key, 10), c = parseInt(hist[key], 10);
                if (!isNaN(s) && !isNaN(c)) { total += c; sum += s * c; }
              }
            }
            meta.textContent = total
              ? ('Ø ' + (Math.round((sum/total)*10)/10) + ' (' + total + ' Stimmen)')
              : 'Noch keine Bewertungen';
          }

          if (pr.userVote) {
            current = pr.userVote;
            highlight(current);
          }

          if (typeof pr.canVote !== 'undefined' && pr.canVote === 0) {
            // Serverseitig verboten → hier deaktivieren
            box.classList.add('whisky-rating--disabled');
            var gls = widget.querySelectorAll('.whisky-glass');
            for (var i2 = 0; i2 < gls.length; i2++) gls[i2].disabled = true;
            if (meta.textContent.indexOf('nicht abstimmen') === -1) {
              meta.textContent += (meta.textContent ? ' • ' : '') + 'Du darfst hier nicht abstimmen.';
            }
          }
        } catch (e) {
          if (window.console && console.error) console.error(e);
          if (meta && !meta.textContent) meta.textContent = 'Bewertungen konnten nicht geladen werden.';
        }
      }).fail(function (xhr) {
        if (window.console && console.error) console.error('Pagerating-Load-Error', xhr);
        if (meta && !meta.textContent) meta.textContent = 'Bewertungen konnten nicht geladen werden.';
      });
    }

    function vote(value) {
      var api = new mw.Api();
      if (meta) meta.textContent = 'Wird gespeichert …';

      var saved = false;
      var failTimer = setTimeout(function () {
        if (!saved && meta) meta.textContent = 'Speichern dauert ungewöhnlich lange … bitte Seite neu laden.';
      }, 8000);

      api.postWithToken('csrf', {
        action: 'ratepage',
        pageid: pageId,
        answer: value,
        contest: contest || undefined,
        format: 'json'
      }).done(function () {
        saved = true;
        clearTimeout(failTimer);
        current = value;
        highlight(current);
        if (meta) meta.textContent = 'Danke! Deine Bewertung: ' + value + ' / ' + scale;
        updateStats();
      }).fail(function (xhr) {
        clearTimeout(failTimer);
        var msg = 'Unbekannter Fehler';
        try {
          var j = xhr && xhr.responseJSON ? xhr.responseJSON : xhr;
          if (j && j.error) {
            msg = (j.error.code ? j.error.code + ': ' : '') + (j.error.info || '');
          }
        } catch(e){}
        if (window.console && console.error) console.error('RatePage-API-Fehler:', xhr);
        if (meta) meta.textContent = 'Speichern fehlgeschlagen: ' + msg;
      });
    }

    updateStats();
  }

  // ---------- Meta-only ----------
  function initMetaOnly(scope) {
    var root = scope || document;
    var nodes = root.querySelectorAll('.whisky-rating__meta-only');
    var i;
    for (i = 0; i < nodes.length; i++) (function(box){
      if (box.getAttribute('data-meta-init') === '1') return;
      box.setAttribute('data-meta-init', '1');

      var pageId = parseInt(box.dataset.ratepagePageid || mw.config.get('wgArticleId'), 10);
      var contest = box.dataset.ratepageContest || undefined;

      new mw.Api().get({
        action: 'query',
        prop: 'pagerating',
        pageids: pageId,
        prcontest: contest || undefined,
        format: 'json'
      }).done(function (data) {
        var pages = get(data, ['query','pages']) || {};
        var keys = []; for (var k in pages) if (Object.prototype.hasOwnProperty.call(pages,k)) keys.push(k);
        var pid = keys.length ? keys[0] : String(pageId);
        var pr = pages[pid] && pages[pid].pagerating;
        if (!pr) { box.textContent = ''; return; }
        if (typeof pr.canSee !== 'undefined' && pr.canSee === 0) { box.textContent = 'Bewertung verborgen'; return; }
        var hist = pr.pageRating || {};
        var total = 0, sum = 0;
        for (var key in hist) {
          if (Object.prototype.hasOwnProperty.call(hist, key)) {
            var s = Number(key), c = Number(hist[key]);
            if (s && c) { total += c; sum += s * c; }
          }
        }
        box.textContent = total ? ('Ø ' + (Math.round((sum/total)*10)/10) + ' (' + total + ' Stimmen)') : 'Noch keine Bewertungen';
      });
    })(nodes[i]);
  }

  // ---------- Summary inkl. Gesamt + Balken ----------
  function renderSummary(container) {
    if (container.getAttribute('data-summary-init') === '1') return;
    container.setAttribute('data-summary-init', '1');

    var pageId   = mw.config.get('wgArticleId');
    var raw      = container.dataset.ratepageContests || 'NASE,GESCHMACK,ABGANG';
    var parts    = raw.split(',');
    var i;

    for (i = 0; i < parts.length; i++) parts[i] = parts[i].replace(/^\s+|\s+$/g, '');

    var nameToId = { 'nase':'NASE', 'geschmack':'GESCHMACK', 'abgang':'ABGANG', 'gesamteindruck':'GESAMTEINDRUCK' };
    var contests = [];
    var seen = {};
    for (i = 0; i < parts.length; i++) {
      var key = parts[i]; if (!key) continue;
      var norm = key.toLowerCase();
      var id = nameToId[norm] ? nameToId[norm] : key;
      if (!seen[id]) { contests.push(id); seen[id] = true; }
    }

    var labels = { NASE: 'Nase', GESCHMACK: 'Geschmack', ABGANG: 'Abgang', GESAMTEINDRUCK: 'Gesamteindruck' };
    container.textContent = 'Lade Bewertungen …';

    function fetchContest(contest) {
      return new mw.Api().get({
        action: 'query',
        prop: 'pagerating',
        pageids: pageId,
        prcontest: contest,
        format: 'json',
        errorformat: 'plaintext'
      }).then(function (data) {
        var pages = get(data, ['query','pages']) || {};
        var keys = []; for (var k in pages) if (Object.prototype.hasOwnProperty.call(pages,k)) keys.push(k);
        var pid = keys.length ? keys[0] : String(pageId);
        var pr = pages[pid] && pages[pid].pagerating;

        if (!pr || (typeof pr.canSee !== 'undefined' && pr.canSee === 0)) {
          return { contest: contest, label: (labels[contest] || contest), avg: null, total: 0 };
        }
        var hist = pr.pageRating || {};
        var total = 0, sum = 0;
        for (var key in hist) {
          if (Object.prototype.hasOwnProperty.call(hist, key)) {
            var s = Number(key), c = Number(hist[key]);
            if (s && c) { total += c; sum += s * c; }
          }
        }
        var avg = total ? Math.round((sum / total) * 10) / 10 : null;
        return { contest: contest, label: (labels[contest] || contest), avg: avg, total: total };
      }, function () {
        return { contest: contest, label: (labels[contest] || contest), avg: null, total: 0, _error: true };
      });
    }

    var promises = [];
    for (i = 0; i < contests.length; i++) promises.push(fetchContest(contests[i]));

    Promise.all(promises).then(function (rows) {
      if (!rows || !rows.length) {
        container.textContent = 'Konnte Bewertungen nicht laden.';
        return;
      }

      var table = document.createElement('table');
      table.className = 'whisky-summary__table';

      var thead = document.createElement('thead');
      thead.innerHTML = '<tr><th>Kategorie</th><th>Ø</th><th>Stimmen</th></tr>';
      table.appendChild(thead);

      var tbody = document.createElement('tbody');

      // Zeilen mit Balken
      var r;
      for (r = 0; r < rows.length; r++) {
        var row = rows[r];
        var totalText = row.total ? String(row.total) : '0';

        var tr = document.createElement('tr');

        var tdLabel = document.createElement('td');
        tdLabel.textContent = row.label;
        tr.appendChild(tdLabel);

        var tdAvg = document.createElement('td');
        if (row.avg !== null) {
          var wrap  = document.createElement('div');  wrap.className = 'whisky-bar';
          var track = document.createElement('div');  track.className = 'whisky-bar__track';
          var fill  = document.createElement('div');  fill.className  = 'whisky-bar__fill';
          fill.style.width = Math.max(0, Math.min(100, (row.avg/10)*100)) + '%';
          var val   = document.createElement('span'); val.className   = 'whisky-bar__value';
          val.textContent = (row.avg.toFixed ? row.avg.toFixed(1) : (Math.round(row.avg*10)/10));
          track.appendChild(fill); wrap.appendChild(track); wrap.appendChild(val); tdAvg.appendChild(wrap);
        } else {
          tdAvg.textContent = '–';
        }
        tr.appendChild(tdAvg);

        var tdCnt = document.createElement('td');
        tdCnt.textContent = totalText;
        tr.appendChild(tdCnt);

        tbody.appendChild(tr);
      }

      // Gesamt
      var present = 0, sumAvg = 0, totalVotes = 0;
      for (r = 0; r < rows.length; r++) {
        if (rows[r].avg !== null) { present++; sumAvg += rows[r].avg; }
        if (rows[r].total) totalVotes += rows[r].total;
      }
      var overall = (present > 0) ? Math.round((sumAvg / present) * 10) / 10 : null;
      var overallText = (overall !== null)
        ? (overall.toFixed ? overall.toFixed(1) : (Math.round(overall*10)/10))
        : '–';

      var trG = document.createElement('tr');

      var tdGL = document.createElement('td');
      tdGL.innerHTML = '<strong>Gesamt</strong>';
      trG.appendChild(tdGL);

      var tdGA = document.createElement('td');
      if (overall !== null) {
        var w  = document.createElement('div');  w.className = 'whisky-bar';
        var t  = document.createElement('div');  t.className = 'whisky-bar__track';
        var f  = document.createElement('div');  f.className = 'whisky-bar__fill';
        f.style.width = Math.max(0, Math.min(100, (overall/10)*100)) + '%';
        var v  = document.createElement('span'); v.className = 'whisky-bar__value';
        v.innerHTML = '<strong>' + overallText + '</strong>';
        t.appendChild(f); w.appendChild(t); w.appendChild(v); tdGA.appendChild(w);
      } else {
        tdGA.innerHTML = '<strong>–</strong>';
      }
      trG.appendChild(tdGA);

      var tdGD = document.createElement('td');
      tdGD.textContent = totalVotes;
      trG.appendChild(tdGD);

      tbody.appendChild(trG);

      table.appendChild(tbody);

      while (container.firstChild) container.removeChild(container.firstChild);
      container.appendChild(table);

      var badge = document.getElementById('whisky-overall-badge');
      if (badge && overall !== null) {
        badge.textContent = overallText;
      }
    }).catch(function(){
      container.textContent = 'Konnte Bewertungen nicht geladen werden.';
    });
  }

});


/* --- Whisky Top-5 (multi-root, recursive, robust, namespaces, ES5) ------- */
mw.loader.using(['mediawiki.api']).then(function () {

  /* ========== Utils ========== */
  function get(obj, path) {
    var cur = obj, i;
    for (i = 0; i < path.length; i++) { if (!cur || typeof cur !== 'object') return; cur = cur[path[i]]; }
    return cur;
  }
  function parseWeights(raw, contests) {
    var map = {}, parts = (raw || '').split(','), i;
    for (i = 0; i < parts.length; i++) {
      var kv = parts[i].split(':'); if (kv.length !== 2) continue;
      var k = kv[0].replace(/^\s+|\s+$/g, ''), v = parseFloat(kv[1]);
      if (!isNaN(v)) map[k] = v;
    }
    for (i = 0; i < contests.length; i++) if (typeof map[contests[i]] !== 'number') map[contests[i]] = 1;
    return map;
  }

  /* ========== Statusbox im Widget ========== */
  function makeStatus(container, keep){
    var box = container.querySelector('.whisky-top5__status');
    if (!box) {
      box = document.createElement('div');
      box.className = 'whisky-top5__status';
      container.insertBefore(box, container.firstChild || null);
    }
    function line(txt){ var row = document.createElement('div'); row.textContent = txt; box.appendChild(row); box.scrollTop = box.scrollHeight; }
    function status(msg, append){ if (!append && !keep) box.innerHTML = ''; line(msg); /* console.log('[Top5]', msg); */ }
    status.done = function(delayMs){ if (keep) return; setTimeout(function(){ if (box && box.parentNode) box.parentNode.removeChild(box); }, typeof delayMs==='number'?delayMs:3000); };
    return status;
  }

  /* ========== Robuste Kategorie-Auflösung ========== */
  function categoryCandidates(name){
    var n = (name || '').replace(/^\s+|\s+$/g,'');
    var v = {}, add=function(s){ if (s && !v[s]) v[s]=1; };
    add(n);
    add(n.replace(/\u2026/g,'...'));  // … -> ...
    add(n.replace(/\.{3}/g,'…'));     // ... -> …
    add(n.replace(/\u2013/g,'-'));    // – -> -
    add(n.replace(/-/g,'–'));         // - -> –
    add(n.replace(/\s+/g,' '));       // Mehrfach-Spaces
    return Object.keys(v);
  }
  function resolveCategoryTitle(api, rawName){
    var cands = categoryCandidates(rawName).map(function(n){ return 'Kategorie:' + n; });
    return api.get({ action:'query', titles: cands.join('|'), format:'json' }).then(function(d){
      var pages = get(d,['query','pages']) || {}, pid, p;
      for (pid in pages) if (Object.prototype.hasOwnProperty.call(pages,pid)) { p = pages[pid]; if (p && p.pageid && p.ns === 14) return p.title; }
      return null;
    });
  }

  /* ========== Kategorien rekursiv einsammeln (inkl. Subkats, Namespaces) ========== */
function fetchCategoryMembersRecursiveSingleResolved(api, catTitle, limit, outSet, pages, nsStr){
  var visited = {}, queue = [catTitle];
  var cmNS = (nsStr && nsStr.trim()) ? nsStr.trim() : '0|14';  // default: Artikel+Kategorie

  function fetchOne(title, cont){
    var params = {
      action: 'query',
      list: 'categorymembers',
      cmtitle: title,
      cmtype: 'page|subcat',
      cmlimit: Math.min(200, limit),
      format: 'json'
    };
    if (cmNS !== '*' && cmNS !== '') {
      params.cmnamespace = cmNS; // z.B. "0|102|14"
    }
    if (cont) params.cmcontinue = cont;

    return api.get(params).then(function(d){
      var cms = (d.query && d.query.categorymembers) || [], i, it;
      for (i = 0; i < cms.length; i++) {
        it = cms[i];
        if (it.ns !== 14) { // alles außer Kategorien zählt als Seite
          var pid = String(it.pageid);
          if (!outSet[pid] && pages.length < limit) {
            outSet[pid] = 1;
            pages.push({ pageid: pid, title: it.title });
          }
        } else {
          var sub = it.title;
          if (!visited[sub]) { visited[sub] = 1; queue.push(sub); }
        }
      }
      var next = d.continue && d.continue.cmcontinue;
      if (next && pages.length < limit) return fetchOne(title, next);
    });
  }

  function loop(){
    if (pages.length >= limit || !queue.length) return Promise.resolve();
    var next = queue.shift();
    if (visited[next]) return loop();
    visited[next] = 1;
    return fetchOne(next).then(loop);
  }

  return loop();
}

  function fetchCategoryMembersRecursiveMulti(rootCats, limit, status, nsStr){
    var api = new mw.Api();
    var pages = [], outSet = {};
    var idx = 0;

    function next(){
      if (idx >= rootCats.length || pages.length >= limit) return Promise.resolve(pages);
      var raw = rootCats[idx++]; if (!raw) return next();

      return resolveCategoryTitle(api, raw).then(function(resolved){
        if (!resolved) { status('Kategorie nicht gefunden: "' + raw + '"', true); return next(); }
        status('Kategorie erkannt: ' + resolved + ' – sammle …', true);
        var before = pages.length;
        return fetchCategoryMembersRecursiveSingleResolved(api, resolved, limit, outSet, pages, nsStr).then(function(){
          var added = pages.length - before;
          status('→ gefunden in "' + resolved + '": ' + added + ' Seiten (kumuliert: ' + pages.length + ')', true);
          return next();
        });
      });
    }

    return next();
  }

  /* ========== Ratings laden / auswerten ========== */
  function fetchRatingsForContest(pageIds, contest, includeHidden) {
    var api = new mw.Api(), res = {}, i, chunk = 50, chunks = [];
    for (i = 0; i < pageIds.length; i += chunk) chunks.push(pageIds.slice(i, i + chunk));

    function step(idx) {
      if (idx >= chunks.length) return Promise.resolve(res);
      var ids = chunks[idx];
      return api.get({
        action:'query', prop:'pagerating', pageids: ids.join('|'),
        prcontest: contest, format:'json'
      }).then(function(d){
        var pages = get(d,['query','pages']) || {}, pid, pr, hist, k, total, sum, s, c;
        for (pid in pages) if (Object.prototype.hasOwnProperty.call(pages,pid)) {
          if (!res[pid]) res[pid] = { avg:null, total:0 };
          pr = pages[pid].pagerating;
          if (pr && (includeHidden || !('canSee' in pr) || pr.canSee !== 0)) {
            hist = pr.pageRating || {}; total=0; sum=0;
            for (k in hist) if (Object.prototype.hasOwnProperty.call(hist,k)) {
              s = parseInt(k,10); c = parseInt(hist[k],10);
              if (!isNaN(s) && !isNaN(c)) { total += c; sum += s*c; }
            }
            if (total > 0) res[pid] = { avg: Math.round((sum/total)*10)/10, total: total };
          }
        }
        return step(idx + 1);
      }, function(){ return step(idx + 1); });
    }
    return step(0);
  }

  function computeOverall(entry, contests, weights) {
    var wSum=0, wAvgSum=0, present=0, totalVotes=0, i, sc, w;
    for (i=0;i<contests.length;i++){
      sc = entry.scores[contests[i]];
      if (sc && sc.avg !== null) { w = (typeof weights[contests[i]]==='number') ? weights[contests[i]] : 1; wSum+=w; wAvgSum+=sc.avg*w; present++; }
      if (sc && sc.total) totalVotes += sc.total;
    }
    entry.totalVotes = totalVotes;
    entry.overall = (present>0 && wSum>0) ? Math.round((wAvgSum/wSum)*10)/10 : null;
  }

  /* ========== Render ========== */
  function renderTopN(container, rows, N, minVotes) {
    var keep = (container.getAttribute && container.getAttribute('data-keep-status') === 'true');
    var statusBox = keep ? container.querySelector('.whisky-top5__status') : null;

    rows = rows.filter(function(r){ return (r.overall !== null) && (r.totalVotes >= minVotes); });
    rows.sort(function(a,b){
      if (a.overall===null && b.overall!==null) return 1;
      if (a.overall!==null && b.overall===null) return -1;
      if (b.overall!==a.overall) return b.overall - a.overall;
      if (b.totalVotes!==a.totalVotes) return b.totalVotes - a.totalVotes;
      return a.title.localeCompare(b.title);
    });
    rows = rows.slice(0, N);

    while (container.firstChild) container.removeChild(container.firstChild);
    if (statusBox) container.appendChild(statusBox);

    if (!rows.length) { container.appendChild(document.createTextNode('Noch keine Bewertungen vorhanden.')); return; }

    var frag = document.createDocumentFragment(), i, r, item, rank, name, a, right, mini, track, fill, val, votes;
    for (i=0;i<rows.length;i++){
      r = rows[i];
      item = document.createElement('div'); item.className = 'whisky-top5__item';

      rank = document.createElement('div'); rank.className = 'whisky-top5__rank'; rank.textContent = (i+1);

      name = document.createElement('div'); name.className = 'whisky-top5__name';
      a = document.createElement('a'); a.href = mw.util.getUrl(r.title); a.textContent = r.title; name.appendChild(a);

      right = document.createElement('div'); right.style.minWidth = '160px';

      mini = document.createElement('div'); mini.className = 'whisky-mini';
      track = document.createElement('div'); track.className = 'whisky-mini__track';
      fill = document.createElement('div'); fill.className = 'whisky-mini__fill';
      fill.style.width = Math.max(0, Math.min(100, (r.overall/10)*100)) + '%';
      val = document.createElement('span'); val.className = 'whisky-mini__val';
      val.textContent = (r.overall.toFixed ? r.overall.toFixed(1) : (Math.round(r.overall*10)/10));

      track.appendChild(fill); mini.appendChild(track); mini.appendChild(val);

      votes = document.createElement('div'); votes.className = 'whisky-top5__votes';
      votes.textContent = r.totalVotes + ' Stimmen';

      right.appendChild(mini); right.appendChild(votes);

      item.appendChild(rank); item.appendChild(name); item.appendChild(right);
      frag.appendChild(item);
    }
    container.appendChild(frag);
  }

  /* ========== Boot ========== */
  function bootTop5(root) {
    var nodes = (root || document).querySelectorAll('.whisky-top5');
    if (!nodes.length) return;

    for (var n=0;n<nodes.length;n++) (function(container){
      if (container.getAttribute('data-top5-init')==='1') return;
      container.setAttribute('data-top5-init','1');

      var rawCats = container.getAttribute('data-categories') || container.getAttribute('data-category') || '';
      var parts = rawCats.split(/\n|;/), rootCats = [], i;
      for (i=0;i<parts.length;i++){ var nm = parts[i].replace(/^\s+|\s+$/g,''); if (nm) rootCats.push(nm); }
      if (!rootCats.length) { container.textContent = 'Keine Kategorien angegeben.'; return; }

      var lim = parseInt(container.getAttribute('data-limit') || '2000', 10);
      var cnt = parseInt(container.getAttribute('data-count') || '5', 10);
      var minVotes = parseInt(container.getAttribute('data-min-votes') || '1', 10);
      var includeHidden = (container.getAttribute('data-include-hidden') === 'true');
      var nsStr = container.getAttribute('data-namespaces') || '0|14';

      var rawC = container.getAttribute('data-contests') || 'NASE,GESCHMACK,ABGANG,GESAMTEINDRUCK';
      var cParts = rawC.split(','), contests = [], seen = {};
      for (i=0;i<cParts.length;i++){ var c = cParts[i].replace(/^\s+|\s+$/g,''); if (c && !seen[c]){ contests.push(c); seen[c]=1; } }
      var weights = parseWeights(container.getAttribute('data-weights') || '', contests);

      var keep = (container.getAttribute('data-keep-status') === 'true');
      function status(){ /* Debug-Ausgabe deaktiviert */ }
      status('Sammle Seiten …');

      fetchCategoryMembersRecursiveMulti(rootCats, lim, status, nsStr).then(function(members){
        status('Gefundene Seiten gesamt: ' + (members ? members.length : 0) + ' – lade Bewertungen …', true);
        if (!members || !members.length) { status('Keine passenden Seiten gefunden.', true); return; }

        var pageIds = [], byId = {}, i;
        for (i=0;i<members.length;i++){ pageIds.push(members[i].pageid); byId[members[i].pageid] = { pageid: members[i].pageid, title: members[i].title, scores: {} }; }

        function loopContest(idx){
          if (idx >= contests.length) return Promise.resolve();
          var contest = contests[idx];
          return fetchRatingsForContest(pageIds, contest, includeHidden).then(function(map){
            var pid; for (pid in map) if (Object.prototype.hasOwnProperty.call(map,pid)) byId[pid].scores[contest] = map[pid];
            return loopContest(idx + 1);
          });
        }

        function buildAndRender(){
          var rows = [], pid, e, withVotes = 0;
          for (pid in byId) if (Object.prototype.hasOwnProperty.call(byId,pid)) {
            e = byId[pid]; computeOverall(e, contests, weights); rows.push(e);
            if (e.totalVotes > 0 && e.overall !== null) withVotes++;
          }
          if (withVotes === 0 && !includeHidden) {
            status('Keine sichtbaren Stimmen – zweiter Versuch (versteckte Ergebnisse mitzählen) …', true);
            includeHidden = true;
            return loopContest(0).then(buildAndRender);
          }
          renderTopN(container, rows, cnt, minVotes);
          status.done(keep ? 0 : 3000);
        }

        loopContest(0).then(buildAndRender).catch(function(){ status('Topliste konnte nicht geladen werden.', true); });
      }).catch(function(){ status('Topliste konnte nicht geladen werden.'); });

    })(nodes[n]);
  }

  if (document.readyState === 'loading') {
    document.addEventListener('DOMContentLoaded', function(){ bootTop5(document); });
  } else { bootTop5(document); }
  mw.hook('wikipage.content').add(function($c){ if($c && $c[0]) bootTop5($c[0]); });

});


/* Render star ratings from data-rating on .rating elements (0..5, step .5) */
mw.hook('wikipage.content').add(function($content){
  $content.find('.rating').each(function(){
    var el = this, val = parseFloat(el.getAttribute('data-rating') || '0');
    if (isNaN(val)) val = 0;
    val = Math.max(0, Math.min(5, val));
    el.style.setProperty('--stars', (val).toString());
    el.setAttribute('aria-label', val + ' von 5 Sternen');
    el.setAttribute('title', val + ' von 5 Sternen');
  });
});


// Force light color-scheme at document level (helps Mobile Safari)
mw.loader.using('mediawiki.util').then(function () {
  var m = document.querySelector('meta[name="color-scheme"]');
  if (!m) {
    m = document.createElement('meta');
    m.name = 'color-scheme';
    m.content = 'light';
    document.head.appendChild(m);
  } else {
    m.content = 'light';
  }
});


function isFlagTrue(v){
  if (v == null) return false;
  v = String(v).trim().toLowerCase();
  return v === 'true' || v === '1' || v === 'yes';
}

// Gesamtzahl unter der Legende einfügen (im Diagramm-Block, nicht im Canvas!)
function addTotalBelowLegend(chart, block) {
  try {
    if (!chart || !block) return;

    const total = chart.data.datasets.reduce((sum, ds) =>
      sum + (ds.data || []).reduce((a, b) => a + (parseFloat(b) || 0), 0)
    , 0);

    const oldInfo = block.querySelector(':scope > .chart-total-info');
    if (oldInfo) oldInfo.remove();

    const info = document.createElement('div');
    info.className = 'chart-total-info';
    info.textContent = 'Gesamte Anzahl aller eigenen Abfüllungen: ' + total;
    info.style.textAlign = 'center';
    info.style.fontWeight = 'bold';
    info.style.fontSize = '1.05em';
    info.style.marginTop = '0.5rem';
    info.style.marginBottom = '0.5rem';
    info.style.color = '#444';
    block.appendChild(info);

  } catch(e) { console.warn('[addTotalBelowLegend]', e); }
}


/* === ADOS Multi-Serien-Chart (Chart.js) ============================= *
 * Lädt Chart.js sicher (asynchron) und baut Diagramme aus Cargo-Tabellen.
 * Benötigt im Artikel: <div class="ados-chart-multi"> + Cargo-Query als |format=table
 * ==================================================================== */

(function () {
  var _chartReady = null;
  function ensureChartJS() {
    if (_chartReady) return _chartReady;
    _chartReady = new Promise(function (resolve, reject) {
      if (window.Chart) return resolve();
      var s = document.createElement('script');
      s.src = 'https://cdn.jsdelivr.net/npm/chart.js';
      s.async = true;
      s.onload = function(){ resolve(); };
      s.onerror = function(){ console.error('Chart.js konnte nicht geladen werden'); reject(); };
      document.head.appendChild(s);
    });
    return _chartReady;
  }

  var ADOS_COLORS = {
    'A Dream of Scotland':                 '#C2410C',
    'A Dream of Ireland':                  '#15803D',
    'A Dream of... – Der Rest der Welt':   '#1D4ED8',
    'Friendly Mr. Z':                      '#9333EA',
    'Die Whisky Elfen':                    '#0891B2',
    'The Fine Art of Whisky':              '#CA8A04'
  };
  var COLOR_CYCLE = ['#2563eb','#16a34a','#f97316','#dc2626','#a855f7','#0ea5e9','#f59e0b','#10b981'];

  function toYear(x){
    var n = parseInt(String(x).replace(/[^\d]/g,''),10);
    return isFinite(n) ? n : null;
  }
  function getColor(name, used){
    if (ADOS_COLORS[name]) return ADOS_COLORS[name];
    var i = used.size % COLOR_CYCLE.length;
    used.add(name);
    return COLOR_CYCLE[i];
  }

  function buildDatasetsFromTable(tbl){
    var rows = Array.from(tbl.querySelectorAll('tr'));
    if (rows.length < 2) return { labels:[], datasets:[] };

    var yearsSet = new Set();
    var bySeries  = new Map();

    rows.slice(1).forEach(function(tr){
      var tds = tr.querySelectorAll('td,th');
      if (tds.length < 3) return;
      var y = toYear(tds[0].textContent.trim());
      var s = tds[1].textContent.trim();
      var v = parseFloat(tds[2].textContent.replace(',','.')) || 0;
      if (y == null) return;
      yearsSet.add(y);
      if (!bySeries.has(s)) bySeries.set(s, new Map());
      bySeries.get(s).set(y, v);
    });

    var years = Array.from(yearsSet).sort(function(a,b){return a-b;});
    var used = new Set();
    var labels = years.map(String);

    var datasets = Array.from(bySeries.entries()).map(function(entry){
      var name = entry[0], yearMap = entry[1];
      var data = years.map(function(y){ return yearMap.get(y) || 0; });
      var color = getColor(name, used);
      return {
        label: name,
        data: data,
        borderColor: color,
        backgroundColor: color + '80',
        tension: 0.25,
        pointRadius: 3
      };
    });

    return { labels: labels, datasets: datasets };
  }

  function renderOne(block){
    if (block.dataset.rendered === '1') return;

    var el = block.nextElementSibling, tbl = null, wrapToHide = null;
    while (el) {
      if (/^H[1-6]$/.test(el.tagName) || (el.classList && el.classList.contains('ados-chart-multi'))) break;
      if (el.tagName === 'TABLE') {
        tbl = el;
      } else if (el.querySelector) {
        var t = el.querySelector('table');
        if (t) tbl = t;
      }
      if (tbl) {
        wrapToHide = tbl.parentElement;
        break;
      }
      el = el.nextElementSibling;
    }
    if (!tbl) return;

    var out = buildDatasetsFromTable(tbl);
    if (!out.labels.length || !out.datasets.length) return;

    var hide = (block.dataset.hideTable || '').toLowerCase() === 'true';
    if (hide) {
      var onlyTable = wrapToHide && wrapToHide.children.length === 1 && wrapToHide.firstElementChild === tbl;
      if (onlyTable) {
        wrapToHide.setAttribute('aria-hidden','true');
        wrapToHide.style.display = 'none';
      } else {
        tbl.setAttribute('aria-hidden','true');
        tbl.style.display = 'none';
      }
    }

    var wrap = document.createElement('div');
    wrap.style.position = 'relative';
    wrap.style.width = '100%';
    wrap.style.height = block.dataset.height || (window.matchMedia('(min-width:768px)').matches ? '450px' : '300px');
    var canvas = document.createElement('canvas');
    wrap.appendChild(canvas);
    block.innerHTML = '';
    block.appendChild(wrap);

    var type  = (block.dataset.type || 'line').toLowerCase();
    var title = block.dataset.title || '';
    var cumulative = (block.dataset.cumulative || '').toLowerCase() === 'true';

    if (cumulative) {
      out.datasets = out.datasets.map(function(ds){
        var acc = 0;
        return Object.assign({}, ds, {
          data: ds.data.map(function(v){ acc += v; return acc; })
        });
      });
    }

    ensureChartJS().then(function(){
      const chart = new Chart(canvas.getContext('2d'), {
        type: type,
        data: { labels: out.labels, datasets: out.datasets },
        options: {
          responsive: true,
          maintainAspectRatio: false,
          interaction: { mode: 'nearest', intersect: false },
          plugins: {
            legend: { position: 'bottom', labels: { font: { size: 13 }, boxWidth: 20 } },
            title:  { display: !!title, text: title, font: { size: 16 } },
            tooltip:{ backgroundColor: 'rgba(0,0,0,0.8)', titleFont: {size:14}, bodyFont: {size:13} }
          },
          scales: {
            x: { ticks: { font: { size: 12 } } },
            y: { beginAtZero: true, ticks: { precision: 0, font: { size: 12 } } }
          }
        }
      });

      const hideTotal = (block.dataset.hideTotal || '').toLowerCase() === 'true';
      const oldInfo = block.querySelector(':scope > .chart-total-info');
      if (oldInfo) oldInfo.remove();

      if (!hideTotal) {
        addTotalBelowLegend(chart, block);

        if (window.ResizeObserver) {
          const obs = new ResizeObserver(() => addTotalBelowLegend(chart, block));
          obs.observe(chart.canvas);
          chart.$adosTotalObserver = obs;
        }
      }

      block.dataset.rendered = '1';
    });
  }

  function boot($scope){
    var blocks = ($scope && $scope[0] ? $scope[0] : document).querySelectorAll('.ados-chart-multi');
    if (!blocks.length) return;
    ensureChartJS().then(function(){ blocks.forEach(renderOne); });
  }

  if (window.mw && mw.hook) {
    mw.hook('wikipage.content').add(boot);
  } else {
    (document.readyState === 'loading')
      ? document.addEventListener('DOMContentLoaded', function(){ boot(); })
      : boot();
  }
})();


// ==========================Scan==================================
mw.loader.using('mediawiki.util').then(function () {
  if (mw.config.get('wgPageName') !== 'LabelScan') return;
  mw.loader.load('/index.php?title=MediaWiki:Gadget-LabelScan.js&action=raw&ctype=text/javascript');
  mw.loader.load('/index.php?title=MediaWiki:Gadget-LabelScan.css&action=raw&ctype=text/css', 'text/css');
});


// ==========================ScanApp==================================
/* ==== PWA: Manifest + Service Worker + Install-Button (ES5) ==== */

/* Manifest einbinden */
(function () {
  var link = document.createElement("link");
  link.rel = "manifest";
  link.href = "/app/labelscan/manifest.webmanifest";
  document.head.appendChild(link);
})();

/* Service Worker registrieren (nur wenn vorhanden) */
(function () {
  if ("serviceWorker" in navigator) {
    navigator.serviceWorker.register("/app/labelscan/sw.js")["catch"](function () {});
  }
})();

/* Install-Button steuern (Button-ID: ados-install) */
(function () {
  var installPrompt = null;

  window.addEventListener("beforeinstallprompt", function (e) {
    try { e.preventDefault(); } catch (ex) {}
    installPrompt = e;
    var btn = document.getElementById("ados-install");
    if (btn) btn.style.display = "inline-block";
  });

  function onReady(fn){ if (document.readyState === "loading") document.addEventListener("DOMContentLoaded", fn); else fn(); }
  onReady(function () {
    var btn = document.getElementById("ados-install");
    if (!btn) return;
    btn.addEventListener("click", function () {
      if (!installPrompt) return;
      try { installPrompt.prompt(); } catch (ex) {}
      installPrompt = null;
      btn.style.display = "none";
    });
  });
})();

if ('serviceWorker' in navigator) {
  navigator.serviceWorker.register('/app/labelscan/sw.js').catch(function(){});
}


// ============================================================

mw.loader.using('mediawiki.util').then(function () {
  function checkNeuBadges() {
    var badges = document.querySelectorAll('.ados-neu-badge');
    var now = new Date();

    badges.forEach(function (badge) {
      var expiry = badge.getAttribute('data-expiry');
      if (!expiry) return;

      var expiryDate = new Date(expiry + "T23:59:59");
      if (now > expiryDate) {
        badge.style.display = "none";
      }
    });
  }

  if (document.readyState === "loading") {
    document.addEventListener("DOMContentLoaded", checkNeuBadges);
  } else {
    checkNeuBadges();
  }
});


// ============================================================


mw.loader.load('/wiki/MediaWiki:WhiskybaseBatch.js?action=raw&ctype=text/javascript');