Difference between revisions of "Module:Spell"

From CrawlWiki
Jump to: navigation, search
(Fixes. Show range 7 as LOS.)
(Fix missing entries in Hated By. Fix spell book sorting and capitalization.)
Line 3: Line 3:
 
local RANGE_LOS = 7
 
local RANGE_LOS = 7
  
local function table_keys_sorted(t)
+
local function table_keys_sorted(t, f)
 
   local keys = {}
 
   local keys = {}
 
   for k in pairs(t) do
 
   for k in pairs(t) do
 
     table.insert(keys, k)
 
     table.insert(keys, k)
 
   end
 
   end
   table.sort(keys)
+
   table.sort(keys, f)
 
   return keys
 
   return keys
 
end
 
end
Line 106: Line 106:
 
   end
 
   end
 
   return ret:sub(1, -3)
 
   return ret:sub(1, -3)
 +
end
 +
 +
local function compare_books(a, b)
 +
  return a:lower():gsub('^book of ', ''):gsub('^a ', ''):gsub('^the ', '') <
 +
    b:lower():gsub('^book of ', ''):gsub('^a ', ''):gsub('^the ', '')
 
end
 
end
  
 
local function format_books(books)
 
local function format_books(books)
 
   local ret = ''
 
   local ret = ''
   for _, book in ipairs(table_keys_sorted(books)) do
+
   for _, book in ipairs(table_keys_sorted(books, compare_books)) do
     ret = ret .. '[[' .. book .. ']]<br>'
+
     ret = ret .. '[[' .. book:gsub('^%l', string.upper) .. ']]<br>'
 
   end
 
   end
 
   return ret:sub(1, -5)
 
   return ret:sub(1, -5)
Line 168: Line 173:
  
 
function p.spell_table_by_school(frame)
 
function p.spell_table_by_school(frame)
   local schools = { 'Air', 'Charms', 'Conjuration', 'Earth', 'Fire', 'Hexes',
+
   local schools = {}
     'Ice', 'Necromancy', 'Poison', 'Summoning', 'Translocation',
+
  for _,spell in pairs(data) do
     'Transmutation' }
+
     for school in pairs(spell['schools']) do
 +
      schools[school] = true
 +
     end
 +
  end
  
 
   local ret = '==Spells==\n' .. spell_table_header()
 
   local ret = '==Spells==\n' .. spell_table_header()
   for _,school in ipairs(schools) do
+
   for _,school in ipairs(table_keys_sorted(schools)) do
 
     ret = ret .. spell_table_section(
 
     ret = ret .. spell_table_section(
 
       frame:expandTemplate{title = 'schoollink', args = {school}})
 
       frame:expandTemplate{title = 'schoollink', args = {school}})
Line 194: Line 202:
  
 
   local ret = '==Spells==\n' .. spell_table_header()
 
   local ret = '==Spells==\n' .. spell_table_header()
   for _,book in ipairs(table_keys_sorted(books)) do
+
   for _,book in ipairs(table_keys_sorted(books, compare_books)) do
     ret = ret .. spell_table_section('[[' .. book .. ']]')
+
     ret = ret ..
 +
      spell_table_section('[[' .. book:gsub('^%l', string.upper) .. ']]')
 
     for _, name in ipairs(names_by_level(data)) do
 
     for _, name in ipairs(names_by_level(data)) do
 
       if data[name]['books'][book] then
 
       if data[name]['books'][book] then
Line 237: Line 246:
 
end
 
end
  
local function format_hated_by(flags)
+
local function format_hated_by(spell)
 
   local output = ''
 
   local output = ''
   if flags.HASTY then
+
   if spell.flags.HASTY then
 
     output = output .. '[[Cheibriados]]<br>'
 
     output = output .. '[[Cheibriados]]<br>'
 
   end
 
   end
   if flags.CORPSE_VIOLATING then
