打开/关闭菜单
打开/关闭个人菜单
未登录
未登录用户的IP地址会在进行任意编辑后公开展示。

模块:FrameChart:修订间差异

来自骷髅女孩Wiki
无编辑摘要
无编辑摘要
 
(未显示同一用户的75个中间版本)
第3行: 第3行:
local html
local html
local frameChartDataHtml
local frameChartDataHtml
local lineOneIndex = 1
local lineTwoIndex = 1
local lineThreeIndex = 1
local lineFourIndex = 1
local isSuperflash = false
local superIndex = { l1 = -1, l2 = -1, l3 = -1, l4 = -1 }
local lineFrameIndex = { l1 = -1, l2 = -1, l3 = -1, l4 = -1 }
local f = ''


function p.drawFrameData( frame )
function p.drawFrameData(frame)
   -- 开始循环,遍历包含{startup, active, recovery}的列表
   -- 开始循环,遍历包含{startup, active, recovery}的列表
   -- 对于每种帧类型,从fd中获取相应的帧数数据并存储在duration中
   -- 对于每种帧类型,从fd中获取相应的帧数数据并存储在duration中
第10行: 第18行:
   -- _, frameKind表示,忽略迭代过程中的第一个返回值(即k,键),只关心第二个返回值(即v,值,循环变量命名为frameKind)
   -- _, frameKind表示,忽略迭代过程中的第一个返回值(即k,键),只关心第二个返回值(即v,值,循环变量命名为frameKind)
   -- inpairs保证以数组顺序(1-n)遍历数组,所以以下代码意为将frameKind依次赋值为s、a、r并进行循环体操作
   -- inpairs保证以数组顺序(1-n)遍历数组,所以以下代码意为将frameKind依次赋值为s、a、r并进行循环体操作
   for _, frameKind in ipairs({"startup", "active", "recovery", "nsc", "linefour"}) do
   for _, frameKind in ipairs({ "startup", "active", "recovery", "nc", "linefour", "linetwo", "iwc" }) do
     local duration
     local duration
     if frame.args[frameKind] ~= nil then -- frame.args[frameKind]表示访问frame表中frameKind键对应的值
     if frame.args[frameKind] ~= nil then                 -- frame.args[frameKind]表示访问frame表中frameKind键对应的值
       duration = frame:preprocess(frame.args[frameKind]) -- frame:preprocess的参数中如有MW模板或变量(如{{Var}}),会自动展开
       duration = frame:preprocess(frame.args[frameKind]) -- frame:preprocess的参数中如有MW模板或变量(如{{Var}}),会自动展开
     end
     end
     -- 如果duration为nil,则跳过这个帧类型
     -- 如果duration为nil,则跳过这个帧类型
     if duration == nil then
     if duration == nil then
    -- 如果duration可以直接转化为数字,则直接赋值给frame.args[frameKind]
      -- 如果duration可以直接转化为数字,则直接赋值给frame.args[frameKind]
     elseif tonumber(duration) ~= nil then
     elseif tonumber(duration) ~= nil then
       frame.args[frameKind] = duration
       frame.args[frameKind] = duration
    -- 如果不是数字,则使用对应的解析函数解析帧数字并存储于
      -- 如果不是数字,则使用对应的解析函数解析帧数字并存储于
     else
     else
       local parsed = nil
       local parsed = nil
第27行: 第35行:
       elseif frameKind == "startup" then
       elseif frameKind == "startup" then
         parsed = parseStartup(duration)
         parsed = parseStartup(duration)
        if string.find(duration, "+") then
          isSuperflash = true
          f = string.match(duration, "^(%d+)%+")
          local i = 1
          while i < 5 do
            superIndex['l' .. tostring(i)] = tonumber(f)
            i = i + 1
          end
        end
       elseif frameKind == "recovery" then
       elseif frameKind == "recovery" then
         parsed = parseRecovery(duration)
         parsed = parseRecovery(duration)
       elseif frameKind == "nsc" then
       elseif frameKind == "nc" then
         parsed = parseNSC(duration)
         parsed = parseNSC(duration)
       elseif frameKind == "linefour" then
       elseif frameKind == "linefour" then
         parsed = parseLineFour(duration)
         parsed = parseLineFour(duration)
      elseif frameKind == "linetwo" then
        if frame.args[frameKind] ~= nil then
          parsed = parseLineTwo(duration)
        end
       end
       end
    -- 如果parsed为nil,表示解析失败,报错
      -- 如果parsed为nil,表示解析失败,报错
       if parsed == nil then
       if parsed == nil then
         return string.format('<span class="error">The number of %s frames is not a simple number (%s). Please specify explicitly.</span>', frameKind, tostring(duration))
         return string.format(
          '<span class="error">The number of %s frames is not a simple number (%s). Please specify explicitly.</span>',
          frameKind, tostring(duration))
       end
       end
    -- 如果解析成功,遍历解析后的表parsed,将每个字段值赋给frame.args[k]
      -- 如果解析成功,遍历解析后的表parsed,将每个字段值赋给frame.args[k]
       for k, v in pairs(parsed) do -- parsed格式会是{startup = s, active = a, recovery = r}的数据格式
       for k, v in pairs(parsed) do -- parsed格式会是{startup = s, active = a, recovery = r}的数据格式
         frame.args[k] = v
         frame.args[k] = v
       end
       end
第49行: 第72行:


function metaFrameData(frame)
function metaFrameData(frame)
currentFrame = 0 --跟踪当前帧数的变量
  currentFrame = 0             --跟踪当前帧数的变量
html = mw.html.create('div') --mw的html方法,创造一个div标签
  html = mw.html.create('div') --mw的html方法,创造一个div标签
html:addClass('frameChart')   --给这个div标签添加frameChart类
  html:addClass('frameChart') --给这个div标签添加frameChart类
-- 如果传递的框架frame中有title参数,则将其值赋给title临时变量
  -- 如果传递的框架frame中有title参数,则将其值赋给title临时变量
local title
  local title
if frame.args['title'] ~= nil then
  if frame.args['title'] ~= nil then
title = frame.args['title']
    title = frame.args['title']
end
  end
-- 从传递的框架frame中获取startup、active、recovery值并赋值给同名变量
  -- 从传递的框架frame中获取startup、active、recovery值并赋值给同名变量
local startup = frame.args['startup']
  local startup = frame.args['startup']
local active = frame.args['active']
  local active = frame.args['active']
   local inactive = frame.args['inactive2']
   local inactive = frame.args['inactive2']
local recovery = frame.args['recovery']
  local recovery = frame.args['recovery']
