Difference between revisions of "Module:Spell"
(Set some categories in spell_info. Show 0 power as N/A.) |
(Restore spell_info, remove flag ignore as it's no longer works, add separate monster_spell_info function) |
||
| (16 intermediate revisions by 6 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 = | + | local monster_data = mw.loadData('Module: Table of monster spells') |
| + | local RANGE_LOS = 8 | ||
local function table_keys_sorted(t, f) | local function table_keys_sorted(t, f) | ||
| Line 83: | Line 84: | ||
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:upper() == 'WL_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 96: | ||
end | end | ||
| − | local function format_flags(flags | + | local function format_flags(flags) |
if flags == nil then | if flags == nil then | ||
return '' | return '' | ||
| Line 101: | Line 102: | ||
local ret = '' | local ret = '' | ||
for _, flag in ipairs(table_keys_sorted(flags)) do | for _, flag in ipairs(table_keys_sorted(flags)) do | ||
| − | + | ret = ret .. format_flag(flag) .. ', ' | |
| − | |||
| − | |||
end | end | ||
return ret:sub(1, -3) | return ret:sub(1, -3) | ||
| Line 251: | Line 250: | ||
if spell.flags.HASTY then | if spell.flags.HASTY then | ||
output = output .. '[[Cheibriados]][[Category:Hasty Spells]]<br>' | output = output .. '[[Cheibriados]][[Category:Hasty Spells]]<br>' | ||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
end | end | ||
if spell.flags.UNHOLY or spell.schools.Necromancy then | if spell.flags.UNHOLY or spell.schools.Necromancy then | ||
output = output .. '[[Elyvilon]]<br>' | output = output .. '[[Elyvilon]]<br>' | ||
end | end | ||
| − | if spell.flags.UNHOLY or spell.schools.Necromancy | + | if spell.flags.UNHOLY or spell.schools.Necromancy then |
output = output .. '[[The Shining One]]<br>' | output = output .. '[[The Shining One]]<br>' | ||
| − | |||
| − | |||
| − | |||
end | end | ||
if spell.flags.CHAOTIC or spell.flags.UNCLEAN or spell.flags.UNHOLY or | if spell.flags.CHAOTIC or spell.flags.UNCLEAN or spell.flags.UNHOLY or | ||
| Line 277: | Line 267: | ||
end | end | ||
if spell.flags.UNHOLY then | if spell.flags.UNHOLY then | ||
| − | output = output .. '[[Category: | + | output = output .. '[[Category:Evil Spells]]' |
end | end | ||
output = output .. '<br>' | output = output .. '<br>' | ||
| Line 328: | Line 318: | ||
args.spellnoise = spell.noise.effect | args.spellnoise = spell.noise.effect | ||
args['power cap'] = spell['power cap'] == 0 and 'N/A' or 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) |
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
local infobox = frame:expandTemplate{title = 'spell', args = args} | local infobox = frame:expandTemplate{title = 'spell', args = args} | ||
| Line 355: | Line 332: | ||
flavour = frame:expandTemplate{title = 'flavour', args = {flavour}} | flavour = frame:expandTemplate{title = 'flavour', args = {flavour}} | ||
return infobox .. '\n' .. flavour | return infobox .. '\n' .. flavour | ||
| + | end | ||
| + | |||
| + | function p.monster_spell_info(frame) | ||
| + | local name = frame.args[1] | ||
| + | if not name or name == '' then | ||
| + | name = mw.title.getCurrentTitle().text | ||
| + | end | ||
| + | local spell = monster_data[name] | ||
| + | if not spell then | ||
| + | return name | ||
| + | end | ||
| + | |||
| + | local args = {} | ||
| + | args.name = spell.name | ||
| + | args.castingnoise = spell.noise.casting | ||
| + | args.spellnoise = spell.noise.effect | ||
| + | args['summons limit'] = spell['summons limit'] | ||
| + | args.targetting = format_targetting(spell.flags) | ||
| + | args.range = format_range(spell.range) | ||
| + | args.flags = format_flags(spell.flags) | ||
| + | |||
| + | local infobox = frame:expandTemplate{title = 'monster 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 | ||
| + | |||
| + | local function names_by_noise() | ||
| + | local names = table_keys_sorted(data) | ||
| + | table.sort(names, function(a, b) | ||
| + | if data[a].noise.effect == data[b].noise.effect then | ||
| + | return a < b | ||
| + | else | ||
| + | return data[a].noise.effect < data[b].noise.effect | ||
| + | end | ||
| + | end) | ||
| + | return names | ||
| + | end | ||
| + | |||
| + | function p.spell_table_by_noise(books) | ||
| + | local result = [=[{|class="prettytable" cellpadding="3" border="1" | ||
| + | |- align="center" | ||
| + | ! Spell !! Effect noise | ||
| + | ]=] | ||
| + | for _, name in ipairs(names_by_noise()) do | ||
| + | local spell = data[name] | ||
| + | if spell.noise.effect > 0 then | ||
| + | result = result .. '|-\n| [[' .. spell.name .. ']]\n| ' .. tostring(spell.noise.effect) .. '\n' | ||
| + | end | ||
| + | end | ||
| + | result = result .. '|}\n' | ||
| + | return result | ||
end | end | ||
return p | return p | ||
Latest revision as of 15:42, 23 May 2026
local p = {}
local data = mw.loadData('Module: Table of spells')
local monster_data = mw.loadData('Module: Table of monster spells')
local RANGE_LOS = 8
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:upper() == 'WL_CHECK' then
return 'Will check'
else
return flag:gsub('_', ' '):lower():gsub('^%l', string.upper)
end
end
local function format_flags(flags)
if flags == nil then
return ''
end
local ret = ''
for _, flag in ipairs(table_keys_sorted(flags)) do
ret = ret .. format_flag(flag) .. ', '
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)
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
function p.monster_spell_info(frame)
local name = frame.args[1]
if not name or name == '' then
name = mw.title.getCurrentTitle().text
end
local spell = monster_data[name]
if not spell then
return name
end
local args = {}
args.name = spell.name
args.castingnoise = spell.noise.casting
args.spellnoise = spell.noise.effect
args['summons limit'] = spell['summons limit']
args.targetting = format_targetting(spell.flags)
args.range = format_range(spell.range)
args.flags = format_flags(spell.flags)
local infobox = frame:expandTemplate{title = 'monster 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
local function names_by_noise()
local names = table_keys_sorted(data)
table.sort(names, function(a, b)
if data[a].noise.effect == data[b].noise.effect then
return a < b
else
return data[a].noise.effect < data[b].noise.effect
end
end)
return names
end
function p.spell_table_by_noise(books)
local result = [=[{|class="prettytable" cellpadding="3" border="1"
|- align="center"
! Spell !! Effect noise
]=]
for _, name in ipairs(names_by_noise()) do
local spell = data[name]
if spell.noise.effect > 0 then
result = result .. '|-\n| [[' .. spell.name .. ']]\n| ' .. tostring(spell.noise.effect) .. '\n'
end
end
result = result .. '|}\n'
return result
end
return p