+
   if spell.schools.Fire then
 +
    output = output .. '[[Dithmenos]]<br>'
 +
  end
 +
  if spell.flags.CORPSE_VIOLATING then
 
     output = output .. '[[Fedhas]]<br>'
 
     output = output .. '[[Fedhas]]<br>'
 
   end
 
   end
   if flags.UNHOLY then
+
   if spell.flags.UNHOLY or spell.schools.Necromancy then
 
     output = output .. '[[Elyvilon]]<br>'
 
     output = output .. '[[Elyvilon]]<br>'
 
   end
 
   end
   if flags.UNHOLY then
+
   if spell.flags.UNHOLY or spell.schools.Necromancy or spell.schools.Poison then
 
     output = output .. '[[The Shining One]]<br>'
 
     output = output .. '[[The Shining One]]<br>'
 
   end
 
   end
   if flags.CHAOTIC or flags.UNCLEAN or flags.UNHOLY then
+
   if spell.name == 'Statue Form' then
 +
    output = output .. '[[Yredelemnul]]<br>'
 +
  end
 +
  if spell.flags.CHAOTIC or spell.flags.UNCLEAN or spell.flags.UNHOLY or
 +
    spell.schools.Necromancy then
 
     output = output .. '[[Zin]]<br>'
 
     output = output .. '[[Zin]]<br>'
 
   end
 
   end
Line 303: Line 319:
 
   args.range = format_range(spell.range)
 
   args.range = format_range(spell.range)
 
   args.rarity = spell.rarity
 
   args.rarity = spell.rarity
   args['hated by'] = format_hated_by(spell.flags)