local nsc = frame.args['nsc']
  local nc = frame.args['nc']
   local spe = frame.args['spe']
   local spe = frame.args['spe']
     local activeData = parseActive(active)
  local inv = frame.args['inv']
     local frameAll = parseStartup(startup).startup + parseActive(active).active + parseRecovery(recovery).recovery
  local super = frame.args['super']
    if inactive ~= nil then
  local iwc = frame.args['iwc']
  if iwc ~= nil then
     local i = 1
    while i < 5 do
      lineFrameIndex['l' .. tostring(i)] = tonumber(iwc)
      i = i + 1
     end
  end
  local frameAll = parseStartup(startup).startup + parseActive(active).active + parseRecovery(recovery).recovery
  if inactive ~= nil then
     local i = 2
     local i = 2
     while (frame.args['inactive' .. tostring(i)] ~= nil) do
     while (frame.args['inactive' .. tostring(i)] ~= nil) do
       frameAll = frameAll + frame.args['inactive' .. tostring(i)] + frame.args['active' .. tostring(i+1)]
       frameAll = frameAll + frame.args['inactive' .. tostring(i)] + frame.args['active' .. tostring(i + 1)]
       i = i + 2
       i = i + 2
      end
     end
     end
     local backWidth = 22 + 12 * frameAll
  end
    local backWidthArg = 'width: ' .. (math.ceil((12 * frameAll - 60) / 60) * 60 + 82) .. 'px'
  if isSuperflash == true then
    local backBlockN = math.ceil((backWidth - 82) / 60 )
     frameAll = frameAll + 1
    local scrollbarArg = 'scrollbar-color: var(--color-base) var(--color-surface-3); scrollbar-width: thin;'
  end
    -- 如果提供title变量,则创建一个包含title的div标签,并添加到HTML结构中,添加frameChart-title类
  local backWidth = 22 + 12 * frameAll
if title ~= nil then
  local backWidthArg = 'width: ' .. (math.ceil((12 * frameAll - 60) / 60) * 60 + 82) .. 'px'
html:tag('div'):addClass('frameChart-title'):wikitext(title):done()
  local backBlockN = math.ceil((backWidth - 82) / 60)
end
  local scrollbarArg = 'scrollbar-color: var(--color-base) var(--color-surface-3); scrollbar-width: thin;'
-- 创建div标签并添加类frameChart-data,将此帧数数据容器添加到HTML主结构中
  -- 如果提供title变量,则创建一个包含title的div标签,并添加到HTML结构中,添加frameChart-title类
    frameChartContainer = mw.html.create('div'):addClass('frameChart-back'):attr('style', scrollbarArg .. backWidthArg):done()
  if title ~= nil then
    html:node(frameChartContainer)
    html:tag('div'):addClass('frameChart-title'):wikitext(title):done()
    frameChartStrap = {frameChartStrap1, frameChartStrap2, frameChartStrap3, frameChartStrap4, frameChartStrap5, frameChartStrap6}
  end
    local frameChartStrapIndex = 1
  -- 创建div标签并添加类frameChart-data,将此帧数数据容器添加到HTML主结构中
    while frameChartStrapIndex < 7 do
  frameChartContainer = mw.html.create('div'):addClass('frameChart-back'):attr('style', scrollbarArg .. backWidthArg)
    frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-strap'):addClass('frameChart-back-strap-pos-' .. frameChartStrapIndex):attr('style', backWidthArg))
      :done()
frameChartStrapIndex = frameChartStrapIndex +1
  html:node(frameChartContainer)
    end
  frameChartStrap = { frameChartStrap1, frameChartStrap2, frameChartStrap3, frameChartStrap4, frameChartStrap5,
    frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-block-first'):done())
    frameChartStrap6 }
    local frameChartBlockIndex = 1
  local frameChartStrapIndex = 1
    while frameChartBlockIndex < (backBlockN + 1) do
  while frameChartStrapIndex < 7 do
    local leftArg = 'left: ' .. (80 + 60 * (frameChartBlockIndex - 1) + 2) .. 'px'
    frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-strap'):addClass(
    frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-block'):attr('style', leftArg))
      'frameChart-back-strap-pos-' .. frameChartStrapIndex):attr('style', backWidthArg))
    frameChartBlockIndex = frameChartBlockIndex + 1
    frameChartStrapIndex = frameChartStrapIndex + 1
    end
  end
    frameChartContainer:node(mw.html.create('div'):addClass('frameChart-startline'):done())
  frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-block-first'):done())
    frameChartBody = mw.html.create('div'):addClass('frameChart-body'):done()
  local frameChartBlockIndex = 1
frameChartContainer:node(frameChartBody)
  while frameChartBlockIndex < (backBlockN + 1) do
frameChartDataHtml1 = mw.html.create('div'):addClass('frameChart-data'):done()
    local leftArg = 'left: ' .. (80 + 60 * (frameChartBlockIndex - 1) + 2) .. 'px'
frameChartBody:node(frameChartDataHtml1)
    frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-block'):attr('style', leftArg))
    -- 调用drawFrame函数,绘制第一行启动帧占位
    frameChartBlockIndex = frameChartBlockIndex + 1
drawFrame(startup, "placeholder", 1)
  end
if isThrow(frame.args['properties']) then
  frameChartContainer:node(mw.html.create('div'):addClass('frameChart-startline'):done())
  frameChartBody = mw.html.create('div'):addClass('frameChart-body'):done()
  frameChartContainer:node(frameChartBody)
  frameChartDataHtml1 = mw.html.create('div'):addClass('frameChart-data'):done()
  frameChartBody:node(frameChartDataHtml1)
  -- 调用drawFrame函数,绘制第一行启动帧占位
  drawFrame(startup, "placeholder", 1)
  if isThrow(frame.args['properties']) then
     drawFrame(active, "throw", 1)
     drawFrame(active, "throw", 1)
   else
   else
     drawFrame(active, "strike", 1)
     drawFrame(active, "strike", 1)
   end
   end
-- 检查是否存在更多的非活跃帧,遍历处理,逻辑类似于上文的取消窗口
  -- 检查是否存在更多的非活跃帧,遍历处理,逻辑类似于上文的取消窗口
index = 2
  index = 2
while tonumber(frame.args['active' .. index]) ~= nil or tonumber(frame.args['inactive' .. index]) ~= nil do
  while tonumber(frame.args['active' .. index]) ~= nil or tonumber(frame.args['inactive' .. index]) ~= nil do
drawFrame(frame.args['active' .. index], "strike", 1)
    drawFrame(frame.args['active' .. index], "strike", 1)
drawFrame(frame.args['inactive' .. index], "placeholder", 1)
    drawFrame(frame.args['inactive' .. index], "placeholder", 1)
index = index + 1
    index = index + 1
end
  end


frameChartDataHtml2 = mw.html.create('div'):addClass('frameChart-data'):done()
  frameChartDataHtml2 = mw.html.create('div'):addClass('frameChart-data'):done()
frameChartBody:node(frameChartDataHtml2)
  frameChartBody:node(frameChartDataHtml2)
   if isInvul(frame.args['properties']) == false and frame.args['vul'] == nil then
   if isInvul(frame.args['properties']) == false and inv == nil then
     drawFrame(startup, "vuln", 2)
     drawFrame(startup, "vuln", 2)
     drawFrame(active, "vuln", 2)
     drawFrame(active, "vuln", 2)
