Toggle menu
Toggle preferences menu
Toggle personal menu
Not logged in
Your IP address will be publicly visible if you make any edits.

Module:Team: Difference between revisions

From eSportsAmaze
No edit summary
No edit summary
Line 5: Line 5:


-- ============================================================
-- ============================================================
-- HELPER: Argument Fetcher (Fixes #invoke vs Template issue)
-- HELPER: Argument Fetcher
-- ============================================================
-- ============================================================
local function getArgs(frame)
local function getArgs(frame)
     local args = {}
     local args = {}
   
    -- 1. Get arguments from the Template (if used)
     if frame:getParent() then
     if frame:getParent() then
         for k, v in pairs(frame:getParent().args) do
         for k, v in pairs(frame:getParent().args) do args[k] = v end
            args[k] = v
        end
     end
     end
   
     for k, v in pairs(frame.args) do args[k] = v end
    -- 2. Override with arguments from the #invoke (Direct use)
     for k, v in pairs(frame.args) do
        args[k] = v
    end
   
     return args
     return args
end
end
Line 79: Line 70:
-- ============================================================
-- ============================================================
function p.infobox(frame)
function p.infobox(frame)
     local args = getArgs(frame) -- FIXED ARG LOADING
     local args = getArgs(frame)
     local team = clean(args.name) or currentTitle.subpageText
     local team = clean(args.name) or currentTitle.subpageText
      
      
Line 129: Line 120:


-- ============================================================
-- ============================================================
-- MAIN 2: HYBRID ROSTER (MANUAL + DB)
-- MAIN 2: HYBRID ROSTER (SORTED BY ROLE)
-- ============================================================
-- ============================================================
function p.roster(frame)
function p.roster(frame)
     local args = getArgs(frame) -- FIXED ARG LOADING
     local args = getArgs(frame)
     local team = clean(args.team) or currentTitle.subpageText
     local team = clean(args.team) or currentTitle.subpageText
      
      
     -- 1. Fetch Database Players (SORTED BY ID)
     -- 1. Fetch Database Players
     local dbPlayers = {}
     local dbPlayers = {}
     if cargo and cargo.query then
     if cargo and cargo.query then
Line 141: Line 132:
         local fields = "id, real_name, role, nationality, image, _pageName"
         local fields = "id, real_name, role, nationality, image, _pageName"
         local where = "current_team = '" .. team:gsub("'", "\\'") .. "' AND status='Active'"
         local where = "current_team = '" .. team:gsub("'", "\\'") .. "' AND status='Active'"
         local results = cargo.query(tables, fields, { where = where, orderBy = "id ASC" })
         local results = cargo.query(tables, fields, { where = where })
          
          
         for _, row in ipairs(results) do
         for _, row in ipairs(results) do
Line 158: Line 149:
     -- 2. Process Manual Inputs
     -- 2. Process Manual Inputs
     local finalRoster = {}
     local finalRoster = {}
   
     for _, pData in pairs(dbPlayers) do table.insert(finalRoster, pData) end
    -- Add DB players
     for _, pData in pairs(dbPlayers) do
        table.insert(finalRoster, pData)
    end
   
    -- Check manual args (player1, player2... player10)
     for i = 1, 10 do
     for i = 1, 10 do
         local mid = clean(args['player' .. i])
         local mid = clean(args['player' .. i])
         if mid and mid ~= "" then
         if mid and mid ~= "" then
            -- Avoid duplicates: Only add if NOT in DB list
             if not dbPlayers[mid] then
             if not dbPlayers[mid] then
                 table.insert(finalRoster, {
                 table.insert(finalRoster, {
Line 183: Line 167:
     end
     end
      
      
     -- 3. Sort Final Roster by ID (Name) Alphabetically
     -- 3. Define Role Hierarchy & Colors
     table.sort(finalRoster, function(a, b)  
    -- Order matches your request
         return (a.id or ""):lower() < (b.id or ""):lower()  
    local roleConfig = {
        { key = "igl",      sort = 1, color = "#eab308" }, -- Gold
        { key = "entry",    sort = 2, color = "#f97316" }, -- Orange (Entry Fragger)
        { key = "assault",  sort = 3, color = "#3b82f6" }, -- Blue
        { key = "fragger",  sort = 3, color = "#3b82f6" }, -- Blue (Variant for Assaulter)
        { key = "support",  sort = 4, color = "#22c55e" }, -- Green
        { key = "medic",    sort = 4, color = "#22c55e" }, -- Green (Variant)
        { key = "scout",    sort = 5, color = "#14b8a6" }, -- Teal
        { key = "sniper",  sort = 6, color = "#ef4444" }, -- Red
        { key = "coach",    sort = 7, color = "#a855f7" }, -- Purple
        { key = "analyst",  sort = 8, color = "#a855f7" }, -- Purple
        { key = "manager",  sort = 9, color = "#1f2937" }  -- Black
    }
   
    -- 4. Assign Sort Priority & Color to each player
    for _, player in ipairs(finalRoster) do
        local r = (player.role or ""):lower()
       
        -- Default values if no match found
        player.sortOrder = 99
        player.color = "#64748b" -- Gray
       
        for _, config in ipairs(roleConfig) do
            if r:find(config.key) then
                player.sortOrder = config.sort
                player.color = config.color
                break -- Stop at the first match (Priority)
            end
        end
    end
   
    -- 5. Sort the Roster Table
     table.sort(finalRoster, function(a, b)
         if a.sortOrder ~= b.sortOrder then
            return a.sortOrder < b.sortOrder -- Sort by Role Priority
        else
            return (a.id or ""):lower() < (b.id or ""):lower() -- Tie-breaker: Name
        end
     end)
     end)
      
      
     -- 4. Render
     -- 6. Render
     local grid = html.create('div'):addClass('hero-roster-grid')
     local grid = html.create('div'):addClass('hero-roster-grid')
      
      
Line 194: Line 215:
         local card = grid:tag('div'):addClass('hero-player-card')
         local card = grid:tag('div'):addClass('hero-player-card')
          
          
         -- Header (Image Area)
         -- Header
         local header = card:tag('div'):addClass('hero-card-image')
         local header = card:tag('div'):addClass('hero-card-image')
         if player.image and player.image ~= "" then
         if player.image and player.image ~= "" then
Line 202: Line 223:
         end
         end
          
          
         -- Info Body
         -- Body
         local body = card:tag('div'):addClass('hero-card-body')
         local body = card:tag('div'):addClass('hero-card-body')
          
          
         -- Role Pill
         -- Role Pill (Using the assigned color)
        local roleColor = "#64748b" -- Gray Default
        local r = (player.role or ""):lower()
        if r:find("igl") then roleColor = "#eab308" -- Gold
        elseif r:find("fragger") or r:find("entry") or r:find("assault") then roleColor = "#ef4444" -- Red
        elseif r:find("support") or r:find("medic") then roleColor = "#22c55e" -- Green
        elseif r:find("sniper") then roleColor = "#3b82f6" -- Blue
        elseif r:find("coach") then roleColor = "#a855f7" -- Purple
        end
       
         body:tag('div'):addClass('hero-role-pill')
         body:tag('div'):addClass('hero-role-pill')
             :css('background-color', roleColor)
             :css('background-color', player.color)
             :wikitext(player.role or "Player")
             :wikitext(player.role or "Player")
              
              
Line 240: Line 252:
-- ============================================================
-- ============================================================
function p.formerPlayers(frame)
function p.formerPlayers(frame)
     local args = getArgs(frame) -- FIXED ARG LOADING
     local args = getArgs(frame)
     local team = clean(args.team) or currentTitle.subpageText
     local team = clean(args.team) or currentTitle.subpageText
      
      
     -- STEP 1: Get the history from the history table (NO JOIN)
     -- STEP 1: Get History
     local history = cargo.query(
     local history = cargo.query(
         "Player_Former_Teams",  
         "Player_Former_Teams",  
Line 258: Line 270:
     end
     end
      
      
     -- STEP 2: Gather all Page Names to fetch Details
     -- STEP 2: Get Page Names
     local pageNames = {}
     local pageNames = {}
   
     for i, row in ipairs(history) do
     for i, row in ipairs(history) do
         local pName = row.player_id
         local pName = row.player_id
         if pName and pName ~= "" then
         if pName and pName ~= "" then table.insert(pageNames, "'" .. pName:gsub("'", "\\'") .. "'") end
            table.insert(pageNames, "'" .. pName:gsub("'", "\\'") .. "'")
        end
     end
     end
      
      
     -- STEP 3: Fetch Real Names & Short IDs from Players Table
     -- STEP 3: Get Details
     local playerDetails = {}
     local playerDetails = {}
     if #pageNames > 0 then
     if #pageNames > 0 then
         local whereClause = "_pageName IN (" .. table.concat(pageNames, ",") .. ")"
         local details = cargo.query("Players", "_pageName, real_name, id", { where = "_pageName IN (" .. table.concat(pageNames, ",") .. ")" })
        local details = cargo.query(
            "Players",
            "_pageName, real_name, id",
            { where = whereClause }
        )
       
        -- Map details for easy lookup
         if details then
         if details then
             for _, d in ipairs(details) do
             for _, d in ipairs(details) do
                 playerDetails[d._pageName] = {
                 playerDetails[d._pageName] = { realName = d.real_name, shortID = d.id }
                    realName = d.real_name,
                    shortID = d.id
                }
             end
             end
         end
         end
     end
     end
      
      
     -- STEP 4: Render the Table (USING FLAT UI CLASSES)
     -- STEP 4: Render
     local tbl = html.create('table'):addClass('flat-data-table sortable')
     local tbl = html.create('table'):addClass('flat-data-table sortable')
   
    -- Header
     local h = tbl:tag('tr')
     local h = tbl:tag('tr')
     h:tag('th'):wikitext('ID')
     h:tag('th'):wikitext('ID'); h:tag('th'):wikitext('Name'); h:tag('th'):wikitext('Role'); h:tag('th'):wikitext('Join Date'); h:tag('th'):wikitext('Leave Date')
    h:tag('th'):wikitext('Name')
    h:tag('th'):wikitext('Role')
    h:tag('th'):wikitext('Join Date')
    h:tag('th'):wikitext('Leave Date')
      
      
     for _, row in ipairs(history) do
     for _, row in ipairs(history) do
         local pName = row.player_id
         local pName = row.player_id
         local details = playerDetails[pName] or {}
         local details = playerDetails[pName] or {}
       
        -- Determine Display ID
         local displayID = details.shortID
         local displayID = details.shortID
         if not displayID or displayID == "" then
         if not displayID or displayID == "" then displayID = pName:gsub("^.*/", "") end
            displayID = pName:gsub("^.*/", "")
        end
          
          
         local tr = tbl:tag('tr')
         local tr = tbl:tag('tr')
       
        -- 1. ID Column (Bold + Link)
         tr:tag('td'):addClass('t-id'):wikitext('[[' .. pName .. '|' .. displayID .. ']]')
         tr:tag('td'):addClass('t-id'):wikitext('[[' .. pName .. '|' .. displayID .. ']]')
       
        -- 2. Name Column
         tr:tag('td'):wikitext(details.realName or "-")
         tr:tag('td'):wikitext(details.realName or "-")
          
          
        -- 3. Role Column (Styled Pill)
         local rCell = tr:tag('td')
         local rCell = tr:tag('td')
         if row.role and row.role ~= "" then
         if row.role and row.role ~= "" then rCell:tag('span'):addClass('t-role'):wikitext(row.role) else rCell:wikitext("-") end
            rCell:tag('span'):addClass('t-role'):wikitext(row.role)
        else
            rCell:wikitext("-")
        end
          
          
        -- 4. Dates (Monospace font)
         tr:tag('td'):addClass('t-date'):wikitext(row.join_date or "?")
         tr:tag('td'):addClass('t-date'):wikitext(row.join_date or "?")
         tr:tag('td'):addClass('t-date'):wikitext(row.leave_date or "?")
         tr:tag('td'):addClass('t-date'):wikitext(row.leave_date or "?")
Line 335: Line 314:


-- ============================================================
-- ============================================================
-- MAIN 4: HISTORY (Preserved)
-- MAIN 4: HISTORY
-- ============================================================
-- ============================================================
function p.history(frame)
function p.history(frame)
     local args = getArgs(frame) -- FIXED ARG LOADING
     local args = getArgs(frame)
     local team = clean(args.team) or mw.title.getCurrentTitle().subpageText
     local team = clean(args.team) or mw.title.getCurrentTitle().subpageText
     local results = cargo.query("StageStandings", "tournament, stage, totalpts, matchesplayed, wwcd, elimpts, lastmatchrank, result", {
     local results = cargo.query("StageStandings", "tournament, stage, totalpts, matchesplayed, wwcd, elimpts, lastmatchrank, result", {

Revision as of 23:42, 29 January 2026

Documentation for this module may be created at Module:Team/doc

local p = {}
local cargo = mw.ext.cargo
local html = mw.html
local currentTitle = mw.title.getCurrentTitle()

-- ============================================================
-- HELPER: Argument Fetcher
-- ============================================================
local function getArgs(frame)
    local args = {}
    if frame:getParent() then
        for k, v in pairs(frame:getParent().args) do args[k] = v end
    end
    for k, v in pairs(frame.args) do args[k] = v end
    return args
end

-- ============================================================
-- HELPER: Sanitize Strings
-- ============================================================
local function clean(s)
    if not s then return nil end
    local cleaned = s:gsub("[\r\n]", ""):gsub("^%s*(.-)%s*$", "%1")
    if cleaned == "" then return nil end
    return cleaned
end

local function formatCurrency(amount)
    if not amount then return "0" end
    local n = tonumber(amount) or 0
    local formatted = tostring(n):reverse():gsub("(%d%d%d)(%d%d%d)","%1,%2"):reverse()
    return '<span class="indian-currency">₹ ' .. formatted .. '</span>'
end

local function getSocials(args)
    local container = html.create('div'):addClass('fib-socials')
    local hasSocials = false
    local platforms = {
        {arg='instagram', file='Icon_instagram.png'},
        {arg='twitter',   file='Icon_twitter.png'},
        {arg='youtube',   file='Icon_youtube.png'},
        {arg='discord',   file='Icon_discord.png'},
        {arg='facebook',  file='Icon_facebook.png'},
        {arg='website',   file='Icon_website.png'}
    }
    for _, p in ipairs(platforms) do
        if args[p.arg] and args[p.arg] ~= "" then
            hasSocials = true
            container:wikitext('[[File:' .. p.file .. '|24px|link=' .. args[p.arg] .. '|class=social-img]]')
        end
    end
    if hasSocials then return tostring(container) else return "" end
end

local function getInfoboxLogo(teamName, image, imageDark)
    local lightFile = (image ~= "" and image) or (teamName .. '.png')
    local darkFile = (imageDark ~= "" and imageDark) or (teamName .. '_dark.png')
    local hasLight = mw.title.new('File:' .. lightFile).exists
    local hasDark = mw.title.new('File:' .. darkFile).exists
    local container = html.create('div'):addClass('fib-image')
    local lSpan = container:tag('span'):addClass('logo-lightmode')
    if hasLight then lSpan:wikitext('[[File:' .. lightFile .. '|220px]]') else lSpan:wikitext('[[File:Shield_team.png|180px]]') end
    local dSpan = container:tag('span'):addClass('logo-darkmode')
    if hasDark then dSpan:wikitext('[[File:' .. darkFile .. '|220px]]') elseif hasLight then dSpan:wikitext('[[File:' .. lightFile .. '|220px]]') else dSpan:wikitext('[[File:Shield_team_dark.png|180px]]') end
    return tostring(container)
end

-- ============================================================
-- MAIN 1: TEAM INFOBOX
-- ============================================================
function p.infobox(frame)
    local args = getArgs(frame)
    local team = clean(args.name) or currentTitle.subpageText
    
    local root = html.create('div'):addClass('flat-infobox')
    root:tag('div'):addClass('fib-header'):tag('div'):addClass('fib-title'):wikitext(team)
    root:wikitext(getInfoboxLogo(team, clean(args.image), clean(args.image_dark)))
    
    local earnings = "0"
    if cargo and cargo.query then
        local results = cargo.query("PrizeMoney", "SUM(prize)=total", { where = "team = '" .. team:gsub("'", "\\'") .. "' AND (player='' OR player IS NULL)" })
        if results and #results > 0 and results[1].total then earnings = results[1].total end
    end
    
    local rankVal = "Unranked"
    if cargo and cargo.query then
        local rResults = cargo.query("Krafton_Rankings", "rank", { where = "name = '" .. team:gsub("'", "\\'") .. "' AND type='Team'", limit = 1 })
        if rResults and #rResults > 0 then rankVal = "#" .. rResults[1].rank end
    end
    
    local statusColor = "#333"
    local statusText = clean(args.status) or "Active"
    if args.status then
        local s = args.status:lower()
        if s == "active" then statusColor = "#16a34a" elseif s == "inactive" or s == "disbanded" then statusColor = "#dc2626" end
    end

    local grid1 = root:tag('div'):addClass('fib-grid')
    grid1:tag('div'):addClass('fib-cell'):tag('div'):addClass('fib-label-sm'):wikitext('Status'):done():tag('div'):addClass('fib-value-sm'):css('color', statusColor):wikitext(statusText):done()
    grid1:tag('div'):addClass('fib-cell'):tag('div'):addClass('fib-label-sm'):wikitext('Krafton Rank'):done():tag('div'):addClass('fib-value-sm'):wikitext(rankVal):done()

    local grid2 = root:tag('div'):addClass('fib-grid')
    grid2:tag('div'):addClass('fib-cell'):tag('div'):addClass('fib-label-sm'):wikitext('Country'):done():tag('div'):addClass('fib-value-sm'):wikitext(clean(args.country) or 'TBD'):done()
    grid2:tag('div'):addClass('fib-cell'):tag('div'):addClass('fib-label-sm'):wikitext('Tag'):done():tag('div'):addClass('fib-value-sm'):wikitext(clean(args.short_code) or '-'):done()

    if earnings and tonumber(earnings) and tonumber(earnings) > 0 then
        root:tag('div'):addClass('fib-prize'):tag('div'):addClass('fib-label-sm'):wikitext('Total Earnings'):done():tag('div'):addClass('fib-prize-val'):wikitext(formatCurrency(earnings)):done()
    end
    
    local list = root:tag('div'):addClass('fib-list')
    local function addRow(label, value)
        if value and value ~= "" then list:tag('div'):addClass('fib-row'):tag('div'):addClass('fib-label'):wikitext(label):done():tag('div'):addClass('fib-data'):wikitext(value):done() end
    end
    
    addRow('Full Name', clean(args.other_team))
    if args.sponsors then addRow('Sponsors', args.sponsors:gsub(",", "<br>")) end
    root:wikitext(getSocials(args))
    return tostring(root)
end

-- ============================================================
-- MAIN 2: HYBRID ROSTER (SORTED BY ROLE)
-- ============================================================
function p.roster(frame)
    local args = getArgs(frame)
    local team = clean(args.team) or currentTitle.subpageText
    
    -- 1. Fetch Database Players
    local dbPlayers = {}
    if cargo and cargo.query then
        local tables = "Players"
        local fields = "id, real_name, role, nationality, image, _pageName"
        local where = "current_team = '" .. team:gsub("'", "\\'") .. "' AND status='Active'"
        local results = cargo.query(tables, fields, { where = where })
        
        for _, row in ipairs(results) do
            dbPlayers[row.id] = {
                id = row.id,
                name = row.real_name,
                role = row.role,
                flag = clean(row.nationality),
                image = clean(row.image),
                link = row._pageName,
                source = "auto"
            }
        end
    end
    
    -- 2. Process Manual Inputs
    local finalRoster = {}
    for _, pData in pairs(dbPlayers) do table.insert(finalRoster, pData) end
    for i = 1, 10 do
        local mid = clean(args['player' .. i])
        if mid and mid ~= "" then
            if not dbPlayers[mid] then
                table.insert(finalRoster, {
                    id = mid,
                    name = clean(args['name' .. i]) or "",
                    role = clean(args['role' .. i]) or "Player",
                    flag = clean(args['flag' .. i]) or "India",
                    image = clean(args['image' .. i]) or "", 
                    link = mid, 
                    source = "manual"
                })
            end
        end
    end
    
    -- 3. Define Role Hierarchy & Colors
    -- Order matches your request
    local roleConfig = {
        { key = "igl",      sort = 1, color = "#eab308" }, -- Gold
        { key = "entry",    sort = 2, color = "#f97316" }, -- Orange (Entry Fragger)
        { key = "assault",  sort = 3, color = "#3b82f6" }, -- Blue
        { key = "fragger",  sort = 3, color = "#3b82f6" }, -- Blue (Variant for Assaulter)
        { key = "support",  sort = 4, color = "#22c55e" }, -- Green
        { key = "medic",    sort = 4, color = "#22c55e" }, -- Green (Variant)
        { key = "scout",    sort = 5, color = "#14b8a6" }, -- Teal
        { key = "sniper",   sort = 6, color = "#ef4444" }, -- Red
        { key = "coach",    sort = 7, color = "#a855f7" }, -- Purple
        { key = "analyst",  sort = 8, color = "#a855f7" }, -- Purple
        { key = "manager",  sort = 9, color = "#1f2937" }  -- Black
    }
    
    -- 4. Assign Sort Priority & Color to each player
    for _, player in ipairs(finalRoster) do
        local r = (player.role or ""):lower()
        
        -- Default values if no match found
        player.sortOrder = 99
        player.color = "#64748b" -- Gray
        
        for _, config in ipairs(roleConfig) do
            if r:find(config.key) then
                player.sortOrder = config.sort
                player.color = config.color
                break -- Stop at the first match (Priority)
            end
        end
    end
    
    -- 5. Sort the Roster Table
    table.sort(finalRoster, function(a, b)
        if a.sortOrder ~= b.sortOrder then
            return a.sortOrder < b.sortOrder -- Sort by Role Priority
        else
            return (a.id or ""):lower() < (b.id or ""):lower() -- Tie-breaker: Name
        end
    end)
    
    -- 6. Render
    local grid = html.create('div'):addClass('hero-roster-grid')
    
    for _, player in ipairs(finalRoster) do
        local card = grid:tag('div'):addClass('hero-player-card')
        
        -- Header
        local header = card:tag('div'):addClass('hero-card-image')
        if player.image and player.image ~= "" then
            header:wikitext('[[File:' .. player.image .. '|link=' .. player.link .. ']]')
        else
            header:wikitext('[[File:Player_Placeholder.png|link=' .. player.link .. ']]')
        end
        
        -- Body
        local body = card:tag('div'):addClass('hero-card-body')
        
        -- Role Pill (Using the assigned color)
        body:tag('div'):addClass('hero-role-pill')
            :css('background-color', player.color)
            :wikitext(player.role or "Player")
            
        -- ID and Name
        body:tag('div'):addClass('hero-player-id'):wikitext('[[' .. player.link .. '|' .. player.id .. ']]')
        if player.name and player.name ~= "" then
            body:tag('div'):addClass('hero-player-name'):wikitext(player.name)
        end
        
        -- Flag
        if player.flag then
            local safeFlag = player.flag:gsub(" ", "_") 
            local flagFile = "Flag_" .. safeFlag .. ".png"
            body:tag('div'):addClass('hero-player-flag'):wikitext('[[File:' .. flagFile .. '|link=]] ' .. player.flag)
        end
    end
    
    return tostring(grid)
end

-- ============================================================
-- MAIN 3: FORMER PLAYERS LIST (NO JOIN + FLAT UI DESIGN)
-- ============================================================
function p.formerPlayers(frame)
    local args = getArgs(frame)
    local team = clean(args.team) or currentTitle.subpageText
    
    -- STEP 1: Get History
    local history = cargo.query(
        "Player_Former_Teams", 
        "player_id, role, join_date, leave_date", 
        {
            where = "team = '" .. team:gsub("'", "\\'") .. "' AND leave_date != ''",
            orderBy = "leave_date DESC",
            limit = 100
        }
    )
    
    if not history or #history == 0 then
        return '<div style="font-style:italic; color:#64748b; padding:10px;">No former players recorded.</div>'
    end
    
    -- STEP 2: Get Page Names
    local pageNames = {}
    for i, row in ipairs(history) do
        local pName = row.player_id
        if pName and pName ~= "" then table.insert(pageNames, "'" .. pName:gsub("'", "\\'") .. "'") end
    end
    
    -- STEP 3: Get Details
    local playerDetails = {}
    if #pageNames > 0 then
        local details = cargo.query("Players", "_pageName, real_name, id", { where = "_pageName IN (" .. table.concat(pageNames, ",") .. ")" })
        if details then
            for _, d in ipairs(details) do
                playerDetails[d._pageName] = { realName = d.real_name, shortID = d.id }
            end
        end
    end
    
    -- STEP 4: Render
    local tbl = html.create('table'):addClass('flat-data-table sortable')
    local h = tbl:tag('tr')
    h:tag('th'):wikitext('ID'); h:tag('th'):wikitext('Name'); h:tag('th'):wikitext('Role'); h:tag('th'):wikitext('Join Date'); h:tag('th'):wikitext('Leave Date')
    
    for _, row in ipairs(history) do
        local pName = row.player_id
        local details = playerDetails[pName] or {}
        local displayID = details.shortID
        if not displayID or displayID == "" then displayID = pName:gsub("^.*/", "") end
        
        local tr = tbl:tag('tr')
        tr:tag('td'):addClass('t-id'):wikitext('[[' .. pName .. '|' .. displayID .. ']]')
        tr:tag('td'):wikitext(details.realName or "-")
        
        local rCell = tr:tag('td')
        if row.role and row.role ~= "" then rCell:tag('span'):addClass('t-role'):wikitext(row.role) else rCell:wikitext("-") end
        
        tr:tag('td'):addClass('t-date'):wikitext(row.join_date or "?")
        tr:tag('td'):addClass('t-date'):wikitext(row.leave_date or "?")
    end
    
    return tostring(tbl)
end

-- ============================================================
-- MAIN 4: HISTORY
-- ============================================================
function p.history(frame)
    local args = getArgs(frame)
    local team = clean(args.team) or mw.title.getCurrentTitle().subpageText
    local results = cargo.query("StageStandings", "tournament, stage, totalpts, matchesplayed, wwcd, elimpts, lastmatchrank, result", {
        where = "team = '" .. team:gsub("'", "\\'") .. "'",
        orderBy = "tournament DESC",
        limit = 50
    })
    
    local root = html.create('div')
    root:wikitext('== Tournament Statistics ==')
    
    if #results > 0 then
        local tbl = root:tag('table'):addClass('modern-history-table')
        local thead = tbl:tag('thead'):tag('tr')
        thead:tag('th'):wikitext('Tournament'); thead:tag('th'):wikitext('Stage'); thead:tag('th'):wikitext('Rank')
        thead:tag('th'):wikitext('MP'); thead:tag('th'):wikitext('WWCD'); thead:tag('th'):wikitext('Elims'); thead:tag('th'):wikitext('Pts')
        
        for _, row in ipairs(results) do
            local tr = tbl:tag('tr')
            local res = (row.result or ""):lower()
            if res == "q" or res == "qualified" then tr:css('background-color', '#E9FFF0')
            elseif res == "e" or res == "eliminated" then tr:css('background-color', '#FFE9E9') end
            
            tr:tag('td'):css('font-weight','bold'):wikitext('[[' .. row.tournament .. ']]')
            tr:tag('td'):wikitext(row.stage)
            tr:tag('td'):css('font-weight','bold'):css('text-align','center'):wikitext('#' .. (row.lastmatchrank or '?'))
            tr:tag('td'):css('text-align','center'):wikitext(row.matchesplayed)
            tr:tag('td'):css('text-align','center'):wikitext(row.wwcd)
            tr:tag('td'):css('text-align','center'):wikitext(row.elimpts)
            tr:tag('td'):css('text-align','center'):css('font-weight','800'):wikitext(row.totalpts)
        end
    else root:wikitext("''No tournament data found for this team.''") end
    return tostring(root)
end

return p