Modul:Link
Erzeugt interne Links mit Hover-Tooltip (Icon, Titel, Kurzbeschreibung) aus Modul:LinkData. Optional kann eine „Detailansicht“ (Wikitable) ausgegeben werden.
Grundsyntax
- Tooltip-Link
 {{Link|Ziel}}
- Tooltip-Link mit eigenem Anzeigenamen
 {{Link|Ziel|Anzeigename}}
- Detailansicht (Tabelle) statt Link
 {{Link|Ziel|view}}
Parameter
| Param | Name | Typ/Standard | Beschreibung | 
|---|---|---|---|
| 1 | Ziel | Pflicht | Seitentitel oder „Titel#Abschnitt“. Wird als Linkziel verwendet. | 
| 2 | Anzeigename / view / Kurz-Suffix | 
optional | 
  | 
| s | s (Suffix) | 
optional | Hängt Text an den Anzeigenamen aus LinkData an (z. B. s=e → „Biere“).
 | 
| key | key | 
optional | Erzwingt den Datensatz-Schlüssel für LinkData (falls Ziel und title nicht übereinstimmen).
 | 
| limit | limit | 
Zahl (220) | Maximale Länge der Tooltip-Beschreibung; bei Überschreitung wird abgeschnitten (…). | 
Verhalten
Anzeigename
- Standardmäßig wird als Anzeigename der deutsche Titel aus Modul:LinkData verwendet (Feld 
title). - Eigener Anzeigename (2. Parameter) überschreibt den Titel.
 - Groß-/Kleinschreibung bleibt genau wie im Wikitext/LinkData.
 
Plural / LinkTrail
MediaWikis LinkTrail funktioniert nur bei direkt geschriebenen Links, nicht bei Vorlagen. Dafür gibt es zwei Wege:
- Suffix-Kurzform im 2. Parameter:
 
- Expliziter Suffix über 
s=: 
{{Link|Bier|s=e}}→Bier
Hinweise:
- Sobald ein eigener Anzeigename gesetzt ist, wird 
s=ignoriert. - Bei „unpiped“ Links (also wirklich 
Biere) greift LinkTrail – das ist mit Vorlagen technisch nicht möglich. Nutze dahers=/+…. 
Anker/Abschnitte
- Bei Zielen mit 
#(z. B.Währungen#Bier) wird immer ein gepipter Link erzeugt (kein LinkTrail / ToolTip möglich). - Die Detailansicht (
view) setzt die Tabellen-IDanhand des Titels (Leerzeichen → Unterstrich). 
Tooltip
- Holt 
title/icon/descaus Modul:LinkData. - Links in der Beschreibung werden im Hover-Tooltip zu normalem Text neutralisiert.
 - Zeilenumbrüche werden unterstützt; lange Texte werden nach 
limitZeichen abgeschnitten. 
Fallbacks
- Gibt es keinen Datensatz in LinkData, wird ein normaler Link ohne Tooltip ausgegeben.
 
Beispiele
- Standard
 {{Link|Bier}}→ Bier