第130行: 第168行:
     drawFrame(recovery, "vuln", 2)
     drawFrame(recovery, "vuln", 2)
   else
   else
     drawFrame(startup, "strike", 2)
     drawFrame(inv, tostring(frame.args['invtype']), 2)
    local invi = 2
    while (frame.args['inv' .. tostring(invi)] ~= nil) do
      drawFrame(frame.args['inv' .. tostring(invi)], tostring(frame.args['invtype' .. tostring(invi)]), 2)
      invi = invi + 1
    end
   end
   end


frameChartDataHtml3 = mw.html.create('div'):addClass('frameChart-data'):done()
  frameChartDataHtml3 = mw.html.create('div'):addClass('frameChart-data'):done()
frameChartBody:node(frameChartDataHtml3)
  frameChartBody:node(frameChartDataHtml3)
    -- 调用drawFrame函数,绘制第一行启动帧占位
  -- 调用drawFrame函数,绘制第一行启动帧占位
drawFrame(nsc, "placeholder", 3)
  if tonumber(frame.args['nc2']) == nil then
-- 检查是否存在更多的非活跃帧,遍历处理,逻辑类似于上文的取消窗口
    drawFrame(nc, "nc", 3)
index = 2
  else
while tonumber(frame.args['nsc' .. index]) ~= nil do
    drawFrame(nc, "placeholder", 3)
drawFrame(frame.args['nsc' .. index], "nsc", 3)
    -- 检查是否存在更多的非活跃帧,遍历处理,逻辑类似于上文的取消窗口
drawFrame(frame.args['nnsc' .. index], "placeholder", 3)
    index = 2
index = index + 1
    while tonumber(frame.args['nc' .. index]) ~= nil do
end
      drawFrame(frame.args['nc' .. index], "nc", 3)
      drawFrame(frame.args['nnc' .. index], "placeholder", 3)
      index = index + 1
    end
  end
  if isSuperflash and nc == nil then
    drawFrame(f + 1, "placeholder", 3)
  end


   if spe ~= nil then
   frameChartDataHtml4 = mw.html.create('div'):addClass('frameChart-data'):done()
    frameChartDataHtml4 = mw.html.create('div'):addClass('frameChart-data'):done()
  frameChartBody:node(frameChartDataHtml4)
    frameChartBody:node(frameChartDataHtml4)
  drawFrame(spe, tostring(frame.args['spetype']), 4)
    if frame.args['spetype'] == nil then
  local spei = 2
      drawFrame(spe, "placeholder", 4)
  while (frame.args['spe' .. tostring(spei)] ~= nil) do
    else
    drawFrame(frame.args['spe' .. tostring(spei)], tostring(frame.args['spetype' .. tostring(spei)]), 4)
      drawFrame(spe, tostring(frame.args['spetype']), 4)
     spei = spei + 1
    end
  end
    local i = 2
  if isSuperflash and spe == nil then
    while frame.args['spe' .. tostring(i)] do
    drawFrame(f + 1, "placeholder", 4)
      drawFrame(frame.args['spe' .. tostring(i)], tostring(frame.args['spetype' .. tostring(i)]), 4)
   end
     end
   end
   -- 将生成的HTML结构转换为字符串,并使用定义在Module:FrameChart/styles.css中的样式渲染
   -- 将生成的HTML结构转换为字符串,并使用定义在Module:FrameChart/styles.css中的样式渲染
return tostring(html) .. mw.getCurrentFrame():extensionTag{
  return tostring(html) -- .. mw.getCurrentFrame():extensionTag { name = 'templatestyles', args = { src = 'Module:FrameChart/styles.css' } }
name = 'templatestyles', args = { src = 'Module:FrameChart/styles.css' }
}
end
end


function drawFrame(frames, frameType, line) -- frames为绘制的帧数,是数字;frameType表示帧类型,如active等
function drawFrame(frames, frameType, line)       -- frames为绘制的帧数,是数字;frameType表示帧类型,如active等
if tonumber(frames) ~= nil then   -- 将frames转化为数字,如果转化成功说明其为一个有效的数字字符串
  if tonumber(frames) ~= nil then                 -- 将frames转化为数字,如果转化成功说明其为一个有效的数字字符串
for i=1, tonumber(frames) do   -- 循环从1到frames(帧数)次
    for i = 1, tonumber(frames) do               -- 循环从1到frames(帧数)次
--currentFrame = currentFrame + 1  -- 每次循环时更新currentFrame数
      local frameDataHtml = mw.html.create('div') --创建新div标签表示单个帧(小方块)
local frameDataHtml = mw.html.create('div') --创建新div标签表示单个帧(小方块)
      local frameDataHtmlSuper = mw.html.create('div')
frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):done() --为其添加类frame-data和类frame-data-frameType
      if tonumber(line) == 1 then
if tonumber(line) == 1 then
        if lineOneIndex - 1 == superIndex['l1'] then
frameChartDataHtml1:node(frameDataHtml) --完成div标签列的构建并添加到frameChartDataHtml1结构中
          frameDataHtmlSuper:addClass('frame-data'):addClass('frame-data-superflash'):attr('id',
elseif tonumber(line) == 2 then
            'framedata-' .. tostring(line) .. '-s'):done()
frameChartDataHtml2:node(frameDataHtml)
          superIndex['l1'] = -1
elseif tonumber(line) == 3 then
          frameChartDataHtml1:node(frameDataHtmlSuper)
frameChartDataHtml3:node(frameDataHtml)
        end
elseif tonumber(line) == 4 then
        if lineOneIndex < lineFrameIndex['l1'] then
frameChartDataHtml4:node(frameDataHtml)
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):addClass('frame-data-opacity'):attr('id',
end
          'framedata-' .. tostring(line) .. '-' .. tostring(lineOneIndex)):done()
end
        else
