Zum Inhalt springen

MediaWiki:Common.js: Unterschied zwischen den Versionen

Aus ADOS Wiki
Keine Bearbeitungszusammenfassung
Markierung: Zurückgesetzt
Keine Bearbeitungszusammenfassung
Markierung: Manuelle Zurücksetzung
Zeile 1: Zeile 1:
/* Das folgende CSS wird für alle Benutzeroberflächen geladen. */
/* 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 ----------
  1) Whisky-Bewertung (Gläser, Balken, Summary-Tabelle)
  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]);
  }


/* Layout für die drei Bewertungsboxen */
  if (document.readyState === 'loading') {
.whisky-rating {
    document.addEventListener('DOMContentLoaded', function(){ boot(document); });
  display: grid;
   } else {
  grid-template-columns: repeat(auto-fit, minmax(260px, 1fr));
    boot(document);
   gap: 14px;
   }
   margin: 1rem 0 1.5rem;
  mw.hook('wikipage.content').add(function($content){
}
    if ($content && $content[0]) boot($content[0]);
  });


.whisky-rating__item {
  // ---------- Interaktives Widget ----------
   border: 1px solid #e5e7eb;
   function setupWidget(box) {
  border-radius: 12px;
    if (box.getAttribute('data-rating-init') === '1') return;
  padding: 12px 14px;
    box.setAttribute('data-rating-init', '1');
  box-shadow: 0 1px 3px rgba(0,0,0,.06);
  background: #fffaf2; /* leicht whisky-farbener Ton */
}


.whisky-rating__label {
    var pageId  = mw.config.get('wgArticleId');
  font-weight: 600;
    var contest = box.dataset.ratepageContest || undefined;
  margin-bottom: .5rem;
    var scale   = parseInt(box.dataset.ratepageScale || '10', 10);
   color: #111827;
}


/* Container für die Gläser */
    var widget  = box.querySelector('.whisky-rating__widget');
.whisky-rating__widget {
    var meta    = box.querySelector('.whisky-rating__meta');
  display: flex;
  flex-wrap: wrap;          /* erlaubt Umbruch bei kleinen Screens */
  align-items: center;
  gap: 6px;
}


/* Gläser (Button) – konsolidierte Definition */
    while (widget.firstChild) widget.removeChild(widget.firstChild);
.whisky-glass {
  display: inline-block;
  width: 36px;
  height: 36px;
  margin: 0 6px 6px 0;
  border: 0;
  padding: 0;
  background-color: transparent;
  background-image: url("/index.php?title=Special:FilePath/Whisky_empty.png");
  background-size: contain;
  background-repeat: no-repeat;
  background-position: center;
  cursor: pointer;
  line-height: 0;
  vertical-align: middle;
  box-sizing: content-box;
  opacity: .8;
  transition: opacity .08s ease, filter .08s ease;
  touch-action: manipulation;
}


/* Aktive (gefüllte) Gläser */
    // Anonyme NICHT blockieren – nur optionaler Hinweis
.whisky-glass.is-active {
    var isAnon  = mw.user.isAnon();
  background-image: url("/index.php?title=Special:FilePath/Whisky_filled.png");
    if (isAnon && meta && !meta.textContent) {
  opacity: 1;
      meta.textContent = 'Bewerte diesen Whisky!';
}
    }


/* Hover/Fokus-Effekt */
    var buttons = [];
.whisky-glass:hover,
    var i;
.whisky-glass:focus {
    for (i = 1; i <= scale; i++) {
  opacity: 1;
      (function(iVal){
  outline: none;
        var btn = document.createElement('button');
  filter: drop-shadow(0 0 0.4px rgba(0,0,0,.25));
        btn.type = 'button';
}
        btn.className = 'whisky-glass';
        btn.setAttribute('aria-label', iVal + ' von ' + scale);
        btn.setAttribute('aria-pressed', 'false');


/* Zustand: nicht eingeloggt */
        // Immer klickbar – egal ob anonym oder eingeloggt
.whisky-rating--disabled .whisky-glass {
        btn.title = iVal + ' / ' + scale;
  cursor: not-allowed;
        btn.addEventListener('mouseenter', function(){ highlight(iVal); });
  filter: grayscale(0.3) opacity(.5);
        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();
        });


/* Textbereich unter den Gläsern */
        widget.appendChild(btn);
.whisky-rating__meta {
        buttons.push(btn);
  margin-top: .4rem;
      })(i);
  font-size: .9rem;
    }
  color: #4b5563;
}


/* Mobile: kleinere Gläser */
    var current = 0;
@media (max-width: 480px) {
    highlight(current);
  .whisky-glass { width: 32px; height: 32px; margin: 0 5px 5px 0; }
}
@media (max-width: 420px) {
  .whisky-glass { width: 28px; height: 28px; }
}


/* Tabelle für Whisky-Bewertungen */
    function highlight(n) {
.whisky-summary__table {
      var j;
  border-collapse: collapse;
      for (j = 0; j < buttons.length; j++) {
  margin-top: 0.5em;
        var active = (j < n);
  width: auto;
        buttons[j].classList.toggle('is-active', active);
}
        buttons[j].setAttribute('aria-pressed', active ? 'true' : 'false');
.whisky-summary__table th,
      }
.whisky-summary__table td {
    }
  padding: 0.4em 0.8em;
  text-align: left;
}
.whisky-summary__table th {
  background: #f5f5f5;
  font-weight: 600;
}
.whisky-summary__table tr:nth-child(even) td {
  background: #fafafa;
}
/* Letzte Zeile (Gesamt) hervorheben */
.whisky-summary__table tr:last-child td {
  padding-top: 0.6em;
  padding-bottom: 0.6em;
  border-top: 2px solid #333;
  font-weight: bold;
  background: #fff9e6;
}


