Difference between revisions of "Module:Spell"
(More fields to spell_info. Fix detecting spells with no flags. Use schoollink template.) |
(no longer hated by yred) |
||
(10 intermediate revisions by 3 users not shown) | |||
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, 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 | ||
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 78: | Line 83: | ||
return '' | return '' | ||
else | else | ||
− | return math.max(noise.casting, noise.effect) | + | return math.max(string.match(noise.casting, "%d+"), string.match(noise.effect, "%d+")) |
end | end | ||
end | end | ||
local function format_flag(flag) | local function format_flag(flag) | ||
− | if flag == ' | + | if flag == 'WILL_CHECK' then |
− | return ' | + | return 'Will check' |
else | else | ||
return flag:gsub('_', ' '):lower():gsub('^%l', string.upper) | return flag:gsub('_', ' '):lower():gsub('^%l', string.upper) | ||
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 101: | 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 | + | 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 116: | Line 126: | ||
'|' .. format_schools(frame, info['schools'], no_link_for) .. '\n' .. | '|' .. format_schools(frame, info['schools'], no_link_for) .. '\n' .. | ||
'|' .. info['level'] .. '\n' .. | '|' .. info['level'] .. '\n' .. | ||
− | '|' .. info['power cap'] .. '\n' .. | + | '|' .. (info['power cap'] == 0 and 'N/A' or info['power cap']) .. |
+ | '\n' .. | ||
'|' .. format_range(info['range']) .. '\n' .. | '|' .. format_range(info['range']) .. '\n' .. | ||
'|' .. format_noise(info['noise']) .. '\n' .. | '|' .. format_noise(info['noise']) .. '\n' .. | ||
Line 163: | Line 174: | ||
function p.spell_table_by_school(frame) | function p.spell_table_by_school(frame) | ||
− | local schools = { | + | 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() | 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 189: | Line 203: | ||
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 217: | Line 232: | ||
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 232: | Line 247: | ||
end | end | ||
− | local function format_hated_by( | + | local function format_hated_by(spell) |
− | + | local output = '' | |
− | + | if spell.flags.HASTY then | |
− | + | output = output .. '[[Cheibriados]][[Category:Hasty Spells]]<br>' | |
+ | end | ||
+ | if spell.flags.UNHOLY or spell.schools.Necromancy then | ||
+ | output = output .. '[[Elyvilon]]<br>' | ||
+ | end | ||
+ | if spell.flags.UNHOLY or spell.schools.Necromancy then | ||
+ | output = output .. '[[The Shining One]]<br>' | ||
+ | end | ||
+ | if spell.flags.CHAOTIC or spell.flags.UNCLEAN or spell.flags.UNHOLY or | ||
+ | spell.schools.Necromancy then | ||
+ | output = output .. '[[Zin]]' | ||
+ | if spell.flags.UNCLEAN then | ||
+ | output = output .. '[[Category:Unclean Spells]]' | ||
end | end | ||
− | if flags. | + | if spell.flags.CHAOTIC then |
− | + | output = output .. '[[Category:Chaotic Spells]]' | |
end | end | ||
− | if flags.UNHOLY then | + | if spell.flags.UNHOLY then |
− | + | output = output .. '[[Category:Evil Spells]]' | |
end | end | ||
− | + | output = output .. '<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 | ||
Line 288: | Line 311: | ||
for i,school in ipairs(table_keys_sorted(spell.schools)) do | for i,school in ipairs(table_keys_sorted(spell.schools)) do | ||
args['school' .. i] = | args['school' .. i] = | ||
− | frame:expandTemplate{title = 'schoollink', args = {school}} | + | frame:expandTemplate{title = 'schoollink', args = {school}} .. |
+ | '[[Category:' .. school .. ' Spells]]' | ||
i = i + 1 | i = i + 1 | ||
end | end | ||
Line 294: | Line 318: | ||
args.castingnoise = spell.noise.casting | args.castingnoise = spell.noise.casting | ||
args.spellnoise = spell.noise.effect | args.spellnoise = spell.noise.effect | ||
− | args['power cap'] = spell['power cap'] | + | args['power cap'] = spell['power cap'] == 0 and 'N/A' or spell['power cap'] |
+ | args['summons limit'] = spell['summons limit'] | ||
args.targetting = format_targetting(spell.flags) | args.targetting = format_targetting(spell.flags) | ||
args.range = format_range(spell.range) | args.range = format_range(spell.range) | ||
− | + | args['hated by'] = format_hated_by(spell) | |
− | args['hated by'] = format_hated_by(spell | ||
args.flags = format_flags(spell.flags, { | args.flags = format_flags(spell.flags, { | ||
CHAOTIC = true, | CHAOTIC = true, |
Latest revision as of 04:20, 16 March 2023
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(string.match(noise.casting, "%d+"), string.match(noise.effect, "%d+")) end end local function format_flag(flag) if flag == 'WILL_CHECK' then return 'Will 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'] == 0 and 'N/A' or 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]][[Category:Hasty Spells]]<br>' end if spell.flags.UNHOLY or spell.schools.Necromancy then output = output .. '[[Elyvilon]]<br>' end if spell.flags.UNHOLY or spell.schools.Necromancy then output = output .. '[[The Shining One]]<br>' end if spell.flags.CHAOTIC or spell.flags.UNCLEAN or spell.flags.UNHOLY or spell.schools.Necromancy then output = output .. '[[Zin]]' if spell.flags.UNCLEAN then output = output .. '[[Category:Unclean Spells]]' end if spell.flags.CHAOTIC then output = output .. '[[Category:Chaotic Spells]]' end if spell.flags.UNHOLY then output = output .. '[[Category:Evil Spells]]' end output = output .. '<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}} .. '[[Category:' .. school .. ' Spells]]' 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'] == 0 and 'N/A' or spell['power cap'] args['summons limit'] = spell['summons limit'] args.targetting = format_targetting(spell.flags) args.range = format_range(spell.range) 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