end
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineOneIndex)):done() --为其添加类frame-data和类frame-data-frameType
        end
        lineOneIndex = lineOneIndex + 1
        frameChartDataHtml1:node(frameDataHtml)                                   --完成div标签列的构建并添加到frameChartDataHtml1结构中
      elseif tonumber(line) == 2 then
        if lineTwoIndex - 1 == superIndex['l2'] then
          frameDataHtmlSuper:addClass('frame-data'):addClass('frame-data-superflash'):attr('id',
            'framedata-' .. tostring(line) .. '-s'):done()
          superIndex['l2'] = -1
          frameChartDataHtml2:node(frameDataHtmlSuper)
        end
        if lineTwoIndex < lineFrameIndex['l2'] then
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):addClass('frame-data-opacity'):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineTwoIndex)):done()
        else
        frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineTwoIndex)):done()
        end
        lineTwoIndex = lineTwoIndex + 1
        frameChartDataHtml2:node(frameDataHtml)
      elseif tonumber(line) == 3 then
        if lineThreeIndex - 1 == superIndex['l3'] then
          frameDataHtmlSuper:addClass('frame-data'):addClass('frame-data-superflash'):attr('id',
            'framedata-' .. tostring(line) .. '-s'):done()
          superIndex['l3'] = -1
          frameChartDataHtml3:node(frameDataHtmlSuper)
        end
        if lineThreeIndex < lineFrameIndex['l3'] then
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):addClass('frame-data-opacity'):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineThreeIndex)):done()
        else
        frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineThreeIndex)):done()
        end
        lineThreeIndex = lineThreeIndex + 1
        frameChartDataHtml3:node(frameDataHtml)
      elseif tonumber(line) == 4 then
        if lineFourIndex - 1 == superIndex['l4'] then
          frameDataHtmlSuper:addClass('frame-data'):addClass('frame-data-superflash'):attr('id',
            'framedata-' .. tostring(line) .. '-s'):done()
          superIndex['l4'] = -1
          frameChartDataHtml4:node(frameDataHtmlSuper)
        end
        if lineFourIndex < lineFrameIndex['l4'] then
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):addClass('frame-data-opacity'):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineFourIndex)):done()
        else
        frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineFourIndex)):done()
        end
        lineFourIndex = lineFourIndex + 1
        frameChartDataHtml4:node(frameDataHtml)
      end
    end
  end
end
end


第187行: 第288行:
   -- simple number
   -- simple number
   if tonumber(duration) ~= nil then
   if tonumber(duration) ~= nil then
     return {active = tonumber(duration)}
     return { active = tonumber(duration) }
   end
   end


第208行: 第309行:
       totalActive = totalActive + tonumber(dur)
       totalActive = totalActive + tonumber(dur)
     end
     end
     if pos ~= string.len(duration)+1 then
     if pos ~= string.len(duration) + 1 then
       return nil -- trailing stuff at the end
       return nil -- trailing stuff at the end
     end
     end
     -- Done.
     -- Done.
     local out = {active = totalActive}
     local out = { active = totalActive }
     return out  
     return out
   elseif mw.ustring.find(duration, "^%d+[x×]%d+$") ~= nil then
   elseif mw.ustring.find(duration, "^%d+[x×]%d+$") ~= nil then
     -- 3x4 format -- also multihit with no gaps
     -- 3x4 format -- also multihit with no gaps
第233行: 第334行:
       end
       end
       out['inactive' .. tostring(ordinal)] = d1
       out['inactive' .. tostring(ordinal)] = d1
       out['active' .. tostring(ordinal+1)] = d2
       out['active' .. tostring(ordinal + 1)] = d2
       ordinal = ordinal + 2
       ordinal = ordinal + 2
       pos = p2
       pos = p2
第250行: 第351行:
   -- simple number
   -- simple number
   if tonumber(duration) ~= nil then
   if tonumber(duration) ~= nil then
     return {startup = tonumber(duration)}
     return { startup = tonumber(duration) }
   end
   end


第256行: 第357行:
   if string.find(duration, "^%d+%+%d+$") ~= nil then
   if string.find(duration, "^%d+%+%d+$") ~= nil then
     local first, second = string.match(duration, "^(%d+)%+(%d+)$")
     local first, second = string.match(duration, "^(%d+)%+(%d+)$")
     return {startup = tonumber(first) + tonumber(second) }
     return { startup = tonumber(first) + tonumber(second) }
   end
   end
   return nil
   return nil
第264行: 第365行:
   -- simple number
   -- simple number
   if tonumber(duration) ~= nil then
   if tonumber(duration) ~= nil then
     return {recovery = tonumber(duration)}
     return { recovery = tonumber(duration) }
   end
   end


第270行: 第371行:
   if string.find(duration, "^Until L%+%d+$") ~= nil then
   if string.find(duration, "^Until L%+%d+$") ~= nil then
     local recovery = string.match(duration, "^Until L%+(%d+)$")
     local recovery = string.match(duration, "^Until L%+(%d+)$")
     return {recovery=0, specialRecovery=recovery }
     return { recovery = 0, specialRecovery = recovery }
   end
   end
   return nil
   return nil
第277行: 第378行:
function isThrow(str)
function isThrow(str)
   if str ~= nil then
   if str ~= nil then
     if string.find(str, "{{特性|}}") then
     if string.find(str, "投") then
       return true
       if string.find(str, "对空投") then
    elseif string.find(str, "{{特性|空投}}") then
        return false
      return true
      elseif string.find(str, "打击投") then
    elseif string.find(str, "{{特性|指令投}}") then
        return false
       return true
       else
        return true
      end
     else
     else
       return false
       return false
第301行: 第404行:
function parseNSC(duration)
function parseNSC(duration)
   if tonumber(duration) ~= nil then
   if tonumber(duration) ~= nil then
     return {nsc = tonumber(duration)}
     return { nc = tonumber(duration) }
   end
   end
   if string.find(duration, "^%d+%(%d+%)") ~= nil then
   if string.find(duration, "^%d+%(%d+%)") ~= nil then
     local out = {}
     local out = {}
     local firstval, pos = string.match(duration, "^(%d+)()")
     local firstval, pos = string.match(duration, "^(%d+)()")
     out['nsc'] = firstval
     out['nc'] = firstval
     local ordinal = 2
     local ordinal = 2
     for p1, d1, d2, p2 in string.gmatch(duration, "()%((%d+)%)(%d+)()") do
     for p1, d1, d2, p2 in string.gmatch(duration, "()%((%d+)%)(%d+)()") do
第312行: 第415行:
         return nil
         return nil
       end
       end
       out['nsc' .. tostring(ordinal)] = d1
       out['nc' .. tostring(ordinal)] = d1
       out['nnsc' .. tostring(ordinal)] = d2
       out['nnc' .. tostring(ordinal)] = d2
       ordinal = ordinal + 1
       ordinal = ordinal + 1
       pos = p2
       pos = p2
第327行: 第430行:
function parseLineFour(duration)
function parseLineFour(duration)
   local out = {}
   local out = {}
   if string.find(duration, "^%a+%d+") == nil then
   if tostring(duration) == nil then
     local firstval, pos = string.match(duration, "^(%d+)()")
     return out
    out['spe'] = firstval
  end
   else
  local firsttype, firstval, pos = string.match(duration, "^(%a+)(%d+)()")
     local firsttype, firstval, pos = string.match(duration, "^(%a+)(%d+)()")
  out['spetype'] = firsttype
     if firstval == nil then
  out['spe'] = firstval
   if string.find(duration, ",") ~= nil then
     local ordinal = 2
    for p1, d1, d2, p2 in string.gmatch(duration, "(),(%a+)(%d+)()") do
      if pos ~= p1 then
        return nil
      end
      out['spetype' .. tostring(ordinal)] = d1
      out['spe' .. tostring(ordinal)] = d2
      ordinal = ordinal + 1
      pos = p2
    end
     if pos ~= string.len(duration) + 1 then
       return nil
       return nil
     end
     end
     out['spe'] = firstval
  end
    out['spetype'] = firsttype
  out['spetype'] = replaceSpeAbbreviations(out['spetype'])
  local i = 2
  while (out['spetype' .. tostring(i)] ~= nil) do
     out['spetype' .. tostring(i)] = replaceSpeAbbreviations(out['spetype' .. tostring(i)])
    i = i + 1
  end
  return out