/* Whisky Ø-Balken */
    function updateStats() {
.whisky-bar {
      var api = new mw.Api();
  display: flex;
      api.get({
  align-items: center;
        action: 'query',
  gap: .5rem;
        prop: 'pagerating',
  min-width: 160px;
        pageids: pageId,
}
        prcontest: contest || undefined,
.whisky-bar__track {
        format: 'json',
  position: relative;
        errorformat: 'plaintext'
  flex: 1 1 auto;
      }).done(function (data) {
  height: 10px;
        try {
  background: #ececec;
          var pages = get(data, ['query','pages']) || {};
  border-radius: 999px;
          var keys = []; for (var k in pages) if (Object.prototype.hasOwnProperty.call(pages, k)) keys.push(k);
  overflow: hidden;
          var pid = keys.length ? keys[0] : String(pageId);
  box-shadow: inset 0 1px 2px rgba(0,0,0,.06);
          var page = pages[pid] || {};
}
          var pr = page.pagerating;
.whisky-bar__fill {
  height: 100%;
  width: 0%;
  border-radius: 999px;
  background: linear-gradient(90deg, #f0b429, #d97706);
  transition: width .35s ease;
}
.whisky-bar__value {
  font-variant-numeric: tabular-nums;
  min-width: 3ch;
  text-align: right;
}


.whisky-overall-badge {
          if (!meta) return;
  display: inline-block;
  margin-left: .5rem;
  padding: .1rem .45rem;
  border-radius: .5rem;
  background: #111827;
  color: #fff;
  font-weight: 600;
  font-size: .95rem;
  line-height: 1.2;
}


          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) {
  2) Top-5 Widget
            current = pr.userVote;
  ========================================================= */
            highlight(current);
          }


.whisky-top5 {
          if (typeof pr.canVote !== 'undefined' && pr.canVote === 0) {
  background: #fffaf2; /* warm statt weiß */
            // Serverseitig verboten → hier deaktivieren
  border: 1px solid #eee;
            box.classList.add('whisky-rating--disabled');
  border-radius: 12px;
            var gls = widget.querySelectorAll('.whisky-glass');
  padding: 12px;
            for (var i2 = 0; i2 < gls.length; i2++) gls[i2].disabled = true;
  box-shadow: 0 2px 8px rgba(0,0,0,.04);
            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.';
      });
    }


/* Strukturierte Items (wenn das JS diese Klassen nutzt) */
    function vote(value) {
.whisky-top5__item {
      var api = new mw.Api();
  display: grid;
      if (meta) meta.textContent = 'Wird gespeichert …';
  grid-template-columns: auto 1fr auto;
  gap: 8px 12px;
  align-items: center;
  padding: 8px 4px;
  border-bottom: 1px dashed #eee;
}
.whisky-top5__item:last-child { border-bottom: 0; }


.whisky-top5__rank {
      var saved = false;
  width: 28px;
      var failTimer = setTimeout(function () {
  height: 28px;
        if (!saved && meta) meta.textContent = 'Speichern dauert ungewöhnlich lange … bitte Seite neu laden.';
  border-radius: 999px;
      }, 8000);
  background: #fff1c7;
  color: #7a4b00;
  font-weight: 700;
  display: flex;
  align-items: center;
  justify-content: center;
}
.whisky-top5__name a {
  text-decoration: none;
  color: #111;
}
.whisky-top5__name a:hover { text-decoration: underline; }


/* kleiner Ø-Balken */
      api.postWithToken('csrf', {
.whisky-mini {
        action: 'ratepage',
  display: flex;
        pageid: pageId,
  align-items: center;
        answer: value,
  gap: .5rem;
        contest: contest || undefined,
}
        format: 'json'
.whisky-mini__track {
      }).done(function () {
  flex: 1 1 140px;
        saved = true;
  height: 8px;
        clearTimeout(failTimer);
  background: #ececec;
        current = value;
  border-radius: 999px;
        highlight(current);
  overflow: hidden;
        if (meta) meta.textContent = 'Danke! Deine Bewertung: ' + value + ' / ' + scale;
  box-shadow: inset 0 1px 2px rgba(0,0,0,.06);
        updateStats();
}
      }).fail(function (xhr) {
.whisky-mini__fill {
        clearTimeout(failTimer);
  height: 100%;
        var msg = 'Unbekannter Fehler';
  width: 0%;
        try {
  background: linear-gradient(90deg, #f0b429, #d97706);
          var j = xhr && xhr.responseJSON ? xhr.responseJSON : xhr;
  transition: width .35s ease;
          if (j && j.error) {
}
            msg = (j.error.code ? j.error.code + ': ' : '') + (j.error.info || '');
.whisky-mini__val {
          }
  min-width: 3.2ch;
        } catch(e){}
  text-align: right;
        if (window.console && console.error) console.error('RatePage-API-Fehler:', xhr);
  font-variant-numeric: tabular-nums;
        if (meta) meta.textContent = 'Speichern fehlgeschlagen: ' + msg;
  color: #222;
      });
}
    }
.whisky-top5__votes {
  color: #666;
  font-size: .9em;
}


/* Generische Fallback-Styles, falls nur <li> erzeugt wird */
    updateStats();
.whisky-top5 .item,
   }
.whisky-top5 li {
  display: flex;
  justify-content: space-between;
  align-items: baseline;
  gap: .6rem;
  padding: .35rem 0;
  border-bottom: 1px dashed #eaeaea;
}
.whisky-top5 .item:last-child,
.whisky-top5 li:last-child {
   border-bottom: 0;
}


  // ---------- 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({
  3) Layout-Helper, Teaser, Sterne-Rating
        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]);
  }


.grid { display: grid; gap: 1rem; }
  // ---------- Summary inkl. Gesamt + Balken ----------
