Difference between revisions of "Module:Spell"
(More fields to spell_info. Fix detecting spells with no flags. Use schoollink template.) |
(Fixes. Show range 7 as LOS.) |
||
Line 1: | Line 1: | ||
local p = {} | local p = {} | ||
local data = mw.loadData('Module: Table of spells') | local data = mw.loadData('Module: Table of spells') | ||
+ | local RANGE_LOS = 7 | ||
local function table_keys_sorted(t) | local function table_keys_sorted(t) | ||
Line 12: | Line 13: | ||
local function empty_table(t) | local function empty_table(t) | ||
− | + | for _ in pairs(t) do | |
− | + | return false | |
− | + | end | |
− | + | return true | |
end | end | ||
Line 53: | Line 54: | ||
local function format_schools(frame, schools, no_link_for) | local function format_schools(frame, schools, no_link_for) | ||
local ret = '' | local ret = '' | ||
− | for school in | + | for _, school in ipairs(table_keys_sorted(schools)) do |
if school == no_link_for then | if school == no_link_for then | ||
ret = ret .. school .. '/' | ret = ret .. school .. '/' | ||
Line 68: | Line 69: | ||
return '' | return '' | ||
elseif range.min == range.max then | elseif range.min == range.max then | ||
− | return range.min | + | if range.min == RANGE_LOS then |
+ | return 'LOS' | ||
+ | else | ||
+ | return range.min | ||
+ | end | ||
else | else | ||
return range.min .. '-' .. range.max | return range.min .. '-' .. range.max | ||
Line 95: | Line 100: | ||
end | end | ||
local ret = '' | local ret = '' | ||
− | for flag in | + | for _, flag in ipairs(table_keys_sorted(flags)) do |
if not ignored or not ignored[flag] then | if not ignored or not ignored[flag] then | ||
ret = ret .. format_flag(flag) .. ', ' | ret = ret .. format_flag(flag) .. ', ' | ||
Line 105: | Line 110: | ||
local function format_books(books) | local function format_books(books) | ||
local ret = '' | local ret = '' | ||
− | for book in | + | for _, book in ipairs(table_keys_sorted(books)) do |
ret = ret .. '[[' .. book .. ']]<br>' | ret = ret .. '[[' .. book .. ']]<br>' | ||
end | end | ||
Line 217: | Line 222: | ||
end | end | ||
− | for _,flag in ipairs(table_keys_sorted(flags)) do | + | for _, flag in ipairs(table_keys_sorted(flags)) do |
ret = ret .. spell_table_section(format_flag(flag)) .. | ret = ret .. spell_table_section(format_flag(flag)) .. | ||
'| colspan=9 |' .. | '| colspan=9 |' .. | ||
Line 233: | Line 238: | ||
local function format_hated_by(flags) | local function format_hated_by(flags) | ||
− | + | local output = '' | |
− | + | if flags.HASTY then | |
− | + | output = output .. '[[Cheibriados]]<br>' | |
− | + | end | |
− | + | if flags.CORPSE_VIOLATING then | |
− | + | output = output .. '[[Fedhas]]<br>' | |
− | + | end | |
− | + | if flags.UNHOLY then | |
− | + | output = output .. '[[Elyvilon]]<br>' | |
− | + | end | |
− | + | if flags.UNHOLY then | |
− | + | output = output .. '[[The Shining One]]<br>' | |
− | + | end | |
− | + | if flags.CHAOTIC or flags.UNCLEAN or flags.UNHOLY then | |
− | + | output = output .. '[[Zin]]<br>' | |
− | + | end | |
− | + | return output:sub(1, -5) | |
end | end | ||
local function format_targetting(flags) | 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 | ||
− | + | end | |
− | + | return style | |
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
end | end | ||
Revision as of 15:56, 9 November 2016
local p = {} local data = mw.loadData('Module: Table of spells') local RANGE_LOS = 7 local function table_keys_sorted(t) local keys = {} for k in pairs(t) do table.insert(keys, k) end table.sort(keys) 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 format_books(books) local ret = '' for _, book in ipairs(table_keys_sorted(books)) do ret = ret .. '[[' .. book .. ']]<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 = { 'Air', 'Charms', 'Conjuration', 'Earth', 'Fire', 'Hexes', 'Ice', 'Necromancy', 'Poison', 'Summoning', 'Translocation', 'Transmutation' } local ret = '==Spells==\n' .. spell_table_header() for _,school in ipairs(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)) do ret = ret .. spell_table_section('[[' .. book .. ']]') 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(flags) local output = '' if flags.HASTY then output = output .. '[[Cheibriados]]<br>' end if flags.CORPSE_VIOLATING then output = output .. '[[Fedhas]]<br>' end if flags.UNHOLY then output = output .. '[[Elyvilon]]<br>' end if flags.UNHOLY then output = output .. '[[The Shining One]]<br>' end if flags.CHAOTIC or flags.UNCLEAN or flags.UNHOLY 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.flags) 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