end
 
function parseLineTwo(duration)
  local out = {}
  if tostring(duration) == nil then
    return out
  end
  local firsttype, firstval, pos = string.match(duration, "^(%a+)(%d+)()")
  out['invtype'] = firsttype
  out['inv'] = firstval
  if string.find(duration, ",") ~= nil then
     local ordinal = 2
     local ordinal = 2
     for p1, type, dur, p2 in string.gmatch(duration, "(),%s*(%a+)(%d+)%s*()") do
     for p1, d1, d2, p2 in string.gmatch(duration, "(),(%a+)(%d+)()") do
       if pos ~= p1 then
       if pos ~= p1 then
         return nil
         return nil
       end
       end
       out['spe' .. tostring(ordinal)] = dur
       out['invtype' .. tostring(ordinal)] = d1
       out['spetype' .. tostring(ordinal)] = type
       out['inv' .. tostring(ordinal)] = d2
       ordinal = ordinal + 1
       ordinal = ordinal + 1
       pos = p2
       pos = p2
      if pos ~= string.len(duration) + 1 then
        return nil
      end
     end
     end
     if out['spetype'] ~= nil then
     if pos ~= string.len(duration) + 1 then
      if string.match(out['spetype'], "ph") ~= nil then
      return nil
        string.gsub(out['spetype'], "ph", "placeholder")
    end
       end
  end
       if string.match(out['spetype'], "arm") ~= nil then
  out['invtype'] = replaceInvAbbreviations(out['invtype'])
        string.gsub(out['spetype'], "arm", "armor")
  local i = 2
      end
  while (out['invtype' .. tostring(i)] ~= nil) do
      if string.match(out['spetype'], "hyp") ~= nil then
    out['invtype' .. tostring(i)] = replaceInvAbbreviations(out['invtype' .. tostring(i)])
        string.gsub(out['spetype'], "hyp", "hyperarmor")
    i = i + 1
      end
  end
      if string.match(out['spetype'], "pro") ~= nil then
  return out
        string.gsub(out['spetype'], "pro", "hyperarmor-proj")
end
       end
 
      if string.match(out['spetype'], "hat") ~= nil then
function replaceInvAbbreviations(field)
        string.gsub(out['spetype'], "hat", "hatredguard")
  if field ~= nil then
      end
    if string.find(field, "ph") ~= nil then
      if string.match(out['spetype'], "noc") ~= nil then
      field = string.gsub(field, "ph", "placeholder")
        string.gsub(out['spetype'], "noc", "nocollision")
    end
      end
    if string.find(field, "vul") ~= nil then
      if string.match(out['spetype'], "nocwh") ~= nil then
       field = string.gsub(field, "vul", "vuln")
        string.gsub(out['spetype'], "nocwh", "nocollision-when-hit")
    end
       end
    if string.find(field, "ful") ~= nil then
       field = string.gsub(field, "ful", "invuln-full")
    end
    if string.find(field, "stk") ~= nil then
      field = string.gsub(field, "stk", "invuln-strike")
    end
    if string.find(field, "thr") ~= nil then
      field = string.gsub(field, "thr", "invuln-throw")
    end
    if string.find(field, "pro") ~= nil then
      field = string.gsub(field, "pro", "invuln-proj")
    end
  end
  return field
end
 
function replaceSpeAbbreviations(field)
  if field ~= nil then
    if string.find(field, "ph") ~= nil then
       field = string.gsub(field, "ph", "placeholder")
    end
    if string.find(field, "arm") ~= nil then
      field = string.gsub(field, "arm", "armor")
    end
    if string.find(field, "hyp") ~= nil then
      field = string.gsub(field, "hyp", "hyperarmor")
    end
    if string.find(field, "pro") ~= nil then
      field = string.gsub(field, "pro", "hyperarmor-proj")
    end
    if string.find(field, "hat") ~= nil then
       field = string.gsub(field, "hat", "hatredguard")
    end
    if string.find(field, "noc") ~= nil then
      field = string.gsub(field, "noc", "nocollision")
     end
     end
     local i = 2
     if string.find(field, "nwh") ~= nil then
    while out['spetype' .. tostring(i)] do
       field = string.gsub(field, "nwh", "nocollision-when-hit")
      if string.match(out['spetype'] .. tostring(i), "ph") ~= nil then
        string.gsub(out['spetype'] .. tostring(i), "ph", "placeholder")
      end
       if string.match(out['spetype' .. tostring(i)], "arm") ~= nil then
        string.gsub(out['spetype' .. tostring(i)], "arm", "armor")
      end
      if string.match(out['spetype' .. tostring(i)], "hyp") ~= nil then
        string.gsub(out['spetype' .. tostring(i)], "hyp", "hyperarmor")
      end
      if string.match(out['spetype' .. tostring(i)], "pro") ~= nil then
        string.gsub(out['spetype' .. tostring(i)], "pro", "hyperarmor-proj")
      end
      if string.match(out['spetype' .. tostring(i)], "hat") ~= nil then
        string.gsub(out['spetype' .. tostring(i)], "hat", "hatredguard")
      end
      if string.match(out['spetype' .. tostring(i)], "noc") ~= nil then
        string.gsub(out['spetype' .. tostring(i)], "noc", "nocollision")
      end
      if string.match(out['spetype' .. tostring(i)], "nocwh") ~= nil then
        string.gsub(out['spetype' .. tostring(i)], "nocwh", "nocollision-when-hit")
      end
      i = i + 1
     end
     end
    return out
   end
   end
   return nil
   return field
end
end


p.drawFrame = drawFrame
p.drawFrame = drawFrame
return p
return p

2025年1月15日 (三) 00:24的最新版本

此模块的文档可以在模块:FrameChart/doc创建

-- FrameChart
local p = {}
local html
local frameChartDataHtml
local lineOneIndex = 1
local lineTwoIndex = 1
local lineThreeIndex = 1
local lineFourIndex = 1
local isSuperflash = false
local superIndex = { l1 = -1, l2 = -1, l3 = -1, l4 = -1 }
local lineFrameIndex = { l1 = -1, l2 = -1, l3 = -1, l4 = -1 }
local f = ''