+
   args['hated by'] = format_hated_by(spell)
 
   args.flags = format_flags(spell.flags, {
 
   args.flags = format_flags(spell.flags, {
 
     CHAOTIC = true,
 
     CHAOTIC = true,

Revision as of 11:35, 14 November 2016

This module generates the spell tables.

Requires: Module:Table of spells


local p = {}
local data = mw.loadData('Module: Table of spells')
local RANGE_LOS = 7

local function table_keys_sorted(t, f)
  local keys = {}
  for k in pairs(t) do
    table.insert(keys, k)
  end
  table.sort(keys, f)
  return keys
end

local function empty_table(t)
  for _ in pairs(t) do
    return false
  end
  return true
end

local function names_by_level()
  local names = table_keys_sorted(data)
  table.sort(names, function(a, b)
    if data[a].level == data[b].level then
      return a < b
    else
      return data[a].level < data[b].level
    end
  end)
  return names
end

local function spell_table_header()
  return [=[{| class="prettytable"
!|Icon
!|Name
!|Schools
!|Level
!|Power<br>cap
!|Range
!|Noise
!|Flags
!|Books
|----
]=]
end

local function spell_table_section(heading)
  return '! colspan=9 style="text-align:left"|\n' ..
         '====' .. heading .. '====\n' ..
         '|----\n'
end

local function format_schools(frame, schools, no_link_for)
  local ret = ''
  for _, school in ipairs(table_keys_sorted(schools)) do
    if school == no_link_for then
      ret = ret .. school .. '/'
    else
      ret = ret ..
        frame:expandTemplate{title = 'schoollink', args = {school}} .. '/'
    end
  end
  return ret:sub(1, -2)
end

local function format_range(range)
  if range == nil then
    return ''
  elseif range.min == range.max then
    if range.min == RANGE_LOS then
      return 'LOS'
    else
      return range.min
    end
  else
    return range.min .. '-' .. range.max
  end
end

local function format_noise(noise)
  if noise == nil then
    return ''
  else
    return math.max(noise.casting, noise.effect)
  end
end

local function format_flag(flag)
  if flag == 'MR_CHECK' then
    return 'MR check'
  else
    return flag:gsub('_', ' '):lower():gsub('^%l', string.upper)
  end
end

local function format_flags(flags, ignored)
  if flags == nil then
    return ''
  end
  local ret = ''
  for _, flag in ipairs(table_keys_sorted(flags)) do
    if not ignored or not ignored[flag] then
      ret = ret .. format_flag(flag) .. ', '
    end
  end
  return ret:sub(1, -3)
end

local function compare_books(a, b)
  return a:lower():gsub('^book of ', ''):gsub('^a ', ''):gsub('^the ', '') <
    b:lower():gsub('^book of ', ''):gsub('^a ', ''):gsub('^the ', '')
end

local function format_books(books)
  local ret = ''
  for _, book in ipairs(table_keys_sorted(books, compare_books)) do
    ret = ret .. '[[' .. book:gsub('^%l', string.upper) .. ']]<br>'
  end
  return ret:sub(1, -5)
end

local function spell_table_line(frame, name, info, no_link_for)
  return '|[[File:' .. name:lower() .. '.png]]\n' ..
         '|style="padding-left:1em"|[[' .. name .. ']]\n' ..
         '|' .. format_schools(frame, info['schools'], no_link_for) .. '\n' ..
         '|' .. info['level'] .. '\n' ..
         '|' .. info['power cap'] .. '\n' ..
         '|' .. format_range(info['range']) .. '\n' ..
         '|' .. format_noise(info['noise']) .. '\n' ..
         '|' .. format_flags(info['flags']) .. '\n' ..
         '|' .. format_books(info['books']) .. '\n' ..
         '|----\n'
end

function p.spell_table(frame)
  local alphabet = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'

  local ret = '==Spells==\n' .. spell_table_header()
  local c = nil
  for _, name in ipairs(table_keys_sorted(data)) do
    if name:sub(1, 1) ~= c then
      c = name:sub(1, 1)
      ret = ret .. spell_table_section(c)
    end
    ret = ret .. spell_table_line(frame, name, data[name])
  end
  return ret .. '|}'
end

function p.spell_table_by_level(frame)
  local school = frame.args[1]
  if school == '' then
    school = nil
  end
  local ret = '==Spells==\n' .. spell_table_header()
  for level=1,9 do
    local found = false
    for _, name in ipairs(table_keys_sorted(data)) do
      if data[name]['level'] == level then
        if school == nil or data[name]['schools'][school] then
          if not found then
            found = true
            ret = ret .. spell_table_section('Level ' .. level)
          end
          ret = ret .. spell_table_line(frame, name, data[name], school)
        end
      end
    end
  end
  return ret .. '|}'
end

function p.spell_table_by_school(frame)
  local schools = {}
  for _,spell in pairs(data) do
    for school in pairs(spell['schools']) do
      schools[school] = true
    end
  end

  local ret = '==Spells==\n' .. spell_table_header()
  for _,school in ipairs(table_keys_sorted(schools)) do
    ret = ret .. spell_table_section(
      frame:expandTemplate{title = 'schoollink', args = {school}})
    for _, name in ipairs(names_by_level(data)) do
      if data[name]['schools'][school] then
        ret = ret .. spell_table_line(frame, name, data[name], school)
      end
    end
  end
  return ret .. '|}'
end

function p.spell_table_by_book(frame)
  local books = {}
  for _,spell in pairs(data) do if spell['books'] ~= nil then
    for book in pairs(spell['books']) do
      books[book] = true
    end
  end end

  local ret = '==Spells==\n' .. spell_table_header()
  for _,book in ipairs(table_keys_sorted(books, compare_books)) do
    ret = ret ..
      spell_table_section('[[' .. book:gsub('^%l', string.upper) .. ']]')
    for _, name in ipairs(names_by_level(data)) do
      if data[name]['books'][book] then
        ret = ret .. spell_table_line(frame, name, data[name])
      end
    end
  end
  return ret .. '|}'
end

function p.spell_table_by_flag(frame)
  local flags = {}
  for _,spell in pairs(data) do if spell['flags'] ~= nil then
    for flag,_ in pairs(spell['flags']) do
      flags[flag] = true
    end
  end end

  local ret = '==Spells==\n' .. spell_table_header()

  ret = ret .. spell_table_section('No flags')
  for _, name in ipairs(names_by_level(data)) do
    if empty_table(data[name]['flags']) then
      ret = ret .. spell_table_line(frame, name, data[name])
    end
  end

  for _, flag in ipairs(table_keys_sorted(flags)) do
    ret = ret .. spell_table_section(format_flag(flag)) ..
      '| colspan=9 |' ..
      frame:expandTemplate{title = 'SpellFlagDesc ' .. format_flag(flag),
        args = {}} ..
      '\n|----\n'
    for _, name in ipairs(names_by_level(data)) do
      if data[name]['flags'][flag] then
        ret = ret .. spell_table_line(frame, name, data[name])
      end
    end
  end
  return ret .. '|}'
end

local function format_hated_by(spell)
  local output = ''
  if spell.flags.HASTY then
    output = output .. '[[Cheibriados]]<br>'
  end
  if spell.schools.Fire then
    output = output .. '[[Dithmenos]]<br>'
  end
  if spell.flags.CORPSE_VIOLATING then
    output = output .. '[[Fedhas]]<br>'
  end
  if spell.flags.UNHOLY or spell.schools.Necromancy then
    output = output .. '[[Elyvilon]]<br>'
  end
  if spell.flags.UNHOLY or spell.schools.Necromancy or spell.schools.Poison then
    output = output .. '[[The Shining One]]<br>'
  end
  if spell.name == 'Statue Form' then
    output = output .. '[[Yredelemnul]]<br>'
  end
  if spell.flags.CHAOTIC or spell.flags.UNCLEAN or spell.flags.UNHOLY or
    spell.schools.Necromancy then
    output = output .. '[[Zin]]<br>'
  end
  return output:sub(1, -5)
end

local function format_targetting(flags)
  local style = nil
  if flags.DIR then
    style = 'Direction'
  elseif flags.DIR_OR_TARGET then
    style = 'Target or direction'
  elseif flags.TARGET then
    style = 'Smite'
  end
  if style then
    if flags.OBJ and flags.NOT_SELF then
      return style .. ' (object, not self)'
    elseif flags.OBJ then
      return style .. ' (object)'
    elseif flags.NOT_SELF then
      return style .. ' (not self)'
    end
  end
  return style
end

function p.spell_info(frame)
  local name = frame.args[1]
  if not name or name == '' then
    name = mw.title.getCurrentTitle().text
  end
  local spell = data[name]
  if not spell then
    return name
  end

  local args = {}
  args.name = spell.name
  args.level = spell.level
  for i,school in ipairs(table_keys_sorted(spell.schools)) do
    args['school' .. i] =
      frame:expandTemplate{title = 'schoollink', args = {school}}
    i = i + 1
  end
  args.sources = format_books(spell.books)
  args.castingnoise = spell.noise.casting
  args.spellnoise = spell.noise.effect
  args['power cap'] = spell['power cap']
  args.targetting = format_targetting(spell.flags)
  args.range = format_range(spell.range)
  args.rarity = spell.rarity
  args['hated by'] = format_hated_by(spell)
  args.flags = format_flags(spell.flags, {
    CHAOTIC = true,
    CORPSE_VIOLATING = true,
    DIR = true,
    DIR_OR_TARGET = true,
    HASTY = true,
    MONS_ABJURE = true,
    NEEDS_TRACER = true,
    NOT_SELF = true,
    OBJ = true,
    TARGET = true,
    UNCLEAN = true,
    UNHOLY = true,
  })

  local infobox = frame:expandTemplate{title = 'spell', args = args}

  local flavour = spell.description
  if spell.quote then
    flavour = flavour .. '\n----\n' .. spell.quote:gsub('\n', '<br>')
  end
  flavour = frame:expandTemplate{title = 'flavour', args = {flavour}}
  return infobox .. '\n' .. flavour
end

return p