Mudanças entre as edições de "Módulo:Character"
Ir para navegação
Ir para pesquisar
(foi adicionado os 3 idiomas (es,en,pl) pro widget de translator) |
m (primeiro passo p add o char transl) |
||
| Linha 1: | Linha 1: | ||
-- Module:Character | |||
-- Module:Character | |||
local p = {} | local p = {} | ||
| Linha 14: | Linha 13: | ||
end | end | ||
local ok, data | local ok, data | ||
ok, data = pcall(function() return require("Módulo:" .. char) end) | ok, data = pcall(function() | ||
if ok and type(data) == "table" then return data end | return require("Módulo:" .. char) | ||
ok, data = pcall(function() return require("Modulo:" .. char) end) | end) | ||
if ok and type(data) == "table" then return data end | if ok and type(data) == "table" then | ||
ok, data = pcall(function() return require("Module:" .. char) end) | return data | ||
if ok and type(data) == "table" then return data end | end | ||
ok, data = pcall(function() | |||
return require("Modulo:" .. char) | |||
end) | |||
if ok and type(data) == "table" then | |||
return data | |||
end | |||
ok, data = pcall(function() | |||
return require("Module:" .. char) | |||
end) | |||
if ok and type(data) == "table" then | |||
return data | |||
end | |||
return nil | return nil | ||
end | end | ||
| Linha 27: | Linha 38: | ||
local ATTR_I18N = { | local ATTR_I18N = { | ||
pt = { | pt = { | ||
cooldown | cooldown = "Recarga", | ||
energy | energy = "Energia", | ||
energy_gain = "Ganho de energia", | energy_gain = "Ganho de energia", | ||
energy_cost = "Custo de energia", | energy_cost = "Custo de energia", | ||
power | power = "Poder", | ||
power_pvp | power_pvp = "Poder PvP", | ||
level | level = "Nível" | ||
}, | }, | ||
en = { | en = { | ||
cooldown | cooldown = "Cooldown", | ||
energy | energy = "Energy", | ||
energy_gain = "Energy Gain", | energy_gain = "Energy Gain", | ||
energy_cost = "Energy Cost", | energy_cost = "Energy Cost", | ||
power | power = "Power", | ||
power_pvp | power_pvp = "PvP Power", | ||
level | level = "Level" | ||
}, | }, | ||
es = { | es = { | ||
cooldown | cooldown = "Recarga", | ||
energy | energy = "Energía", | ||
energy_gain = "Ganancia de energía", | energy_gain = "Ganancia de energía", | ||
energy_cost = "Costo de energía", | energy_cost = "Costo de energía", | ||
power | power = "Poder", | ||
power_pvp | power_pvp = "Poder PvP", | ||
level | level = "Nivel" | ||
}, | }, | ||
pl = { | pl = { | ||
cooldown | cooldown = "Czas odnowienia", | ||
energy | energy = "Energia", | ||
energy_gain = "Przyrost energii", | energy_gain = "Przyrost energii", | ||
energy_cost = "Koszt energii", | energy_cost = "Koszt energii", | ||
power | power = "Moc", | ||
power_pvp | power_pvp = "Moc PvP", | ||
level | level = "Poziom" | ||
} | |||
} | |||
-- ===== I18N: labels das abas (Skills/Skins) ===== | |||
local TAB_I18N = { | |||
pt = { | |||
skills = "Habilidades", | |||
skins = "Skins", | |||
skins_title = "SKINS & SPOTLIGHTS" | |||
}, | }, | ||
en = { | |||
skills = "Skills", | |||
skins = "Skins", | |||
skins_title = "SKINS & SPOTLIGHTS" | |||
}, | |||
es = { | |||
skills = "Habilidades", | |||
skins = "Aspectos", | |||
skins_title = "ASPECTOS Y DESTACADOS" | |||
}, | |||
pl = { | |||
skills = "Umiejętności", | |||
skins = "Skórki", | |||
skins_title = "SKÓRKI I PREZENTACJE" | |||
} | |||
} | } | ||
| Linha 68: | Linha 102: | ||
local a = frame.args | local a = frame.args | ||
local obj = { | local obj = { | ||
sprite | sprite = a.sprite or '', | ||
background = a.background or '', | background = a.background or '', | ||
tooltip | tooltip = a.tooltip or '', | ||
youtube | youtube = a.youtube or '', | ||
source | source = a.source or '' | ||
} | } | ||
return mw.text.jsonEncode(obj) | return mw.text.jsonEncode(obj) | ||
| Linha 89: | Linha 123: | ||
-- ===== tier / classes (auto-resolve) ===== | -- ===== tier / classes (auto-resolve) ===== | ||
local rawTier | local rawTier = trim(frame:preprocess(args.tier or "")) | ||
local rawClasse = trim(frame:preprocess(args.classe or "")) | local rawClasse = trim(frame:preprocess(args.classe or "")) | ||
| Linha 98: | Linha 132: | ||
if rawClasse ~= "" and mw.ustring.lower(rawClasse):sub(1, 5) == "class" then | if rawClasse ~= "" and mw.ustring.lower(rawClasse):sub(1, 5) == "class" then | ||
local data = requireCharModule(nomeChar) or {} | local data = requireCharModule(nomeChar) or {} | ||
local arr | local arr = (data.tags_i18n and data.tags_i18n.pt) or data.tags | ||
if type(arr) == "table" then rawClasse = table.concat(arr, " / ") end | if type(arr) == "table" then | ||
rawClasse = table.concat(arr, " / ") | |||
end | |||
end | end | ||
| Linha 109: | Linha 145: | ||
if rawClasse == "" then | if rawClasse == "" then | ||
local arr = (data.tags_i18n and data.tags_i18n.pt) or data.tags | local arr = (data.tags_i18n and data.tags_i18n.pt) or data.tags | ||
if type(arr) == "table" then rawClasse = table.concat(arr, " / ") | if type(arr) == "table" then | ||
else rawClasse = arr or "" end | rawClasse = table.concat(arr, " / ") | ||
else | |||
rawClasse = arr or "" | |||
end | |||
end | end | ||
end | end | ||
| Linha 118: | Linha 157: | ||
local tierKey = mw.ustring.lower(tierRaw or "") | local tierKey = mw.ustring.lower(tierRaw or "") | ||
local tierMap = { | local tierMap = { | ||
bronze = "tier-bronze", bronce = "tier-bronze", | bronze = "tier-bronze", | ||
silver = "tier-silver", prata | bronce = "tier-bronze", | ||
gold | silver = "tier-silver", | ||
diamond = "tier-diamond", diamante = "tier-diamond" | prata = "tier-silver", | ||
gold = "tier-gold", | |||
ouro = "tier-gold", | |||
diamond = "tier-diamond", | |||
diamante = "tier-diamond" | |||
} | } | ||
local tierClass = tierMap[tierKey] | local tierClass = tierMap[tierKey] | ||
| Linha 138: | Linha 181: | ||
if url and url ~= "" then | if url and url ~= "" then | ||
box:attr('data-bg-file', raw) | box:attr('data-bg-file', raw) | ||
box:attr('data-bg-url', | box:attr('data-bg-url', url) | ||
end | end | ||
end | end | ||
end | end | ||
local function trim2(s2) return (s2 or ''):match('^%s*(.-)%s*$') end | local function trim2(s2) | ||
return (s2 or ''):match('^%s*(.-)%s*$') | |||
end | |||
local function fileURL2(name) | local function fileURL2(name) | ||
if not name or name == '' then return nil end | if not name or name == '' then | ||
return nil | |||
end | |||
name = name:gsub('^Arquivo:', ''):gsub('^File:', '') | name = name:gsub('^Arquivo:', ''):gsub('^File:', '') | ||
return mw.title.new('Special:FilePath/' .. name):fullUrl() | return mw.title.new('Special:FilePath/' .. name):fullUrl() | ||
| Linha 150: | Linha 197: | ||
do | do | ||
local bgParam = trim2(args.background or '') | local bgParam = trim2(args.background or '') | ||
if bgParam == '' then bgParam = trim2(args.banner or '') end | if bgParam == '' then | ||
bgParam = trim2(args.banner or '') | |||
end | |||
if bgParam ~= '' then | if bgParam ~= '' then | ||
local url = fileURL2(bgParam) | local url = fileURL2(bgParam) | ||
if url then | if url then | ||
local style = string.format("--character-bg: url('%s'); background-image: url('%s') !important;", url, url) | local style = string.format("--character-bg: url('%s'); background-image: url('%s') !important;", url, | ||
url) | |||
local old = box:getAttr('style') | local old = box:getAttr('style') | ||
box:attr('style', old and (old .. ' ' .. style) or style) | box:attr('style', old and (old .. ' ' .. style) or style) | ||
| Linha 168: | Linha 218: | ||
box:attr('style', style) | box:attr('style', style) | ||
end | end | ||
if tierClass and tierClass ~= "" then box:addClass(tierClass) end | if tierClass and tierClass ~= "" then | ||
box:addClass(tierClass) | |||
end | |||
-- ===== header / topbar ===== | -- ===== header / topbar ===== | ||
| Linha 182: | Linha 234: | ||
local classTags = nameGroup:tag('div'):addClass('class-tags') | local classTags = nameGroup:tag('div'):addClass('class-tags') | ||
if tierRaw and tierRaw ~= "" then classTags:tag('div'):addClass('class-tag tier'):wikitext(tierRaw) end | if tierRaw and tierRaw ~= "" then | ||
classTags:tag('div'):addClass('class-tag tier'):wikitext(tierRaw) | |||
end | |||
local classeString = rawClasse or "" | local classeString = rawClasse or "" | ||
for classe in mw.text.gsplit(classeString, '/', true) do | for classe in mw.text.gsplit(classeString, '/', true) do | ||
local clean = mw.text.trim(classe or '') | local clean = mw.text.trim(classe or '') | ||
if clean ~= '' then classTags:tag('div'):addClass('class-tag'):wikitext(clean) end | if clean ~= '' then | ||
classTags:tag('div'):addClass('class-tag'):wikitext(clean) | |||
end | |||
end | end | ||
header:tag('div'):addClass('topbar-description'):wikitext(args.desc or '') | header:tag('div'):addClass('topbar-description'):wikitext(args.desc or '') | ||
local rawLang = trim(args.lang or "pt") | |||
rawLang = mw.ustring.lower(rawLang) | |||
local base = rawLang:match("^([a-z][a-z])") or rawLang | |||
local TAB = TAB_I18N[rawLang] or TAB_I18N[base] or TAB_I18N.pt | |||
local tabs = header:tag('div'):addClass('character-tabs') | local tabs = header:tag('div'):addClass('character-tabs') | ||
tabs:tag('div'):addClass('tab-btn active'):attr('data-tab', 'skills'):wikitext( | tabs:tag('div'):addClass('tab-btn active'):attr('data-tab', 'skills'):wikitext(TAB.skills) | ||
tabs:tag('div'):addClass('tab-btn'):attr('data-tab', 'skins'):wikitext( | tabs:tag('div'):addClass('tab-btn'):attr('data-tab', 'skins'):wikitext(TAB.skins) | ||
-- ===== SKILLS ===== | -- ===== SKILLS ===== | ||
local skillsTab = box:tag('div'):addClass('tab-content active'):attr('id', 'skills') | local skillsTab = box:tag('div'):addClass('tab-content active'):attr('id', 'skills') | ||
local iconBar | local iconBar = skillsTab:tag('div'):addClass('icon-bar') | ||
local skillsContainer = skillsTab:tag('div'):addClass('skills-container') | local skillsContainer = skillsTab:tag('div'):addClass('skills-container') | ||
local details | local details = skillsContainer:tag('div'):addClass('skills-details') | ||
local descBox | local descBox = details:tag('div'):addClass('desc-box') | ||
-- I18N: injeta o mapa inteiro para o JS usar (botão de idioma) | -- I18N: injeta o mapa inteiro para o JS usar (botão de idioma) | ||
skillsTab:attr('data-i18n-attrs', mw.text.jsonEncode(ATTR_I18N)) | skillsTab:attr('data-i18n-attrs', mw.text.jsonEncode(ATTR_I18N)) | ||
if args.lang and trim(args.lang) ~= "" then | if args.lang and trim(args.lang) ~= "" then | ||
skillsTab:attr('data-i18n-default', mw.ustring.lower(trim(args.lang))) | skillsTab:attr('data-i18n-default', mw.ustring.lower(trim(args.lang))) | ||
| Linha 214: | Linha 274: | ||
if ok and type(sk) == "table" and (sk.name or sk.nome) and (sk.name ~= '' or sk.nome ~= '') then | if ok and type(sk) == "table" and (sk.name or sk.nome) and (sk.name ~= '' or sk.nome ~= '') then | ||
idx = idx + 1 | idx = idx + 1 | ||
local name | local name = sk.name or sk.nome or '' | ||
local icon | local icon = sk.icon or '' | ||
local desc | local desc = sk.desc or '' | ||
local level = tostring(sk.level or '') | local level = tostring(sk.level or '') | ||
-- ordem esperada pelo JS: PVE, PVP, Energia, Recarga | -- ordem esperada pelo JS: PVE, PVP, Energia, Recarga | ||
local attrs = table.concat({ | local attrs = table.concat({sk.powerpve or '-', sk.powerpvp or '-', sk.energy or '-', sk.cooldown or '-'}, | ||
", ") | |||
local videoURL = (sk.video and sk.video ~= '') and fileURL(sk.video) or '' | local videoURL = (sk.video and sk.video ~= '') and fileURL(sk.video) or '' | ||
local iconWrap = iconBar:tag('div') | local iconWrap = iconBar:tag('div'):addClass('skill-icon'):attr('data-index', idx):attr('data-nome', name) | ||
:attr('data-desc', desc):attr('data-atr', attrs):attr('data-video', videoURL):attr('data-video-preload', | |||
'auto') | |||
:attr('data-desc', | |||
-- aplica data-level quando houver (e não for placeholder) | -- aplica data-level quando houver (e não for placeholder) | ||
| Linha 251: | Linha 305: | ||
local skinsTab = box:tag('div'):addClass('tab-content'):attr('id', 'skins') | local skinsTab = box:tag('div'):addClass('tab-content'):attr('id', 'skins') | ||
local cardSkins = skinsTab:tag('div'):addClass('card-skins') | local cardSkins = skinsTab:tag('div'):addClass('card-skins') | ||
cardSkins:tag('span'):addClass('card-skins-title'):wikitext( | cardSkins:tag('span'):addClass('card-skins-title'):wikitext(TAB.skins_title) | ||
local wrapper | local wrapper = cardSkins:tag('div'):addClass('skins-carousel-wrapper') | ||
wrapper:tag('div'):addClass('skins-arrow left'):wikitext('«') | wrapper:tag('div'):addClass('skins-arrow left'):wikitext('«') | ||
local carousel = wrapper:tag('div'):addClass('skins-carousel') | local carousel = wrapper:tag('div'):addClass('skins-carousel') | ||
| Linha 261: | Linha 315: | ||
local ok, sk = pcall(mw.text.jsonDecode, obj) | local ok, sk = pcall(mw.text.jsonDecode, obj) | ||
if ok and type(sk) == "table" then | if ok and type(sk) == "table" then | ||
local bannerFile | local bannerFile = sk.background or '' | ||
local imageFile | local imageFile = sk.sprite or '' | ||
local tooltipRaw | local tooltipRaw = sk.tooltip or '' | ||
local tooltipHtml = tooltipRaw:gsub("'''([^']+)'''", "<b>%1</b>"):gsub("\n", "<br>") | local tooltipHtml = tooltipRaw:gsub("'''([^']+)'''", "<b>%1</b>"):gsub("\n", "<br>") | ||
| Linha 269: | Linha 323: | ||
local yt = (sk.youtube or '') | local yt = (sk.youtube or '') | ||
if yt ~= '' then skinCard:attr('data-youtube', yt):addClass('is-clickable') end | if yt ~= '' then | ||
skinCard:attr('data-youtube', yt):addClass('is-clickable') | |||
end | |||
skinCard:tag('div'):addClass('skin-banner') | skinCard:tag('div'):addClass('skin-banner'):wikitext(bannerFile ~= '' and | ||
string.format('[[Arquivo:%s|link=]]', bannerFile) or | |||
''):attr('alt', 'banner') | |||
skinCard:tag('div'):addClass('skin-sprite') | skinCard:tag('div'):addClass('skin-sprite'):wikitext(imageFile ~= '' and | ||
string.format('[[Arquivo:%s|link=]]', imageFile) or | |||
''):attr('alt', 'skin') | |||
end | end | ||
end | end | ||
Edição das 03h28min de 10 de setembro de 2025
A documentação para este módulo pode ser criada em Módulo:Character/doc
-- Module:Character
local p = {}
-- ===== util =====
local function trim(s)
return (tostring(s or ""):gsub("^%s+", ""):gsub("%s+$", ""))
end
local function requireCharModule(char)
char = trim(char or "")
if char == "" then
return nil
end
local ok, data
ok, data = pcall(function()
return require("Módulo:" .. char)
end)
if ok and type(data) == "table" then
return data
end
ok, data = pcall(function()
return require("Modulo:" .. char)
end)
if ok and type(data) == "table" then
return data
end
ok, data = pcall(function()
return require("Module:" .. char)
end)
if ok and type(data) == "table" then
return data
end
return nil
end
-- ===== I18N: labels de atributos + "Nível" =====
-- Este mapa é injetado em data-i18n-attrs no #skills para o JS consumir.
local ATTR_I18N = {
pt = {
cooldown = "Recarga",
energy = "Energia",
energy_gain = "Ganho de energia",
energy_cost = "Custo de energia",
power = "Poder",
power_pvp = "Poder PvP",
level = "Nível"
},
en = {
cooldown = "Cooldown",
energy = "Energy",
energy_gain = "Energy Gain",
energy_cost = "Energy Cost",
power = "Power",
power_pvp = "PvP Power",
level = "Level"
},
es = {
cooldown = "Recarga",
energy = "Energía",
energy_gain = "Ganancia de energía",
energy_cost = "Costo de energía",
power = "Poder",
power_pvp = "Poder PvP",
level = "Nivel"
},
pl = {
cooldown = "Czas odnowienia",
energy = "Energia",
energy_gain = "Przyrost energii",
energy_cost = "Koszt energii",
power = "Moc",
power_pvp = "Moc PvP",
level = "Poziom"
}
}
-- ===== I18N: labels das abas (Skills/Skins) =====
local TAB_I18N = {
pt = {
skills = "Habilidades",
skins = "Skins",
skins_title = "SKINS & SPOTLIGHTS"
},
en = {
skills = "Skills",
skins = "Skins",
skins_title = "SKINS & SPOTLIGHTS"
},
es = {
skills = "Habilidades",
skins = "Aspectos",
skins_title = "ASPECTOS Y DESTACADOS"
},
pl = {
skills = "Umiejętności",
skins = "Skórki",
skins_title = "SKÓRKI I PREZENTACJE"
}
}
-- ===== skin serializer (mantido) =====
function p.skin(frame)
local a = frame.args
local obj = {
sprite = a.sprite or '',
background = a.background or '',
tooltip = a.tooltip or '',
youtube = a.youtube or '',
source = a.source or ''
}
return mw.text.jsonEncode(obj)
end
-- ===== componente principal =====
function p.generate(frame)
local args = frame:getParent().args
local html = mw.html.create('div')
local function fileURL(name)
return tostring(mw.uri.fullUrl('Special:FilePath/' .. (name or '')))
end
local nomeChar = args.nome or ""
-- ===== tier / classes (auto-resolve) =====
local rawTier = trim(frame:preprocess(args.tier or ""))
local rawClasse = trim(frame:preprocess(args.classe or ""))
if rawTier ~= "" and mw.ustring.lower(rawTier):sub(1, 4) == "tier" then
local data = requireCharModule(nomeChar) or {}
rawTier = (data.tier_i18n and data.tier_i18n.pt) or data.tier or rawTier
end
if rawClasse ~= "" and mw.ustring.lower(rawClasse):sub(1, 5) == "class" then
local data = requireCharModule(nomeChar) or {}
local arr = (data.tags_i18n and data.tags_i18n.pt) or data.tags
if type(arr) == "table" then
rawClasse = table.concat(arr, " / ")
end
end
if rawTier == "" or rawClasse == "" then
local data = requireCharModule(nomeChar) or {}
if rawTier == "" then
rawTier = (data.tier_i18n and data.tier_i18n.pt) or data.tier or ""
end
if rawClasse == "" then
local arr = (data.tags_i18n and data.tags_i18n.pt) or data.tags
if type(arr) == "table" then
rawClasse = table.concat(arr, " / ")
else
rawClasse = arr or ""
end
end
end
-- ===== Tier -> classe css =====
local tierRaw = rawTier
local tierKey = mw.ustring.lower(tierRaw or "")
local tierMap = {
bronze = "tier-bronze",
bronce = "tier-bronze",
silver = "tier-silver",
prata = "tier-silver",
gold = "tier-gold",
ouro = "tier-gold",
diamond = "tier-diamond",
diamante = "tier-diamond"
}
local tierClass = tierMap[tierKey]
-- ===== raiz =====
local box = html:tag('div'):addClass('character-box')
-- BG via data-attr
do
local raw = frame:preprocess(args.background or "")
raw = (raw or ""):gsub("^%s+", ""):gsub("%s+$", "")
if raw ~= "" then
raw = raw:gsub("^Arquivo:", ""):gsub("^File:", "")
local title = mw.title.new('Especial:FilePath/' .. raw)
local url = title and title:fullUrl() or nil
if url and url ~= "" then
box:attr('data-bg-file', raw)
box:attr('data-bg-url', url)
end
end
end
local function trim2(s2)
return (s2 or ''):match('^%s*(.-)%s*$')
end
local function fileURL2(name)
if not name or name == '' then
return nil
end
name = name:gsub('^Arquivo:', ''):gsub('^File:', '')
return mw.title.new('Special:FilePath/' .. name):fullUrl()
end
do
local bgParam = trim2(args.background or '')
if bgParam == '' then
bgParam = trim2(args.banner or '')
end
if bgParam ~= '' then
local url = fileURL2(bgParam)
if url then
local style = string.format("--character-bg: url('%s'); background-image: url('%s') !important;", url,
url)
local old = box:getAttr('style')
box:attr('style', old and (old .. ' ' .. style) or style)
box:attr('data-bg-file', bgParam)
end
end
end
local bgFile = trim(frame:preprocess(args.background or ""))
if bgFile ~= "" then
local bgUrl = fileURL(bgFile)
local style = string.format("--character-bg: url('%s'); background-image: url('%s');", bgUrl, bgUrl)
box:attr('style', style)
end
if tierClass and tierClass ~= "" then
box:addClass(tierClass)
end
-- ===== header / topbar =====
local header = box:tag('div'):addClass('character-header')
local topbar = header:tag('div'):addClass('character-topbar')
local nameBox = topbar:tag('div'):addClass('character-name-box')
local avatarImg = args.avatar or 'Franky_ts_medal.png'
nameBox:wikitext(string.format('[[Arquivo:%s|class=topbar-icon|link=|alt=Avatar]]', avatarImg))
local nameGroup = nameBox:tag('div'):addClass('character-name-group')
nameGroup:tag('div'):addClass('character-name'):wikitext(args.nome or '')
local classTags = nameGroup:tag('div'):addClass('class-tags')
if tierRaw and tierRaw ~= "" then
classTags:tag('div'):addClass('class-tag tier'):wikitext(tierRaw)
end
local classeString = rawClasse or ""
for classe in mw.text.gsplit(classeString, '/', true) do
local clean = mw.text.trim(classe or '')
if clean ~= '' then
classTags:tag('div'):addClass('class-tag'):wikitext(clean)
end
end
header:tag('div'):addClass('topbar-description'):wikitext(args.desc or '')
local rawLang = trim(args.lang or "pt")
rawLang = mw.ustring.lower(rawLang)
local base = rawLang:match("^([a-z][a-z])") or rawLang
local TAB = TAB_I18N[rawLang] or TAB_I18N[base] or TAB_I18N.pt
local tabs = header:tag('div'):addClass('character-tabs')
tabs:tag('div'):addClass('tab-btn active'):attr('data-tab', 'skills'):wikitext(TAB.skills)
tabs:tag('div'):addClass('tab-btn'):attr('data-tab', 'skins'):wikitext(TAB.skins)
-- ===== SKILLS =====
local skillsTab = box:tag('div'):addClass('tab-content active'):attr('id', 'skills')
local iconBar = skillsTab:tag('div'):addClass('icon-bar')
local skillsContainer = skillsTab:tag('div'):addClass('skills-container')
local details = skillsContainer:tag('div'):addClass('skills-details')
local descBox = details:tag('div'):addClass('desc-box')
-- I18N: injeta o mapa inteiro para o JS usar (botão de idioma)
skillsTab:attr('data-i18n-attrs', mw.text.jsonEncode(ATTR_I18N))
if args.lang and trim(args.lang) ~= "" then
skillsTab:attr('data-i18n-default', mw.ustring.lower(trim(args.lang)))
end
local skillsPacked = args.skills or '' -- sequência de {} gerados por {{Skill}} (via Módulo:Info)
local idx = 0
for obj in skillsPacked:gmatch("%b{}") do
local ok, sk = pcall(mw.text.jsonDecode, obj)
if ok and type(sk) == "table" and (sk.name or sk.nome) and (sk.name ~= '' or sk.nome ~= '') then
idx = idx + 1
local name = sk.name or sk.nome or ''
local icon = sk.icon or ''
local desc = sk.desc or ''
local level = tostring(sk.level or '')
-- ordem esperada pelo JS: PVE, PVP, Energia, Recarga
local attrs = table.concat({sk.powerpve or '-', sk.powerpvp or '-', sk.energy or '-', sk.cooldown or '-'},
", ")
local videoURL = (sk.video and sk.video ~= '') and fileURL(sk.video) or ''
local iconWrap = iconBar:tag('div'):addClass('skill-icon'):attr('data-index', idx):attr('data-nome', name)
:attr('data-desc', desc):attr('data-atr', attrs):attr('data-video', videoURL):attr('data-video-preload',
'auto')
-- aplica data-level quando houver (e não for placeholder)
if level ~= '' and mw.ustring.upper(level) ~= 'NIVEL' then
iconWrap:attr('data-level', level)
end
iconWrap:wikitext(string.format('[[Arquivo:%s|class=skill-icon-img|link=]]', icon))
descBox:tag('div'):addClass('skill-desc'):attr('data-index', idx)
end
end
details:done()
skillsContainer:tag('div'):addClass('video-container'):done()
skillsTab:done()
-- ===== SKINS =====
local skinsTab = box:tag('div'):addClass('tab-content'):attr('id', 'skins')
local cardSkins = skinsTab:tag('div'):addClass('card-skins')
cardSkins:tag('span'):addClass('card-skins-title'):wikitext(TAB.skins_title)
local wrapper = cardSkins:tag('div'):addClass('skins-carousel-wrapper')
wrapper:tag('div'):addClass('skins-arrow left'):wikitext('«')
local carousel = wrapper:tag('div'):addClass('skins-carousel')
local skinsPacked = args.skins or ''
for obj in skinsPacked:gmatch("%b{}") do
local ok, sk = pcall(mw.text.jsonDecode, obj)
if ok and type(sk) == "table" then
local bannerFile = sk.background or ''
local imageFile = sk.sprite or ''
local tooltipRaw = sk.tooltip or ''
local tooltipHtml = tooltipRaw:gsub("'''([^']+)'''", "<b>%1</b>"):gsub("\n", "<br>")
local skinCard = carousel:tag('div'):addClass('skin-card'):attr('data-skin-tooltip', tooltipHtml)
local yt = (sk.youtube or '')
if yt ~= '' then
skinCard:attr('data-youtube', yt):addClass('is-clickable')
end
skinCard:tag('div'):addClass('skin-banner'):wikitext(bannerFile ~= '' and
string.format('[[Arquivo:%s|link=]]', bannerFile) or
''):attr('alt', 'banner')
skinCard:tag('div'):addClass('skin-sprite'):wikitext(imageFile ~= '' and
string.format('[[Arquivo:%s|link=]]', imageFile) or
''):attr('alt', 'skin')
end
end
wrapper:tag('div'):addClass('skins-arrow right'):wikitext('»')
return tostring(html)
end
return p