.grid-2 { grid-template-columns: 1fr; }
  function renderSummary(container) {
@media (min-width: 900px) {
    if (container.getAttribute('data-summary-init') === '1') return;
  .grid-2 { grid-template-columns: 1.2fr 0.8fr; }
    container.setAttribute('data-summary-init', '1');
}
.col { display: flex; flex-direction: column; gap: 1rem; }


/* News-Listen & kompakte Listen */
    var pageId   = mw.config.get('wgArticleId');
.list.compact .news-item {
    var raw      = container.dataset.ratepageContests || 'NASE,GESCHMACK,ABGANG';
  display: flex;
    var parts    = raw.split(',');
   gap: .6rem;
    var i;
  padding: .25rem 0;
  border-bottom: 1px dashed #eee;
}
.list.compact .news-item:last-child { border-bottom: 0; }
.news-item .date {
  font-variant-numeric: tabular-nums;
  color:#666;
  min-width: 7ch;
}


/* Pillen / Kategoriechips (Basis – Farbe später bei Links) */
    for (i = 0; i < parts.length; i++) parts[i] = parts[i].replace(/^\s+|\s+$/g, '');
.pill-list a {
  display:inline-block;
  margin:.25rem;
  padding:.35rem .7rem;
  border:1px solid #ddd;
  border-radius:999px;
  text-decoration:none;
}
.pill-list a:hover { background:#f8f8f8; }


/* Whisky cards */
    var nameToId = { 'nase':'NASE', 'geschmack':'GESCHMACK', 'abgang':'ABGANG', 'gesamteindruck':'GESAMTEINDRUCK' };
.whisky-card {
    var contests = [];
  display: grid;
    var seen = {};
  grid-template-columns: 140px 1fr;
    for (i = 0; i < parts.length; i++) {
  gap: .9rem;
      var key = parts[i]; if (!key) continue;
  align-items: start;
      var norm = key.toLowerCase();
}
      var id = nameToId[norm] ? nameToId[norm] : key;
.whisky-card .whisky-media img { border-radius: 10px; }
      if (!seen[id]) { contests.push(id); seen[id] = true; }
.whisky-card h3 { margin:.2rem 0; }
    }
.whisky-card .meta { color:#666; font-size:.95rem; }
.whisky-card .note { margin:.4rem 0 0; }


.whisky-mini { display: block; padding:.35rem 0; }
    var labels = { NASE: 'Nase', GESCHMACK: 'Geschmack', ABGANG: 'Abgang', GESAMTEINDRUCK: 'Gesamteindruck' };
.whisky-mini .dim { color:#666; }
    container.textContent = 'Lade Bewertungen …';


/* Teaser grid (3 Spalten auf Desktop) */
    function fetchContest(contest) {
.teaser-grid { display:grid; grid-template-columns:1fr; gap:.25rem; }
      return new mw.Api().get({
@media (min-width: 700px){
        action: 'query',
  .teaser-grid { grid-template-columns:1fr 1fr 1fr; }
        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;


/* Recent changes block */
        if (!pr || (typeof pr.canSee !== 'undefined' && pr.canSee === 0)) {
.recent-changes {
          return { contest: contest, label: (labels[contest] || contest), avg: null, total: 0 };
  max-height: 360px;
        }
  overflow:auto;
        var hist = pr.pageRating || {};
  border:1px solid #eee;
        var total = 0, sum = 0;
  border-radius:10px;
        for (var key in hist) {
  padding:.5rem;
          if (Object.prototype.hasOwnProperty.call(hist, key)) {
  background:#fafafa;
            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 };
      });
    }


/* Simple star rating (rendered via JS) */
    var promises = [];
.rating {
    for (i = 0; i < contests.length; i++) promises.push(fetchContest(contests[i]));
  --stars: 0;
  position: relative;
  display:inline-block;
  font-size: 1.1rem;
  line-height: 1;
}
.rating::before {
  content: "★★★★★";
  letter-spacing: .1rem;
  color:#ddd;
}
.rating::after  {
  content: "★★★★★";
  letter-spacing: .1rem;
  color:#f5b50a;
  position:absolute;
  left:0;
  top:0;
  width: calc(var(--stars) * 20%);
  overflow:hidden;
}


/* Typo tweaks */
    Promise.all(promises).then(function (rows) {
h2, h3 { scroll-margin-top: 80px; }
      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');
  4) ADOS: Mobile-first Layout & Basis-Farben
  ========================================================= */


/* Typografie & Basis */
      // Zeilen mit Balken
:root {
      var r;
  --bg: #faf8f3;
      for (r = 0; r < rows.length; r++) {
  --card: #fffaf2;
        var row = rows[r];
  --muted: #f6f0e6;
        var totalText = row.total ? String(row.total) : '0';
  --border: #e6e6e6;
  --text: #222;
  --text-sub: #666;
  --link: #2d6ca2;
  --link-hover: #1c4d7d;
  --radius: 14px;
  --shadow: 0 1px 3px rgba(0,0,0,.06);
}


body {
        var tr = document.createElement('tr');
  background: var(--bg);
  color: var(--text);
  line-height: 1.6;
}
a {
  color: var(--link);
  text-decoration: none;
}
a:hover {
  color: var(--link-hover);
  text-decoration: underline;
}
img {
  max-width: 100%;
  height: auto;
  border-radius: 8px;
}


/* Grid (1 Spalte mobil, 2 Spalten ab 900px) */
        var tdLabel = document.createElement('td');
.ados-grid {
        tdLabel.textContent = row.label;
  display: grid;
        tr.appendChild(tdLabel);
  gap: 12px;
}
.col {
  display: flex;
  flex-direction: column;
  gap: 10px;
}
@media (min-width: 900px) {
  .ados-grid {
    grid-template-columns: 1.15fr .85fr;
    align-items: start;
  }
}


/* Karten (globale Verwendung) */
        var tdAvg = document.createElement('td');
.card {
        if (row.avg !== null) {
  background: var(--card);
          var wrap  = document.createElement('div');  wrap.className = 'whisky-bar';
  border: 1px solid var(--border);
          var track = document.createElement('div');  track.className = 'whisky-bar__track';
  border-radius: var(--radius);
          var fill  = document.createElement('div');  fill.className  = 'whisky-bar__fill';
   box-shadow: var(--shadow);
          fill.style.width = Math.max(0, Math.min(100, (row.avg/10)*100)) + '%';
   padding: 12px 14px;
          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));
.card h2 {
          track.appendChild(fill); wrap.appendChild(track); wrap.appendChild(val); tdAvg.appendChild(wrap);
  margin: 0 0 .5rem;
        } else {
  padding-bottom: .25rem;
          tdAvg.textContent = '–';
  border-bottom: 1px solid #eee;
        }
  font-size: 1.12rem;
        tr.appendChild(tdAvg);
}
.card-muted { background: var(--muted); }


/* Meta/Hinweis */
        var tdCnt = document.createElement('td');
.meta,
        tdCnt.textContent = totalText;
.hint {
        tr.appendChild(tdCnt);
  font-size: 90%;
  color: var(--text-sub);
  margin-top: .35rem;
}


/* Navigation Einzeiler */
        tbody.appendChild(tr);
.navline { margin: 0; }
      }


/* Footer */
      // Gesamt
.ados-footerdate {
      var present = 0, sumAvg = 0, totalVotes = 0;
  font-size: 80%;
      for (r = 0; r < rows.length; r++) {
  color: #777;
        if (rows[r].avg !== null) { present++; sumAvg += rows[r].avg; }
  margin-top: .6rem;
        if (rows[r].total) totalVotes += rows[r].total;
}
      }
.ados-footerinfo {
      var overall = (present > 0) ? Math.round((sumAvg / present) * 10) / 10 : null;
  border-top: 1px solid var(--border);
      var overallText = (overall !== null)
  margin-top: .8rem;
        ? (overall.toFixed ? overall.toFixed(1) : (Math.round(overall*10)/10))
  padding-top: .5rem;
        : '–';
  font-size: 92%;
  color: var(--text-sub);
}
.ados-footernote {
  text-align: center;
  font-size: 90%;
  margin-top: .8rem;
  color: var(--text-sub);
}


/* Touch-friendly auf sehr kleinen Displays */
      var trG = document.createElement('tr');
@media (max-width: 480px) {
  .card { padding: 12px; }
  .card h2 { font-size: 1.05rem; }
}


/* Leichtes Hover-Feedback (nur Desktop) */
      var tdGL = document.createElement('td');
@media (pointer: fine) {
      tdGL.innerHTML = '<strong>Gesamt</strong>';
  .card:hover {
      trG.appendChild(tdGL);
    box-shadow: 0 6px 18px rgba(0,0,0,.08);
    transition: box-shadow .2s;
  }
}


/* Bevorzugt dunkles Farbschema */
      var tdGA = document.createElement('td');
@media (prefers-color-scheme: dark) {
      if (overall !== null) {
  :root {
        var w  = document.createElement('div'); w.className = 'whisky-bar';
    --bg: #121212;
        var t  = document.createElement('div'); t.className = 'whisky-bar__track';
    --card: #1c1c1c;
        var f  = document.createElement('div'); f.className = 'whisky-bar__fill';
    --muted: #181818;
        f.style.width = Math.max(0, Math.min(100, (overall/10)*100)) + '%';
    --border: #313131;
        var v  = document.createElement('span'); v.className = 'whisky-bar__value';
    --text: #e8e8e8;
        v.innerHTML = '<strong>' + overallText + '</strong>';
    --text-sub: #b4b4b4;
        t.appendChild(f); w.appendChild(t); w.appendChild(v); tdGA.appendChild(w);
    --link: #86c2ff;
      } else {
    --link-hover: #c5e1ff;
        tdGA.innerHTML = '<strong>–</strong>';
    --shadow: 0 1px 4px rgba(0,0,0,.6);
      }
  }
      trG.appendChild(tdGA);
  img { filter: brightness(.95) contrast(1.05); }
}


/* Respektiere reduzierte Animationen */
      var tdGD = document.createElement('td');
@media (prefers-reduced-motion: reduce) {
      tdGD.textContent = totalVotes;
  * { animation: none !important; transition: none !important; }
      trG.appendChild(tdGD);
}


      tbody.appendChild(trG);


      table.appendChild(tbody);


/* =========================================================
      while (container.firstChild) container.removeChild(container.firstChild);
  5) Hard Fix: Force Light Mode on iOS/Safari
      container.appendChild(table);
  ========================================================= */


html { color-scheme: light !important; }
      var badge = document.getElementById('whisky-overall-badge');
@media (prefers-color-scheme: dark) {
      if (badge && overall !== null) {
  html, body {
        badge.textContent = overallText;
    background: #ffffff !important;
      }
     color: #222 !important;
    }).catch(function(){
      container.textContent = 'Konnte Bewertungen nicht geladen werden.';
     });
   }
   }
  .card,
 
  .card-muted,
});
  .whisky-rating__item,
  .whisky-top5,
  .mw-popup-modal,
  .whisky-summary__table th,
  .whisky-summary__table td {
    background: #ffffff !important;
    color: #222 !important;
    border-color: #e6e6e6 !important;
    box-shadow: 0 1px 3px rgba(0,0,0,.04) !important;
  }
  .whisky-summary__table tr:nth-child(even) td { background: #fafafa !important; }
  .list.compact .news-item { border-bottom-color: #eee !important; }
  a { color: #2d6ca2 !important; }
  a:hover { color: #1c4d7d !important; }
  img { filter: none !important; }
}






/* =========================================================
  6) Modal / News-Popup
  ========================================================= */


.mwnews-overlay {
  position:fixed;
  inset:0;
  background:rgba(0,0,0,.45);
  z-index:10000;
}
.mwnews-modal {
  position:fixed;
  top:50%;
  left:50%;
  transform:translate(-50%,-50%);
  background:#fff;
  color:#111;
  max-width:96%;
  width:760px;
  padding:22px 26px 26px;
  border-radius:14px;
  z-index:10001;
  text-align:center;
  box-shadow:0 10px 28px rgba(0,0,0,.35);
  max-height:94vh;
  overflow-y:auto;
}
.mwnews-modal h2 {
  margin:8px 0 6px;
  font-size:1.35em;
}
.mwnews-intro {
  margin:.5em 0 1em;
  font-size:1.06em;
  line-height:1.5;
}


/* Karten im Popup */
.mwnews-cards {
  display:flex;
  gap:12px;
  flex-wrap:wrap;
  justify-content:center;
}
.mwnews-card {
  display:block;
  width:300px;
  text-decoration:none;
  color:inherit;
  background:#f8f9fa;
  border:1px solid #e3e6e8;
  border-radius:10px;
  overflow:hidden;
  transition:transform .15s ease, box-shadow .15s ease, border-color .15s ease;
  box-shadow:0 3px 8px rgba(0,0,0,.12);
}
.mwnews-card:hover {
  transform:translateY(-2px);
  box-shadow:0 8px 18px rgba(0,0,0,.18);
  border-color:#d5dadd;
}
.mwnews-thumb img {
  display:block;
  width:100%;
  height:210px;
  object-fit:cover;
}
.mwnews-meta { padding:10px 12px; }
.mwnews-title {
  font-weight:700;
  font-size:1.02em;
  margin-bottom:6px;
}
.mwnews-cta {
  color:#36c;
  font-weight:600;
}


/* Buttons */
/* --- Whisky Top-5 (multi-root, recursive, robust, namespaces, ES5) ------- */
.mwnews-btnrow {
mw.loader.using(['mediawiki.api']).then(function () {
  display:flex;
  justify-content:center;
  gap:10px;
  margin-top:12px;
}
.mwnews-close {
  padding:10px 16px;
  border:0;
  background:#36c;
  color:#fff;
  border-radius:6px;
  cursor:pointer;
  font-size:1em;
  font-weight:600;
}
.mwnews-close:hover { background:#258; }


/* Mobil */
  /* ========== Utils ========== */
@media (max-width:640px){
  function get(obj, path) {
  .mwnews-modal{
    var cur = obj, i;
    width:calc(100% - 20px);
    for (i = 0; i < path.length; i++) { if (!cur || typeof cur !== 'object') return; cur = cur[path[i]]; }
     padding:16px;
     return cur;
   }
   }
   .mwnews-card{ width:100%; }
   function parseWeights(raw, contests) {
  .mwnews-thumb img{
    var map = {}, parts = (raw || '').split(','), i;
     height:240px;
    for (i = 0; i < parts.length; i++) {
     object-fit:contain;
      var kv = parts[i].split(':'); if (kv.length !== 2) continue;
     background:#000;
      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) ========== */
  7) Links, Navigationslisten, Chips
// NIMMT nsStr (z. B. "*", "0|102|14"). Bei "*" wird cmnamespace NICHT gesetzt.
  ========================================================= */
function fetchCategoryMembersRecursiveSingleResolved(api, catTitle, limit, outSet, pages, nsStr){
  var visited = {}, queue = [catTitle];
  var cmNS = (nsStr && nsStr.trim()) ? nsStr.trim() : '0|14';  // default: Artikel+Kategorie


/* Grundlegende Linksichtbarkeit */
  function fetchOne(title, cont){
.mw-parser-output a {
    var params = {
  color: #2d6ca2;
      action: 'query',
  text-decoration: underline;
      list: 'categorymembers',
  text-underline-offset: 2px;
      cmtitle: title,
}
      // cmnamespace nur setzen, wenn NICHT "*"
.mw-parser-output a:hover,
      cmtype: 'page|subcat',
.mw-parser-output a:focus {
      cmlimit: Math.min(200, limit),
  color: #1c4d7d;
      format: 'json'
  text-decoration-thickness: 2px;
    };
}
    if (cmNS !== '*' && cmNS !== '') {
.mw-parser-output a:visited { color: #5a3fa8; }
      params.cmnamespace = cmNS; // z.B. "0|102|14"
    }
    if (cont) params.cmcontinue = cont;


/* Tastaturfokus */
    return api.get(params).then(function(d){
.mw-parser-output a:focus-visible {
      var cms = (d.query && d.query.categorymembers) || [], i, it;
  outline: 3px solid #99c2ff;
      for (i = 0; i < cms.length; i++) {
  outline-offset: 2px;
        it = cms[i];
  border-radius: 4px;
        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);
    });
  }


/* Tap-Feedback auf Touch-Geräten */
  function loop(){
@media (hover:none) and (pointer:coarse) {
    if (pages.length >= limit || !queue.length) return Promise.resolve();
  .mw-parser-output a:active {
    var next = queue.shift();
     background: rgba(45,108,162,.08);
     if (visited[next]) return loop();
     border-radius: 6px;
     visited[next] = 1;
    return fetchOne(next).then(loop);
   }
   }
}


/* Link-Listen / Navigationslisten */
   return loop();
.list-links {
   list-style: none;
  padding-left: 0;
  margin: .4rem 0;
}
.list-links li {
  display: flex;
  align-items: center;
  gap: .5rem;
  padding: .45rem .6rem;
  border: 1px solid #e6e6e6;
  border-radius: 10px;
  background: #fffaf2;
  margin: .35rem 0;
  transition: transform .06s ease, box-shadow .12s ease, background .12s ease;
}
.list-links li::before {
  content: "›";
  font-weight: 700;
  color: #888;
}
.list-links li:hover {
  background:#f8fbff;
  transform: translateY(-1px);
  box-shadow: 0 2px 8px rgba(0,0,0,.06);
}
.list-links li a {
  text-decoration: none;
}
.list-links li a:hover {
  text-decoration: underline;
}
}


/* Chips / Kategorie-Pillen mit warmem Hintergrund */
   function fetchCategoryMembersRecursiveMulti(rootCats, limit, status, nsStr){
.pill-row a,
    var api = new mw.Api();
.pill-list a {
    var pages = [], outSet = {};
   display: inline-block;
    var idx = 0;
  border: 1px solid #e0e0e0;
  padding: .35rem .7rem;
  border-radius: 999px;
  background: #fffaf2;
  text-decoration: none;
  transition: background .12s ease, box-shadow .12s ease, transform .06s ease;
}
.pill-row a:hover,
.pill-list a:hover {
  background:#f2f7ff;
  box-shadow: 0 1px 6px rgba(0,0,0,.06);
  transform: translateY(-1px);
}
.pill-row a:focus-visible,
.pill-list a:focus-visible {
  outline: 3px solid #99c2ff;
  outline-offset: 2px;
}


/* Ganze Karten klickbar machen */
    function next(){
.card .card-link {
      if (idx >= rootCats.length || pages.length >= limit) return Promise.resolve(pages);
  position: static;
      var raw = rootCats[idx++]; if (!raw) return next();
  font-weight: 600;
}
.card.card-clickable { cursor: pointer; }
.card.card-clickable:hover {
  box-shadow: 0 6px 18px rgba(0,0,0,.08);
}
.card .card-overlay {
  position:absolute;
  inset:0;
  z-index:1;
  text-indent:-9999px;
  background: transparent;
}


/* Hinweise-Icons */
      return resolveCategoryTitle(api, raw).then(function(resolved){
a.external::after { content:" "; font-size:.9em; }
        if (!resolved) { status('Kategorie nicht gefunden: "' + raw + '"', true); return next(); }
a.category-link::after { content:" ⟶"; }
        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();
        });
      });
    }


/* Dark-Mode Anpassungen für Link-Kacheln */
    return next();
@media (prefers-color-scheme: dark) {
  .mw-parser-output a { color:#86c2ff; }
  .mw-parser-output a:hover,
  .mw-parser-output a:focus { color:#c5e1ff; }
  .list-links li {
    background:#1c1c1c;
    border-color:#313131;
   }
   }
   .list-links li:hover { background:#20242b; }
 
  .pill-row a,
  /* ========== Ratings laden / auswerten ========== */
  .pill-list a {
   function fetchRatingsForContest(pageIds, contest, includeHidden) {
     background:#1c1c1c;
    var api = new mw.Api(), res = {}, i, chunk = 50, chunks = [];
     border-color:#313131;
    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) {
    // Statusbox parken, falls keep aktiv
    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); });
  8) Cargo / Diagramme
    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);


.cargoNVD3chart svg {
    while (container.firstChild) container.removeChild(container.firstChild);
  min-height: 400px !important;
    if (statusBox) container.appendChild(statusBox);
}
.nvd3 .nv-line path {
  stroke-width: 3px !important;
  stroke: #f0b429 !important;
}
.nvd3 text {
  fill: #111 !important;
}


/* Diagramm flexibel */
    if (!rows.length) { container.appendChild(document.createTextNode('Noch keine Bewertungen vorhanden.')); return; }
.ados-chart-multi { width: 100%; max-width: 100%; }
@media (max-width: 480px) {
  .ados-chart-multi { min-height: 280px; }
}


.chart-total-info {
    var frag = document.createDocumentFragment(), i, r, item, rank, name, a, right, mini, track, fill, val, votes;
  text-align: center;
    for (i=0;i<rows.length;i++){
  font-weight: bold;
      r = rows[i];
  font-size: 1.05em;
      item = document.createElement('div'); item.className = 'whisky-top5__item';
  color: #444;
  margin-top: 0.4em;
}
.ados-chart-multi { margin-bottom: .75rem; }


      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';
  9) Diagramm-Popup (Canvas)
  ========================================================= */


.mw-popup-overlay {
      mini = document.createElement('div'); mini.className = 'whisky-mini';
  position: fixed;
      track = document.createElement('div'); track.className = 'whisky-mini__track';
  inset: 0;
      fill = document.createElement('div'); fill.className = 'whisky-mini__fill';
  background: rgba(0,0,0,0.45);
      fill.style.width = Math.max(0, Math.min(100, (r.overall/10)*100)) + '%';
  z-index: 10000;
      val = document.createElement('span'); val.className = 'whisky-mini__val';
}
      val.textContent = (r.overall.toFixed ? r.overall.toFixed(1) : (Math.round(r.overall*10)/10));
.mw-popup-modal {
  position: fixed;
  top: 50%;
  left: 50%;
  transform: translate(-50%, -50%);
  background: #fff;
  max-width: 96%;
  width: 720px;
  padding: 22px 26px 26px;
  border-radius: 14px;
  z-index: 10001;
  text-align: center;
  box-shadow: 0 10px 28px rgba(0,0,0,0.35);
  max-height: 94vh;
  overflow-y: auto;
}


/* Canvas-Bühne */
      track.appendChild(fill); mini.appendChild(track); mini.appendChild(val);
.mw-fw-canvas-wrap {
  position: relative;
  height: 240px;
  margin: -6px -6px 12px;
  border-radius: 10px;
  overflow: hidden;
  background: radial-gradient(ellipse at center, #0b1c38 0%, #061025 60%, #03060d 100%);
  box-shadow: inset 0 0 30px rgba(0,0,0,0.65);
}
.mw-fw-canvas { display:block; width:100%; height:100%; }


/* Text */
      votes = document.createElement('div'); votes.className = 'whisky-top5__votes';
.mw-popup-modal h2 {
      votes.textContent = r.totalVotes + ' Stimmen';
  margin: 8px 0 6px;
  font-size: 1.38em;
}
.mw-popup-content p {
  margin: 0.55em 0;
  font-size: 1.06em;
  line-height: 1.55;
}


/* Buttons */
      right.appendChild(mini); right.appendChild(votes);
.mw-popup-button-row {
  display: flex;
  justify-content: center;
  gap: 10px;
  margin-top: 12px;
  flex-wrap: wrap;
}
.mw-popup-wiki-button {
  padding: 10px 16px;
  background: #f60;
  color: #fff;
  border-radius: 6px;
  text-decoration: none;
  font-weight: 700;
}
.mw-popup-wiki-button:hover { background: #d85000; }
.mw-popup-close {
  padding: 10px 16px;
  border: none;
  background: #36c;
  color: #fff;
  border-radius: 6px;
  cursor: pointer;
  font-weight: 700;
}
.mw-popup-close:hover { background: #258; }


/* Mobil */
      item.appendChild(rank); item.appendChild(name); item.appendChild(right);
@media (max-width: 640px) {
      frag.appendChild(item);
  .mw-popup-modal {
     }
    width: calc(100% - 20px);
    container.appendChild(frag);
     padding: 16px;
   }
   }
  .mw-fw-canvas-wrap {
    height: 220px;
    margin: -6px -6px 10px;
  }
}


  /* ========== 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');
      // Kategorien (Zeilenumbruch ODER Semikolon getrennt)
      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'; // z. B. "*", "0|102|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 …');
      // 1) Artikel einsammeln
      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(){
  10) Minerva: Abschnitte offen halten
          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);
        }


.skin-minerva .collapsible-block {
        loopContest(0).then(buildAndRender).catch(function(){ status('Topliste konnte nicht geladen werden.', true); });
  display: block !important;
      }).catch(function(){ status('Topliste konnte nicht geladen werden.'); });
}
.skin-minerva button.section-toggle {
  display: none !important;
}


    })(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]); });


/* =========================================================
});
  11) LabelScan / Scan-UI
  ========================================================= */


/* Grund-Layout */
.ados-scan {
  border:1px solid #e6e6e6;
  border-radius:14px;
  padding:12px;
  background:#fffaf2;
  box-shadow:0 1px 3px rgba(0,0,0,.04);
}
.ados-scan h2 { margin:.2rem 0 .4rem; }


.ados-scan__controls {
  display:flex;
  gap:8px;
  flex-wrap:wrap;
  align-items:center;
  margin:.4rem 0;
}
.ados-scan__preview img {
  max-width: 100%;
  border-radius: 8px;
}
.ados-scan__status {
  color:#666;
  font-size:90%;
  margin:.4rem 0;
}
.ados-scan__results {
  display:grid;
  grid-template-columns: 1fr;
  gap:8px;
}
@media (min-width:700px){
  .ados-scan__results { grid-template-columns: 1fr 1fr; }
}
.ados-hit {
  border:1px solid #eee;
  border-radius:10px;
  padding:10px;
  background:#fafafa;
}
.ados-hit b {
  display:block;
  margin-bottom:4px;
}


/* Uploader / Dropzone */
.ados-scan__uploader {
  display:flex;
  justify-content:center;
}
.ados-scan__btn {
  display:flex;
  flex-direction:column;
  align-items:center;
  gap:.35rem;
  padding:18px 20px;
  border:2px dashed #cfcfcf;
  border-radius:16px;
  background:#fafafa;
  cursor:pointer;
  width:100%;
  max-width:520px;
  text-align:center;
  transition:background .15s, border-color .15s, transform .05s;
}
.ados-scan__btn:hover {
  background:#f6f6f6;
  border-color:#bbb;
}
.ados-scan__btn:active { transform:scale(.99); }
.ados-scan__btn svg {
  width:44px;
  height:44px;
  fill:#ff7a00;
  opacity:.9;
}
.ados-scan__btn span { font-weight:700; }
.ados-scan__btn small { color:#666; }
#ados-scan-drop.dragover .ados-scan__btn {
  border-color:#ff7a00;
  background:#fff8f1;
}


/* Fortschritt */
/* Render star ratings from data-rating on .rating elements (0..5, step .5) */
.ados-scan__progress {
mw.hook('wikipage.content').add(function($content){
   margin:.6rem 0 .3rem;
   $content.find('.rating').each(function(){
  display:flex;
    var el = this, val = parseFloat(el.getAttribute('data-rating') || '0');
  align-items:center;
    if (isNaN(val)) val = 0;
  gap:10px;
    // clamp 0..5
  flex-wrap:wrap;
    val = Math.max(0, Math.min(5, val));
}
    // set CSS variable for width percentage (0..5 -> 0..5 stars)
#ados-scan-progress {
    el.style.setProperty('--stars', (val).toString());
  width:220px;
    el.setAttribute('aria-label', val + ' von 5 Sternen');
  height:10px;
    el.setAttribute('title', val + ' von 5 Sternen');
   accent-color:#ff7a00;
   });
}
});


/* Ergebnis-Vorschau */
.ados-scan__preview {
  margin:.5rem 0;
}
.ados-scan__preview img {
  max-width:100%;
  border-radius:10px;
  box-shadow:0 2px 10px rgba(0,0,0,.06);
}


/* Buttons */
// --------------------------------
.ados-scan__actions {
  margin:.4rem 0 .6rem;
}
.ados-scan__run {
  padding:.55rem .9rem;
  border:1px solid #ddd;
  border-radius:999px;
  background:#fff;
  cursor:pointer;
  font-weight:600;
}
.ados-scan__run[disabled] {
  opacity:.6;
  cursor:not-allowed;
}


/* Dark-Mode-Anpassungen */
// Force light color-scheme at document level (helps Mobile Safari)
@media (prefers-color-scheme: dark){
mw.loader.using('mediawiki.util').then(function () {
   .ados-scan {
   var m = document.querySelector('meta[name="color-scheme"]');
    background:#1c1c1c;
   if (!m) {
    border-color:#313131;
     m = document.createElement('meta');
   }
     m.name = 'color-scheme';
  .ados-scan__btn {
    m.content = 'light';
     background:#181818;
     document.head.appendChild(m);
     border-color:#343434;
   } else {
  }
    m.content = 'light';
  .ados-scan__btn:hover {
    background:#1b1b1b;
     border-color:#555;
   }
  .ados-scan__status { color:#bbb; }
  .ados-hit {
    background:#202020;
    border-color:#2e2e2e;
   }
   }
}
});


/* LabelScan: Button sicher klickbar */
// -------------------------------------------------
#ados-labelscan { position: relative; }
#ados-scan-run,
#ados-scan-bigbtn {
  position: relative;
  z-index: 9999;
  pointer-events: auto;
}
#ados-labelscan * { pointer-events: auto; }
#ados-labelscan [aria-hidden="true"] { pointer-events: none; }


/* Häufige Störenfriede (Vector sticky header) */
.mw-body-content .sticky-header,
.mw-body-content .vector-sticky-header {
  pointer-events: none;
}


/* LabelScan – Mobile optimiert */
function isFlagTrue(v){
#ados-labelscan {
   if (v == null) return false;
   max-width: 640px;
   v = String(v).trim().toLowerCase();
   margin: 0 auto;
   return v === 'true' || v === '1' || v === 'yes';
  padding: 1rem;
}
#ados-labelscan .ados-scan__drop {
  background: #ffffff;
  border-radius: 12px;
  padding: 1.5rem 1rem;
   text-align: center;
  border: 2px dashed #ccc;
  color: #444;
}
#ados-labelscan button,
#ados-labelscan .mw-ui-button {
  font-size: 1rem !important;
  border-radius: 30px !important;
  padding: 0.7rem 1.4rem !important;
}
#ados-labelscan .ados-scan__form {
  display: flex;
  flex-direction: column;
  gap: 1rem;
}
#ados-labelscan h2 {
  font-size: 1.8rem;
  margin-bottom: 1rem;
}
#ados-scan-preview img {
  border-radius: 14px;
  margin-top: 0.6rem;
}
}


/* Dropzone hell halten (explizit) */
// Gesamtzahl unter der Legende einfügen (im Diagramm-Block, nicht im Canvas!)
#ados-labelscan .ados-scan__drop,
function addTotalBelowLegend(chart, block) {
#ados-scan-drop,
   try {
.ados-scan__drop {
    if (!chart || !block) return;
   background: #ffffff !important;
  color: #444 !important;
  border: 2px dashed #ccc !important;
  box-shadow: none !important;
}
#ados-labelscan .ados-scan__drop img,
#ados-scan-drop img {
  filter: none !important;
}


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


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


/* =========================================================
    // Neue Anzeige einfügen
  12) ADOS Timer-Balken oben
    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);


#ados-timer-bar {
  } catch(e) { console.warn('[addTotalBelowLegend]', e); }
  position: fixed;
  top: 0;
  left: 0;
  right: 0;
  z-index: 2000;
  background: #111827;
  color: #f9fafb;
  font-family: system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;
  font-size: 14px;
  line-height: 1.4;
  padding: 6px 12px;
  display: none;
  box-shadow: 0 2px 8px rgba(0,0,0,.35);
}
#ados-timer-inner {
  max-width: 1200px;
  margin: 0 auto;
  display: flex;
  align-items: center;
  gap: 8px;
}
#ados-timer-message { font-weight: 600; }
#ados-timer-countdown {
  font-weight: 700;
  letter-spacing: 0.05em;
  padding: 2px 8px;
  border-radius: 999px;
  background: #f59e0b;
  color: #111827;
  white-space: nowrap;
}
#ados-timer-close {
  margin-left: auto;
  cursor: pointer;
  border: none;
  background: transparent;
  color: #9ca3af;
  font-size: 16px;
  padding: 0 4px;
}
}
#ados-timer-close:hover { color: #f9fafb; }


/* Inhalt nach unten schieben, wenn Timer sichtbar */
body.has-ados-timer { padding-top: 38px; }


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


/* === 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 () {
  13) ADOS-Hero – whiskyfarben, Fade, keine weißen Boxen
  // 1) Chart.js nur 1x laden und erst dann rendern
  ========================================================= */
  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'; // UMD-Bundle
      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;
  }


/* Äußerer Hero-Block mit Fade nach unten ins Weiße */
  // 2) Farben je Serie (gut unterscheidbar)
.mw-parser-output .ados-hero {
  var ADOS_COLORS = {
  background: linear-gradient(
    'A Dream of Scotland':                 '#C2410C', // Kupferbraun
     180deg,
     'A Dream of Ireland':                  '#15803D', // Flaschengrün
     #e7cfa4 0%,
     'A Dream of... – Der Rest der Welt':  '#1D4ED8', // Mittelblau
     #e1c297 40%,
     'Friendly Mr. Z':                      '#9333EA', // Violett
     #f6efe4 75%,
     'Die Whisky Elfen':                    '#0891B2', // Türkis
     #ffffff 100%
     'The Fine Art of Whisky':              '#CA8A04'  // Goldgelb
   );
   };
   padding: 1.2rem 1.4rem 1.5rem;
   var COLOR_CYCLE = ['#2563eb','#16a34a','#f97316','#dc2626','#a855f7','#0ea5e9','#f59e0b','#10b981'];
  margin: 0 0 1.6rem;
  border-radius: 12px;
  box-shadow: 0 4px 12px rgba(0,0,0,.05);
  border: 1px solid rgba(120,80,40,.15);
}


/* auf Desktop etwas breiter wirken lassen */
  function toYear(x){
@media (min-width: 960px) {
    var n = parseInt(String(x).replace(/[^\d]/g,''),10);
  .mw-parser-output .ados-hero {
    return isFinite(n) ? n : null;
     margin-left: -1.5rem;
  }
     margin-right: -1.5rem;
  function getColor(name, used){
     border-radius: 0 0 12px 12px;
    if (ADOS_COLORS[name]) return ADOS_COLORS[name];
     var i = used.size % COLOR_CYCLE.length;
     used.add(name);
     return COLOR_CYCLE[i];
   }
   }
}


/* Grid: Titel & Text nebeneinander, Spalten unten drunter */
  // 3) Tabelle (Jahr | Serie | Anzahl) -> {labels, datasets}
.ados-hero__grid {
   function buildDatasetsFromTable(tbl){
   max-width: 1200px;
    var rows = Array.from(tbl.querySelectorAll('tr'));
  margin: 0 auto;
    if (rows.length < 2) return { labels:[], datasets:[] };
  display: grid;
 
  grid-template-columns: minmax(0, 0.9fr) minmax(0, 1.3fr);
     var yearsSet = new Set();
  grid-template-areas:
     var bySeries = new Map(); // serie -> Map(year -> count)
     "title center"
     "cols cols";
  gap: 1.1rem 1.6rem;
  align-items: center;
}


/* Linke Seite: Titel */
    rows.slice(1).forEach(function(tr){
.ados-hero__title {
      var tds = tr.querySelectorAll('td,th');
  grid-area: title;
      if (tds.length < 3) return;
  font-family: "Georgia","Times New Roman",serif;
      var y = toYear(tds[0].textContent.trim());
  font-size: 1.7rem;
      var s = tds[1].textContent.trim();
  font-weight: 700;
      var v = parseFloat(tds[2].textContent.replace(',','.')) || 0;
  line-height: 1.2;
      if (y == null) return;
  color: #4b2b14;
      yearsSet.add(y);
}
      if (!bySeries.has(s)) bySeries.set(s, new Map());
.ados-hero__title span { display:block; }
      bySeries.get(s).set(y, v);
.ados-hero__title hr {
    });
  width: 55%;
  margin-top: .55rem;
  border: 0;
  border-top: 1px solid rgba(75,43,20,.26);
}


/* Mitte: Beschreibung, ohne Box */
    var years = Array.from(yearsSet).sort(function(a,b){return a-b;});
.ados-hero__center {
    var used = new Set();
  grid-area: center;
    var labels = years.map(String);
  display: flex;
  justify-content: center;
}
.ados-hero__center-box {
  font-family: "Courier New", ui-monospace, monospace;
  font-size: .96rem;
  line-height: 1.6;
  color: #3a2918;
  text-align: center;
  padding: .15rem .25rem;
}
.ados-hero__center-box strong {
  color: #4b2b14;
  font-weight: 700;
}


/* Unten: Spalten nebeneinander über die gesamte Breite */
    var datasets = Array.from(bySeries.entries()).map(function(entry){
.ados-hero__cols {
      var name = entry[0], yearMap = entry[1];
  grid-area: cols;
      var data = years.map(function(y){ return yearMap.get(y) || 0; });
  display: flex;
      var color = getColor(name, used);
  justify-content: center;
      return {
  gap: 2.2rem;
        label: name,
  padding-top: .4rem;
        data: data,
  border-top: 1px solid rgba(75,43,20,.16);
        borderColor: color,
}
        backgroundColor: color + '80',
        tension: 0.25,
        pointRadius: 3
      };
    });


.ados-hero__col {
    return { labels: labels, datasets: datasets };
  min-width: 140px;
   }
  font-size: .95rem;
}
.ados-hero__col-title {
  font-weight: 700;
  margin-bottom: .2rem;
   color: #4b2b14;
  font-size: .94rem;
}
.ados-hero__col ul {
  margin: 0;
  padding-left: 1.05rem;
}
.ados-hero__col li {
  margin: .08rem 0;
}


/* Sicherheit: im Hero niemals weiße Karten-Hintergründe */
  // 4) Einen Chart-Container rendern: nimmt die NÄCHSTE Tabelle als Datenquelle
.ados-hero .card,
function renderOne(block){
.ados-hero .whisky-top5,
  // schon gerendert? (z. B. durch AJAX/Minerva-Reloads)
.ados-hero .whisky-rating__item {
  if (block.dataset.rendered === '1') return;
  background: transparent !important;
  border: 0 !important;
  box-shadow: none !important;
}


/* ---------------- MOBIL ---------------- */
  // nächste Tabelle (auch wenn sie in einem Wrapper steckt) finden
@media (max-width: 720px) {
  var el = block.nextElementSibling, tbl = null, wrapToHide = null;
  .mw-parser-output .ados-hero {
  while (el) {
     border-radius: 0;
    if (/^H[1-6]$/.test(el.tagName) || (el.classList && el.classList.contains('ados-chart-multi'))) break;
     margin-left: -1rem;
    if (el.tagName === 'TABLE') {
     margin-right: -1rem;
      tbl = el;
     padding: 1.1rem 1rem 1.3rem;
     } else if (el.querySelector) {
      var t = el.querySelector('table');
      if (t) tbl = t;
     }
    if (tbl) {
      // 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;


   .ados-hero__grid {
   var out = buildDatasetsFromTable(tbl);
     display: flex;
  if (!out.labels.length || !out.datasets.length) return;
     flex-direction: column;
 
    gap: .8rem;
  // Tabelle verstecken, wenn gewünscht (nur die Tabelle – oder den Wrapper, falls er sonst leer ist)
     text-align: center;
  var hide = (block.dataset.hideTable || '').toLowerCase() === 'true';
  if (hide) {
     // Hat der Wrapper außer der Tabelle noch sichtbaren Inhalt?
    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';
    }
   }
   }


   .ados-hero__title {
   // Zeichenfläche einsetzen (mobil/desktop)
     font-size: 1.5rem;
  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';
 
  // optional: kumulative Werte pro Serie bauen
  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; })
      });
    });
   }
   }
   .ados-hero__title hr {
 
     margin-left: auto;
ensureChartJS().then(function(){
     margin-right: auto;
   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 } } }
      }
     }
  });
 
  // --- NEU ---
  const hideTotal = (block.dataset.hideTotal || '').toLowerCase() === 'true';
 
  // existierende Anzeige entfernen (falls Seite neu gerendert wird)
  const oldInfo = block.querySelector(':scope > .chart-total-info');
  if (oldInfo) oldInfo.remove();
 
  if (!hideTotal) {
    // Gesamtzahl einfügen
    addTotalBelowLegend(chart, block);
 
     // bei Größenänderung (mobil <-> desktop) neu berechnen
    if (window.ResizeObserver) {
      const obs = new ResizeObserver(() => addTotalBelowLegend(chart, block));
      obs.observe(chart.canvas);
      chart.$adosTotalObserver = obs;
    }
   }
   }


   .ados-hero__center-box {
   block.dataset.rendered = '1';
     max-width: 34rem;
});
     margin: 0 auto;
 
}
 
  // 5) Start: erst wenn DOM fertig, dann Chart.js laden, dann rendern
  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); });
   }
   }


   .ados-hero__cols {
   if (window.mw && mw.hook) {
     flex-direction: row;
     mw.hook('wikipage.content').add(boot);
    flex-wrap: wrap;
  } else {
     justify-content: center;
     // Fallback
     gap: 1.2rem 2rem;
     (document.readyState === 'loading')
    border-top: 1px solid rgba(75,43,20,.18);
      ? document.addEventListener('DOMContentLoaded', function(){ boot(); })
    padding-top: .6rem;
      : 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) ==== */


   .ados-hero__col {
/* Manifest einbinden */
    min-width: 130px;
(function () {
    text-align: center;
   var link = document.createElement("link");
   }
  link.rel = "manifest";
  link.href = "/app/labelscan/manifest.webmanifest";
   document.head.appendChild(link);
})();


   .ados-hero__col ul {
/* Service Worker registrieren (nur wenn vorhanden) */
     padding-left: 0;
(function () {
    list-style-position: inside;
   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";
  });


/* =========================================================
  // Sicherstellen, dass DOM existiert, bevor wir Handler setzen
  14) ADOS-Hauptseite – flexibles 2-Spalten-Layout
  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";
    });
  });
})();


.ados-mainpage {
  max-width: 1200px;                      /* Gesamtbreite der Hauptseitev2 */
  margin: 0 auto 2.5rem;                  /* zentriert, unten Luft */
  display: grid;
  grid-template-columns: minmax(0, 2fr) minmax(0, 1fr);  /* links breiter als rechts */
  gap: 1.6rem 2.2rem;
  align-items: flex-start;
}


.ados-main-left,
if ('serviceWorker' in navigator) {
.ados-main-right {
   navigator.serviceWorker.register('/app/labelscan/sw.js').catch(function(){});
   display: flex;
  flex-direction: column;
  gap: 1.1rem;
}
}


/* Boxen innerhalb der Hauptseite – etwas „luftiger“ */
.ados-mainpage .ados-box {
  background: #ffffff;
  border: 1px solid #e4e4e4;
  border-radius: 12px;
  padding: 12px 14px;
  box-shadow: 0 1px 3px rgba(0,0,0,.04);
}


/* Überschriften in den Boxen */
// ------------------------------------------------------------
.ados-mainpage .ados-box h2,
// ADOS: Countdown zur nächsten Messe (Bottle Market Bremen)
.ados-mainpage .ados-box h3 {
// ------------------------------------------------------------
   margin: 0 0 .5rem;
(function () {
   padding-bottom: .25rem;
   // Zieldatum: 12.12.2025, 9:00 Uhr, Bremen (MEZ = +01:00)
   border-bottom: 1px solid #eeeeee;
   // Wenn Uhrzeit ändern: einfach die 10:00:00 anpassen.
  font-size: 1.08rem;
   var adosTimerTarget = new Date('2025-12-12T10:00:00+01:00');
}


/* Mobil / schmal: alles untereinander */
  // Nur im Haupt-Namespace anzeigen (Abfüllungs-/Artikel-Seiten)
@media (max-width: 900px) {
   if (typeof mw !== 'undefined' && mw.config.get('wgNamespaceNumber') !== 0) {
   .ados-mainpage {
     return;
     display: block;              /* keine Spalten, nur Flow */
    max-width: 100%;
    margin: 0 0 2rem;
   }
   }


   .ados-main-right {
   // Falls Datum kaputt -> nichts tun
     margin-top: 1.5rem;
  if (isNaN(adosTimerTarget.getTime())) {
     return;
   }
   }


   .ados-mainpage .ados-box {
   function domReady(fn) {
    border-radius: 0;
    if (document.readyState === 'loading') {
     border-left: 0;
      document.addEventListener('DOMContentLoaded', fn);
     border-right: 0;
     } else {
      fn();
     }
   }
   }
}
 
  domReady(function () {
    // Timer-Bar einbauen
    var bar = document.createElement('div');
    bar.id = 'ados-timer-bar';
    bar.innerHTML =
      '<div id="ados-timer-inner">' +
        '<span id="ados-timer-message">Nächste Messe: Bottle Market Bremen in&nbsp;</span>' +
        '<span id="ados-timer-countdown">–:–:–:–</span>' +
        '<button id="ados-timer-close" type="button" title="Ausblenden">×</button>' +
      '</div>';
 
    document.body.appendChild(bar);
    document.body.className += ' has-ados-timer';
 
    var countdownEl = document.getElementById('ados-timer-countdown');
    var closeBtn    = document.getElementById('ados-timer-close');
 
    bar.style.display = 'block';
 
    if (closeBtn) {
      closeBtn.onclick = function () {
        bar.style.display = 'none';
        document.body.className = document.body.className.replace(/\bhas-ados-timer\b/g, '');
      };
    }
 
    function pad(num) {
      return num < 10 ? '0' + num : '' + num;
    }
 
    function updateTimer() {
      var now  = new Date();
      var diff  = adosTimerTarget.getTime() - now.getTime();
 
      if (diff <= 0) {
        countdownEl.innerHTML = '🔔 Jetzt auf dem Bottle Market!';
        clearInterval(intervalId);
        return;
      }
 
      var secTotal = Math.floor(diff / 1000);
      var days  = Math.floor(secTotal / 86400);
      var hours = Math.floor((secTotal % 86400) / 3600);
      var mins  = Math.floor((secTotal % 3600) / 60);
      var secs  = secTotal % 60;
 
      var text;
      if (days > 0) {
        text = days + 'T ' + pad(hours) + ':' + pad(mins) + ':' + pad(secs);
      } else {
        text = pad(hours) + ':' + pad(mins) + ':' + pad(secs);
      }
 
      countdownEl.textContent = text;
    }
 
    // Direkt initialisieren & dann jede Sekunde aktualisieren
    updateTimer();
    var intervalId = window.setInterval(updateTimer, 1000);
  });
})();

Version vom 23. November 2025, 19:03 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) ========== */
// NIMMT nsStr (z. B. "*", "0|102|14"). Bei "*" wird cmnamespace NICHT gesetzt.
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,
      // cmnamespace nur setzen, wenn NICHT "*"
      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) {
    // Statusbox parken, falls keep aktiv
    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');

      // Kategorien (Zeilenumbruch ODER Semikolon getrennt)
      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'; // z. B. "*", "0|102|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 …');


      // 1) Artikel einsammeln
      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;
    // clamp 0..5
    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.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;

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

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

    // Neue Anzeige einfügen
    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 () {
  // 1) Chart.js nur 1x laden und erst dann rendern
  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'; // UMD-Bundle
      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;
  }

  // 2) Farben je Serie (gut unterscheidbar)
  var ADOS_COLORS = {
    'A Dream of Scotland':                 '#C2410C', // Kupferbraun
    'A Dream of Ireland':                  '#15803D', // Flaschengrün
    'A Dream of... – Der Rest der Welt':   '#1D4ED8', // Mittelblau
    'Friendly Mr. Z':                      '#9333EA', // Violett
    'Die Whisky Elfen':                    '#0891B2', // Türkis
    'The Fine Art of Whisky':              '#CA8A04'  // Goldgelb
  };
  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];
  }

  // 3) Tabelle (Jahr | Serie | Anzahl) -> {labels, datasets}
  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(); // serie -> Map(year -> count)

    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 };
  }

  // 4) Einen Chart-Container rendern: nimmt die NÄCHSTE Tabelle als Datenquelle