function p.drawFrameData(frame)
  -- 开始循环,遍历包含{startup, active, recovery}的列表
  -- 对于每种帧类型,从fd中获取相应的帧数数据并存储在duration中
  -- 如果frame参数中有对应的帧类型字段,使用frame:preprocess方法处理该字段值
  -- _, frameKind表示,忽略迭代过程中的第一个返回值(即k,键),只关心第二个返回值(即v,值,循环变量命名为frameKind)
  -- inpairs保证以数组顺序(1-n)遍历数组,所以以下代码意为将frameKind依次赋值为s、a、r并进行循环体操作
  for _, frameKind in ipairs({ "startup", "active", "recovery", "nc", "linefour", "linetwo", "iwc" }) do
    local duration
    if frame.args[frameKind] ~= nil then                 -- frame.args[frameKind]表示访问frame表中frameKind键对应的值
      duration = frame:preprocess(frame.args[frameKind]) -- frame:preprocess的参数中如有MW模板或变量(如{{Var}}),会自动展开
    end
    -- 如果duration为nil,则跳过这个帧类型
    if duration == nil then
      -- 如果duration可以直接转化为数字,则直接赋值给frame.args[frameKind]
    elseif tonumber(duration) ~= nil then
      frame.args[frameKind] = duration
      -- 如果不是数字,则使用对应的解析函数解析帧数字并存储于
    else
      local parsed = nil
      if frameKind == "active" then
        parsed = parseActive(duration)
      elseif frameKind == "startup" then
        parsed = parseStartup(duration)
        if string.find(duration, "+") then
          isSuperflash = true
          f = string.match(duration, "^(%d+)%+")
          local i = 1
          while i < 5 do
            superIndex['l' .. tostring(i)] = tonumber(f)
            i = i + 1
          end
        end
      elseif frameKind == "recovery" then
        parsed = parseRecovery(duration)
      elseif frameKind == "nc" then
        parsed = parseNSC(duration)
      elseif frameKind == "linefour" then
        parsed = parseLineFour(duration)
      elseif frameKind == "linetwo" then
        if frame.args[frameKind] ~= nil then
          parsed = parseLineTwo(duration)
        end
      end
      -- 如果parsed为nil,表示解析失败,报错
      if parsed == nil then
        return string.format(
          '<span class="error">The number of %s frames is not a simple number (%s). Please specify explicitly.</span>',
          frameKind, tostring(duration))
      end
      -- 如果解析成功,遍历解析后的表parsed,将每个字段值赋给frame.args[k]
      for k, v in pairs(parsed) do -- parsed格式会是{startup = s, active = a, recovery = r}的数据格式
        frame.args[k] = v
      end
    end
  end
  -- 将此模块中p.drawFrameData的参数处理后输入给调用Module:MetaFrameChart的drawFrameData函数
  return metaFrameData(frame)
end

function metaFrameData(frame)
  currentFrame = 0             --跟踪当前帧数的变量
  html = mw.html.create('div') --mw的html方法,创造一个div标签
  html:addClass('frameChart')  --给这个div标签添加frameChart类
  -- 如果传递的框架frame中有title参数,则将其值赋给title临时变量
  local title
  if frame.args['title'] ~= nil then
    title = frame.args['title']
  end
  -- 从传递的框架frame中获取startup、active、recovery值并赋值给同名变量
  local startup = frame.args['startup']
  local active = frame.args['active']
  local inactive = frame.args['inactive2']
  local recovery = frame.args['recovery']
  local nc = frame.args['nc']
  local spe = frame.args['spe']
  local inv = frame.args['inv']
  local super = frame.args['super']
  local iwc = frame.args['iwc']
  if iwc ~= nil then
    local i = 1
    while i < 5 do
      lineFrameIndex['l' .. tostring(i)] = tonumber(iwc)
      i = i + 1
    end
  end
  local frameAll = parseStartup(startup).startup + parseActive(active).active + parseRecovery(recovery).recovery
  if inactive ~= nil then
    local i = 2
    while (frame.args['inactive' .. tostring(i)] ~= nil) do
      frameAll = frameAll + frame.args['inactive' .. tostring(i)] + frame.args['active' .. tostring(i + 1)]
      i = i + 2
    end
  end
  if isSuperflash == true then
    frameAll = frameAll + 1
  end
  local backWidth = 22 + 12 * frameAll
  local backWidthArg = 'width: ' .. (math.ceil((12 * frameAll - 60) / 60) * 60 + 82) .. 'px'
  local backBlockN = math.ceil((backWidth - 82) / 60)
  local scrollbarArg = 'scrollbar-color: var(--color-base) var(--color-surface-3); scrollbar-width: thin;'
  -- 如果提供title变量,则创建一个包含title的div标签,并添加到HTML结构中,添加frameChart-title类
  if title ~= nil then
    html:tag('div'):addClass('frameChart-title'):wikitext(title):done()
  end
  -- 创建div标签并添加类frameChart-data,将此帧数数据容器添加到HTML主结构中
  frameChartContainer = mw.html.create('div'):addClass('frameChart-back'):attr('style', scrollbarArg .. backWidthArg)
      :done()
  html:node(frameChartContainer)
  frameChartStrap = { frameChartStrap1, frameChartStrap2, frameChartStrap3, frameChartStrap4, frameChartStrap5,
    frameChartStrap6 }
  local frameChartStrapIndex = 1
  while frameChartStrapIndex < 7 do
    frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-strap'):addClass(
      'frameChart-back-strap-pos-' .. frameChartStrapIndex):attr('style', backWidthArg))
    frameChartStrapIndex = frameChartStrapIndex + 1
  end
  frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-block-first'):done())
  local frameChartBlockIndex = 1
  while frameChartBlockIndex < (backBlockN + 1) do
    local leftArg = 'left: ' .. (80 + 60 * (frameChartBlockIndex - 1) + 2) .. 'px'
    frameChartContainer:node(mw.html.create('div'):addClass('frameChart-back-block'):attr('style', leftArg))
    frameChartBlockIndex = frameChartBlockIndex + 1
  end
  frameChartContainer:node(mw.html.create('div'):addClass('frameChart-startline'):done())
  frameChartBody = mw.html.create('div'):addClass('frameChart-body'):done()
  frameChartContainer:node(frameChartBody)
  frameChartDataHtml1 = mw.html.create('div'):addClass('frameChart-data'):done()
  frameChartBody:node(frameChartDataHtml1)
  -- 调用drawFrame函数,绘制第一行启动帧占位
  drawFrame(startup, "placeholder", 1)
  if isThrow(frame.args['properties']) then
    drawFrame(active, "throw", 1)
  else
    drawFrame(active, "strike", 1)
  end
  -- 检查是否存在更多的非活跃帧,遍历处理,逻辑类似于上文的取消窗口
  index = 2
  while tonumber(frame.args['active' .. index]) ~= nil or tonumber(frame.args['inactive' .. index]) ~= nil do
    drawFrame(frame.args['active' .. index], "strike", 1)
    drawFrame(frame.args['inactive' .. index], "placeholder", 1)
    index = index + 1
  end

  frameChartDataHtml2 = mw.html.create('div'):addClass('frameChart-data'):done()
  frameChartBody:node(frameChartDataHtml2)
  if isInvul(frame.args['properties']) == false and inv == nil then
    drawFrame(startup, "vuln", 2)
    drawFrame(active, "vuln", 2)
    index = 2
    while tonumber(frame.args['active' .. index]) ~= nil or tonumber(frame.args['inactive' .. index]) ~= nil do
      drawFrame(frame.args['active' .. index], "vuln", 2)
      drawFrame(frame.args['inactive' .. index], "vuln", 2)
      index = index + 1
    end
    drawFrame(recovery, "vuln", 2)
  else
    drawFrame(inv, tostring(frame.args['invtype']), 2)
    local invi = 2
    while (frame.args['inv' .. tostring(invi)] ~= nil) do
      drawFrame(frame.args['inv' .. tostring(invi)], tostring(frame.args['invtype' .. tostring(invi)]), 2)
      invi = invi + 1
    end
  end

  frameChartDataHtml3 = mw.html.create('div'):addClass('frameChart-data'):done()
  frameChartBody:node(frameChartDataHtml3)
  -- 调用drawFrame函数,绘制第一行启动帧占位
  if tonumber(frame.args['nc2']) == nil then
    drawFrame(nc, "nc", 3)
  else
    drawFrame(nc, "placeholder", 3)
    -- 检查是否存在更多的非活跃帧,遍历处理,逻辑类似于上文的取消窗口
    index = 2
    while tonumber(frame.args['nc' .. index]) ~= nil do
      drawFrame(frame.args['nc' .. index], "nc", 3)
      drawFrame(frame.args['nnc' .. index], "placeholder", 3)
      index = index + 1
    end
  end
  if isSuperflash and nc == nil then
    drawFrame(f + 1, "placeholder", 3)
  end

  frameChartDataHtml4 = mw.html.create('div'):addClass('frameChart-data'):done()
  frameChartBody:node(frameChartDataHtml4)
  drawFrame(spe, tostring(frame.args['spetype']), 4)
  local spei = 2
  while (frame.args['spe' .. tostring(spei)] ~= nil) do
    drawFrame(frame.args['spe' .. tostring(spei)], tostring(frame.args['spetype' .. tostring(spei)]), 4)
    spei = spei + 1
  end
  if isSuperflash and spe == nil then
    drawFrame(f + 1, "placeholder", 4)
  end
  -- 将生成的HTML结构转换为字符串,并使用定义在Module:FrameChart/styles.css中的样式渲染
  return tostring(html) -- .. mw.getCurrentFrame():extensionTag { name = 'templatestyles', args = { src = 'Module:FrameChart/styles.css' } }