- Eigener Anzeigename
 {{Link|Währungen#Bier|Bier}}→ Bier
- Plural via Suffix (Kurzform)
 {{Link|Bier|+e}}→ +e
- Plural via 
s= {{Link|Bier|s=en}}→ Bier
- Detailansicht (Wikitable)
 {{Link|Bier|view}}→
- Limit anpassen
 {{Link|Bier|limit=50}}→ Bier
- Abweichender Datensatz (engl. key)
 {{Link|Währungen|key=Beer}}→ WährungenBierBier schaltet sich auf Charakterlevel 15 frei. Biere droppen mit einer Rate von 1 Bier pro 28,8 Sekunden auf dem Schlachtfeld (entspricht 3000 Bier pro Tag). Zusätzlich ist Bier als mögliche Belohnung aus T…
-- Modul:ToolTip
local p = {}
local data = mw.loadData('Modul:LinkData')
local US = mw.ustring
local DEFAULT_LIMIT = 220
local function trim(s) return type(s)=='string' and (s:gsub('^%s+',''):gsub('%s+$','')) or '' end
local function normalize_key(s)
  s = US.lower(trim(s or '')):gsub('_',' '):gsub('%s+',' ')
  return s
end
local function strip_links(txt)
  txt = txt or ''
  txt = txt:gsub('%[%[([^%]|%]]-)|([^\]]-)%]%]', '%2')
  txt = txt:gsub('%[%[([^%]]-)%]%]', function(inner) return inner:gsub('_',' ') end)
  txt = txt:gsub('%[(https?://[^%s%]]+)%s+([^\]]-)%]', '%2')
  txt = txt:gsub('%[(https?://[^%]]+)%]', '%1')
  return txt
end
local function shorten(txt, limit)
  limit = tonumber(limit) or DEFAULT_LIMIT
  local len = US.len(txt or '')
  if len <= limit then return txt end
  local cut = US.sub(txt, 1, limit)
  local lastSpace = cut:match('^.*()%s')
  if lastSpace and lastSpace > 15 then cut = US.sub(cut, 1, lastSpace - 1) end
  return cut .. '…'
end
local function htmlize_plain(txt)
  txt = mw.text.encode(tostring(txt or ''))
  return txt:gsub('\n','<br/>')
end
local function make_link(target, display)
  target  = trim(target)
  display = trim(display or '')
  return display ~= '' and ('[['..target..'|'..display..']]') or ('[['..target..']]')
end
local function render_hover(link_wt, info, limit)
  if not info or (not info.title and not info.desc and not info.icon) then
    return link_wt
  end
  local wrap   = mw.html.create('span'):addClass('fs-tt')
  wrap:wikitext(link_wt)
  local content = mw.html.create('span'):addClass('tt-content')
  local box     = mw.html.create('span'):addClass('tt-box')
  -- Icon (ohne Link)
  if info.icon and info.icon ~= '' then
    box:wikitext('[[File:' .. info.icon .. '|48x48px|link=none|class=tt-icon]]')
  else
    box:wikitext('<span class="tt-icon tt-icon--empty"></span>')
  end
  -- Titel + Beschreibung (gekürzt & link-bereinigt) als _fertigen_ HTML-String
  local title = info.title or ''
  local desc  = info.desc  or ''
  desc = shorten(strip_links(desc), limit)
  local function esc(s)
    s = tostring(s or '')
    s = mw.text.encode(s)       -- HTML-sicher
    s = s:gsub('\n','<br/>')
    return s
  end
  local text_html = table.concat({
    '<span class="tt-text">',
      '<span class="tt-title">', esc(title), '</span>',
      '<span class="tt-desc">',  esc(desc),  '</span>',
    '</span>'
  })
  box:wikitext(text_html)
  content:node(box)
  wrap:node(content)
  return tostring(wrap)
end
-- Helper: Anchor-ID nur aus dem Title, Spaces -> "_"
local function anchor_id_from_title(title)
  local t = mw.text.trim(title or '')
  if t == '' then return nil end
  -- Alle Whitespaces zu Unterstrichen; Mehrfach-WS -> einzelner "_"
  t = mw.ustring.gsub(t, '%s+', '_')
  return t
end
-- VIEW: wikitable + <span id="Title_mit_Unterstrichen">
local function render_view(info)
  local title = (info and info.title) or ''
  local desc  = (info and info.desc)  or ''
  local icon  = (info and info.icon)  or ''
  local out = {}
  -- Anchor voranstellen (ID aus Title, z. B. "Voild Crystal" -> "Voild_Crystal")
  local anchor_id = anchor_id_from_title(title)
  if anchor_id then
    local anchor = mw.html.create('span'):attr('id', anchor_id)
    table.insert(out, tostring(anchor))
  end
  -- Tabelle: class="wikitable" width="100%"
  local tbl = mw.html.create('table')
    :addClass('wikitable')
    :attr('width', '100%')
  
  -- Header-Zeile, colspan=2
  local trh = tbl:tag('tr')
  trh:tag('th')
     :attr('colspan', '2')
     :wikitext(title ~= '' and title or ' ')
  -- Inhalt: Icon links (110 breit, zentriert), Text rechts (volle Desc inkl. Links)
  local tr = tbl:tag('tr')
  local tdIcon = tr:tag('td')
    :attr('width', '110')
    :attr('align', 'center')
  if icon ~= '' then
    -- Icon ohne Link
    tdIcon:wikitext('[[File:' .. icon .. '|x64px|link=]]')
  else
    tdIcon:wikitext(' ')
  end
  tr:tag('td'):wikitext(desc or '')
  table.insert(out, tostring(tbl))
  return table.concat(out, '\n')
end
function p.main(frame)
  local parent = frame:getParent()
  local args = parent and parent.args or frame.args
  local target = trim(args[1] or '')
  if target == '' then return '' end
  local p2 = trim(args[2] or '')
  local mode_view = (US.lower(p2) == 'view')
  -- Key: Anzeigename (Param 2) > Ziel (Param 1); überschreibbar via |key=
  local key = trim(args.key or (mode_view and target or (p2 ~= '' and p2 or target)))
  local info = data[normalize_key(key)] or data[normalize_key(key):gsub(' ','_')]
  local limit = tonumber(args.limit or '') or DEFAULT_LIMIT
  if mode_view then
    return render_view(info or {})
  else
    return render_hover(make_link(target, (p2 ~= '' and p2 or nil)), info or {}, limit)
  end
end
return p