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 |
||
| (15 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 34: | Line 18: | ||
local _STAT_ORDER = { "str", "vit", "dex", "agi", "int", "luk" } | local _STAT_ORDER = { "str", "vit", "dex", "agi", "int", "luk" } | ||
local _STAT_GRID_ORDER = { "str", "agi", "vit", "int", "dex", "luk" } | |||
local _STAT_LABELS = { | local _STAT_LABELS = { | ||
| Line 60: | Line 45: | ||
["stat|int"] = "int", | ["stat|int"] = "int", | ||
["stat|luk"] = "luk", | ["stat|luk"] = "luk", | ||
} | |||
local _STAT_ICON_FILES = { | |||
str = "File:Strength.png", | |||
vit = "File:Vitality.png", | |||
dex = "File:Dexterity.png", | |||
agi = "File:Agility.png", | |||
int = "File:Intelligence.png", | |||
luk = "File:Luck.png", | |||
} | } | ||
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 78: | 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 96: | 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 | ||
local function _err(msg, preview) | local function _err(msg, preview) | ||
local box = mw.html.create("div"):addClass("sv-gi-error") | local box = mw.html.create("div") | ||
:addClass("sv-card") | |||
:addClass("sv-gi-error") | |||
box:wikitext("GameInfo/Skills error: " .. tostring(msg)) | box:wikitext("GameInfo/Skills error: " .. tostring(msg)) | ||
if preview and preview ~= "" then | if preview and preview ~= "" then | ||
| Line 218: | Line 226: | ||
end | end | ||
local function | local function _split_def_token(s) | ||
s = _trim(s) | s = _trim(s) | ||
if s == "" then return nil end | if s == "" then return nil, nil end | ||
local bar = s:find("|", 1, true) | local bar = s:find("|", 1, true) | ||
if not bar then | if not bar then return nil, nil end | ||
local domain = _trim(s:sub(1, bar - 1)) | local domain = _trim(s:sub(1, bar - 1)) | ||
local key | local key = _trim(s:sub(bar + 1)) | ||
if domain == "" or key == "" then | if domain == "" or key == "" then return nil, nil end | ||
return domain, key | |||
end | |||
local function _render_def_token(frame, s, noicon, extra_args) | |||
s = _trim(s) | |||
if s == "" then return nil end | |||
local domain, key = _split_def_token(s) | |||
if not domain or not key then | |||
return mw.text.nowiki(s) | return mw.text.nowiki(s) | ||
end | end | ||
| Line 246: | 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 266: | 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 wt = _render_def_token(frame, token, true, { pill = "1", fill = "1" }) | |||
if not wt then return false end | |||
parent:tag("span") | |||
:addClass(class_name) | |||
:attr("aria-hidden", "true") | |||
:wikitext(wt) | |||
return | return true | ||
end | end | ||
| Line 439: | Line 467: | ||
:addClass("sv-tip-btn") | :addClass("sv-tip-btn") | ||
:addClass("sv-tip-btn--icon") | :addClass("sv-tip-btn--icon") | ||
:addClass("sv-hover-lift") | |||
:attr("role", "button") | :attr("role", "button") | ||
:attr("tabindex", "0") | :attr("tabindex", "0") | ||
| Line 533: | Line 562: | ||
:addClass("sv-tip-btn--pill") | :addClass("sv-tip-btn--pill") | ||
:addClass("sv-overcap-btn") | :addClass("sv-overcap-btn") | ||
:addClass("sv-hover-lift") | |||
:attr("role", "button") | :attr("role", "button") | ||
:attr("tabindex", "0") | :attr("tabindex", "0") | ||
| Line 587: | Line 617: | ||
local s = _trim(it) | local s = _trim(it) | ||
if s == "" then return nil end | if s == "" then return nil end | ||
return { text = s } | return { text = s, page = s } | ||
end | end | ||
| Line 596: | Line 626: | ||
if text == "" then return nil end | if text == "" then return nil end | ||
local page = _safe_str(it.page or "", "") | local page = _safe_str(it.page or "", "") | ||
if page == "" then page = text end | |||
local suffix = _safe_str(it.suffix or "", "") | local suffix = _safe_str(it.suffix or "", "") | ||
return { text = text, page = (page ~= "" and page or nil), suffix = (suffix ~= "" and suffix or nil) } | return { text = text, page = (page ~= "" and page or nil), suffix = (suffix ~= "" and suffix or nil) } | ||
| Line 647: | Line 678: | ||
:addClass("sv-disclose-btn") | :addClass("sv-disclose-btn") | ||
:addClass("sv-disclose-btn--compact") | :addClass("sv-disclose-btn--compact") | ||
:addClass("sv-hover-lift") | |||
:attr("role", "button") | :attr("role", "button") | ||
:attr("tabindex", "0") | :attr("tabindex", "0") | ||
| Line 726: | Line 758: | ||
local root = mw.html.create("div"):addClass("sv-skill-head") | local root = mw.html.create("div"):addClass("sv-skill-head") | ||
local icon_div = root:tag("div"):addClass("sv-skill-icon") | local icon_div = root:tag("div") | ||
:addClass("sv-skill-icon") | |||
:addClass("sv-tile") | |||
local sprite = _safe_tbl(identity.sprite) | local sprite = _safe_tbl(identity.sprite) | ||
local sprite_page = _safe_str(sprite.page, "") | local sprite_page = _safe_str(sprite.page, "") | ||
| Line 752: | Line 787: | ||
return root | 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") | |||
local icon_page = _safe_str(spec.icon_page, "") | |||
local icon_alt = _safe_str(spec.icon_alt, "") | |||
if icon_page ~= "" then | |||
icon_div:node(_render_file_image(icon_page, icon_alt, 24)) | |||
else | |||
icon_div:node(_question_badge()) | |||
end | |||
local wrap = card:tag("div"):addClass("sv-meta-textwrap") | |||
local text = wrap:tag("div"):addClass("sv-meta-text") | |||
local lines = _safe_tbl(spec.lines) | |||
if #lines > 0 then | |||
text:node(_meta_lines_def(frame, lines)) | |||
else | |||
text:wikitext("—") | |||
end | |||
_build_hitbox(card, "sv-meta-hit", frame, spec.hit_tok) | |||
end | end | ||
| Line 760: | 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 791: | 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 805: | Line 851: | ||
end | end | ||
local function | local function _build_meta_row_native(frame, skill_meta) | ||
skill_meta = _safe_tbl(skill_meta) | skill_meta = _safe_tbl(skill_meta) | ||
| Line 811: | 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, "") | |||
local | |||
if #lines == 0 and hit_tok ~= "" then | |||
lines = { hit_tok } | |||
if # | |||
end | end | ||
_build_meta_card(frame, meta, { | |||
icon_page = _safe_str(cell.icon, ""), | |||
icon_alt = "", | |||
lines = lines, | |||
hit_tok = hit_tok, | |||
}) | |||
end | end | ||
| Line 868: | Line 892: | ||
end | end | ||
local function | local function _normalize_level_native(level_obj) | ||
level_obj = _safe_tbl(level_obj) | level_obj = _safe_tbl(level_obj) | ||
| Line 896: | Line 920: | ||
local root = mw.html.create("div") | local root = mw.html.create("div") | ||
:addClass("sv-card") | |||
:addClass("sv-skill-level") | :addClass("sv-skill-level") | ||
:attr("data-sv-level-boundary", "1") | :attr("data-sv-level-boundary", "1") | ||
| Line 947: | Line 972: | ||
scaling = _safe_tbl(scaling) | scaling = _safe_tbl(scaling) | ||
local root = mw.html.create("div"):addClass("sv-skill-scaling") | local root = mw.html.create("div") | ||
:addClass("sv-card") | |||
:addClass("sv-skill-scaling") | |||
local row = root:tag("div"):addClass("sv-scaling-row") | local row = root:tag("div"):addClass("sv-scaling-row") | ||
local grid = row:tag("div"):addClass("sv-scaling-grid") | local grid = row:tag("div"):addClass("sv-scaling-grid") | ||
| Line 967: | Line 995: | ||
local val = _safe_str(it.value or "", "") | local val = _safe_str(it.value or "", "") | ||
if tok ~= "" and val ~= "" then | if tok ~= "" and val ~= "" then | ||
local item = list:tag("div"):addClass("sv-scaling-item sv-scaling-item--stat") | local item = list:tag("div") | ||
:addClass("sv-tile") | |||
:addClass("sv-scaling-item") | |||
:addClass("sv-scaling-item--stat") | |||
local wt = _render_def_token(frame, tok, false) or mw.text.nowiki(tok) | local wt = _render_def_token(frame, tok, false) or mw.text.nowiki(tok) | ||
item:tag("span"):addClass("sv-scale-stat"):wikitext(wt) | item:tag("span"):addClass("sv-scale-stat"):wikitext(wt) | ||
| Line 975: | 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 986: | Line 1,019: | ||
core_stats = _safe_tbl(core_stats) | core_stats = _safe_tbl(core_stats) | ||
local root = mw.html.create("div"):addClass("sv-skill-core") | local root = mw.html.create("div") | ||
:addClass("sv-card") | |||
:addClass("sv-skill-core") | |||
local row = root:tag("div"):addClass("sv-core-row") | local row = root:tag("div"):addClass("sv-core-row") | ||
local grid = row:tag("div"):addClass("sv-core-grid") | local grid = row:tag("div"):addClass("sv-core-grid") | ||
| Line 993: | Line 1,029: | ||
cell = _safe_tbl(cell) | cell = _safe_tbl(cell) | ||
local c = grid:tag("div"):addClass("sv-core-cell") | local c = grid:tag("div") | ||
:addClass("sv-core-cell") | |||
:addClass("sv-tile") | |||
local top = c:tag("div"):addClass("sv-core-top") | local top = c:tag("div"):addClass("sv-core-top") | ||
| Line 1,015: | Line 1,053: | ||
local label = _STAT_LABELS[stat_key] or stat_key:upper() | local label = _STAT_LABELS[stat_key] or stat_key:upper() | ||
local def_tok = _STAT_DEF_TOKENS[stat_key] or label | local def_tok = _STAT_DEF_TOKENS[stat_key] or label | ||
local icon_file = _STAT_ICON_FILES[stat_key] | |||
local active = _value_has_content(raw_val) | local active = _value_has_content(raw_val) | ||
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,023: | Line 1,064: | ||
:addClass("sv_skill_scaling__stat-pill--" .. stat_key) | :addClass("sv_skill_scaling__stat-pill--" .. stat_key) | ||
:attr("data-stat-key", stat_key) | :attr("data-stat-key", stat_key) | ||
:attr("aria-label", label) | |||
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") | ||
local visual = main:tag("span"):addClass("sv_skill_scaling__stat-visual") | |||
if active then | if active then | ||
local | local icon = visual:tag("span"):addClass("sv_skill_scaling__stat-icon") | ||
if icon_file and icon_file ~= "" then | |||
icon:node(_render_file_image(icon_file, label, 14)) | |||
else | |||
icon:node(_question_badge()) | |||
end | |||
: | |||
: | |||
end | end | ||
local val = | local val = visual:tag("span"):addClass("sv_skill_scaling__stat-value") | ||
_apply_value(val, raw_val, level) | _apply_value(val, raw_val, level) | ||
if active then | |||
_build_hitbox(pill, "sv_skill_scaling__stat-hit", frame, def_tok) | |||
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") | ||
:addClass("sv-tile") | |||
:addClass("sv_skill_scaling__pill") | :addClass("sv_skill_scaling__pill") | ||
:addClass("sv_skill_scaling__pill--core") | :addClass("sv_skill_scaling__pill--core") | ||
: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,069: | 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 | ||
local function | local function _build_skill_scaling_native(frame, root_id, skill_scaling, default_level, max_level, min_level, overcap_obj) | ||
skill_scaling = _safe_tbl(skill_scaling) | skill_scaling = _safe_tbl(skill_scaling) | ||
local root = mw.html.create("div"):addClass("sv_skill_scaling") | local root = mw.html.create("div") | ||
:addClass("sv-card") | |||
:addClass("sv_skill_scaling") | |||
local level_panel, actual_default = _build_level( | local level_panel, actual_default = _build_level( | ||
| Line 1,094: | Line 1,149: | ||
do | do | ||
local primary = | local primary_group = body:tag("div") | ||
:addClass("sv_skill_scaling__column") | :addClass("sv_skill_scaling__group") | ||
:addClass("sv_skill_scaling__group--primary-stats") | |||
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 | |||
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 | ||
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,130: | 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 | ||
return root, actual_default | return root, actual_default | ||
end | |||
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 has_def = _has_pipe(spec.label_tok) | |||
local has_value = _value_has_content(spec.value) | |||
local pill = parent:tag("div") | |||
:addClass("sv-tile") | |||
:addClass("sv-mech-mod-pill") | |||
:attr("aria-label", spec.label_text) | |||
if has_value then | |||
pill:addClass("is-active") | |||
else | |||
pill:addClass("is-inactive") | |||
end | |||
if has_def or has_value then | |||
pill:addClass("sv-hover-lift") | |||
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 | |||
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 | |||
return { | |||
label_tok = tok, | |||
label_text = label, | |||
} | |||
end | |||
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 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 | end | ||
| Line 1,141: | Line 1,304: | ||
local mechanics = _safe_tbl(tabs_obj.mechanics) | local mechanics = _safe_tbl(tabs_obj.mechanics) | ||
local effects = _safe_tbl(tabs_obj.effects) | 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) | local effects_cards = _safe_tbl(effects.cards) | ||
local | if #mechanics_mods == 0 then | ||
local | 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 | |||
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 | |||
effects_cards = merged | |||
end | |||
local mechanics_count = _to_int( | |||
mechanics.count, | |||
#mechanics_mods + #mechanics_keywords | |||
) | |||
local effects_count = _to_int( | |||
effects.count, | |||
#effects_cards | |||
) | |||
local tab_specs = { | local tab_specs = { | ||
{ key = "mechanics", title = "Mechanics" | { key = "mechanics", title = "Mechanics (" .. tostring(mechanics_count) .. ")" }, | ||
{ key = "effects", title = "Effects (" .. tostring(effects_count) .. ")" }, | { key = "effects", title = "Effects (" .. tostring(effects_count) .. ")" }, | ||
} | } | ||
| Line 1,180: | Line 1,368: | ||
end | end | ||
local function render_ref_icon(parent, icon) | |||
icon = _safe_tbl(icon) | |||
local page = _safe_str(icon.page, "") | |||
local alt = _safe_str(icon.alt, "") | |||
if page ~= "" then | |||
parent:node(_render_file_image(page, alt, 52)) | |||
local | 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 | end | ||
return | |||
end | end | ||
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 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 | local function render_ref_card(parent, card) | ||
card = _safe_tbl(card) | card = _safe_tbl(card) | ||
local kind = _safe_str(card.kind, "") | |||
local title = _safe_str(card.title, "—") | local title = _safe_str(card.title, "—") | ||
local page = _safe_str(card.page, "") | local page = _safe_str(card.page, "") | ||
local icon = _safe_tbl(card.icon) | local icon = _safe_tbl(card.icon) | ||
local container = parent:tag("div"):addClass("sv-ref-card") | 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 | |||
local ico = container:tag("div"):addClass("sv-ref-ico") | local ico = container:tag("div") | ||
:addClass("sv-ref-ico") | |||
:addClass("sv-tile") | |||
render_ref_icon(ico, icon) | render_ref_icon(ico, icon) | ||
local text = container:tag("div"):addClass("sv-ref-text") | local text = container:tag("div"):addClass("sv-ref-text") | ||
local title_div = text:tag("div"):addClass("sv-ref-title") | local title_div = text:tag("div"):addClass("sv-ref-title") | ||
if page ~= "" then | if page ~= "" then | ||
| Line 1,240: | Line 1,445: | ||
end | end | ||
render_effect_stats(text, card.stats, kind) | |||
end | end | ||
| Line 1,258: | Line 1,451: | ||
local panel = panels:tag("div") | local panel = panels:tag("div") | ||
:addClass("sv-tabpanel") | :addClass("sv-tabpanel") | ||
:addClass("sv-mech-panel") | |||
:attr("role", "tabpanel") | :attr("role", "tabpanel") | ||
:attr("data-panel", " | :attr("data-panel", "mechanics") | ||
: | |||
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 | |||
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 | ||
| Line 1,271: | Line 1,494: | ||
local panel = panels:tag("div") | local panel = panels:tag("div") | ||
:addClass("sv-tabpanel") | :addClass("sv-tabpanel") | ||
:addClass("sv-hidden") | |||
:attr("role", "tabpanel") | :attr("role", "tabpanel") | ||
:attr("data-panel", " | :attr("data-panel", "effects") | ||
:attr("hidden", "hidden") | :attr("hidden", "hidden") | ||
local grid = panel:tag("div"):addClass("sv-ref-grid") | 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 | ||
| Line 1,348: | Line 1,579: | ||
end | end | ||
local function | 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 req_groups = _map_to_groups(requirements_map, { "Skills", "Weapon Types", "Stances" }) | ||
local usr_groups = _map_to_groups(users_map, { "Classes", "Summons" }) | 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) | ||
| Line 1,382: | Line 1,613: | ||
end | end | ||
local function | 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 | |||
local function _render_legacy(frame, payload, notes_wt) | |||
local identity = _safe_tbl(payload.identity) | local identity = _safe_tbl(payload.identity) | ||
local display_name = _safe_str(identity.display_name, "Unknown Skill") | local display_name = _safe_str(identity.display_name, "Unknown Skill") | ||
| Line 1,408: | 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,423: | Line 1,682: | ||
end | end | ||
local function | local function _render_native(frame, payload, notes_wt) | ||
local identity = _safe_tbl(payload.identity) | local identity = _safe_tbl(payload.identity) | ||
local display_name = _safe_str(identity.display_name, "Unknown Skill") | local display_name = _safe_str(identity.display_name, "Unknown Skill") | ||
| Line 1,435: | Line 1,694: | ||
end | end | ||
local min_level, default_level, max_level = | local min_level, default_level, max_level = _normalize_level_native(payload.level) | ||
local box = GI.box({ | local box = GI.box({ | ||
| Line 1,458: | 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 reqrow = | local meta_block = _build_meta_row_native(frame, payload.skill_meta) | ||
if | local reqrow = _build_req_users_native(root_id, payload.requirements, payload.users) | ||
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 = | local scaling_block, actual_default = _build_skill_scaling_native( | ||
frame, | frame, | ||
root_id, | root_id, | ||
| Line 1,490: | Line 1,750: | ||
if payload.schema == 2 then | if payload.schema == 2 then | ||
return | return _render_native(frame, payload, notes_wt) | ||
end | end | ||
return | return _render_legacy(frame, payload, notes_wt) | ||
end | end | ||
return p | return p | ||