end

function drawFrame(frames, frameType, line)       -- frames为绘制的帧数,是数字;frameType表示帧类型,如active等
  if tonumber(frames) ~= nil then                 -- 将frames转化为数字,如果转化成功说明其为一个有效的数字字符串
    for i = 1, tonumber(frames) do                -- 循环从1到frames(帧数)次
      local frameDataHtml = mw.html.create('div') --创建新div标签表示单个帧(小方块)
      local frameDataHtmlSuper = mw.html.create('div')
      if tonumber(line) == 1 then
        if lineOneIndex - 1 == superIndex['l1'] then
          frameDataHtmlSuper:addClass('frame-data'):addClass('frame-data-superflash'):attr('id',
            'framedata-' .. tostring(line) .. '-s'):done()
          superIndex['l1'] = -1
          frameChartDataHtml1:node(frameDataHtmlSuper)
        end
        if lineOneIndex < lineFrameIndex['l1'] then
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):addClass('frame-data-opacity'):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineOneIndex)):done()
        else
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineOneIndex)):done() --为其添加类frame-data和类frame-data-frameType
        end
        lineOneIndex = lineOneIndex + 1
        frameChartDataHtml1:node(frameDataHtml)                                   --完成div标签列的构建并添加到frameChartDataHtml1结构中
      elseif tonumber(line) == 2 then
        if lineTwoIndex - 1 == superIndex['l2'] then
          frameDataHtmlSuper:addClass('frame-data'):addClass('frame-data-superflash'):attr('id',
            'framedata-' .. tostring(line) .. '-s'):done()
          superIndex['l2'] = -1
          frameChartDataHtml2:node(frameDataHtmlSuper)
        end
        if lineTwoIndex < lineFrameIndex['l2'] then
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):addClass('frame-data-opacity'):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineTwoIndex)):done()
        else
        frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineTwoIndex)):done()
        end
        lineTwoIndex = lineTwoIndex + 1
        frameChartDataHtml2:node(frameDataHtml)
      elseif tonumber(line) == 3 then
        if lineThreeIndex - 1 == superIndex['l3'] then
          frameDataHtmlSuper:addClass('frame-data'):addClass('frame-data-superflash'):attr('id',
            'framedata-' .. tostring(line) .. '-s'):done()
          superIndex['l3'] = -1
          frameChartDataHtml3:node(frameDataHtmlSuper)
        end
        if lineThreeIndex < lineFrameIndex['l3'] then
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):addClass('frame-data-opacity'):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineThreeIndex)):done()
        else
        frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineThreeIndex)):done()
        end
        lineThreeIndex = lineThreeIndex + 1
        frameChartDataHtml3:node(frameDataHtml)
      elseif tonumber(line) == 4 then
        if lineFourIndex - 1 == superIndex['l4'] then
          frameDataHtmlSuper:addClass('frame-data'):addClass('frame-data-superflash'):attr('id',
            'framedata-' .. tostring(line) .. '-s'):done()
          superIndex['l4'] = -1
          frameChartDataHtml4:node(frameDataHtmlSuper)
        end
        if lineFourIndex < lineFrameIndex['l4'] then
          frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):addClass('frame-data-opacity'):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineFourIndex)):done()
        else
        frameDataHtml:addClass('frame-data'):addClass('frame-data-' .. frameType):attr('id',
          'framedata-' .. tostring(line) .. '-' .. tostring(lineFourIndex)):done()
        end
        lineFourIndex = lineFourIndex + 1
        frameChartDataHtml4:node(frameDataHtml)
      end
    end
  end
end

