Module:GameInfo/Skills: Difference between revisions
From SpiritVale Wiki
More actions
No edit summary Tags: Mobile edit Mobile web edit |
No edit summary Tags: Mobile edit Mobile web edit |
||
| (8 intermediate revisions by the same user not shown) | |||
| Line 1: | Line 1: | ||
-- Module:GameInfo/Skills | -- Module:GameInfo/Skills | ||
-- Phase 4. | -- Phase 4.2 — Final native-first rendering (schema 1 + schema 2) | ||
-- | -- | ||
-- | -- Notes: | ||
-- - | -- - JSON Definition tokens must be plain strings that decode to Domain|Key. | ||
-- - Interactivity is provided by sitewide popup systems in Common.js. | |||
-- - Schema 2 is the primary layout target; schema 1 remains supported. | |||
-- - Mechanics / keywords are JSON-driven; the renderer does not infer future | |||
-- mechanic definition tokens from labels. | |||
-- - | |||
-- - | |||
-- - | |||
-- | |||
local p = {} | local p = {} | ||
| Line 86: | Line 57: | ||
local _CORE_ORDER = { | local _CORE_ORDER = { | ||
{ field = "cost", label = "Cost" }, | { field = "cost", label = "Cost", def_tok = "Skill|Cost" }, | ||
{ field = "cast_time", label = "Cast Time" }, | { field = "cast_time", label = "Cast Time", def_tok = "Skill|CastTime" }, | ||
{ field = "cooldown", label = "Cooldown" }, | { field = "cooldown", label = "Cooldown", def_tok = "Skill|Cooldown" }, | ||
{ field = "range", label = "Range" }, | { field = "range", label = "Range", def_tok = "Skill|Range" }, | ||
{ field = "area", label = "Area" }, | { field = "area", label = "Area", def_tok = "Skill|Area" }, | ||
{ field = "duration", label = "Duration" }, | { field = "duration", label = "Duration", def_tok = "Skill|Duration" }, | ||
} | } | ||
| Line 101: | Line 72: | ||
end | end | ||
local function _safe_tbl(t) return type(t) == "table" and t or {} end | local function _safe_tbl(t) | ||
return type(t) == "table" and t or {} | |||
end | |||
local function _safe_str(s, fallback) | local function _safe_str(s, fallback) | ||
| Line 119: | Line 92: | ||
s = s:gsub("[^%w]+", "-"):gsub("%-+", "-"):gsub("^%-", ""):gsub("%-$", "") | s = s:gsub("[^%w]+", "-"):gsub("%-+", "-"):gsub("^%-", ""):gsub("%-$", "") | ||
if s == "" then s = "item" end | if s == "" then s = "item" end | ||
return s | |||
end | |||
local function _humanize_key(s) | |||
s = _trim(s) | |||
if s == "" then return "" end | |||
if s:find("%s") then return s end | |||
s = s:gsub("([A-Z]+)([A-Z][a-z])", "%1 %2") | |||
s = s:gsub("([a-z0-9])([A-Z])", "%1 %2") | |||
return s | return s | ||
end | end | ||
| Line 242: | Line 224: | ||
s = _trim(s) | s = _trim(s) | ||
return s ~= "" and s:find("|", 1, true) ~= nil | return s ~= "" and s:find("|", 1, true) ~= nil | ||
end | |||
local function _split_def_token(s) | |||
s = _trim(s) | |||
if s == "" then return nil, nil end | |||
local bar = s:find("|", 1, true) | |||
if not bar then return nil, nil end | |||
local domain = _trim(s:sub(1, bar - 1)) | |||
local key = _trim(s:sub(bar + 1)) | |||
if domain == "" or key == "" then return nil, nil end | |||
return domain, key | |||
end | end | ||
| Line 248: | Line 244: | ||
if s == "" then return nil end | if s == "" then return nil end | ||
local | local domain, key = _split_def_token(s) | ||
if not | if not domain or not key then | ||
return mw.text.nowiki(s) | return mw.text.nowiki(s) | ||
end | end | ||
local args = { domain, key } | |||
local args = { domain, key } | |||
if noicon then | if noicon then | ||
args.noicon = "1" | args.noicon = "1" | ||
| Line 272: | Line 262: | ||
return frame:expandTemplate{ title = "def", args = args } | return frame:expandTemplate{ title = "def", args = args } | ||
end | |||
local function _display_text_from_token(s) | |||
s = _trim(s) | |||
if s == "" then return "" end | |||
local _, key = _split_def_token(s) | |||
if key then return _humanize_key(key) end | |||
return s | |||
end | end | ||
| Line 292: | Line 291: | ||
end | end | ||
local function | local function _build_hitbox(parent, class_name, frame, token) | ||
token = _safe_str(token, "") | |||
if | if not _has_pipe(token) then return false end | ||
local | local wt = _render_def_token(frame, token, true, { pill = "1", fill = "1" }) | ||
if not | if not wt then return false end | ||
parent:tag("span") | |||
:addClass(class_name) | |||
:attr("aria-hidden", "true") | |||
:wikitext(wt) | |||
return | return true | ||
end | end | ||
local function | local function _is_blankish(v) | ||
if v == nil then return true end | |||
local s = _trim(v) | |||
return s == "" or s == "—" or s == "-" or s == "--" | |||
end | |||
local function _value_has_content(val) | |||
if | if val == nil then return false end | ||
return | |||
if type(val) ~= "table" then | |||
return not _is_blankish(val) | |||
end | end | ||
return | local series = val.series | ||
end | if type(series) == "table" then | ||
for _, item in ipairs(series) do | |||
if not _is_blankish(item) then | |||
return true | |||
end | |||
end | |||
return false | |||
end | |||
if val.text ~= nil then | |||
return not _is_blankish(val.text) | |||
end | |||
return false | |||
end | |||
local function _stat_key_from_token(token) | |||
token = _trim(token):lower() | |||
if token == "" then return nil end | |||
return _STAT_TOKEN_TO_KEY[token] | |||
end | |||
local function _normalize_stat_scaling_slots(stat_scaling) | |||
local slots = {} | |||
for _, key in ipairs(_STAT_ORDER) do | |||
slots[key] = nil | |||
end | end | ||
for _, it in ipairs(_safe_tbl(stat_scaling)) do | |||
it = _safe_tbl(it) | |||
local | local legacy_tok = _safe_str(it.stat_wt or it.stat or "", "") | ||
if legacy_tok ~= "" then | |||
local key = _stat_key_from_token(legacy_tok) | |||
if key then | |||
slots[key] = it.value | |||
end | |||
end | |||
for k, v in pairs(it) do | |||
local lk = _trim(k):lower() | |||
if _STAT_LABELS[lk] ~= nil then | |||
slots[lk] = v | |||
if | |||
end | end | ||
end | end | ||
end | end | ||
return slots | |||
end | end | ||
local function | local function _split_lines(s) | ||
if s == nil then return {} end | |||
if | s = tostring(s) | ||
return | s = s:gsub("\r\n", "\n"):gsub("\r", "\n") | ||
if mw.text and mw.text.split then | |||
return mw.text.split(s, "\n", true) | |||
end | |||
local out = {} | |||
for line in (s .. "\n"):gmatch("(.-)\n") do | |||
table.insert(out, line) | |||
end | |||
return out | |||
end | end | ||
local function | local function _render_notes_body(parent, notes_wt) | ||
notes_wt = _trim(notes_wt) | |||
if notes_wt == "" then | |||
parent:wikitext("—") | |||
return | |||
end | end | ||
notes_wt = notes_wt:gsub("(^|\n)%s+([*#])", "%1%2") | |||
local lines = _split_lines(notes_wt) | |||
local i = 1 | |||
local wrote_any = false | |||
local function is_list_line(ln) | |||
return type(ln) == "string" and ln:match("^[*#]") ~= nil | |||
end | end | ||
while i <= #lines do | |||
local ln = lines[i] or "" | |||
if _trim(ln) == "" then | |||
i = i + 1 | |||
elseif is_list_line(ln) then | |||
local mark = ln:sub(1, 1) | |||
local tagname = (mark == "#") and "ol" or "ul" | |||
local | local items = {} | ||
while i <= #lines do | |||
local l = lines[i] or "" | |||
if _trim(l) == "" then | |||
i = i + 1 | |||
break | |||
end | |||
if not l:match("^" .. mark) then break end | |||
end | |||
local | local txt = l:gsub("^" .. mark .. "+%s*", "") | ||
txt = _trim(txt) | |||
if txt ~= "" then table.insert(items, txt) end | |||
i = i + 1 | |||
end | |||
if #items > 0 then | |||
local list = parent:tag(tagname) | |||
for _, txt in ipairs(items) do | |||
list:tag("li"):wikitext(txt) | |||
end | |||
wrote_any = true | |||
end | |||
else | |||
local block = {} | |||
while i <= #lines do | |||
local l = lines[i] or "" | |||
if _trim(l) == "" then | |||
i = i + 1 | |||
break | |||
end | |||
if is_list_line(l) then break end | |||
table.insert(block, l) | |||
i = i + 1 | |||
end | |||
if #block > 0 then | |||
local joined = table.concat(block, "<br />\n") | |||
parent:tag("div"):addClass("sv-note-par"):wikitext(joined) | |||
wrote_any = true | |||
end | |||
end | |||
end | |||
if not wrote_any then | |||
parent:wikitext(mw.text.nowiki(notes_wt)) | |||
end | end | ||
end | |||
local function _build_notes_tip(frame, root_id, notes_wt) | |||
notes_wt = _trim(notes_wt) | |||
if notes_wt == "" then return nil end | |||
local tip_id = root_id .. "-notes" | |||
local wrap = mw.html.create("div"):addClass("sv-tip") | |||
local btn = wrap:tag("span") | |||
:addClass("sv-tip-btn") | |||
:addClass("sv-tip-btn--icon") | |||
:addClass("sv-hover-lift") | |||
:attr("role", "button") | |||
:attr("tabindex", "0") | |||
:attr("data-sv-toggle", "1") | |||
:attr("data-sv-pop", "hover") | |||
:attr("data-sv-pop-size", "sm") | |||
:attr("aria-label", "Notes") | |||
:attr("aria-controls", tip_id) | |||
:attr("aria-expanded", "false") | |||
btn:tag("span") | |||
:addClass("sv-ico") | |||
:addClass("sv-ico--info") | |||
:attr("aria-hidden", "true") | |||
:wikitext("i") | |||
local pop = wrap:tag("div") | |||
:addClass("sv-tip-pop") | |||
:addClass("sv-hidden") | |||
:attr("hidden", "hidden") | |||
:attr("id", tip_id) | |||
:attr("aria-label", "Notes") | |||
local head = pop:tag("div"):addClass("sv-tip-pop-head") | |||
head:tag("div"):addClass("sv-tip-pop-title"):wikitext("Notes") | |||
local body = pop:tag("div"):addClass("sv-tip-pop-body") | |||
_render_notes_body(body, notes_wt) | |||
return wrap | |||
end | end | ||
local function | local function _normalize_overcap_sources(sources, total_bonus) | ||
sources = _safe_tbl(sources) | |||
local | local out = {} | ||
local | local missing = 0 | ||
local explicit_sum = 0 | |||
for _, s in ipairs(sources) do | |||
local page = "" | |||
local lv = nil | |||
if type(s) == "string" then | |||
page = _trim(s) | |||
elseif type(s) == "table" then | |||
page = _safe_str(s.page or s[1] or "", "") | |||
lv = _to_int(s.levels or s.level or s[2], nil) | |||
end | |||
if page ~= "" then | |||
if lv and lv > 0 then | |||
explicit_sum = explicit_sum + lv | |||
else | |||
lv = nil | |||
missing = missing + 1 | |||
end | |||
table.insert(out, { page = page, levels = lv }) | |||
end | |||
end | |||
local | local remaining = (_to_int(total_bonus, 0) or 0) - explicit_sum | ||
if remaining < 0 then remaining = 0 end | |||
if missing == 1 and remaining > 0 then | |||
for _, it in ipairs(out) do | |||
if it.levels == nil then | |||
it.levels = remaining | |||
break | |||
end | |||
end | |||
end | |||
return | return out | ||
end | end | ||
local function | local function _build_overcap_tip(root_id, default_cap, max_cap, sources) | ||
default_cap = _to_int(default_cap, 1) or 1 | |||
max_cap = _to_int(max_cap, default_cap) or default_cap | |||
if max_cap <= default_cap then return nil end | |||
local | local total = max_cap - default_cap | ||
local | local tip_id = root_id .. "-overcap" | ||
local norm = _normalize_overcap_sources(sources, total) | |||
local show_unknown = (#norm == 0) | |||
local wrap = mw.html.create("span") | |||
:addClass("sv-tip") | |||
:addClass("sv-overcap-tip") | |||
local btn = wrap:tag("span") | |||
:addClass("sv-tip-btn") | |||
:addClass("sv-tip-btn--pill") | |||
:addClass("sv-overcap-btn") | |||
:addClass("sv-hover-lift") | |||
:attr("role", "button") | |||
:attr("tabindex", "0") | |||
:attr("data-sv-toggle", "1") | |||
:attr("data-sv-pop", "click") | |||
:attr("data-sv-pop-size", "sm") | |||
:attr("aria-label", "Overcap") | |||
:attr("aria-controls", tip_id) | |||
:attr("aria-expanded", "false") | |||
:wikitext(mw.text.nowiki("Overcap +" .. tostring(total))) | |||
local | local pop = wrap:tag("div") | ||
:addClass("sv-tip-pop") | |||
:addClass("sv-hidden") | |||
:attr("hidden", "hidden") | |||
:attr("id", tip_id) | |||
:attr("aria-label", "Overcap") | |||
local head = pop:tag("div"):addClass("sv-tip-pop-head") | |||
head:tag("div"):addClass("sv-tip-pop-title"):wikitext("Overcap sources") | |||
local body = pop:tag("div"):addClass("sv-tip-pop-body") | |||
body:tag("div"):addClass("sv-overcap-summary"):wikitext(mw.text.nowiki( | |||
"Raises max from " .. tostring(default_cap) .. " to " .. tostring(max_cap) .. " (+" .. tostring(total) .. ")." | |||
)) | |||
local | local ul = body:tag("ul"):addClass("sv-disclose-list") | ||
local | if show_unknown then | ||
ul:tag("li"):wikitext("Source unknown") | |||
else | |||
for _, it in ipairs(norm) do | |||
local li = ul:tag("li") | |||
local page = _safe_str(it.page, "") | |||
if page ~= "" then | |||
li:wikitext(string.format("[[%s|%s]]", page, mw.text.nowiki(page))) | |||
else | |||
li:wikitext("—") | |||
end | |||
if it.levels ~= nil and it.levels > 0 then | |||
li:wikitext(mw.text.nowiki(" (+" .. tostring(it.levels) .. ")")) | |||
else | |||
li:wikitext(mw.text.nowiki(" (+?)")) | |||
end | |||
end | |||
end | |||
return wrap | |||
end | |||
local function _normalize_disclose_item(it) | |||
if type(it) == "string" then | |||
local s = _trim(it) | |||
if s == "" then return nil end | |||
return { text = s, page = s } | |||
end | |||
if type(it) ~= "table" then return nil end | |||
local | if it.text ~= nil or it.page ~= nil or it.suffix ~= nil or it.name ~= nil or it.label ~= nil then | ||
local text = _safe_str(it.text or it.label or it.name or "", "") | |||
if text == "" then return nil end | |||
local page = _safe_str(it.page or "", "") | |||
if page == "" then page = text end | |||
local suffix = _safe_str(it.suffix or "", "") | |||
return { text = text, page = (page ~= "" and page or nil), suffix = (suffix ~= "" and suffix or nil) } | |||
end | |||
local | if #it >= 1 then | ||
local name = _safe_str(it[1], "") | |||
if name == "" then return nil end | |||
local suffix = _safe_str(it[2], "") | |||
return { | |||
text = name, | |||
page = name, | |||
suffix = (suffix ~= "" and suffix or nil) | |||
} | |||
end | |||
local | local lines = _safe_tbl(it.label_lines) | ||
local text = "" | |||
for _, ln in ipairs(lines) do | |||
local s = _trim(ln) | |||
if s ~= "" then | |||
if text ~= "" then text = text .. " " end | |||
text = text .. s | |||
end | |||
end | |||
if text == "" then return nil end | |||
local link = _safe_tbl(it.link) | |||
local page = _safe_str(link.page or it.page or "", "") | |||
return { text = text, page = (page ~= "" and page or nil) } | |||
end | |||
local function _count_group_items(groups) | |||
local n = 0 | |||
for _, g in ipairs(_safe_tbl(groups)) do | |||
for _, it in ipairs(_safe_tbl(g.items)) do | |||
if _normalize_disclose_item(it) then n = n + 1 end | |||
end | end | ||
end | end | ||
return n | |||
return | |||
end | end | ||
local function | local function _build_grouped_disclose(root_id, key, label, groups) | ||
local total = _count_group_items(groups) | |||
if total == 0 then return nil, 0 end | |||
local pop_id = root_id .. "-" .. key | |||
local wrap = mw.html.create("div"):addClass("sv-disclose") | |||
local btn = wrap:tag("span") | |||
:addClass("sv-disclose-btn") | |||
:addClass("sv-disclose-btn--compact") | |||
:addClass("sv-hover-lift") | |||
:attr("role", "button") | |||
:attr("tabindex", "0") | |||
:attr("data-sv-toggle", "1") | |||
:attr("data-sv-pop", "click") | |||
:attr("data-sv-pop-size", "sm") | |||
:attr("aria-label", label) | |||
:attr("aria-controls", pop_id) | |||
:attr("aria-expanded", "false") | |||
btn:tag("span"):addClass("sv-disclose-label"):wikitext(mw.text.nowiki(label)) | |||
btn:tag("span"):addClass("sv-disclose-count"):wikitext(" (" .. tostring(total) .. ")") | |||
local | local pop = wrap:tag("div") | ||
:addClass("sv-disclose-pop") | |||
:addClass("sv-hidden") | |||
:attr("hidden", "hidden") | |||
:attr("id", pop_id) | |||
:attr("aria-label", label) | |||
local | local head = pop:tag("div"):addClass("sv-disclose-pop-head") | ||
head:tag("div"):addClass("sv-disclose-pop-title"):wikitext(mw.text.nowiki(label)) | |||
local | local body = pop:tag("div"):addClass("sv-disclose-pop-body") | ||
local | local secs = body:tag("div"):addClass("sv-disclose-sections") | ||
local | local function render_item(parent, norm) | ||
local item = parent:tag("span"):addClass("sv-disclose-item") | |||
if norm.page then | |||
item:wikitext(string.format( | |||
"[[%s|%s]]", | |||
norm.page, | |||
mw.text.nowiki(norm.text) | |||
)) | |||
else | |||
item:wikitext(mw.text.nowiki(norm.text)) | |||
: | end | ||
local sfx = _safe_str(norm.suffix or "", "") | |||
if sfx ~= "" then | |||
if not sfx:match("^%s") then sfx = " " .. sfx end | |||
item:tag("span"):addClass("sv-disclose-sfx"):wikitext(mw.text.nowiki(sfx)) | |||
end | |||
end | |||
for _, g in ipairs(_safe_tbl(groups)) do | |||
local title = _safe_str(g.title, "—") | |||
local items = _safe_tbl(g.items) | |||
if #items > 0 then | |||
local sec = secs:tag("div"):addClass("sv-disclose-sec") | |||
sec:tag("div"):addClass("sv-disclose-sec-title"):wikitext(mw.text.nowiki(title)) | |||
local line = sec:tag("div"):addClass("sv-disclose-sec-items") | |||
local norms = {} | |||
for _, it in ipairs(items) do | |||
local norm = _normalize_disclose_item(it) | |||
if norm then table.insert(norms, norm) end | |||
end | |||
if #norms == 0 then | |||
line:tag("span"):addClass("sv-disclose-item"):wikitext("—") | |||
else | |||
for i, norm in ipairs(norms) do | |||
render_item(line, norm) | |||
if i < #norms then line:wikitext(", ") end | |||
end | |||
end | |||
end | end | ||
end | end | ||
return wrap, total | |||
end | |||
local function _build_identity(frame, root_id, identity, notes_wt) | |||
identity = _safe_tbl(identity) | |||
local root = mw.html.create("div"):addClass("sv-skill-head") | |||
local icon_div = root:tag("div") | |||
:addClass("sv-skill-icon") | |||
:addClass("sv-tile") | |||
local sprite = _safe_tbl(identity.sprite) | |||
local sprite_page = _safe_str(sprite.page, "") | |||
local sprite_alt = _safe_str(sprite.alt, _safe_str(identity.display_name, "")) | |||
if sprite_page ~= "" then | |||
icon_div:node(_render_file_image(sprite_page, sprite_alt, 64)) | |||
else | |||
icon_div:node(_question_badge()) | |||
end | end | ||
local headtext = root:tag("div"):addClass("sv-skill-headtext") | |||
local | local title_row = headtext:tag("div"):addClass("sv-skill-title-row") | ||
title_row:tag("div") | |||
:addClass("sv-skill-title") | |||
:wikitext(mw.text.nowiki(_safe_str(identity.display_name, "Unknown Skill"))) | |||
local | local tip = _build_notes_tip(frame, root_id, notes_wt) | ||
if tip then title_row:node(tip) end | |||
local | headtext:tag("div") | ||
local | :addClass("sv-skill-desc") | ||
local | :wikitext(mw.text.nowiki(_safe_str(identity.description, ""))) | ||
return root | |||
end | |||
local function _build_meta_card(frame, parent, spec) | |||
spec = _safe_tbl(spec) | |||
local card = parent:tag("div") | |||
:addClass("sv-tile") | |||
:addClass("sv-meta-card") | |||
:addClass("sv-hover-lift") | |||
local icon_div = card:tag("div") | |||
:addClass("sv-meta-icon") | |||
:addClass("sv-tile") | |||
if | local icon_page = _safe_str(spec.icon_page, "") | ||
icon_div:node(_render_file_image( | local icon_alt = _safe_str(spec.icon_alt, "") | ||
if icon_page ~= "" then | |||
icon_div:node(_render_file_image(icon_page, icon_alt, 24)) | |||
else | else | ||
icon_div:node(_question_badge()) | icon_div:node(_question_badge()) | ||
end | end | ||
local | local wrap = card:tag("div"):addClass("sv-meta-textwrap") | ||
local text = wrap:tag("div"):addClass("sv-meta-text") | |||
local | local lines = _safe_tbl(spec.lines) | ||
if #lines > 0 then | |||
: | text:node(_meta_lines_def(frame, lines)) | ||
:wikitext( | else | ||
text:wikitext("—") | |||
end | |||
_build_hitbox(card, "sv-meta-hit", frame, spec.hit_tok) | |||
end | |||
end | |||
local function _build_meta_row(frame, meta_row) | local function _build_meta_row(frame, meta_row) | ||
| Line 833: | Line 828: | ||
for i = 1, 4 do | for i = 1, 4 do | ||
local cell = _safe_tbl(meta_row[i]) | local cell = _safe_tbl(meta_row[i]) | ||
local icon = _safe_tbl(cell.icon) | local icon = _safe_tbl(cell.icon) | ||
local lines = _safe_tbl(cell.label_lines) | local lines = _safe_tbl(cell.label_lines) | ||
local hit_tok = "" | |||
for _, ln in ipairs(lines) do | for _, ln in ipairs(lines) do | ||
local s = _safe_str(ln, "") | local s = _safe_str(ln, "") | ||
| Line 869: | Line 839: | ||
end | end | ||
end | end | ||
_build_meta_card(frame, meta, { | |||
icon_page = _safe_str(icon.page, ""), | |||
icon_alt = _safe_str(icon.alt, ""), | |||
lines = lines, | |||
hit_tok = hit_tok, | |||
}) | |||
end | end | ||
| Line 889: | Line 857: | ||
for i = 1, 4 do | for i = 1, 4 do | ||
local cell = _safe_tbl(skill_meta[i]) | local cell = _safe_tbl(skill_meta[i]) | ||
local | local lines = _safe_tbl(cell.label_lines) | ||
local hit_tok = _safe_str(cell.label_wt, "") | |||
if #lines == 0 and hit_tok ~= "" then | |||
lines = { hit_tok } | |||
end | end | ||
_build_meta_card(frame, meta, { | |||
icon_page = _safe_str(cell.icon, ""), | |||
icon_alt = "", | |||
lines = lines, | |||
hit_tok = hit_tok, | |||
}) | |||
end | end | ||
| Line 1,055: | Line 996: | ||
if tok ~= "" and val ~= "" then | if tok ~= "" and val ~= "" then | ||
local item = list:tag("div") | local item = list:tag("div") | ||
:addClass("sv-tile") | |||
:addClass("sv-scaling-item") | :addClass("sv-scaling-item") | ||
:addClass("sv-scaling-item--stat") | :addClass("sv-scaling-item--stat") | ||
| Line 1,064: | Line 1,006: | ||
for _, ln in ipairs(_safe_tbl(scaling.scaling_lines)) do | for _, ln in ipairs(_safe_tbl(scaling.scaling_lines)) do | ||
local item = list:tag("div"):addClass("sv-scaling-item") | local item = list:tag("div") | ||
:addClass("sv-tile") | |||
:addClass("sv-scaling-item") | |||
_apply_value(item, ln, level) | _apply_value(item, ln, level) | ||
end | end | ||
| Line 1,113: | Line 1,057: | ||
local pill = parent:tag("div") | local pill = parent:tag("div") | ||
:addClass("sv-pill") | |||
:addClass("sv-pill--compact") | |||
:addClass("sv_skill_scaling__pill") | :addClass("sv_skill_scaling__pill") | ||
:addClass("sv_skill_scaling__pill--stat") | :addClass("sv_skill_scaling__pill--stat") | ||
| Line 1,121: | Line 1,067: | ||
pill:addClass(active and "is-active" or "is-inactive") | pill:addClass(active and "is-active" or "is-inactive") | ||
if active then pill:addClass("sv-hover-lift") end | |||
local main = pill:tag("div"):addClass("sv_skill_scaling__stat-main") | local main = pill:tag("div"):addClass("sv_skill_scaling__stat-main") | ||
| Line 1,138: | Line 1,085: | ||
if active then | if active then | ||
_build_hitbox(pill, "sv_skill_scaling__stat-hit", frame, def_tok) | |||
end | end | ||
end | end | ||
local function _build_skill_scaling_core_pill(parent, spec, fld, level) | local function _build_skill_scaling_core_pill(frame, parent, spec, fld, level) | ||
fld = _safe_tbl(fld) | fld = _safe_tbl(fld) | ||
local active = _value_has_content(fld.value) | local active = _value_has_content(fld.value) | ||
local resolved_label = _safe_str(fld.label or fld.key, spec.label) | |||
local resolved_tok = _safe_str(fld.label_wt or fld.def_wt, spec.def_tok) | |||
local pill = parent:tag("div") | local pill = parent:tag("div") | ||
| Line 1,158: | Line 1,101: | ||
:addClass("sv_skill_scaling__core-pill") | :addClass("sv_skill_scaling__core-pill") | ||
:addClass("sv_skill_scaling__core-pill--" .. spec.field) | :addClass("sv_skill_scaling__core-pill--" .. spec.field) | ||
:addClass("sv-hover-lift") | |||
:attr("data-core-key", spec.field) | :attr("data-core-key", spec.field) | ||
:attr("aria-label", resolved_label) | |||
pill:addClass(active and "is-active" or "is-inactive") | pill:addClass(active and "is-active" or "is-inactive") | ||
local top = pill:tag("div"):addClass("sv_skill_scaling__core-main") | local top = pill:tag("div"):addClass("sv_skill_scaling__core-main") | ||
local value = top:tag("span"):addClass("sv_skill_scaling__core-value") | local value = top:tag("span"):addClass("sv_skill_scaling__core-value") | ||
_apply_value(value, fld.value, level) | _apply_value(value, fld.value, level) | ||
| Line 1,176: | Line 1,120: | ||
pill:tag("div") | pill:tag("div") | ||
:addClass("sv_skill_scaling__core-label") | :addClass("sv_skill_scaling__core-label") | ||
:wikitext(mw.text.nowiki( | :wikitext(mw.text.nowiki(resolved_label)) | ||
_build_hitbox(pill, "sv_skill_scaling__core-hit", frame, resolved_tok) | |||
end | end | ||
| Line 1,197: | Line 1,143: | ||
local body = root:tag("div"):addClass("sv_skill_scaling__body") | local body = root:tag("div"):addClass("sv_skill_scaling__body") | ||
local damage = _safe_tbl(skill_scaling.damage) | local damage = _safe_tbl(skill_scaling.damage) | ||
| Line 1,205: | Line 1,149: | ||
do | do | ||
local | local primary_group = body:tag("div") | ||
:addClass(" | :addClass("sv_skill_scaling__group") | ||
:addClass(" | :addClass("sv_skill_scaling__group--primary-stats") | ||
local | local cluster = primary_group:tag("div"):addClass("sv_skill_scaling__cluster") | ||
do | |||
local primary = cluster:tag("div") | |||
:addClass("sv_skill_scaling__column") | |||
:addClass("sv_skill_scaling__column--primary") | |||
:addClass("sv_skill_scaling__primary") | |||
local v = primary:tag("div"):addClass("sv_skill_scaling__primary-value") | |||
_apply_value(v, damage_value, actual_default) | |||
primary:tag("div") | |||
:addClass("sv_skill_scaling__primary-label") | |||
:wikitext(mw.text.nowiki(damage_label)) | |||
end | end | ||
do | |||
local stats_col = cluster:tag("div") | |||
:addClass("sv_skill_scaling__column") | |||
:addClass("sv_skill_scaling__column--stats") | |||
:addClass("sv_skill_scaling__stats") | |||
local stats_grid = stats_col:tag("div"):addClass("sv_skill_scaling__stats-grid") | |||
local stat_slots = _normalize_stat_scaling_slots(skill_scaling.stat_scaling) | |||
for _, stat_key in ipairs(_STAT_GRID_ORDER) do | |||
_build_skill_scaling_stat_pill(frame, stats_grid, stat_key, stat_slots[stat_key], actual_default) | |||
end | |||
end | |||
end | |||
do | do | ||
local core_col = | local core_group = body:tag("div") | ||
:addClass("sv_skill_scaling__group") | |||
:addClass("sv_skill_scaling__group--core") | |||
local core_col = core_group:tag("div") | |||
:addClass("sv_skill_scaling__column") | :addClass("sv_skill_scaling__column") | ||
:addClass("sv_skill_scaling__column--core") | :addClass("sv_skill_scaling__column--core") | ||
| Line 1,243: | Line 1,197: | ||
for _, spec in ipairs(_CORE_ORDER) do | for _, spec in ipairs(_CORE_ORDER) do | ||
_build_skill_scaling_core_pill(core_grid, spec, skill_scaling[spec.field], actual_default) | _build_skill_scaling_core_pill(frame, core_grid, spec, skill_scaling[spec.field], actual_default) | ||
end | end | ||
end | end | ||
| Line 1,250: | Line 1,204: | ||
end | end | ||
local function | local function _normalize_modifier_spec(item) | ||
item = _safe_tbl(item) | |||
local explicit_tok = _safe_str(item.label_wt or item.def_wt, "") | |||
local label_text = _safe_str(item.label or item.key or item.text, "") | |||
if label_text == "" and explicit_tok ~= "" then | |||
label_text = _display_text_from_token(explicit_tok) | |||
end | |||
return { | |||
label_tok = explicit_tok, | |||
label_text = label_text, | |||
value = item.value, | |||
} | |||
end | |||
local function _build_modifier_pill(frame, parent, item, level) | |||
local spec = _normalize_modifier_spec(item) | |||
if spec.label_text == "" then return end | |||
local | local has_def = _has_pipe(spec.label_tok) | ||
local | local has_value = _value_has_content(spec.value) | ||
local | local pill = parent:tag("div") | ||
:addClass("sv-tile") | |||
:addClass("sv-mech-mod-pill") | |||
:attr("aria-label", spec.label_text) | |||
if | if has_value then | ||
pill:addClass("is-active") | |||
else | |||
pill:addClass("is-inactive") | |||
end | end | ||
if | |||
if has_def or has_value then | |||
pill:addClass("sv-hover-lift") | |||
end | end | ||
pill:tag("div") | |||
:addClass("sv-mech-mod-pill__label") | |||
:wikitext(mw.text.nowiki(spec.label_text)) | |||
local value = pill:tag("div"):addClass("sv-mech-mod-pill__value") | |||
_apply_value(value, spec.value, level) | |||
if has_def then | |||
_build_hitbox(pill, "sv-mech-mod-pill__hit", frame, spec.label_tok) | |||
end | |||
end | |||
local function _normalize_keyword_spec(item) | |||
if type(item) == "string" then | |||
local tok = _safe_str(item, "") | |||
local label = tok | |||
if _has_pipe(tok) then | |||
label = _display_text_from_token(tok) | |||
end | end | ||
return { | |||
label_tok = (_has_pipe(tok) and tok or ""), | |||
label_text = label, | |||
} | |||
end | |||
item = _safe_tbl(item) | |||
local tok = _safe_str(item.label_wt or item.def_wt, "") | |||
local label = _safe_str(item.label or item.text or item.key, "") | |||
if label == "" and tok ~= "" then | |||
label = _display_text_from_token(tok) | |||
end | end | ||
return { | |||
label_tok = tok, | |||
label_text = label, | |||
} | |||
end | |||
local | local function _build_keyword_pill(frame, parent, item) | ||
local spec = _normalize_keyword_spec(item) | |||
local display_text = _safe_str(spec.label_text, "") | |||
local hit_tok = _safe_str(spec.label_tok, "") | |||
if display_text == "" then return end | |||
local | local has_def = _has_pipe(hit_tok) | ||
local pill = parent:tag("span") | |||
:addClass("sv-pill") | |||
:addClass("sv-pill--value") | |||
:addClass("sv-mech-keyword-pill") | |||
if has_def then pill:addClass("sv-hover-lift") end | |||
pill:tag("span") | |||
:addClass("sv-mech-keyword-pill__label") | |||
:wikitext(mw.text.nowiki(display_text)) | |||
if has_def then | |||
_build_hitbox(pill, "sv-mech-keyword-pill__hit", frame, hit_tok) | |||
end | |||
end | |||
local function _build_tabs(frame, root_id, tabs_obj, level) | |||
tabs_obj = _safe_tbl(tabs_obj) | |||
local mechanics = _safe_tbl(tabs_obj.mechanics) | |||
local effects = _safe_tbl(tabs_obj.effects) | |||
local mechanics_mods = _safe_tbl(mechanics.mods) | |||
local mechanics_keywords = _safe_tbl(mechanics.keywords) | |||
local effects_cards = _safe_tbl(effects.cards) | |||
if #mechanics_mods == 0 then | |||
mechanics_mods = _safe_tbl(mechanics.grid) | |||
end | |||
if #mechanics_keywords == 0 then | |||
local old_keywords = _safe_tbl(tabs_obj.keywords) | |||
mechanics_keywords = _safe_tbl(old_keywords.pills) | |||
end | end | ||
if #effects_cards == 0 then | |||
local old_events = _safe_tbl(tabs_obj.events) | |||
local merged = {} | |||
for _, card in ipairs(_safe_tbl(effects.cards)) do | |||
table.insert(merged, card) | |||
end | |||
for _, card in ipairs(_safe_tbl(old_events.cards)) do | |||
table.insert(merged, card) | |||
end | end | ||
effects_cards = merged | |||
end | end | ||
local | local mechanics_count = _to_int( | ||
mechanics.count, | |||
#mechanics_mods + #mechanics_keywords | |||
) | |||
local effects_count = _to_int( | |||
effects.count, | |||
#effects_cards | |||
) | |||
local tab_specs = { | |||
{ key = "mechanics", title = "Mechanics (" .. tostring(mechanics_count) .. ")" }, | |||
{ key = "effects", title = "Effects (" .. tostring(effects_count) .. ")" }, | |||
} | |||
local root = mw.html.create("div"):addClass("sv-skill-tabs") | |||
local tabs = root:tag("div") | |||
:addClass("sv-tabs") | |||
:attr("data-tabs", "1") | |||
:attr("data-tabs-root", root_id) | |||
local list = tabs:tag("div"):addClass("sv-tabs-list"):attr("role", "tablist") | |||
local panels = tabs:tag("div"):addClass("sv-tabs-panels") | |||
for i, spec in ipairs(tab_specs) do | |||
local active = (i == 1) | |||
list:tag("span") | |||
:addClass("sv-tab") | |||
:attr("role", "tab") | |||
:attr("tabindex", active and "0" or "-1") | |||
:attr("data-tab", spec.key) | |||
:attr("aria-selected", active and "true" or "false") | |||
:wikitext(mw.text.nowiki(spec.title)) | |||
end | end | ||
local function | local function render_ref_icon(parent, icon) | ||
icon = _safe_tbl(icon) | |||
local page = _safe_str(icon.page, "") | |||
local alt = _safe_str(icon.alt, "") | |||
local page = _safe_str( | |||
local | |||
if page ~= "" then | if page ~= "" then | ||
parent:node(_render_file_image(page, alt, 52)) | |||
else | else | ||
parent:node(_question_badge()) | |||
end | end | ||
end | end | ||
local function render_effect_stats(parent, stats, kind) | |||
stats = _safe_tbl(stats) | |||
kind = _safe_str(kind, ""):lower() | |||
local | if kind == "event" then | ||
local sub = parent:tag("div"):addClass("sv-ref-sub") | |||
if stats.text ~= nil then | |||
_apply_value(sub, stats, level) | |||
else | |||
_apply_value(sub, { text = "—" }, level) | |||
end | |||
return | |||
end | |||
local | if next(stats) ~= nil then | ||
local srow = parent:tag("div"):addClass("sv-ref-stats") | |||
local d = srow:tag("span") | |||
:addClass("sv-pill") | |||
:addClass("sv-pill--value") | |||
:addClass("sv-ref-stat") | |||
_apply_value(d, stats.duration, level) | |||
local | local c = srow:tag("span") | ||
:addClass("sv-pill") | |||
:addClass("sv-pill--value") | |||
:addClass("sv-ref-stat") | |||
_apply_value(c, stats.chance, level) | |||
local st = srow:tag("span") | |||
:addClass("sv-pill") | |||
:addClass("sv-pill--value") | |||
:addClass("sv-ref-stat") | |||
_apply_value(st, stats.stacks, level) | |||
end | end | ||
end | |||
local function render_ref_card(parent, card) | |||
card = _safe_tbl(card) | |||
local kind = _safe_str(card.kind, "") | |||
local title = _safe_str(card.title, "—") | |||
local page = _safe_str(card.page, "") | |||
local icon = _safe_tbl(card.icon) | |||
local container = parent:tag("div") | |||
:addClass("sv-tile") | |||
:addClass("sv-hover-lift") | |||
:addClass("sv-ref-card") | |||
if kind ~= "" then | |||
container:addClass("sv-ref-card--" .. _slugify(kind)) | |||
end | end | ||
local ico = container:tag("div") | |||
local | :addClass("sv-ref-ico") | ||
:addClass("sv- | :addClass("sv-tile") | ||
: | render_ref_icon(ico, icon) | ||
local text = container:tag("div"):addClass("sv-ref-text") | |||
local title_div = text:tag("div"):addClass("sv-ref-title") | |||
if page ~= "" then | |||
title_div:wikitext(string.format("[[%s|%s]]", page, mw.text.nowiki(title))) | |||
else | else | ||
title_div:wikitext(mw.text.nowiki(title)) | |||
end | end | ||
render_effect_stats(text, card.stats, kind) | |||
end | end | ||
do | |||
local panel = panels:tag("div") | |||
:addClass("sv-tabpanel") | |||
:addClass("sv-mech-panel") | |||
:attr("role", "tabpanel") | |||
:attr("data-panel", "mechanics") | |||
local | local mods_wrap = panel:tag("div") | ||
:addClass("sv-mech-panel__group") | |||
:addClass("sv-mech-panel__group--mods") | |||
:addClass("sv-mech-panel__mods") | |||
local keys_wrap = panel:tag("div") | |||
:addClass("sv-mech-panel__group") | |||
:addClass("sv-mech-panel__group--keywords") | |||
:addClass("sv-mech-panel__keywords") | |||
if #mechanics_mods > 0 then | |||
mods_wrap:tag("div"):addClass("sv-tab-section-title"):wikitext("Modifiers") | |||
local gridwrap = mods_wrap:tag("div"):addClass("sv-mech-mod-grid") | |||
for _, item in ipairs(mechanics_mods) do | |||
_build_modifier_pill(frame, gridwrap, item, level) | |||
end | |||
end | end | ||
if #mechanics_keywords > 0 then | |||
keys_wrap:tag("div"):addClass("sv-tab-section-title"):wikitext("Keywords") | |||
local pills = keys_wrap:tag("div"):addClass("sv-tab-pills") | |||
for _, kw in ipairs(mechanics_keywords) do | |||
_build_keyword_pill(frame, pills, kw) | |||
end | |||
end | |||
if #mechanics_mods == 0 and #mechanics_keywords == 0 then | |||
panel:tag("div") | |||
:addClass("sv-tile") | |||
:addClass("sv-tab-empty") | |||
:wikitext("—") | |||
end | end | ||
end | end | ||
if | do | ||
local panel = panels:tag("div") | |||
:addClass("sv-tabpanel") | |||
:addClass("sv-hidden") | |||
:attr("role", "tabpanel") | |||
:attr("data-panel", "effects") | |||
:attr("hidden", "hidden") | |||
if #effects_cards > 0 then | |||
local grid = panel:tag("div"):addClass("sv-ref-grid") | |||
for _, card in ipairs(effects_cards) do | |||
render_ref_card(grid, card) | |||
end | |||
else | |||
panel:tag("div") | |||
:addClass("sv-tile") | |||
:addClass("sv-tab-empty") | |||
:wikitext("—") | |||
end | end | ||
end | end | ||
local | return root | ||
end | |||
if | local function _map_to_groups(map_tbl, preferred_order) | ||
local m = _safe_tbl(map_tbl) | |||
local function normalize_items(v) | |||
if type(v) == "table" then | |||
return v | |||
end | |||
if v == nil then | |||
return {} | |||
end | end | ||
return { v } | |||
end | end | ||
local | local used = {} | ||
local group_keys = {} | |||
local function push_key(k) | |||
table.insert( | k = tostring(k) | ||
if k ~= "" and not used[k] and m[k] ~= nil then | |||
used[k] = true | |||
table.insert(group_keys, k) | |||
end | end | ||
end | |||
local explicit = m.__order or m.order | |||
if type(explicit) == "table" then | |||
for _, k in ipairs(explicit) do | |||
push_key(k) | |||
end | |||
end | end | ||
if type(preferred_order) == "table" then | |||
end | for _, k in ipairs(preferred_order) do | ||
push_key(k) | |||
end | |||
end | |||
local function _build_req_users_native(root_id, requirements_map, users_map) | local rest = {} | ||
local req_groups = _map_to_groups(requirements_map, { "Skills", "Weapon Types", "Stances" }) | for k, _ in pairs(m) do | ||
local usr_groups = _map_to_groups(users_map, { "Classes", "Summons" }) | k = tostring(k) | ||
if k ~= "__order" and k ~= "order" and not used[k] then | |||
table.insert(rest, k) | |||
end | |||
end | |||
table.sort(rest) | |||
for _, k in ipairs(rest) do | |||
table.insert(group_keys, k) | |||
end | |||
local out_groups = {} | |||
for _, title in ipairs(group_keys) do | |||
local items = normalize_items(m[title]) | |||
local out_items = {} | |||
for _, it in ipairs(_safe_tbl(items)) do | |||
table.insert(out_items, it) | |||
end | |||
table.insert(out_groups, { title = title, items = out_items }) | |||
end | |||
return out_groups | |||
end | |||
local function _build_req_users_native(root_id, requirements_map, users_map) | |||
local req_groups = _map_to_groups(requirements_map, { "Skills", "Weapon Types", "Stances" }) | |||
local usr_groups = _map_to_groups(users_map, { "Classes", "Monsters", "Summons" }) | |||
local req_box, req_n = _build_grouped_disclose(root_id, "req", "Requirements", req_groups) | local req_box, req_n = _build_grouped_disclose(root_id, "req", "Requirements", req_groups) | ||
local usr_box, usr_n = _build_grouped_disclose(root_id, "usr", "Users", usr_groups) | local usr_box, usr_n = _build_grouped_disclose(root_id, "usr", "Users", usr_groups) | ||
if req_n == 0 and usr_n == 0 then return nil end | if req_n == 0 and usr_n == 0 then return nil end | ||
local row = mw.html.create("div"):addClass("sv-reqrow") | local row = mw.html.create("div"):addClass("sv-reqrow") | ||
if req_box then row:node(req_box) end | if req_box then row:node(req_box) end | ||
if usr_box then row:node(usr_box) end | if usr_box then row:node(usr_box) end | ||
return row | return row | ||
end | end | ||
local function _build_req_users(root_id, requirements, users) | local function _build_req_users(root_id, requirements, users) | ||
requirements = _safe_tbl(requirements) | requirements = _safe_tbl(requirements) | ||
users = _safe_tbl(users) | users = _safe_tbl(users) | ||
local req_box, req_n = _build_grouped_disclose( | local req_box, req_n = _build_grouped_disclose( | ||
root_id, "req", "Requirements", _safe_tbl(requirements.groups) | root_id, "req", "Requirements", _safe_tbl(requirements.groups) | ||
) | ) | ||
local usr_box, usr_n = _build_grouped_disclose( | local usr_box, usr_n = _build_grouped_disclose( | ||
root_id, "usr", "Users", _safe_tbl(users.groups) | root_id, "usr", "Users", _safe_tbl(users.groups) | ||
) | ) | ||
if req_n == 0 and usr_n == 0 then return nil end | if req_n == 0 and usr_n == 0 then return nil end | ||
local row = mw.html.create("div"):addClass("sv-reqrow") | local row = mw.html.create("div"):addClass("sv-reqrow") | ||
if req_box then row:node(req_box) end | if req_box then row:node(req_box) end | ||
if usr_box then row:node(usr_box) end | if usr_box then row:node(usr_box) end | ||
return row | return row | ||
end | |||
local function _append_meta_req_bubble(wrap, mod, node) | |||
if not node then return false end | |||
wrap:tag("div") | |||
:addClass("sv-skill-meta-block__bubble") | |||
:addClass("sv-skill-meta-block__bubble--" .. mod) | |||
:node(node) | |||
return true | |||
end | |||
local function _build_meta_req_block(meta_node, req_node) | |||
if not meta_node and not req_node then return nil end | |||
local wrap = mw.html.create("div") | |||
:addClass("sv-skill-meta-block") | |||
local count = 0 | |||
if _append_meta_req_bubble(wrap, "meta", meta_node) then count = count + 1 end | |||
if _append_meta_req_bubble(wrap, "req", req_node) then count = count + 1 end | |||
if count == 1 then | |||
wrap:addClass("sv-skill-meta-block--single") | |||
end | |||
return wrap | |||
end | end | ||
| Line 1,592: | Line 1,667: | ||
box.top:node(_build_identity(frame, root_id, identity, notes_wt)) | box.top:node(_build_identity(frame, root_id, identity, notes_wt)) | ||
local meta_block = _build_meta_row(frame, payload.meta_row) | |||
local reqrow = _build_req_users(root_id, payload.requirements, payload.users) | local reqrow = _build_req_users(root_id, payload.requirements, payload.users) | ||
if | local meta_req_block = _build_meta_req_block(meta_block, reqrow) | ||
if meta_req_block then box.top:node(meta_req_block) end | |||
local level_panel, actual_default = _build_level(root_id, default_level, max_level, 1, nil) | local level_panel, actual_default = _build_level(root_id, default_level, max_level, 1, nil) | ||
box.bottom:node(level_panel) | box.bottom:node(level_panel) | ||
box.bottom:node(_build_scaling_top(frame, payload.scaling_top, actual_default)) | box.bottom:node(_build_scaling_top(frame, payload.scaling_top, actual_default)) | ||
| Line 1,642: | Line 1,717: | ||
box.top:node(_build_identity(frame, root_id, identity, notes_wt)) | box.top:node(_build_identity(frame, root_id, identity, notes_wt)) | ||
local meta_block = _build_meta_row_native(frame, payload.skill_meta) | |||
local reqrow = _build_req_users_native(root_id, payload.requirements, payload.users) | local reqrow = _build_req_users_native(root_id, payload.requirements, payload.users) | ||
if | local meta_req_block = _build_meta_req_block(meta_block, reqrow) | ||
if meta_req_block then box.top:node(meta_req_block) end | |||
local scaling_block, actual_default = _build_skill_scaling_native( | local scaling_block, actual_default = _build_skill_scaling_native( | ||