function renderOne(block){
  // 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;
  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) {
      // 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);
  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';
  if (hide) {
    // Hat der Wrapper außer der Tabelle noch sichtbaren Inhalt?
    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';
    }
  }

  // Zeichenfläche einsetzen (mobil/desktop)
  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';

  // optional: kumulative Werte pro Serie bauen
  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 } } }
      }
    }
  });

  // --- NEU ---
  const hideTotal = (block.dataset.hideTotal || '').toLowerCase() === 'true';

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

  if (!hideTotal) {
    // Gesamtzahl einfügen
    addTotalBelowLegend(chart, block);

    // bei Größenänderung (mobil <-> desktop) neu berechnen
    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){
    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 {
    // Fallback
    (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";
  });

  // Sicherstellen, dass DOM existiert, bevor wir Handler setzen
   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(){});
}


// ------------------------------------------------------------
// ADOS: Countdown zur nächsten Messe (Bottle Market Bremen)
// ------------------------------------------------------------
(function () {
  // Zieldatum: 12.12.2025, 9:00 Uhr, Bremen (MEZ = +01:00)
  // Wenn Uhrzeit ändern: einfach die 10:00:00 anpassen.
  var adosTimerTarget = new Date('2025-12-12T10:00:00+01:00');

  // Nur im Haupt-Namespace anzeigen (Abfüllungs-/Artikel-Seiten)
  if (typeof mw !== 'undefined' && mw.config.get('wgNamespaceNumber') !== 0) {
    return;
  }

  // Falls Datum kaputt -> nichts tun
  if (isNaN(adosTimerTarget.getTime())) {
    return;
  }

  function domReady(fn) {
    if (document.readyState === 'loading') {
      document.addEventListener('DOMContentLoaded', fn);
    } else {
      fn();
    }
  }

  domReady(function () {
    // Timer-Bar einbauen
    var bar = document.createElement('div');
    bar.id = 'ados-timer-bar';
    bar.innerHTML =
      '<div id="ados-timer-inner">' +
        '<span id="ados-timer-message">Nächste Messe: Bottle Market Bremen in&nbsp;</span>' +
        '<span id="ados-timer-countdown">–:–:–:–</span>' +
        '<button id="ados-timer-close" type="button" title="Ausblenden">×</button>' +
      '</div>';

    document.body.appendChild(bar);
    document.body.className += ' has-ados-timer';

    var countdownEl = document.getElementById('ados-timer-countdown');
    var closeBtn    = document.getElementById('ados-timer-close');

    bar.style.display = 'block';

    if (closeBtn) {
      closeBtn.onclick = function () {
        bar.style.display = 'none';
        document.body.className = document.body.className.replace(/\bhas-ados-timer\b/g, '');
      };
    }

    function pad(num) {
      return num < 10 ? '0' + num : '' + num;
    }

    function updateTimer() {
      var now   = new Date();
      var diff  = adosTimerTarget.getTime() - now.getTime();

      if (diff <= 0) {
        countdownEl.innerHTML = '🔔 Jetzt auf dem Bottle Market!';
        clearInterval(intervalId);
        return;
      }

      var secTotal = Math.floor(diff / 1000);
      var days  = Math.floor(secTotal / 86400);
      var hours = Math.floor((secTotal % 86400) / 3600);
      var mins  = Math.floor((secTotal % 3600) / 60);
      var secs  = secTotal % 60;

      var text;
      if (days > 0) {
        text = days + 'T ' + pad(hours) + ':' + pad(mins) + ':' + pad(secs);
      } else {
        text = pad(hours) + ':' + pad(mins) + ':' + pad(secs);
      }

      countdownEl.textContent = text;
    }

    // Direkt initialisieren & dann jede Sekunde aktualisieren
    updateTimer();
    var intervalId = window.setInterval(updateTimer, 1000);
  });
})();