-- 解析函数,类似MetaFrameChart模块中的功能
function parseActive(duration)
  -- simple number
  if tonumber(duration) ~= nil then
    return { active = tonumber(duration) }
  end

  if string.find(duration, "%d+%s*,") ~= nil then
    -- 1,2,3,4 format -- multihit with no gaps
    local totalActive = 0

    -- first match - just a number
    local firstval, pos = string.match(duration, "^(%d+)%s*()")
    if firstval == nil then
      return nil
    end
    -- subsequent matches - coma, then number. Might have spaces between them
    totalActive = totalActive + tonumber(firstval)
    for p1, dur, p2 in string.gmatch(duration, "(),%s*(%d+)%s*()") do
      if pos ~= p1 then
        return nil
      end
      pos = p2
      totalActive = totalActive + tonumber(dur)
    end
    if pos ~= string.len(duration) + 1 then
      return nil -- trailing stuff at the end
    end
    -- Done.
    local out = { active = totalActive }
    return out
  elseif mw.ustring.find(duration, "^%d+[x×]%d+$") ~= nil then
    -- 3x4 format -- also multihit with no gaps
    local a, b = mw.ustring.match(duration, "^(%d+)[x×](%d+)$")
    local out = { active = tonumber(a) * tonumber(b) }
    return out
  elseif string.find(duration, "^%d+%(%d+%)") ~= nil then
    -- 1(2)3(4)5 format -- multihit with gaps
    local out = {}
    -- special handling for the first number
    local firstval, pos = string.match(duration, "^(%d+)()")
    out['active'] = firstval

    local ordinal = 2
    -- then we just have a groups of "(inactive)active"
    for p1, d1, d2, p2 in string.gmatch(duration, "()%((%d+)%)(%d+)()") do
      if pos ~= p1 then
        return nil -- not directly following the previous match, so basically not what we expect
      end
      out['inactive' .. tostring(ordinal)] = d1
      out['active' .. tostring(ordinal + 1)] = d2
      ordinal = ordinal + 2
      pos = p2
    end
    if pos ~= string.len(duration) + 1 then
      return nil -- trailing stuff at the end
    end
    -- Done.
    return out
  end
  -- unrecognized format
  return nil
end

function parseStartup(duration)
  -- simple number
  if tonumber(duration) ~= nil then
    return { startup = tonumber(duration) }
  end

  -- 1+2 -- common for supers
  if string.find(duration, "^%d+%+%d+$") ~= nil then
    local first, second = string.match(duration, "^(%d+)%+(%d+)$")
    return { startup = tonumber(first) + tonumber(second) }
  end
  return nil
end

function parseRecovery(duration)
  -- simple number
  if tonumber(duration) ~= nil then
    return { recovery = tonumber(duration) }
  end

  -- "Until L+10" -- aerial moves, such as grabs.
  if string.find(duration, "^Until L%+%d+$") ~= nil then
    local recovery = string.match(duration, "^Until L%+(%d+)$")
    return { recovery = 0, specialRecovery = recovery }
  end
  return nil
end

function isThrow(str)
  if str ~= nil then
    if string.find(str, "投") then
      if string.find(str, "对空投") then
        return false
      elseif string.find(str, "打击投") then
        return false
      else
        return true
      end
    else
      return false
    end
  end
end

function isInvul(str)
  if str ~= nil then
    if string.find(str, "无敌") then
      return true
    else
      return false
    end
  end
end

function parseNSC(duration)
  if tonumber(duration) ~= nil then
    return { nc = tonumber(duration) }
  end
  if string.find(duration, "^%d+%(%d+%)") ~= nil then
    local out = {}
    local firstval, pos = string.match(duration, "^(%d+)()")
    out['nc'] = firstval
    local ordinal = 2
    for p1, d1, d2, p2 in string.gmatch(duration, "()%((%d+)%)(%d+)()") do
      if pos ~= p1 then
        return nil
      end
      out['nc' .. tostring(ordinal)] = d1
      out['nnc' .. tostring(ordinal)] = d2
      ordinal = ordinal + 1
      pos = p2
    end
    if pos ~= string.len(duration) + 1 then
      return nil -- trailing stuff at the end
    end
    return out
  end
  return nil
end

function parseLineFour(duration)
  local out = {}
  if tostring(duration) == nil then
    return out
  end
  local firsttype, firstval, pos = string.match(duration, "^(%a+)(%d+)()")
  out['spetype'] = firsttype
  out['spe'] = firstval
  if string.find(duration, ",") ~= nil then
    local ordinal = 2
    for p1, d1, d2, p2 in string.gmatch(duration, "(),(%a+)(%d+)()") do
      if pos ~= p1 then
        return nil
      end
      out['spetype' .. tostring(ordinal)] = d1
      out['spe' .. tostring(ordinal)] = d2
      ordinal = ordinal + 1
      pos = p2
    end
    if pos ~= string.len(duration) + 1 then
      return nil
    end
  end
  out['spetype'] = replaceSpeAbbreviations(out['spetype'])
  local i = 2
  while (out['spetype' .. tostring(i)] ~= nil) do
    out['spetype' .. tostring(i)] = replaceSpeAbbreviations(out['spetype' .. tostring(i)])
    i = i + 1
  end
  return out
end

function parseLineTwo(duration)
  local out = {}
  if tostring(duration) == nil then
    return out
  end
  local firsttype, firstval, pos = string.match(duration, "^(%a+)(%d+)()")
  out['invtype'] = firsttype
  out['inv'] = firstval
  if string.find(duration, ",") ~= nil then
    local ordinal = 2
    for p1, d1, d2, p2 in string.gmatch(duration, "(),(%a+)(%d+)()") do
      if pos ~= p1 then
        return nil
      end
      out['invtype' .. tostring(ordinal)] = d1
      out['inv' .. tostring(ordinal)] = d2
      ordinal = ordinal + 1
      pos = p2
    end
    if pos ~= string.len(duration) + 1 then
      return nil
    end
  end
  out['invtype'] = replaceInvAbbreviations(out['invtype'])
  local i = 2
  while (out['invtype' .. tostring(i)] ~= nil) do
    out['invtype' .. tostring(i)] = replaceInvAbbreviations(out['invtype' .. tostring(i)])
    i = i + 1
  end
  return out
end

function replaceInvAbbreviations(field)
  if field ~= nil then
    if string.find(field, "ph") ~= nil then
      field = string.gsub(field, "ph", "placeholder")
    end
    if string.find(field, "vul") ~= nil then
      field = string.gsub(field, "vul", "vuln")
    end
    if string.find(field, "ful") ~= nil then
      field = string.gsub(field, "ful", "invuln-full")
    end
    if string.find(field, "stk") ~= nil then
      field = string.gsub(field, "stk", "invuln-strike")
    end
    if string.find(field, "thr") ~= nil then
      field = string.gsub(field, "thr", "invuln-throw")
    end
    if string.find(field, "pro") ~= nil then
      field = string.gsub(field, "pro", "invuln-proj")
    end
  end
  return field
end

function replaceSpeAbbreviations(field)
  if field ~= nil then
    if string.find(field, "ph") ~= nil then
      field = string.gsub(field, "ph", "placeholder")
    end
    if string.find(field, "arm") ~= nil then
      field = string.gsub(field, "arm", "armor")
    end
    if string.find(field, "hyp") ~= nil then
      field = string.gsub(field, "hyp", "hyperarmor")
    end
    if string.find(field, "pro") ~= nil then
      field = string.gsub(field, "pro", "hyperarmor-proj")
    end
    if string.find(field, "hat") ~= nil then
      field = string.gsub(field, "hat", "hatredguard")
    end
    if string.find(field, "noc") ~= nil then
      field = string.gsub(field, "noc", "nocollision")
    end
    if string.find(field, "nwh") ~= nil then
      field = string.gsub(field, "nwh", "nocollision-when-hit")
    end
  end
  return field
end

p.drawFrame = drawFrame
return p