Module:Item table: Difference between revisions

From Path of Exile 2 Wiki
Jump to navigation Jump to search
>Illviljan
(Add rarity and rarity_id. And remove a bunch of whitespaces.)
(temp inherent skills)
 
(69 intermediate revisions by 10 users not shown)
Line 1: Line 1:
-- Item table
-------------------------------------------------------------------------------
--
--
-- Creates various item tables from cargo queries.
--                            Module:Item table
--
--  
--
-- This module implements [[Template:Item table]] and other templates that query
-- Todo list
-- and display tables or lists of items.
-- ---------
-------------------------------------------------------------------------------
-- * Handle table columns that can have multiple cargo rows, preferably
 
--   in one or two queries. Not per column AND row.
require('Module:No globals')
-- * Handle template include size? Remove item link when getting close
local m_util = require('Module:Util')
--   to the limit?
 
-- * Add a wrapper around f_item_link to be able to disable all
-- Should we use the sandbox version of our submodules?
--   hoverboxes, to avoid running into template expansion size issues.
local use_sandbox = m_util.misc.maybe_sandbox('Item table')


local m_cargo = use_sandbox and require('Module:Cargo/sandbox') or require('Module:Cargo')
local m_game = use_sandbox and mw.loadData('Module:Game/sandbox') or mw.loadData('Module:Game')


-- ----------------------------------------------------------------------------
-- Lazy loading
-- Imports
local m_item_util -- require('Module:Item util')
-- ----------------------------------------------------------------------------
local f_item_link -- require('Module:Item link').item_link
local m_util = require('Module:Util')
local f_skill_link -- require('Module:Skill link').skill_link
local getArgs = require('Module:Arguments').getArgs
local m_game = require('Module:Game')
local f_item_link = require('Module:Item link').item_link
local f_skill_link = require('Module:Skill link').skill_link
local m_cargo = require('Module:Cargo')


-- ----------------------------------------------------------------------------
-- The cfg table contains all localisable strings and configuration, to make it
-- Globals
-- easier to port this module to another wiki.
-- ----------------------------------------------------------------------------
local cfg = use_sandbox and mw.loadData('Module:Item table/config/sandbox') or mw.loadData('Module:Item table/config')


local c = {}
local i18n = cfg.i18n
c.query_default = 50
c.query_max = 300


-- ----------------------------------------------------------------------------
-- ----------------------------------------------------------------------------
-- Strings
-- Helper functions
-- ----------------------------------------------------------------------------
-- ----------------------------------------------------------------------------
-- This section contains strings used by this module.
-- Add new strings here instead of in-code directly, this will help other
-- people to correct spelling mistakes easier and help with translation to
-- other PoE wikis.


local i18n = {
local h = {}
    categories = {
h.string = {}
        -- maintenance cats
        query_limit = 'Item tables hitting query limit',
        query_hard_limit = 'Item tables hitting hard query limit',
        no_results = 'Item tables without results',
    },


     -- Used by the item table
function h.string.format(str, vars)
     item_table = {
     --[[
        item = 'Item',
    Allow string replacement using named arguments.
        skill_gem = 'Skill gem',
   
     TODO:
    * Support %d ?
    * Support 0.2f ?


        physical_dps = m_util.html.abbr('pDPS', 'physical damage per second'),
    Parameters
        fire_dps = m_util.html.abbr('Fire DPS', 'fire damage per second'),
    ----------
        cold_dps = m_util.html.abbr('Cold DPS', 'cold damage per second'),
    str : String to replace.  
        lightning_dps = m_util.html.abbr('Light. DPS', 'lightning damage per second'),
    vars : Table of arguments.
        chaos_dps = m_util.html.abbr('Chaos DPS', 'chaos damage per second'),
        elemental_dps = m_util.html.abbr('eDPS', 'elemental damage (i.e. fire/cold/lightning) per second'),
        poison_dps = m_util.html.abbr('Poison DPS', 'poison damage (i.e. physical/chaos) per second'),
        dps = m_util.html.abbr('DPS', 'total damage (i.e. physical/fire/cold/lightning/chaos) per second'),
        base_item = 'Base Item',
        metadata_id = 'Metadata ID',
        item_class = 'Item Class',
        rarity = 'Rarity',
        rarity_id = 'Rarity ID',
        essence_level = 'Essence<br>Level',
        drop_level = 'Drop<br>Level',
        release_version = 'Release<br>Version',
        removal_version = 'Removal<br>Version',
        version_link = '[[Version %s|%s]]',
        drop_enabled = m_util.html.abbr('Drop<br>Enabled', 'If an item is drop disabled, it can not be normally obtained, but still may be available under specific conditions (like trading via standard league or limited time events'),
        drop_leagues = 'Drop Leagues',
        drop_leagues_link = '[[%s league|%s]]',
        drop_areas = 'Drop Areas',
        drop_text = 'Additional<br>Drop Restrictions',
        stack_size = 'Stack<br>Size',
        stack_size_currency_tab = m_util.html.abbr('Tab<br>Stack<br>Size', 'Stack size in the currency stash tab'),
        armour = m_util.html.abbr('AR', 'Armour'),
        evasion = m_util.html.abbr('EV', 'Evasion Rating'),
        energy_shield = m_util.html.abbr('ES', 'Energy Shield'),
        block = m_util.html.abbr('Block', 'Chance to Block'),
        damage = m_util.html.abbr('Damage', 'Colour coded damage'),
        attacks_per_second = m_util.html.abbr('APS', 'Attacks per second'),
        local_critical_strike_chance = m_util.html.abbr('Crit', 'Local weapon critical strike chance'),
        flask_life = m_util.html.abbr('Life', 'Life regenerated over the flask duration'),
        flask_mana = m_util.html.abbr('Mana', 'Mana regenerated over the flask duration'),
        flask_duration = 'Duration',
        flask_charges_per_use = m_util.html.abbr('Usage', 'Number of charges consumed on use'),
        flask_maximum_charges = m_util.html.abbr('Capacity', 'Maximum number of flask charges held'),
        item_limit = 'Limit',
        jewel_radius = 'Radius',
        map_tier = 'Map<br>Tier',
        map_level = 'Map<br>Level',
        map_guild_character = m_util.html.abbr('Char', 'Character for the guild tag'),
        atlas_tier = 'Atlas map tier<br>based on [[atlas region|region]]',
        atlas_level = 'Atlas map level<br>based on [[atlas region|region]]',
        master_level_requirement = '[[Image:Level up icon small.png‎|link=|Master level]]',
        master = 'Master',
        master_favour_cost = 'Favour<br>Cost',
        variation_count = 'Variations',
        buff_effects = 'Buff Effects',
        stats = 'Stats',
        quality_stats = 'Stats per 1% [[Quality]]',
        effects = 'Effect(s)',
        incubator_effect = 'Incubation Effect',
        flavour_text = 'Flavour Text',
        prediction_text = 'Prediction',
        help_text = 'Help Text',
        seal_cost = m_util.html.abbr('Seal<br>Cost', 'Silver Coin cost of sealing this prophecies into an item'),
        objective = 'Objective',
        reward = 'Reward',
        buff_icon = 'Buff<br>Icon',
        quest_name = 'Quest',
        quest_act = 'Quest<br>Act',
        purchase_costs = m_util.html.abbr('Purchase Cost', 'Cost of purchasing an item of this type at NPC vendors. This does not indicate whether NPCs actually sell the item.'),
        sell_price = m_util.html.abbr('Sell Price', 'Items or currency received when selling this item at NPC vendors. Certain vendor recipes may override this value.'),
        sextant = m_util.html.abbr('Sextant', 'Maps within the sextant radius.'),
        boss_name = 'Boss',
        boss_number = 'Number of bosses',
        legacy = m_util.html.abbr('Legacy stats', 'Compare legacy variants to the current one.&#10;• Bright text indicates modifiers that are different from the latest variant.&#10;• Strike-through text indicates modifiers that do not exist on legacy variants.'),
        granted_skills = 'Granted skills',
        granted_skills_level_label = 'Level',
        granted_skills_level_pattern = '{granted_skills_level_label}%s*(%d+)',
        granted_skills_level_format = '{granted_skills_level_label} {level_number} ',
        granted_skills_skill_output_format = '{level}{sl}',
        granted_skills_gem_output_format = '{level}{il}',
        alternate_art = 'Alternate<br>Arts',


        -- Skills
    Examples
        support_gem_letter = m_util.html.abbr('L', 'Support gem letter.'),
    --------
        skill_icon = 'Icon',
    = h.string.format('{foo} is {bar}.', {foo='Dinner', bar='nice'})
        description = 'Description',
        skill_critical_strike_chance = m_util.html.abbr('Crit', 'Critical Strike Chance'),
        cast_time = m_util.html.abbr('Cast<br>Time', 'Casting time of the skill in seconds'),
        attack_speed_multiplier = m_util.html.abbr('ASPD', 'Attack Speed Multiplier'),
        damage_effectiveness = m_util.html.abbr('Dmg.<br>Eff.', 'Effectiveness of Added Damage'),
        mana_cost_multiplier = m_util.html.abbr('MCM', 'Mana cost multiplier - missing values indicate it changes on gem level'),
        mana_cost = m_util.html.abbr('Mana', 'Mana cost'),
        reserves_mana_suffix = m_util.html.abbr('R', 'reserves mana'),
        vaal_souls_requirement = m_util.html.abbr('Souls', 'Vaal souls requirement (1.5x in part 2, 2x in maps)'),
        stored_uses = m_util.html.abbr('Uses', 'Maximum number of stored uses'),
        primary_radius = m_util.html.abbr('R1', 'Primary radius'),
        secondary_radius = m_util.html.abbr('R2', 'Secondary radius'),
        tertiary_radius = m_util.html.abbr('R3', 'Tertiary radius'),
        vendor_rewards = m_util.html.abbr('Vendor rewards', 'Vendor rewards after quest completion'),
        vendor_rewards_row_format = '[[Act %s]] after [[%s]] from [[%s]] with %s.',
        vendor_rewards_any_classes = 'any character',
        quest_rewards = m_util.html.abbr('Quest rewards', 'Rewards after quest completion'),
        quest_rewards_row_format = '[[Act %s]] after [[%s]] with %s.',
        quest_rewards_any_classes = 'any character',
    },


     prophecy_description = {
     References
        objective = 'Objective',
    ----------
        reward = 'Reward',
    http://lua-users.org/wiki/StringInterpolation
     },
     ]]


     item_disambiguation = {
     if not vars then
         original='the original variant',
         vars = str
         drop_enabled='the current drop enabled variant',
         str = vars[1]
         drop_disabled='a legacy variant',
    end
        known_release = ' that was introduced in [[%s|%s]]',
   
         list_pattern='%s, %s%s.'
    return (string.gsub(str, "({([^}]+)})",
    },
         function(whole, i)
          return vars[i] or whole
         end))
end


     errors = {
-- Lazy loading for Module:Item link
         generic_argument_parameter = 'Unrecognized %s parameter "%s"',
function h.item_link(args)
        invalid_item_table_mode = 'Invalid mode for item table',
     if not f_item_link then
     },
         f_item_link = use_sandbox and require('Module:Item link/sandbox').item_link or require('Module:Item link').item_link
}
    end
     return f_item_link(args)
end


 
-- Lazy loading for Module:Skill link
-- ----------------------------------------------------------------------------
function h.skill_link(args)
-- Helper & utility functions
     if not f_skill_link then
-- ----------------------------------------------------------------------------
         f_skill_link = use_sandbox and require('Module:Skill link/sandbox').skill_link or require('Module:Skill link').skill_link
 
local h = {}
 
function h.na_or_val(tr, value, func)
     if value == nil or value == '' then
         tr:wikitext(m_util.html.td.na())
    else
        local raw_value = value
        if func ~= nil then
            value = func(value)
        end
        tr
            :tag('td')
                :attr('data-sort-value', raw_value)
                :wikitext(value)
                :done()
     end
     end
    return f_skill_link(args)
end
end


h.tbl = {}
function h.range_fields_factory(args)
 
    -- Returns a function that gets the range fields for the given keys
function h.tbl.range_fields(field)
    local suffixes = {'maximum', 'text', 'colour'}
     return function()
    if args.full then
        suffixes[#suffixes+1] = 'minimum'
    end
     return function ()
         local fields = {}
         local fields = {}
         for _, partial_field in ipairs({'maximum', 'text', 'colour'}) do
         for _, field in ipairs(args.fields) do
            fields[#fields+1] = string.format('%s_range_%s', field, partial_field)
            for _, suffix in ipairs(suffixes) do
                fields[#fields+1] = string.format('%s_range_%s', field, suffix)
            end
         end
         end
         return fields
         return fields
Line 207: Line 100:
end
end


h.tbl.display = {}
function h.display_value_factory(args)
function h.tbl.display.na_or_val(tr, value, data)
     args.fmt_options = args.fmt_options or {}
    return h.na_or_val(tr, value)
end
 
function h.tbl.display.seconds(tr, value, data)
    return h.na_or_val(tr, value, function(value)
        return string.format('%ss', value)
    end)
end
 
function h.tbl.display.percent(tr, value, data)
    return h.na_or_val(tr, value, function(value)
        return string.format('%s%%', value)
    end)
end
 
function h.tbl.display.wikilink(tr, value, data)
    return h.na_or_val(tr, value, function(value)
        return string.format('[[%s]]', value)
    end)
end
 
h.tbl.display.factory = {}
function h.tbl.display.factory.value(args)
     args.options = args.options or {}
 
     return function(tr, data, fields, data2)
     return function(tr, data, fields, data2)
         local values = {}
         local values = {}
         local fmt_values = {}
         local fmt_values = {}
        local sdata = data2.skill_levels[data['items._pageName']]
         for index, field in ipairs(fields) do
         for index, field in ipairs(fields) do
             local value = {
             local value = {
Line 245: Line 111:
                 base=data[field],
                 base=data[field],
             }
             }
             if sdata then
            local sdata = data2 and data2.skill_levels[data['items._pageName']]
             if sdata then --For skill data
                -- Use the fixed value, or the first-level value.
                 value.min = value.min or sdata['0'][field] or sdata['1'][field]
                 value.min = value.min or sdata['0'][field] or sdata['1'][field]
                -- Fall back to the fixed value, and then the max level value.
                 value.max = value.max or sdata['0'][field] or sdata[data['skill.max_level']][field]
                 value.max = value.max or sdata['0'][field] or sdata[data['skill.max_level']][field]
             end
             end
             if value.min then
             if value.min then
                 values[#values+1] = value.max
                 values[#values+1] = value.max
                 local opts = args.options[index] or {}
                 local options = args.fmt_options[index] or {}
                 -- global colour is set, no overrides
                 -- global color is set, no overrides
                 if args.colour ~= nil then
                 if args.color ~= nil then
                     opts.no_color = true
                     options.color = false
                 end
                 end
                 fmt_values[#fmt_values+1] = m_util.html.format_value(nil, nil, value, opts)
                 fmt_values[#fmt_values+1] = m_util.html.format_value(nil, value, options)
             end
             end
         end
         end
         if #values == 0 then
         if #values == 0 then
             tr:wikitext(m_util.html.td.na())
             tr:node(m_util.html.table_cell('na'))
         else
         else
             local td = tr:tag('td')
             local td = tr:tag('td')
             td:attr('data-sort-value', table.concat(values, ', '))
             td
            td:wikitext(table.concat(fmt_values, ', '))
                :attr('data-sort-value', table.concat(values, ', '))
             if args.colour then
                :wikitext(table.concat(fmt_values, ', '))
                 td:attr('class', 'tc -' .. args.colour)
             if args.color then
                 td:attr('class', 'tc -' .. args.color)
             end
             end
         end
         end
Line 273: Line 142:
end
end


function h.tbl.display.factory.range(args)
function h.display_range_factory(args)
     -- args: table
     -- args: table
     --  property
     --  property
Line 286: Line 155:
end
end


function h.tbl.display.factory.descriptor_value(args)
function h.display_range_composite_factory(args)
    -- division by default
    if args.func == nil then
        args.func = function (a, b)
            if b == 0 then
                return 'fail'
            end
            return a / b
        end
    end
   
    return function(tr, data, fields)
        local field = {}
        for i=1, 2 do
            local fieldn = args['field' .. i]
            field[i] = {
                min = tonumber(data[string.format('%s_range_minimum', fieldn)]) or 0,
                max = tonumber(data[string.format('%s_range_maximum', fieldn)]) or 0,
                color = data[string.format('%s_range_colour', fieldn)] or 'default',
            }
        end
       
        field.min = args.func(field[1].min, field[2].min)
        field.max = args.func(field[1].max, field[2].max)
       
        if field.min == 'fail' or field.max == 'fail' then
            field.text = ''
            field.color = 'default'
        else
            for i=1, 2 do
                if field[i].color ~= 'default' then
                    field.color = field[i].color
                    break
                end
            end
            if field.min == field.max then
                field.text = string.format('%.2f', field.min)
            else
                field.text = string.format('(%.2f-%.2f)', field.min, field.max)
            end
        end
   
        tr
            :tag('td')
                :attr('data-sort-value', field.max)
                :attr('class', 'tc -' .. field.color)
                :wikitext(field.text)
                :done()
    end
end
 
function h.display_descriptor_value_factory(args)
     -- Arguments:
     -- Arguments:
     --  key
     --  key
     --  tbl
     --  tbl
     args = args or {}
     args = args or {}
     return function (tpl_args, frame, value)
     return function (value)
        args.tbl = args.tbl or tpl_args
         if args.tbl[args.key] then
         if args.tbl[args.key] then
             value = m_util.html.abbr(value, args.tbl[args.key])
             value = m_util.html.tooltip(value, args.tbl[args.key])
         end
         end
         return value
         return value
Line 300: Line 219:
end
end


function h.tbl.display.factory.atlas_tier(args)
function h.display_atlas_tier_factory(args)
     args = args or {}
     args = args or {}
     return function (tr, data)
     return function (tr, data)
Line 326: Line 245:
end
end


-- ----------------------------------------------------------------------------
function h.display_yesno_factory(args)
-- Data mappings
    -- args:
-- ----------------------------------------------------------------------------
    --  field
    --  condition
    args = args or {}
    args.condition = args.condition or function (value)
        return m_util.cast.boolean(value, {cast_nil=false})
    end
    return function (tr, data)
        local type = args.condition(data[args.field]) and 'yes' or 'no'
        tr:node(m_util.html.table_cell(type))
    end
end
 
function h.value_greater_than_zero(value)
    value = m_util.cast.number(value, {default=0})
    if value > 0 then
        return true
    end
    return false
end
 
function h.na_or_val(tr, value)
    if value == nil or value == '' then
        tr:node(m_util.html.table_cell('na'))
    else
        tr
            :tag('td')
                :attr('data-sort-value', value)
                :wikitext(value)
    end
end
 
-- ----------------------------------------------------------------------------
-- Data mappings
-- ----------------------------------------------------------------------------


local data_map = {}
local data_map = {}
Line 336: Line 288:
data_map.generic_item = {
data_map.generic_item = {
     {
     {
         arg = 'base_item',
         order = 1000,
        args = {'base_item'},
         header = i18n.item_table.base_item,
         header = i18n.item_table.base_item,
         fields = {'items.base_item', 'items.base_item_page'},
         fields = {'items.base_item', 'items.base_item_page'},
Line 345: Line 298:
                     :wikitext(string.format('[[%s|%s]]', data['items.base_item_page'], data['items.base_item']))
                     :wikitext(string.format('[[%s|%s]]', data['items.base_item_page'], data['items.base_item']))
         end,
         end,
        order = 1000,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = 'class',
         order = 1001,
        args = {'class'},
         header = i18n.item_table.item_class,
         header = i18n.item_table.item_class,
         fields = {'items.class'},
         fields = {'items.class'},
         display = h.tbl.display.factory.value{options = {
         display = h.display_value_factory{
            [1] = {
            fmt_options = {
                fmt='[[%s]]',
                [1] = {
                    fmt = '[[%s]]',
                },
             },
             },
         }},
         },
        order = 1001,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = 'rarity',
         order = 1002,
        args = {'rarity'},
         header = i18n.item_table.rarity,
         header = i18n.item_table.rarity,
         fields = {'items.rarity'},
         fields = {'items.rarity'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
        order = 1002,
     },
     },
     {
     {
         arg = 'rarity_id',
         order = 1003,
        args = {'rarity_id'},
         header = i18n.item_table.rarity_id,
         header = i18n.item_table.rarity_id,
         fields = {'items.rarity_id'},
         fields = {'items.rarity_id'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
        order = 1003,
     },
     },
     {
     {
         arg = 'metadata_id',
         order = 1004,
        args = {'metadata_id'},
         header = i18n.item_table.metadata_id,
         header = i18n.item_table.metadata_id,
         fields = {'items.metadata_id'},
         fields = {'items.metadata_id'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
         order = 1004,
    },
    {
        order = 1005,
        args = {'size'},
        header = i18n.item_table.inventory_size,
        fields = {'items.size_x', 'items.size_y'},
         display = function (tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', data['items.size_x'] * data['items.size_y'])
                    :wikitext(string.format('%s×%s', data['items.size_x'], data['items.size_y']))
        end,
     },
     },
     {
     {
         arg = 'essence',
         order = 1100,
        args = {'essence'},
         header = i18n.item_table.essence_level,
         header = i18n.item_table.essence_level,
         fields = {'essences.level'},
         fields = {'essences.level'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
        order = 2000,
     },
     },
     {
     {
         arg = {'drop', 'drop_level'},
         order = 1300,
        header = i18n.item_table.drop_level,
         args = {'stack_size'},
         fields = {'items.drop_level'},
        display = h.tbl.display.factory.value{},
        order = 3000,
    },
    {
        arg = 'stack_size',
         header = i18n.item_table.stack_size,
         header = i18n.item_table.stack_size,
         fields = {'stackables.stack_size'},
         fields = {'stackables.stack_size'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
        order = 4000,
     },
     },
     {
     {
         arg = 'stack_size_currency_tab',
         order = 1301,
        args = {'stack_size_currency_tab'},
         header = i18n.item_table.stack_size_currency_tab,
         header = i18n.item_table.stack_size_currency_tab,
         fields = {'stackables.stack_size_currency_tab'},
         fields = {'stackables.stack_size_currency_tab'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
         order = 4001,
    },
    -- Requirements
    {
        order = 1400,
        args = {'level'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.level_icon,
                i18n.item_table.required_level
            ),
            i18n.item_table.required_level
        ),
        fields = h.range_fields_factory{fields={'items.required_level'}},
        display = h.display_range_factory{field='items.required_level'},
    },
    {
        order = 1401,
        args = {'str'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.str_icon,
                i18n.item_table.required_str
            ),
            i18n.item_table.required_str
        ),
        fields = h.range_fields_factory{fields={'items.required_strength'}},
        display = h.display_range_factory{field='items.required_strength'},
    },
    {
         order = 1402,
        args = {'dex'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.dex_icon,
                i18n.item_table.required_dex
            ),
            i18n.item_table.required_dex
        ),
        fields = h.range_fields_factory{fields={'items.required_dexterity'}},
        display = h.display_range_factory{field='items.required_dexterity'},
     },
     },
     {
     {
         arg = 'level',
         order = 1403,
         header = m_game.level_requirement.icon,
        args = {'int'},
         fields = h.tbl.range_fields('items.required_level'),
         header = m_util.html.tooltip(
         display = h.tbl.display.factory.range{field='items.required_level'},
            string.format(
        order = 5000,
                '[[%s|link=|alt=%s]]',
                i18n.item_table.int_icon,
                i18n.item_table.required_int
            ),
            i18n.item_table.required_int
        ),
         fields = h.range_fields_factory{fields={'items.required_intelligence'}},
         display = h.display_range_factory{field='items.required_intelligence'},
     },
     },
    -- Armour 15xx
     {
     {
         arg = 'ar',
         order = 1500,
        args = {'ar'},
         header = i18n.item_table.armour,
         header = i18n.item_table.armour,
         fields = h.tbl.range_fields('armours.armour'),
         fields = h.range_fields_factory{fields={'armours.armour'}},
         display = h.tbl.display.factory.range{field='armours.armour'},
         display = h.display_range_factory{field='armours.armour'},
        order = 6000,
     },
     },
     {
     {
         arg = 'ev',
         order = 1501,
        args = {'ev'},
         header =i18n.item_table.evasion,
         header =i18n.item_table.evasion,
         fields = h.tbl.range_fields('armours.evasion'),
         fields = h.range_fields_factory{fields={'armours.evasion'}},
         display = h.tbl.display.factory.range{field='armours.evasion'},
         display = h.display_range_factory{field='armours.evasion'},
        order = 6001,
     },
     },
     {
     {
         arg = 'es',
         order = 1502,
        args = {'es'},
         header = i18n.item_table.energy_shield,
         header = i18n.item_table.energy_shield,
         fields = h.tbl.range_fields('armours.energy_shield'),
         fields = h.range_fields_factory{fields={'armours.energy_shield'}},
         display = h.tbl.display.factory.range{field='armours.energy_shield'},
         display = h.display_range_factory{field='armours.energy_shield'},
        order = 6002,
     },
     },
     {
     {
         arg = 'block',
         order = 1503,
        args = {'wd'},
        header = i18n.item_table.ward,
        fields = h.range_fields_factory{fields={'armours.ward'}},
        display = h.display_range_factory{field='armours.ward'},
    },
    {
        order = 1504,
        args = {'block'},
         header = i18n.item_table.block,
         header = i18n.item_table.block,
         fields = h.tbl.range_fields('shields.block'),
         fields = h.range_fields_factory{fields={'shields.block'}},
        display = h.tbl.display.factory.range{field='shields.block'},
         display = h.display_range_factory{field='shields.block'},
        order = 6003,
    },
    --[[{
        arg = 'physical_damage_min',
        header = m_util.html.abbr('Min', 'Local minimum weapon damage'),
        fields = h.tbl.range_fields('minimum physical damage'),
         display = h.tbl.display.factory.range{field='minimum physical damage'},
        order = 7000,
     },
     },
    -- Weapons 16xx
     {
     {
        arg = 'physical_damage_max',
         order = 1600,
        header = m_util.html.abbr('Max', 'Local maximum weapon damage'),
         args = {'weapon', 'damage'},
        fields = h.tbl.range_fields('maximum physical damage'),
        display = h.tbl.display.factory.range{field='maximum physical damage'},
         order = 7001,
 
    },]]--
    {
         arg = {'weapon', 'damage'},
         header = i18n.item_table.damage,
         header = i18n.item_table.damage,
         fields = {'weapons.damage_html', 'weapons.damage_avg'},
         fields = {'weapons.damage_html', 'weapons.damage_avg'},
Line 469: Line 473:
                     :wikitext(data['weapons.damage_html'])
                     :wikitext(data['weapons.damage_html'])
         end,
         end,
        order = 8000,
     },
     },
     {
     {
         arg = {'weapon', 'aps'},
         order = 1601,
        args = {'weapon', 'aps'},
         header = i18n.item_table.attacks_per_second,
         header = i18n.item_table.attacks_per_second,
         fields = h.tbl.range_fields('weapons.attack_speed'),
         fields = h.range_fields_factory{fields={'weapons.attack_speed'}},
         display = h.tbl.display.factory.range{field='weapons.attack_speed'},
         display = h.display_range_factory{field='weapons.attack_speed'},
        order = 8001,
     },
     },
     {
     {
         arg = {'weapon', 'crit'},
         order = 1602,
        args = {'weapon', 'crit'},
         header = i18n.item_table.local_critical_strike_chance,
         header = i18n.item_table.local_critical_strike_chance,
         fields = h.tbl.range_fields('weapons.critical_strike_chance'),
         fields = h.range_fields_factory{fields={'weapons.critical_strike_chance'}},
         display = h.tbl.display.factory.range{field='weapons.critical_strike_chance'},
         display = h.display_range_factory{field='weapons.critical_strike_chance'},
        order = 8002,
     },
     },
     {
     {
         arg = {'physical_dps'},
         order = 1603,
        args = {'physical_dps'},
         header = i18n.item_table.physical_dps,
         header = i18n.item_table.physical_dps,
         fields = h.tbl.range_fields('weapons.physical_dps'),
         fields = h.range_fields_factory{fields={'weapons.physical_dps'}},
         display = h.tbl.display.factory.range{field='weapons.physical_dps'},
         display = h.display_range_factory{field='weapons.physical_dps'},
        order = 8100,
     },
     },
     {
     {
         arg = {'lightning_dps'},
         order = 1604,
        args = {'lightning_dps'},
         header = i18n.item_table.lightning_dps,
         header = i18n.item_table.lightning_dps,
         fields = h.tbl.range_fields('weapons.lightning_dps'),
         fields = h.range_fields_factory{fields={'weapons.lightning_dps'}},
         display = h.tbl.display.factory.range{field='weapons.lightning_dps'},
         display = h.display_range_factory{field='weapons.lightning_dps'},
        order = 8101,
     },
     },
     {
     {
         arg = {'cold_dps'},
         order = 1605,
        args = {'cold_dps'},
         header = i18n.item_table.cold_dps,
         header = i18n.item_table.cold_dps,
         fields = h.tbl.range_fields('weapons.cold_dps'),
         fields = h.range_fields_factory{fields={'weapons.cold_dps'}},
         display = h.tbl.display.factory.range{field='weapons.cold_dps'},
         display = h.display_range_factory{field='weapons.cold_dps'},
        order = 8102,
     },
     },
     {
     {
         arg = {'fire_dps'},
         order = 1606,
        args = {'fire_dps'},
         header = i18n.item_table.fire_dps,
         header = i18n.item_table.fire_dps,
         fields = h.tbl.range_fields('weapons.fire_dps'),
         fields = h.range_fields_factory{fields={'weapons.fire_dps'}},
         display = h.tbl.display.factory.range{field='weapons.fire_dps'},
         display = h.display_range_factory{field='weapons.fire_dps'},
        order = 8103,
     },
     },
     {
     {
         arg = {'chaos_dps'},
         order = 1607,
        args = {'chaos_dps'},
         header = i18n.item_table.chaos_dps,
         header = i18n.item_table.chaos_dps,
         fields = h.tbl.range_fields('weapons.chaos_dps'),
         fields = h.range_fields_factory{fields={'weapons.chaos_dps'}},
         display = h.tbl.display.factory.range{field='weapons.chaos_dps'},
         display = h.display_range_factory{field='weapons.chaos_dps'},
        order = 8104,
     },
     },
     {
     {
         arg = {'elemental_dps'},
         order = 1608,
        args = {'elemental_dps'},
         header = i18n.item_table.elemental_dps,
         header = i18n.item_table.elemental_dps,
         fields = h.tbl.range_fields('weapons.elemental_dps'),
         fields = h.range_fields_factory{fields={'weapons.elemental_dps'}},
         display = h.tbl.display.factory.range{field='weapons.elemental_dps'},
         display = h.display_range_factory{field='weapons.elemental_dps'},
        order = 8105,
     },
     },
     {
     {
         arg = {'poison_dps'},
         order = 1609,
        args = {'poison_dps'},
         header = i18n.item_table.poison_dps,
         header = i18n.item_table.poison_dps,
         fields = h.tbl.range_fields('weapons.poison_dps'),
         fields = h.range_fields_factory{fields={'weapons.poison_dps'}},
         display = h.tbl.display.factory.range{field='weapons.poison_dps'},
         display = h.display_range_factory{field='weapons.poison_dps'},
        order = 8106,
     },
     },
     {
     {
         arg = {'dps'},
         order = 1610,
        args = {'dps'},
         header = i18n.item_table.dps,
         header = i18n.item_table.dps,
         fields = h.tbl.range_fields('weapons.dps'),
         fields = h.range_fields_factory{fields={'weapons.dps'}},
         display = h.tbl.display.factory.range{field='weapons.dps'},
         display = h.display_range_factory{field='weapons.dps'},
        order = 8107,
     },
     },
     {
     {
         arg = 'flask_life',
         order = 1611,
        args = {'reload_time'},
        header = i18n.item_table.reload_time,
        fields = h.range_fields_factory{fields={'weapons.reload_time'}},
        display = h.display_range_factory{field='weapons.reload_time'},
    },
    -- Flasks 17xx
    {
        order = 1700,
        args = {'flask_life'},
         header = i18n.item_table.flask_life,
         header = i18n.item_table.flask_life,
         fields = h.tbl.range_fields('flasks.life'),
         fields = h.range_fields_factory{fields={'flasks.life'}},
         display = h.tbl.display.factory.range{field='flasks.life'},
        display = h.display_range_factory{field='flasks.life'},
         order = 9000,
    },
    {
        order = 1701,
        args = {'flask_life_per_second'},
        header = i18n.item_table.flask_life_per_second,
        fields = h.range_fields_factory{fields={'flasks.life', 'flasks.duration'}, full=true},
         display = h.display_range_composite_factory{field1='flasks.life', field2='flasks.duration'},
    },
    {
        order = 1702,
        args = {'flask_life_per_charge'},
        header = i18n.item_table.flask_life_per_charge,
        fields = h.range_fields_factory{fields={'flasks.life', 'flasks.charges_per_use'}, full=true},
         display = h.display_range_composite_factory{field1='flasks.life', field2='flasks.charges_per_use'},
     },
     },
     {
     {
         arg = 'flask_mana',
         order = 1703,
        args = {'flask_mana'},
         header = i18n.item_table.flask_mana,
         header = i18n.item_table.flask_mana,
         fields = h.tbl.range_fields('flasks.mana'),
         fields = h.range_fields_factory{fields={'flasks.mana'}},
         display = h.tbl.display.factory.range{field='flasks.mana'},
        display = h.display_range_factory{field='flasks.mana'},
         order = 9001,
    },
    {
        order = 1704,
        args = {'flask_mana_per_second'},
        header = i18n.item_table.flask_mana_per_second,
        fields = h.range_fields_factory{fields={'flasks.mana', 'flasks.duration'}, full=true},
         display = h.display_range_composite_factory{field1='flasks.mana', field2='flasks.duration'},
    },
    {
        order = 1705,
        args = {'flask_mana_per_charge'},
        header = i18n.item_table.flask_mana_per_charge,
        fields = h.range_fields_factory{fields={'flasks.mana', 'flasks.charges_per_use'}, full=true},
         display = h.display_range_composite_factory{field1='flasks.mana', field2='flasks.charges_per_use'},
     },
     },
     {
     {
         arg = 'flask',
         order = 1706,
        args = {'flask'},
         header = i18n.item_table.flask_duration,
         header = i18n.item_table.flask_duration,
         fields = h.tbl.range_fields('flasks.duration'),
         fields = h.range_fields_factory{fields={'flasks.duration'}},
         display = h.tbl.display.factory.range{field='flasks.duration'},
         display = h.display_range_factory{field='flasks.duration'},
        order = 9002,
     },
     },
     {
     {
         arg = 'flask',
         order = 1707,
        args = {'flask'},
         header = i18n.item_table.flask_charges_per_use,
         header = i18n.item_table.flask_charges_per_use,
         fields = h.tbl.range_fields('flasks.charges_per_use'),
         fields = h.range_fields_factory{fields={'flasks.charges_per_use'}},
         display = h.tbl.display.factory.range{field='flasks.charges_per_use'},
         display = h.display_range_factory{field='flasks.charges_per_use'},
        order = 9003,
     },
     },
     {
     {
         arg = 'flask',
         order = 1708,
        args = {'flask'},
         header = i18n.item_table.flask_maximum_charges,
         header = i18n.item_table.flask_maximum_charges,
         fields = h.tbl.range_fields('flasks.charges_max'),
         fields = h.range_fields_factory{fields={'flasks.charges_max'}},
         display = h.tbl.display.factory.range{field='flasks.charges_max'},
         display = h.display_range_factory{field='flasks.charges_max'},
        order = 9004,
     },
     },
    -- Jewels 18xx
     {
     {
         arg = 'item_limit',
         order = 1800,
         header = i18n.item_table.item_limit,
        args = {'jewel_limit'},
         fields = {'jewels.item_limit'},
         header = i18n.item_table.jewel_limit,
         display = h.tbl.display.factory.value{},
         fields = {'jewels.jewel_limit'},
        order = 10000,
         display = h.display_value_factory{},
     },
     },
     {
     {
         arg = 'jewel_radius',
         order = 1801,
        args = {'jewel_radius'},
         header = i18n.item_table.jewel_radius,
         header = i18n.item_table.jewel_radius,
         fields = {'jewels.radius_html'},
         fields = {'jewels.radius_html'},
Line 592: Line 633:
                     :wikitext(data['jewels.radius_html'])
                     :wikitext(data['jewels.radius_html'])
         end,
         end,
        order = 10001,
     },
     },
    -- Maps 19xx
     {
     {
         arg = 'map_tier',
         order = 1900,
        args = {'map_tier'},
         header = i18n.item_table.map_tier,
         header = i18n.item_table.map_tier,
         fields = {'maps.tier'},
         fields = {'maps.tier'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
        order = 11000,
     },
     },
     {
     {
         arg = 'map_level',
         order = 1901,
        args = {'map_level'},
         header = i18n.item_table.map_level,
         header = i18n.item_table.map_level,
         fields = {'maps.area_level'},
         fields = {'maps.area_level'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
        order = 11010,
     },
     },
     {
     {
         arg = 'map_guild_character',
         order = 1902,
        args = {'map_guild_character'},
         header = i18n.item_table.map_guild_character,
         header = i18n.item_table.map_guild_character,
         fields = {'maps.guild_character'},
         fields = {'maps.guild_character'},
         display = h.tbl.display.factory.value{},
         display = h.display_value_factory{},
        order = 11020,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = 'atlas_tier',
         order = 1903,
        args = {'map_series'},
        header = i18n.item_table.map_series,
        fields = {'maps.series'},
        display = h.display_value_factory{},
    },
    {
        order = 1904,
        args = {'atlas_tier'},
         header = i18n.item_table.atlas_tier,
         header = i18n.item_table.atlas_tier,
        fields = {'atlas_maps.map_tier0', 'atlas_maps.map_tier1', 'atlas_maps.map_tier2', 'atlas_maps.map_tier3', 'atlas_maps.map_tier4'},
        display = h.display_atlas_tier_factory{is_level=false},
         colspan = 5,
         colspan = 5,
        fields = {'atlas_maps.map_tier0', 'atlas_maps.map_tier1', 'atlas_maps.map_tier2', 'atlas_maps.map_tier3', 'atlas_maps.map_tier4'},
        display = h.tbl.display.factory.atlas_tier{is_level=false},
        order = 11050,
     },
     },
     {
     {
         arg = 'atlas_level',
         order = 1905,
        args = {'atlas_level'},
         header = i18n.item_table.atlas_level,
         header = i18n.item_table.atlas_level,
        fields = {'atlas_maps.map_tier0', 'atlas_maps.map_tier1', 'atlas_maps.map_tier2', 'atlas_maps.map_tier3', 'atlas_maps.map_tier4'},
        display = h.display_atlas_tier_factory{is_level=true},
         colspan = 5,
         colspan = 5,
        fields = {'atlas_maps.map_tier0', 'atlas_maps.map_tier1', 'atlas_maps.map_tier2', 'atlas_maps.map_tier3', 'atlas_maps.map_tier4'},
        display = h.tbl.display.factory.atlas_tier{is_level=true},
        order = 11060,
     },
     },
    -- Map fragments 20xx
     {
     {
         arg = {'doodad', 'master_level_requirement'},
         order = 2000,
         header = i18n.item_table.master_level_requirement,
        args = {'map_fragment', 'map_fragment_limit'},
         fields = {'hideout_doodads.level_requirement'},
         header = i18n.item_table.map_fragment_limit,
         display = h.tbl.display.factory.value{},
         fields = {'map_fragments.map_fragment_limit'},
        order = 11100,
         display = h.display_value_factory{},
     },
     },
    -- Hideout decorations 21xx
     {
     {
        arg = {'doodad', 'master'},
         order = 2100,
        header = i18n.item_table.master,
         args = {'doodad', 'variation_count'},
        fields = {'hideout_doodads.master'},
        display = h.tbl.display.factory.value{},
        order = 11101,
        sort_type = 'text',
    },
    {
        arg = {'doodad', 'master_favour_cost'},
        header = i18n.item_table.master_favour_cost,
        fields = {'hideout_doodads.favour_cost'},
        display = h.tbl.display.factory.value{colour='currency'},
         order = 11102,
    },
    {
         arg = {'doodad', 'variation_count'},
         header = i18n.item_table.variation_count,
         header = i18n.item_table.variation_count,
         fields = {'hideout_doodads.variation_count'},
         fields = {'hideout_doodads.variation_count'},
         display = h.tbl.display.factory.value{colour='mod'},
         display = h.display_value_factory{
         order = 11105,
            color = 'mod',
         },
     },
     },
    -- Corpse items 22xx
     {
     {
         arg = 'buff',
         order = 2200,
         header = i18n.item_table.buff_effects,
        args = {'corpse', 'monster_category'},
         fields = {'item_buffs.stat_text'},
         header = i18n.item_table.monster_category,
         display = h.tbl.display.factory.value{colour='mod'},
         fields = {'corpse_items.monster_category', 'corpse_items.monster_category_html'},
        order = 12000,
         display = function (tr, data)
         sort_type = 'text',
            tr
                :tag('td')
                    :attr('data-sort-value', m_game.constants.monster.categories[data['corpse_items.monster_category']].id)
                    :wikitext(data['corpse_items.monster_category_html'])
         end,
     },
     },
     {
     {
         arg = 'stat',
         order = 2201,
         header = i18n.item_table.stats,
        args = {'corpse', 'monster_abilities'},
         fields = {'items.stat_text'},
         header = i18n.item_table.monster_abilities,
         display = h.tbl.display.factory.value{colour='mod'},
         fields = {'corpse_items.monster_abilities'},
         order = 12001,
         display = h.display_value_factory{
            color = 'mod',
         },
         sort_type = 'text',
         sort_type = 'text',
     },
     },
    -- Seed data 91xx,
     {
     {
         arg = 'description',
         order = 9100,
         header = i18n.item_table.effects,
        args = {'seed', 'seed_type'},
         fields = {'items.description'},
         header = i18n.item_table.seed_type,
         display = h.tbl.display.factory.value{colour='mod'},
         fields = {'harvest_seeds.type', 'harvest_seeds.type_id'},
        order = 12002,
         display = function (tr, data)
         sort_type = 'text',
            tr
                :tag('td')
                    :attr('table-sort-value', data['harvest_seeds.type'])
                    :attr('class', 'tc -' .. data['harvest_seeds.type_id'])
                    :wikitext(data['harvest_seeds.type'])
         end,
     },
     },
     {
     {
         arg = 'incubator_effect',
         order = 9101,
         header = i18n.item_table.incubator_effect,
        args = {'seed', 'seed_tier'},
         fields = {'incubators.effect'},
         header = i18n.item_table.seed_tier,
         display = h.tbl.display.factory.value{colour='crafted'},
         fields = {'harvest_seeds.tier'},
        order = 12004,
         display = h.display_value_factory{},
        sort_type = 'text',
     },
     },
     {
     {
         arg = 'flavour_text',
         order = 9102,
         header = i18n.item_table.flavour_text,
        args = {'seed', 'seed_growth_cycles'},
         fields = {'items.flavour_text'},
         header = i18n.item_table.seed_growth_cycles,
         display = h.tbl.display.factory.value{colour='flavour'},
         fields = {'harvest_seeds.growth_cycles'},
        order = 12006,
         display = h.display_value_factory{},
        sort_type = 'text',
     },
     },
     {
     {
         arg = 'help_text',
         order = 9110,
         header = i18n.item_table.help_text,
        args = {'seed', 'seed_cosumed_lifeforce', 'seed_consumed_lifeforce_percentage', 'seed_consumed_primal_lifeforce_percentage'},
         fields = {'items.help_text'},
         header = i18n.item_table.seed_consumed_primal_lifeforce_percentage,
         display = h.tbl.display.factory.value{colour='help'},
         fields = {'harvest_seeds.consumed_primal_lifeforce_percentage'},
         order = 12008,
         display = h.display_value_factory{
        sort_type = 'text',
            color = 'primal',
         },
     },
     },
     {
     {
         arg = {'prophecy', 'objective'},
         order = 9110,
         header = i18n.item_table.objective,
        args = {'seed', 'seed_cosumed_lifeforce', 'seed_consumed_lifeforce_percentage', 'seed_consumed_vivid_lifeforce_percentage'},
         fields = {'prophecies.objective'},
         header = i18n.item_table.seed_consumed_vivid_lifeforce_percentage,
         display = h.tbl.display.factory.value{},
         fields = {'harvest_seeds.consumed_vivid_lifeforce_percentage'},
         order = 13002,
         display = h.display_value_factory{
            color = 'vivid',
         },
     },
     },
         {
    {
         arg = {'prophecy', 'reward'},
         order = 9110,
         header = i18n.item_table.reward,
         args = {'seed', 'seed_cosumed_lifeforce', 'seed_consumed_lifeforce_percentage', 'seed_consumed_wild_lifeforce_percentage'},
         fields = {'prophecies.reward'},
         header = i18n.item_table.seed_consumed_wild_lifeforce_percentage,
         display = h.tbl.display.factory.value{},
         fields = {'harvest_seeds.consumed_wild_lifeforce_percentage'},
         order = 13001,
         display = h.display_value_factory{
            color = 'wild',
         },
     },
     },
     {
     {
         arg = {'prophecy', 'seal_cost'},
         order = 9113,
         header = i18n.item_table.seal_cost,
        args = {'seed', 'seed_required_nearby_seeds', 'seed_required_nearby_seed_amount'},
         fields = {'prophecies.seal_cost'},
         header = i18n.item_table.seed_required_nearby_seed_amount,
         display = h.tbl.display.factory.value{colour='currency'},
         fields = {'harvest_seeds.required_nearby_seed_amount'},
        order = 13002,
         display = h.display_value_factory{},
     },
     },
     {
     {
         arg = {'prediction_text'},
         order = 9114,
         header = i18n.item_table.prediction_text,
        args = {'seed', 'seed_required_nearby_seeds', 'seed_required_nearby_seed_tier'},
         fields = {'prophecies.prediction_text'},
         header = i18n.item_table.seed_required_nearby_seed_tier,
         display = h.tbl.display.factory.value{colour='value'},
         fields = {'harvest_seeds.required_nearby_seed_tier'},
        order = 12004,
         display = h.display_value_factory{},
        sort_type = 'text',
     },
     },
     {
     {
         arg = 'buff_icon',
         order = 12000,
         header = i18n.item_table.buff_icon,
        args = {'buff'},
         fields = {'item_buffs.icon'},
         header = i18n.item_table.buff_effects,
         display = h.tbl.display.factory.value{options = {
         fields = {'item_buffs.stat_text'},
             [1] = {
         display = h.display_value_factory{
                fmt='[[%s]]',
             color = 'mod',
            },
         },
         }},
        order = 14000,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'version', 'release_version'},
         order = 12001,
         header = i18n.item_table.release_version,
        args = {'stat'},
         fields = {'items.release_version'},
         header = i18n.item_table.stats,
         display = function(tr, data)
         fields = {'items.stat_text'},
            tr
         display = h.display_value_factory{
                :tag('td')
            color = 'mod',
                    :wikitext(
        },
                        string.format(
        sort_type = 'text',
                            i18n.item_table.version_link,
    },
                            data['items.release_version'],
    {
                            data['items.release_version']
        order = 12002,
                        )
        args = {'inherent_skills'},
                    )
        header = i18n.item_table.inherent_skills,
         end,
        fields = {'inherent_skills.inherent_skills_text'},
         order = 15000,
        display = h.display_value_factory{
            color = 'mod',
         },
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'version', 'removal_version'},
         order = 12003,
         header = i18n.item_table.removal_version,
        args = {'description'},
         fields = {'items.removal_version'},
         header = i18n.item_table.description,
         display = function(tr, data)
         fields = {'items.description'},
             tr
         display = h.display_value_factory{
                :tag('td')
             color = 'mod',
                    :wikitext(
         },
                        string.format(
         sort_type = 'text',
                            i18n.item_table.version_link,
                            data['items.removal_version'],
                            data['items.removal_version']
                        )
                    )
         end,
         order = 15001,
     },
     },
     {
     {
         arg = {'drop', 'drop_enabled'},
         order = 12004,
         header = i18n.item_table.drop_enabled,
        args = {'seed', 'seed_effect'},
         fields = {'items.drop_enabled'},
         header = i18n.item_table.seed_effects,
         display = h.tbl.display.factory.value{},
         fields = {'harvest_seeds.effect'},
         order = 15002,
         display = h.display_value_factory{
            color = 'crafted',
        },
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'drop', 'drop_leagues'},
         order = 12100,
         header = i18n.item_table.drop_leagues,
        args = {'flavour_text'},
         fields = {'items.drop_leagues'},
         header = i18n.item_table.flavour_text,
         display = function(tr, data)
         fields = {'items.flavour_text'},
            local s = m_util.string.split(data['items.drop_leagues'], ',')
         display = h.display_value_factory{
             for i, v in ipairs(s) do
             color = 'flavour',
                s[i] = string.format(i18n.item_table.drop_leagues_link, v, v)
         },
            end
            tr
                :tag('td')
                    :wikitext(table.concat(s, '<br>'))
         end,
        order = 15003,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'drop', 'drop_areas'},
         order = 12200,
         header = i18n.item_table.drop_areas,
        args = {'help_text'},
         fields = {'items.drop_areas_html'},
         header = i18n.item_table.help_text,
         display = h.tbl.display.factory.value{},
         fields = {'items.help_text'},
         order = 15004,
         display = h.display_value_factory{
            color = 'help',
         },
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'drop', 'drop_text'},
         order = 12300,
         header = i18n.item_table.drop_text,
        args = {'buff_icon'},
         fields = {'items.drop_text'},
         header = i18n.item_table.buff_icon,
         display = h.tbl.display.factory.value{},
         fields = {'item_buffs.icon'},
         order = 15005,
         display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '[[%s]]',
                },
            },
         },
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'quest'},
         order = 13000,
         header = i18n.item_table.quest_rewards,
        args = {'prophecy', 'objective'},
         fields = {'items._pageName'},
         header = i18n.item_table.objective,
         display = function(tr, data, na, results2)
         fields = {'prophecies.objective'},
            if results2['quest_query'] == nil then
         display = h.display_value_factory{},
                results2['quest_query'] = m_cargo.query(
    },
                    {'items', 'quest_rewards'},
    {
                    {
        order = 13001,
                        'quest_rewards.reward',
        args = {'prophecy', 'reward'},
                        'quest_rewards.classes',
        header = i18n.item_table.reward,
                        'quest_rewards.act',
        fields = {'prophecies.reward'},
                        'quest_rewards.quest'
        display = h.display_value_factory{},
                    },
    },
                    {
    {
                        join='items._pageName=quest_rewards.reward',
        order = 13002,
                        where=string.format(
        args = {'prophecy', 'seal_cost'},
                            'items._pageID IN (%s) AND quest_rewards.reward IS NOT NULL',
        header = i18n.item_table.seal_cost,
                            table.concat(results2.pageIDs, ', ')
        fields = {'prophecies.seal_cost'},
                        ),
        display = h.display_value_factory{
                        orderBy='quest_rewards.act, quest_rewards.quest',
            color = 'currency',
                    }
        },
                )
    },
                results2['quest_query'] = m_cargo.map_results_to_id{
    {
                    results=results2['quest_query'],
        order = 13003,
                    field='quest_rewards.reward',
        args = {'prediction_text'},
                }
        header = i18n.item_table.prediction_text,
            end
        fields = {'prophecies.prediction_text'},
 
        display = h.display_value_factory{
            local results = results2['quest_query'][data['items._pageName']] or {}
            color = 'value',
            local tbl = {}
        },
            for _, v in ipairs(results) do
        sort_type = 'text',
                local classes = table.concat(m_util.string.split(v['quest_rewards.classes'] or '', '�'), ', ')
    },
                 if classes == '' or classes == nil then
    {
                     classes = i18n.item_table.quest_rewards_any_classes
        order = 14000,
                end
        args = {'version', 'release_version'},
 
        header = i18n.item_table.release_version,
                tbl[#tbl+1] = string.format(
        fields = {'items.release_version'},
                    i18n.item_table.quest_rewards_row_format,
        display = function(tr, data)
                    v['quest_rewards.act'],
            tr
                    v['quest_rewards.quest'],
                 :tag('td')
                     classes
                     :wikitext(
                )
                        string.format(
            end
                            i18n.item_table.version_link,
 
                            data['items.release_version'],
            value = table.concat(tbl, '<br>')
                            data['items.release_version']
            if value == nil or value == '' then
                        )
                tr:wikitext(m_util.html.td.na())
                     )
            else
        end,
                tr
    },
                    :tag('td')
    {
                        :attr('style', 'text-align:left')
        order = 15000,
                        :wikitext(value)
        args = {'drop', 'drop_level'},
            end
        header = i18n.item_table.drop_level,
         end,
        fields = {'items.drop_level'},
         order = 16001,
        display = h.display_value_factory{},
         sort_type = 'text',
    },
    {
        order = 15001,
        args = {'drop_level_maximum'},
         header = i18n.item_table.drop_level_maximum,
         fields = {'items.drop_level_maximum'},
         display = h.display_value_factory{},
     },
     },
     {
     {
         arg = {'vendor'},
         order = 15002,
         header = i18n.item_table.vendor_rewards,
        args = {'version', 'removal_version'},
         fields = {'items._pageName'},
         header = i18n.item_table.removal_version,
         display = function(tr, data, na, results2)
         fields = {'items.removal_version'},
             if results2['vendor_query'] == nil then
         display = function(tr, data)
                 results2['vendor_query'] = m_cargo.query(
             tr
                    {'items', 'vendor_rewards'},
                 :tag('td')
                     {
                     :wikitext(
                        'vendor_rewards.reward',
                         string.format(
                        'vendor_rewards.classes',
                             i18n.item_table.version_link,
                        'vendor_rewards.act',
                             data['items.removal_version'],
                        'vendor_rewards.npc',
                            data['items.removal_version']
                         'vendor_rewards.quest',
                        )
                    },
                     )
                    {
         end,
                        join='items._pageName=vendor_rewards.reward',
                        where=string.format(
                             'items._pageID IN (%s) AND vendor_rewards.reward IS NOT NULL',
                             table.concat(results2.pageIDs, ', ')
                        ),
                        orderBy='vendor_rewards.act, vendor_rewards.quest',
                    }
                )
                results2['vendor_query'] = m_cargo.map_results_to_id{
                    results=results2['vendor_query'],
                    field='vendor_rewards.reward',
                }
            end
            local results = results2['vendor_query'][data['items._pageName']] or {}
 
            local tbl = {}
            for _, v in ipairs(results) do
                local classes = table.concat(m_util.string.split(v['vendor_rewards.classes'] or '', '�'), ', ')
                if classes == '' or classes == nil then
                    classes = i18n.item_table.vendor_rewards_any_classes
                end
 
                tbl[#tbl+1] = string.format(
                    i18n.item_table.vendor_rewards_row_format,
                    v['vendor_rewards.act'],
                     v['vendor_rewards.quest'],
                    v['vendor_rewards.npc'],
                    classes
                )
            end
 
            value = table.concat(tbl, '<br>')
            if value == nil or value == '' then
                tr:wikitext(m_util.html.td.na())
            else
                tr
                    :tag('td')
                        :attr('style', 'text-align:left')
                        :wikitext(value)
            end
         end,
        order = 17001,
        sort_type = 'text',
     },
     },
     {
     {
         arg = {'price', 'purchase_cost'},
         order = 15003,
         header = i18n.item_table.purchase_costs,
        args = {'drop', 'drop_enabled'},
         fields = {'item_purchase_costs.name', 'item_purchase_costs.amount'},
         header = i18n.item_table.drop_enabled,
         display = function (tr, data)
         fields = {'items.drop_enabled'},
            -- Can purchase costs have multiple currencies and rows?
        display = h.display_value_factory{},
            -- Just switch to the same method as in sell_price then.
         display = h.display_yesno_factory{
             local tbl = {}
             field = 'items.drop_enabled',
            if data['item_purchase_costs.name'] ~= nil then
        },
                tbl[#tbl+1] = string.format(
        sort_type = 'text',
                        '%sx %s',
    },
                        data['item_purchase_costs.amount'],
    {
                        f_item_link{data['item_purchase_costs.name']}
        order = 15004,
                    )
        args = {'drop', 'drop_areas'},
            end
        header = i18n.item_table.drop_areas,
            h.na_or_val(tr, table.concat(tbl, '<br>'))
        fields = {'items.drop_areas_html'},
        end,
         display = h.display_value_factory{},
         order = 18001,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'price', 'sell_price'},
         order = 15005,
         header = i18n.item_table.sell_price,
        args = {'drop', 'drop_monsters'},
         fields = {'item_sell_prices.name', 'item_sell_prices.amount'},
         header = i18n.item_table.drop_monsters,
         fields = {'items.drop_monsters'},
         display = function(tr, data, na, results2)
         display = function(tr, data, na, results2)
             if results2['sell_price_query'] == nil then
             if results2['drop_monsters_query'] == nil then
                 results2['sell_price_query'] = m_cargo.query(
                 results2['drop_monsters_query'] = m_cargo.query(
                     {'items', 'item_sell_prices'},
                     {'items', 'monsters', 'main_pages'},
                     {
                     {
                         'item_sell_prices.name',
                         'items._pageName',
                         'item_sell_prices.amount',
                        'monsters._pageName',
                         'item_sell_prices._pageID'
                         'monsters.name',
                         'main_pages._pageName',
                     },
                     },
                     {
                     {
                         join='items._pageID=item_sell_prices._pageID',
                         join=[[
                         where=string.format(
                            items.drop_monsters HOLDS monsters.metadata_id,
                             'items._pageID IN (%s) AND item_sell_prices._pageID IS NOT NULL',
                            monsters.metadata_id=main_pages.id
                        ]],
                         where=string.format([[
                             items._pageID IN (%s)
                            AND monsters.metadata_id IS NOT NULL
                        ]],
                             table.concat(results2.pageIDs, ', ')
                             table.concat(results2.pageIDs, ', ')
                         ),
                         ),
                         orderBy='item_sell_prices.name',
                         orderBy='items.drop_monsters',
                     }
                     }
                 )
                 )
                 results2['sell_price_query'] = m_cargo.map_results_to_id{
 
                     results=results2['sell_price_query'],
                 results2['drop_monsters_query'] = m_cargo.map_results_to_id{
                     field='item_sell_prices._pageID',
                     results=results2['drop_monsters_query'],
                     field='items._pageName',
                 }
                 }
             end
             end
             local results = results2['sell_price_query'][data['items._pageID']] or {}
             local results = results2['drop_monsters_query'][data['items._pageName']] or {}
 
             local tbl = {}
             local tbl = {}
             for _,v in ipairs(results) do
             for _,v in ipairs(results) do
                 tbl[#tbl+1] = string.format(
                local page = v['main_pages._pageName'] or v['monsters._pageName'] or ''
                    '%sx %s',
                local name = v['monsters.name'] or v['items.drop_monsters'] or ''
                    v['item_sell_prices.amount'],
                 tbl[#tbl+1] = string.format('[[%s|%s]]', page, name)
                    f_item_link{v['item_sell_prices.name']}
                )
             end
             end
             h.na_or_val(tr, table.concat(tbl, '<br>'))
             h.na_or_val(tr, table.concat(tbl, '<br>'))
         end,
         end,
        order = 18002,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'boss', 'boss_name'},
         order = 15006,
         header = i18n.item_table.boss_name,
        args = {'drop', 'drop_text'},
         fields = {'maps.area_id'},
        header = i18n.item_table.drop_text,
        fields = {'items.drop_text'},
        display = h.display_value_factory{},
        sort_type = 'text',
    },
    {
        order = 16000,
        args = {'quest'},
         header = i18n.item_table.quest_rewards,
         fields = {'items._pageName'},
         display = function(tr, data, na, results2)
         display = function(tr, data, na, results2)
             if results2['boss_query'] == nil then
             if results2['quest_query'] == nil then
                 results2['boss_query'] = m_cargo.query(
                 results2['quest_query'] = m_cargo.query(
                     {'items', 'maps', 'areas', 'monsters', 'main_pages'},
                     {'items', 'quest_rewards'},
                     {
                     {
                         'items._pageName',
                         'quest_rewards._pageName',
                         'maps.area_id',
                         'quest_rewards.classes',
                         'areas.id',
                         'quest_rewards.act',
                         'areas.boss_monster_ids',
                         'quest_rewards.quest'
                        'monsters._pageName',
                     },
                        'monsters.name',
                        'main_pages._pageName',
                     },
                     {
                     {
                         join=[[
                         join='items._pageName=quest_rewards._pageName',
                            items._pageID=maps._pageID,
                         where=string.format(
                            maps.area_id=areas.id,
                             'items._pageID IN (%s) AND quest_rewards._pageName IS NOT NULL',
                            areas.boss_monster_ids HOLDS monsters.metadata_id,
                            monsters.metadata_id=main_pages.id
                        ]],
                         where=string.format([[
                             items._pageID IN (%s)
                            AND maps.area_id IS NOT NULL
                            AND areas.boss_monster_ids HOLDS LIKE "%%"
                        ]],
                             table.concat(results2.pageIDs, ', ')
                             table.concat(results2.pageIDs, ', ')
                         ),
                         ),
                         orderBy='areas.boss_monster_ids',
                         orderBy='quest_rewards.act, quest_rewards.quest',
                     }
                     }
                 )
                 )
 
                 results2['quest_query'] = m_cargo.map_results_to_id{
                 results2['boss_query'] = m_cargo.map_results_to_id{
                     results=results2['quest_query'],
                     results=results2['boss_query'],
                     field='quest_rewards._pageName',
                     field='items._pageName',
                 }
                 }
             end
             end
             local results = results2['boss_query'][data['items._pageName']] or {}
 
             local results = results2['quest_query'][data['items._pageName']] or {}
             local tbl = {}
             local tbl = {}
             for _,v in ipairs(results) do
             for _, v in ipairs(results) do
                 local page = v['main_pages._pageName'] or v['monsters._pageName'] or ''
                 local classes = table.concat(m_util.string.split(v['quest_rewards.classes'] or '', ',%s*'), ', ')
                 local name = v['monsters.name'] or v['areas.boss_monster_ids'] or ''
                 if classes == '' or classes == nil then
                 tbl[#tbl+1] = string.format('[[%s|%s]]', page, name)
                    classes = i18n.item_table.quest_rewards_any_classes
                end
 
                 tbl[#tbl+1] = string.format(
                    i18n.item_table.quest_rewards_row_format,
                    v['quest_rewards.act'],
                    v['quest_rewards.quest'],
                    classes
                )
            end
 
            local value = table.concat(tbl, '<br>')
            if value == nil or value == '' then
                tr:node(m_util.html.table_cell('na'))
            else
                tr
                    :tag('td')
                        :attr('style', 'text-align:left')
                        :wikitext(value)
             end
             end
            h.na_or_val(tr, table.concat(tbl, '<br>'))
         end,
         end,
        order = 19001,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'boss', 'boss_number'},
         order = 17000,
         header = i18n.item_table.boss_number,
        args = {'vendor'},
         fields = {'maps.area_id'},
         header = i18n.item_table.vendor_rewards,
         fields = {'items._pageName'},
         display = function(tr, data, na, results2)
         display = function(tr, data, na, results2)
             if results2['boss_query'] == nil then
             if results2['vendor_query'] == nil then
                 results2['boss_query'] = m_cargo.query(
                 results2['vendor_query'] = m_cargo.query(
                     {'items', 'maps', 'areas', 'monsters', 'main_pages'},
                     {'items', 'vendor_rewards'},
                     {
                     {
                         'items._pageName',
                         'vendor_rewards._pageName',
                         'maps.area_id',
                         'vendor_rewards.classes',
                         'areas.id',
                         'vendor_rewards.act',
                         'areas.boss_monster_ids',
                         'vendor_rewards.npc',
                         'monsters._pageName',
                         'vendor_rewards.quest',
                        'monsters.name',
                        'main_pages._pageName',
                     },
                     },
                     {
                     {
                         join=[[
                         join='items._pageName=vendor_rewards._pageName',
                            items._pageID=maps._pageID,
                         where=string.format(
                            maps.area_id=areas.id,
                             'items._pageID IN (%s) AND vendor_rewards._pageName IS NOT NULL',
                            areas.boss_monster_ids HOLDS monsters.metadata_id,
                             table.concat(results2.pageIDs, ', ')
                            monsters.metadata_id=main_pages.id
                         ),
                        ]],
                         orderBy='vendor_rewards.act, vendor_rewards.quest',
                         where=string.format([[
                     }
                             items._pageID IN (%s)
                            AND maps.area_id IS NOT NULL
                            AND areas.boss_monster_ids HOLDS LIKE "%%"
                        ]],
                             table.concat(results2.pageIDs, ', ')
                         ),
                         orderBy='areas.boss_monster_ids',
                     }
                 )
                 )
 
                 results2['vendor_query'] = m_cargo.map_results_to_id{
                 results2['boss_query'] = m_cargo.map_results_to_id{
                     results=results2['vendor_query'],
                     results=results2['boss_query'],
                     field='vendor_rewards._pageName',
                     field='items._pageName',
                 }
                 }
             end
             end
             local results = results2['boss_query'][data['items._pageName']] or {}
             local results = results2['vendor_query'][data['items._pageName']] or {}
 
             local tbl = {}
             local tbl = {}
             for _,v in ipairs(results) do
             for _, v in ipairs(results) do
                 tbl[#tbl+1] = v['areas.boss_monster_ids']
                 local classes = table.concat(m_util.string.split(v['vendor_rewards.classes'] or '', ',%s*'), ', ')
            end
                if classes == '' or classes == nil then
            h.na_or_val(tr, #tbl)
                    classes = i18n.item_table.vendor_rewards_any_classes
        end,
                 end
        order = 19002,
 
    },
                tbl[#tbl+1] = string.format(
    {
                     i18n.item_table.vendor_rewards_row_format,
        arg = {'sextant'},
                    v['vendor_rewards.act'],
        header = i18n.item_table.sextant,
                    v['vendor_rewards.quest'],
        fields = {'items.name', 'atlas_maps.x', 'atlas_maps.y', 'maps.series'},
                     v['vendor_rewards.npc'],
        display = function(tr, data, na, results2)
                     classes
            if results2['sextant_query'] == nil then
                 results2['sextant_query'] = m_cargo.query(
                     {'items', 'maps', 'atlas_maps'},
                    {
                        'items._pageName',
                        'items.name',
                        'maps.series',
                        'atlas_maps.x',
                        'atlas_maps.y',
                        'atlas_maps.connections'
                    },
                     {
                        join='items._pageID=atlas_maps._pageID, items._pageID=maps._pageID, maps._pageID=atlas_maps._pageID',
                        where='atlas_maps.x IS NOT NULL',
                        orderBy='maps.tier, items._pageName',
                     }
                 )
                 )
             end
             end
            local results = results2['sextant_query'] or {}


             local sextant_radius = 55 -- Should be in Module:Game?
             local value = table.concat(tbl, '<br>')
             local x_center = data['atlas_maps.x']
             if value == nil or value == '' then
             local y_center = data['atlas_maps.y']
                tr:node(m_util.html.table_cell('na'))
 
             else
            if not (x_center and y_center) then
                tr
                return
                    :tag('td')
                        :attr('style', 'text-align:left')
                        :wikitext(value)
             end
             end
            local tbl = {}
            for _, v in ipairs(results) do
                local x = v['atlas_maps.x']
                local y = v['atlas_maps.y']
                local r = ((x-x_center)^2 + (y-y_center)^2)^0.5
                if (sextant_radius >= r) and (data['items._pageName'] ~= v['items._pageName']) then --
                    tbl[#tbl+1] = f_item_link({page=v['items._pageName']})
                end
            end
            h.na_or_val(tr, table.concat(tbl, '<br>'))
         end,
         end,
        order = 20001,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'legacy'},
         order = 18000,
         header = i18n.item_table.legacy,
        args = {'price', 'purchase_cost'},
         fields = {'items.name'},
        header = i18n.item_table.purchase_costs,
        fields = {'item_purchase_costs.name', 'item_purchase_costs.amount'},
        display = function (tr, data)
            -- Can purchase costs have multiple currencies and rows?
            -- Just switch to the same method as in sell_price then.
            local tbl = {}
            if data['item_purchase_costs.name'] ~= nil then
                tbl[#tbl+1] = string.format(
                        '%sx %s',
                        data['item_purchase_costs.amount'],
                        h.item_link{data['item_purchase_costs.name']}
                    )
            end
            h.na_or_val(tr, table.concat(tbl, '<br>'))
        end,
        sort_type = 'text',
    },
    {
        order = 18001,
        args = {'price', 'sell_price'},
         header = i18n.item_table.sell_price,
         fields = {'item_sell_prices.name', 'item_sell_prices.amount'},
         display = function(tr, data, na, results2)
         display = function(tr, data, na, results2)
             if results2['legacy_query'] == nil then
             if results2['sell_price_query'] == nil then
                 results2['legacy_query'] = m_cargo.query(
                 results2['sell_price_query'] = m_cargo.query(
                     {'items', 'legacy_variants'},
                     {'items', 'item_sell_prices'},
                     {
                     {
                         'items._pageID',
                         'item_sell_prices.name',
                         'items._pageName',
                         'item_sell_prices.amount',
                         'items.frame_type',
                         'item_sell_prices._pageID'
                        'legacy_variants.removal_version',
                        'legacy_variants.implicit_stat_text',
                        'legacy_variants.explicit_stat_text',
                        'legacy_variants.stat_text',
                        'legacy_variants.base_item',
                        'legacy_variants.required_level'
                     },
                     },
                     {
                     {
                         join='items._pageID=legacy_variants._pageID',
                         join='items._pageID=item_sell_prices._pageID',
                        where='legacy_variants.removal_version IS NOT NULL',
                         where=string.format(
                         where=string.format(
                             'items._pageID IN (%s) AND legacy_variants.removal_version IS NOT NULL',
                             'items._pageID IN (%s) AND item_sell_prices._pageID IS NOT NULL',
                             table.concat(results2.pageIDs, ', ')
                             table.concat(results2.pageIDs, ', ')
                         ),
                         ),
                         orderBy='items._pageName',
                         orderBy='item_sell_prices.name',
                     }
                     }
                 )
                 )
 
                 results2['sell_price_query'] = m_cargo.map_results_to_id{
                 results2['legacy_query'] = m_cargo.map_results_to_id{
                     results=results2['sell_price_query'],
                     results=results2['legacy_query'],
                     field='item_sell_prices._pageID',
                     field='items._pageName',
                 }
                 }
             end
             end
             local results = results2['legacy_query'][data['items._pageName']] or {}
             local results = results2['sell_price_query'][data['items._pageID']] or {}


             local tbl = mw.html.create('table')
             local tbl = {}
                :attr('width', '100%')
             for _,v in ipairs(results) do
             for _, v in ipairs(results) do
                 tbl[#tbl+1] = string.format(
                 local cell = {}
                    '%sx %s',
                local l = {
                     v['item_sell_prices.amount'],
                     'legacy_variants.base_item',
                     h.item_link{v['item_sell_prices.name']}
                     'legacy_variants.stat_text'
                 )
                 }
            end
 
            h.na_or_val(tr, table.concat(tbl, '<br>'))
                -- Clean up data:
        end,
                for _, k in ipairs(l) do
        sort_type = 'text',
                    if v[k] ~= nil then
    },
                        local s = m_util.string.split(v[k], '*')
    {
                        local s_flt = {}
        order = 19000,
                         for _, sss in ipairs(s) do
        args = {'boss', 'boss_name'},
                            if sss ~= nil and sss ~= '' then
        header = i18n.item_table.boss_name,
                                s_flt[#s_flt+1] = string.gsub(sss, '\n', '')
        fields = {'maps.area_id'},
                            end
        display = function(tr, data, na, results2)
                         end
            if results2['boss_query'] == nil then
 
                results2['boss_query'] = m_cargo.query(
                         cell[#cell+1] = table.concat(s_flt, '<br>')
                    {'items', 'maps', 'areas', 'monsters', 'main_pages'},
                     end
                    {
                 end
                         'items._pageName',
 
                        'maps.area_id',
                 local sep = string.format(
                        'areas.id',
                     '<span class="item-stat-separator -%s"></span>',
                        'areas.boss_monster_ids',
                     v['items.frame_type']
                        'monsters._pageName',
                 )
                        'monsters.name',
                tbl
                         'main_pages._pageName',
                    :tag('tr')
                    },
                        :attr('class', 'upgraded-from-set')
                    {
                        :tag('td')
                         join=[[
                            :wikitext(
                            items._pageID=maps._pageID,
                                v['legacy_variants.removal_version']
                            maps.area_id=areas.id,
                            )
                            areas.boss_monster_ids HOLDS monsters.metadata_id,
                            :done()
                            monsters.metadata_id=main_pages.id
                        :tag('td')
                        ]],
                            :attr('class', 'group legacy-stats plainlist')
                        where=string.format([[
                            :wikitext(table.concat(cell, sep))
                            items._pageID IN (%s)
                            :done()
                            AND maps.area_id IS NOT NULL
                    :done()
                            AND areas.boss_monster_ids HOLDS LIKE "%%"
             end
                        ]],
             tr
                            table.concat(results2.pageIDs, ', ')
                :tag('td')
                        ),
                    :node(tbl)
                        orderBy='areas.boss_monster_ids',
                     }
                 )
 
                 results2['boss_query'] = m_cargo.map_results_to_id{
                     results=results2['boss_query'],
                     field='items._pageName',
                 }
            end
            local results = results2['boss_query'][data['items._pageName']] or {}
            local tbl = {}
            for _,v in ipairs(results) do
                local page = v['main_pages._pageName'] or v['monsters._pageName'] or ''
                local name = v['monsters.name'] or v['areas.boss_monster_ids'] or ''
                tbl[#tbl+1] = string.format('[[%s|%s]]', page, name)
             end
             h.na_or_val(tr, table.concat(tbl, '<br>'))
         end,
         end,
        order = 21001,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = {'granted_skills'},
         order = 19001,
         header = i18n.item_table.granted_skills,
        args = {'boss', 'boss_number'},
         fields = {'items.name'},
         header = i18n.item_table.boss_number,
         fields = {'maps.area_id'},
         display = function(tr, data, na, results2)
         display = function(tr, data, na, results2)
             if results2['granted_skills_query'] == nil then
             if results2['boss_query'] == nil then
                 results2['granted_skills_query'] = m_cargo.query(
                 results2['boss_query'] = m_cargo.query(
                     {'items', 'item_mods', 'mods', 'skill', 'items=items2'},
                     {'items', 'maps', 'areas', 'monsters', 'main_pages'},
                     {
                     {
                         'items._pageName',
                         'items._pageName',
                         'items.name',
                         'maps.area_id',
                        'item_mods.id',
                         'areas.id',
                        'mods._pageName',
                         'areas.boss_monster_ids',
                         'mods.id',
                         'monsters._pageName',
                         'mods.granted_skill',
                         'monsters.name',
                         'mods.stat_text_raw',
                         'main_pages._pageName',
                        'skill._pageName',
                         'skill.skill_id',
                        'skill.active_skill_name',
                        'skill.skill_icon',
                        'skill.stat_text',
                        'items2.class',
                        'items2.name',
                         'items2.inventory_icon',
                        'items2.size_x',
                        'items2.size_y',
                        'items2.html',
                     },
                     },
                     {
                     {
                         join='items._pageID=item_mods._pageID, item_mods.id=mods.id, mods.granted_skill=skill.skill_id, skill._pageID=items2._pageID',
                         join=[[
                         where=string.format(
                            items._pageID=maps._pageID,
                             'items._pageID IN (%s) AND mods.granted_skill IS NOT NULL',
                            maps.area_id=areas.id,
                            areas.boss_monster_ids HOLDS monsters.metadata_id,
                            monsters.metadata_id=main_pages.id
                        ]],
                         where=string.format([[
                             items._pageID IN (%s)
                            AND maps.area_id IS NOT NULL
                            AND areas.boss_monster_ids HOLDS LIKE "%%"
                        ]],
                             table.concat(results2.pageIDs, ', ')
                             table.concat(results2.pageIDs, ', ')
                         ),
                         ),
                        orderBy='areas.boss_monster_ids',
                     }
                     }
                 )
                 )


                 results2['granted_skills_query'] = m_cargo.map_results_to_id{
                 results2['boss_query'] = m_cargo.map_results_to_id{
                     results=results2['granted_skills_query'],
                     results=results2['boss_query'],
                     field='items._pageName',
                     field='items._pageName',
                 }
                 }
             end
             end
             local results = results2['granted_skills_query'][data['items._pageName']] or {}
             local results = results2['boss_query'][data['items._pageName']] or {}
 
             local tbl = {}
             local tbl = {}
             for _, v in ipairs(results) do
             for _,v in ipairs(results) do
 
                tbl[#tbl+1] = v['areas.boss_monster_ids']
                 -- Check if a level for the skill is specified in the
            end
                -- mod stat text.
            tr
                -- Stat ids have unreliable naming convention so using
                 :tag('td')
                -- the mod stat text instead.
                    :attr('data-sort-value', #tbl)
                local level = ''
                    :wikitext(#tbl)
                local stat_text = v['mods.stat_text_raw'] or ''
        end,
                local level_number = string.match(
    },
                     stat_text:lower(),
    {
                     m_util.string.format(
        order = 20000,
                         i18n.item_table.granted_skills_level_pattern,
        args = {'legacy'},
                         {
        header = i18n.item_table.legacy,
                            granted_skills_level_label = i18n.item_table.granted_skills_level_label:lower()
        fields = {'items.name'},
                         }
        display = function(tr, data, na, results2)
                     )
            if results2['legacy_query'] == nil then
                )
                results2['legacy_query'] = m_cargo.query(
 
                     {'items', 'legacy_variants'},
                -- If a level number was specified in the stat text
                     {
                -- then add it to the cell:
                        'items._pageID',
                if level_number then
                        'items._pageName',
                    level = m_util.string.format(
                        'items.frame_type',
                        i18n.item_table.granted_skills_level_format,
                         'legacy_variants.removal_version',
                        {
                        'legacy_variants.implicit_stat_text',
                             granted_skills_level_label = i18n.item_table.granted_skills_level_label,
                         'legacy_variants.explicit_stat_text',
                            level_number = level_number,
                        'legacy_variants.stat_text',
                         }
                        'legacy_variants.base_item',
                     )
                         'legacy_variants.required_level'
                 end
                    },
                     {
                        join='items._pageID=legacy_variants._pageID',
                        where='legacy_variants.removal_version IS NOT NULL',
                        where=string.format(
                            'items._pageID IN (%s) AND legacy_variants.removal_version IS NOT NULL',
                             table.concat(results2.pageIDs, ', ')
                        ),
                         orderBy='items._pageName',
                     }
                 )


                 -- Use different formats depending on if it's a gem or
                 results2['legacy_query'] = m_cargo.map_results_to_id{
                -- not:
                     results=results2['legacy_query'],
                if v['items2.class'] == nil  then
                    field='items._pageName',
                     tbl[#tbl+1] = m_util.string.format(
                }
                        i18n.item_table.granted_skills_skill_output_format,
            end
                        {
            local results = results2['legacy_query'][data['items._pageName']] or {}
                            level = level,
 
                            sl = f_skill_link{
            local tbl = mw.html.create('table')
                                skip_query=true,
                :attr('width', '100%')
                                page = v['skill.active_skill_name']
            for _, v in ipairs(results) do
                                    or v['skill._pageName']
                local cell = {}
                                    or v['mods._pageName']
                local l = {
                                    or '',
                    'legacy_variants.base_item',
                                name = v['skill.active_skill_name']
                    'legacy_variants.stat_text'
                                    or v['skill.stat_text']
                }
                                    or v['mods.granted_skill'],
 
                                icon = v['skill.skill_icon'],
                -- Clean up data:
                            },
                 for _, k in ipairs(l) do
                        }
                     if v[k] ~= nil then
                    )
                         local s = m_util.string.split(v[k], '*')
                 else
                         local s_flt = {}
                     local il_args = {
                        for _, sss in ipairs(s) do
                         skip_query=true,
                            if sss ~= nil and sss ~= '' then
                        page=v['items2._pageName'],
                                s_flt[#s_flt+1] = string.gsub(sss, '\n', '')
                         name=v['items2.name'],
                            end
                        inventory_icon=v['items2.inventory_icon'],
                        end
                        width=v['items2.size_x'],
                        height=v['items2.size_y'],
                    }


                    -- TODO: add in tpl_args.
                        cell[#cell+1] = table.concat(s_flt, '<br>')
                    if no_html == nil then
                        il_args.html = v['items2.html']
                     end
                     end
                    tbl[#tbl+1] = m_util.string.format(
                end
                         i18n.item_table.granted_skills_gem_output_format,
 
                         {
                local sep = string.format(
                             level = level,
                    '<span class="item-stat-separator -%s"></span>',
                             il = f_item_link(il_args),
                    v['items.frame_type']
                        }
                )
                     )
                tbl
                end
                    :tag('tr')
                         :attr('class', 'upgraded-from-set')
                        :tag('td')
                            :wikitext(
                                v['legacy_variants.removal_version']
                            )
                            :done()
                         :tag('td')
                             :attr('class', 'group legacy-stats plainlist')
                             :wikitext(table.concat(cell, sep))
                            :done()
                     :done()
             end
             end
             h.na_or_val(tr, table.concat(tbl, '<br>'))
             tr
                :tag('td')
                    :node(tbl)
         end,
         end,
        order = 22001,
         sort_type = 'text',
         sort_type = 'text',
     },
     },
     {
     {
         arg = 'alternate_art',
         order = 21000,
         header = i18n.item_table.alternate_art,
        args = {'granted_skills'},
         fields = {'items.alternate_art_inventory_icons'},
         header = i18n.item_table.granted_skills,
         display = function (tr, data)
         fields = {'items.name'},
             local alt_art = m_util.string.split(
         display = function(tr, data, na, results2)
                data['items.alternate_art_inventory_icons'],
             if results2['granted_skills_query'] == nil then
                ','
                results2['granted_skills_query'] = m_cargo.query(
            )
                    {'items', 'item_mods', 'mods', 'skill', 'items=items2'},
 
                    {
            -- TODO: Use il instead to handle size?
                        'items._pageName',
            -- local size = 39
                        'items.name',
            local out = {}
                        'item_mods.id',
            for i,v in ipairs(alt_art) do
                        'mods._pageName',
                out[#out+1] = string.format(
                        'mods.id',
                    '[[%s|link=|%s]]',
                        'mods.granted_skill',
                    v,
                        'mods.stat_text_raw',
                     v
                        'skill._pageName',
                        'skill.skill_id',
                        'skill.active_skill_name',
                        'skill.skill_icon',
                        'skill.stat_text',
                        'items2.class',
                        'items2.name',
                        'items2.inventory_icon',
                        'items2.size_x',
                        'items2.size_y',
                        'items2.html',
                    },
                    {
                        join='items._pageID=item_mods._pageID, item_mods.id=mods.id, mods.granted_skill=skill.skill_id, skill._pageID=items2._pageID',
                        where=string.format(
                            'items._pageID IN (%s) AND mods.granted_skill IS NOT NULL',
                            table.concat(results2.pageIDs, ', ')
                        ),
                     }
                 )
                 )
                results2['granted_skills_query'] = m_cargo.map_results_to_id{
                    results=results2['granted_skills_query'],
                    field='items._pageName',
                }
             end
             end
            local results = results2['granted_skills_query'][data['items._pageName']] or {}


             tr
             local tbl = {}
                :tag('td')
            for _, v in ipairs(results) do
                    :wikitext(table.concat(out, ''))
        end,
        order = 23000,
        sort_type = 'text',
    },
}


data_map.skill_gem_new = {
                -- Check if a level for the skill is specified in the
    {
                -- mod stat text.
        arg = 'icon',
                -- Stat ids have unreliable naming convention so using
        header = i18n.item_table.support_gem_letter,
                -- the mod stat text instead.
        fields = {'skill_gems.support_gem_letter_html'},
                local level = ''
        display = h.tbl.display.factory.value{},
                local stat_text = v['mods.stat_text_raw'] or ''
        order = 1000,
                local level_number = string.match(
        sort_type = 'text',
                    stat_text:lower(),
    },
                    h.string.format(
    {
                        i18n.item_table.granted_skills_level_pattern,
        arg = 'skill_icon',
                        {
        header = i18n.item_table.skill_icon,
                            granted_skills_level_label = i18n.item_table.granted_skills_level_label:lower()
        fields = {'skill.skill_icon'},
                        }
        display = h.tbl.display.factory.value{options = {
                    )
            [1] = {
                 )
                 fmt='[[%s]]',
 
            },
                -- If a level number was specified in the stat text
        }},
                -- then add it to the cell:
        order = 1001,
                if level_number then
        sort_type = 'text',
                    level = h.string.format(
    },
                        i18n.item_table.granted_skills_level_format,
    {
                        {
        arg = {'stat', 'stat_text'},
                            granted_skills_level_label = i18n.item_table.granted_skills_level_label,
        header = i18n.item_table.stats,
                            level_number = level_number,
        fields = {'skill.stat_text'},
                        }
        display = h.tbl.display.factory.value{},
                    )
        order = 2000,
                end
        sort_type = 'text',
 
    },
                -- Use different formats depending on if it's a gem or
    {
                -- not:
        arg = {'quality', 'quality_stat_text'},
                if v['items2.class'] == nil  then
        header = i18n.item_table.quality_stats,
                    tbl[#tbl+1] = h.string.format(
        fields = {'skill.quality_stat_text'},
                        i18n.item_table.granted_skills_skill_output_format,
        display = h.tbl.display.factory.value{},
                        {
        order = 2001,
                            level = level,
        sort_type = 'text',
                            sl = h.skill_link{
    },
                                skip_query=true,
    {
                                page = v['skill.active_skill_name']
        arg = 'description',
                                    or v['skill._pageName']
        header = i18n.item_table.description,
                                    or v['mods._pageName']
        fields = {'skill.description'},
                                    or '',
        display = h.tbl.display.factory.value{},
                                name = v['skill.active_skill_name']
        order = 2100,
                                    or v['skill.stat_text']
        sort_type = 'text',
                                    or v['mods.granted_skill'],
    },
                                icon = v['skill.skill_icon'],
    {
                            },
        arg = 'level',
                        }
        header = m_game.level_requirement.icon,
                    )
        fields = h.tbl.range_fields('items.required_level'),
                else
        display = h.tbl.display.factory.range{field='items.required_level'},
                    local il_args = {
        order = 3004,
                        skip_query=true,
    },
                        page=v['items2._pageName'],
    {
                        name=v['items2.name'],
        arg = 'crit',
                        inventory_icon=v['items2.inventory_icon'],
        header = i18n.item_table.skill_critical_strike_chance,
                        width=v['items2.size_x'],
        fields = {'skill_levels.critical_strike_chance'},
                        height=v['items2.size_y'],
        display = h.tbl.display.factory.value{options = {
                    }
            [1] = {
 
                fmt='%s%%',
                    -- TODO: add in tpl_args.
                skill_levels = true,
                    if no_html == nil then
            },
                        il_args.html = v['items2.html']
        }},
                    end
        order = 4000,
                    tbl[#tbl+1] = h.string.format(
        options = {
                        i18n.item_table.granted_skills_gem_output_format,
            [1] = {
                        {
                skill_levels = true,
                            level = level,
            },
                            il = h.item_link(il_args),
        },
                        }
    },
                    )
    {
                end
        arg = 'cast_time',
            end
        header = i18n.item_table.cast_time,
             h.na_or_val(tr, table.concat(tbl, '<br>'))
        fields = {'skill.cast_time'},
        end,
        display = h.tbl.display.factory.value{options = {
         sort_type = 'text',
        }},
        order = 4001,
        options = {
        },
    },
    {
        arg = {'aspd', 'attack_speed', 'attack_speed_multiplier'},
        header = i18n.item_table.attack_speed_multiplier,
        fields = {'skill_levels.attack_speed_multiplier'},
        display = h.tbl.display.factory.value{options = {
            [1] = {
                fmt='%s%%',
                skill_levels = true,
            },
        }},
        order = 4002,
        options = {
             [1] = {
                skill_levels = true,
            },
         },
     },
     },
     {
     {
         arg = 'dmgeff',
         order = 23000,
         header = i18n.item_table.damage_effectiveness,
         args = {'alternate_art'},
        fields = {'skill_levels.damage_effectiveness'},
         header = i18n.item_table.alternate_art,
        display = h.tbl.display.factory.value{options = {
         fields = {'items.alternate_art_inventory_icons'},
            [1] = {
         display = function (tr, data)
                fmt='%s%%',
             local alt_art = m_util.string.split(
                skill_levels = true,
                 data['items.alternate_art_inventory_icons'],
            },
                 ','
        }},
             )
        order = 4003,
 
        options = {
            -- TODO: Use il instead to handle size?
            [1] = {
            -- local size = 39
                skill_levels = true,
             local out = {}
            },
             for i,v in ipairs(alt_art) do
        },
                out[#out+1] = string.format(
    },
                    '[[%s|link=|%s]]',
    {
                    v,
        arg = 'mcm',
                    v
         header = i18n.item_table.mana_cost_multiplier,
                 )
         fields = {'skill_levels.mana_multiplier'},
         display = h.tbl.display.factory.value{options = {
             [1] = {
                 fmt='%s%%',
                 skill_levels = true,
             },
        }},
        order = 5000,
        options = {
             [1] = {
                skill_levels = true,
             },
        },
    },
    {
        arg = 'mana',
        header = i18n.item_table.mana_cost,
        fields = {'skill_levels.mana_cost', 'skill.has_percentage_mana_cost', 'skill.has_reservation_mana_cost'},
        display = function (tr, data, fields, data2)
            local appendix = ''
            if m_util.cast.boolean(data['skill.has_percentage_mana_cost']) then
                appendix = appendix .. '%%'
            end
            if m_util.cast.boolean(data['skill.has_reservation_mana_cost']) then
                 appendix = appendix .. ' ' .. i18n.item_table.reserves_mana_suffix
             end
             end


             h.tbl.display.factory.value{options = {
             tr
                 [1] = {
                 :tag('td')
                    fmt='%d' .. appendix,
                     :wikitext(table.concat(out, ''))
                     skill_levels = true,
                },
            }}(tr, data, {'skill_levels.mana_cost'}, data2)
         end,
         end,
         order = 5001,
         sort_type = 'text',
        options = {
            [1] = {
                skill_levels = true,
            },
        },
     },
     },
}
data_map.skill_gem = {
     {
     {
         arg = 'vaal',
         order = 1000,
         header = i18n.item_table.vaal_souls_requirement,
        args = {'gem_tier'},
         fields = {'skill_levels.vaal_souls_requirement'},
         header = i18n.item_table.tier,
         display = h.tbl.display.factory.value{options = {
         fields = {'skill_gems.gem_tier'},
            [1] = {
         display = h.display_value_factory{},
                skill_levels = true,
            },
        }},
        order = 6000,
        options = {
            [1] = {
                skill_levels = true,
            },
        },
     },
     },
     {
     {
         arg = 'vaal',
         order = 1001,
         header = i18n.item_table.stored_uses,
        args = {'skill_icon'},
         fields = {'skill_levels.vaal_stored_uses'},
         header = i18n.item_table.skill_icon,
         display = h.tbl.display.factory.value{options = {
         fields = {'skill.skill_icon'},
            [1] = {
         display = h.display_value_factory{
                skill_levels = true,
            fmt_options = {
            },
                [1] = {
        }},
                    fmt = '[[%s]]',
        order = 6001,
                 },
        options = {
            [1] = {
                 skill_levels = true,
             },
             },
         },
         },
        sort_type = 'text',
     },
     },
     {
     {
         arg = 'radius',
         order = 2000,
         header = i18n.item_table.primary_radius,
        args = {'stat', 'stat_text'},
         fields = {'skill.radius', 'skill.radius_description'},
         header = i18n.item_table.stats,
        options = {[2] = {optional = true}},
         fields = {'skill.stat_text'},
         display = function (tr, data)
         display = h.display_value_factory{},
            tr
         sort_type = 'text',
                :tag('td')
                    :attr('data-sort-value', data['skill.radius'])
                    :wikitext(h.tbl.display.factory.descriptor_value{tbl=data, key='skill.radius_description'}(nil, nil, data['skill.radius']))
        end,
         order = 7000,
     },
     },
     {
     {
         arg = 'radius',
         order = 2001,
         header = i18n.item_table.secondary_radius,
        args = {'quality', 'quality_stat_text'},
         fields = {'skill.radius_secondary', 'skill.radius_secondary_description'},
         header = i18n.item_table.quality_stats,
        options = {[2] = {optional = true}},
         fields = {'skill.quality_stat_text'},
         display = function (tr, data)
         display = h.display_value_factory{},
            tr
         sort_type = 'text',
                :tag('td')
                    :attr('data-sort-value', data['skill.radius_secondary'])
                    :wikitext(h.tbl.display.factory.descriptor_value{tbl=data, key='skill.radius_secondary_description'}(nil, nil, data['skill.radius_secondary']))
        end,
         order = 7001,
     },
     },
     {
     {
         arg = 'radius',
         order = 2100,
         header = i18n.item_table.tertiary_radius,
        args = {'description'},
         fields = {'skill.radius_tertiary', 'skill.radius_tertiary_description'},
         header = i18n.item_table.description,
         options = {[2] = {optional = true}},
         fields = {'skill.description'},
         display = function (tr, data)
        display = h.display_value_factory{},
             tr
        sort_type = 'text',
                 :tag('td')
    },
                    :attr('data-sort-value', data['skill.radius_tertiary'])
    {
                  :wikitext(h.tbl.display.factory.descriptor_value{tbl=data, key='skill.radius_tertiary_description'}(nil, nil, data['skill.radius_tertiary']))
        order = 3000,
         end,
         args = {'level'},
         order = 7002,
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.level_icon,
                i18n.item_table.gem_level_requirement
            ),
            i18n.item_table.gem_level_requirement
        ),
        fields = h.range_fields_factory{fields={'items.required_level'}},
         display = h.display_range_factory{field='items.required_level'},
    },
    {
        order = 3001,
        args = {'str'},
        header = m_util.html.tooltip(
             string.format(
                 '[[%s|link=|alt=%s]]',
                i18n.item_table.str_icon,
                i18n.item_table.str_gem
            ),
            i18n.item_table.str_gem
        ),
        fields = {'skill_gems.strength_percent'},
        display = h.display_yesno_factory{
            field = 'skill_gems.strength_percent',
            condition = h.value_greater_than_zero,
        },
        sort_type = 'text',
    },
    {
        order = 3002,
        args = {'dex'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.dex_icon,
                i18n.item_table.dex_gem
            ),
            i18n.item_table.dex_gem
        ),
        fields = {'skill_gems.dexterity_percent'},
        display = h.display_yesno_factory{
            field = 'skill_gems.dexterity_percent',
            condition = h.value_greater_than_zero,
         },
         sort_type = 'text',
     },
     },
}
     {
 
         order = 3003,
for i, attr in ipairs(m_game.constants.attribute_order) do
         args = {'int'},
     local attr_data = m_game.constants.attributes[attr]
         header = m_util.html.tooltip(
    table.insert(data_map.generic_item, 7, {
             string.format(
         arg = attr_data.arg,
                '[[%s|link=|alt=%s]]',
         header = attr_data.icon,
                i18n.item_table.int_icon,
        fields = h.tbl.range_fields(string.format('items.required_%s', attr)),
                i18n.item_table.int_gem
        display = h.tbl.display.factory.range{field=string.format('items.required_%s', attr)},
            ),
        order = 5000+i,
            i18n.item_table.int_gem
    })
         ),
    table.insert(data_map.skill_gem_new, 1, {
         fields = {'skill_gems.intelligence_percent'},
        arg = attr_data.arg,
         display = h.display_yesno_factory{
         header = attr_data.icon,
             field = 'skill_gems.intelligence_percent',
        fields = {string.format('skill_gems.%s_percent', attr)},
             condition = h.value_greater_than_zero,
        display = function (tr, data)
             tr
                :tag('td')
                    :attr('data-sort-value', data[string.format('skill_gems.%s_percent', attr)])
                    :wikitext('[[File:Yes.png|yes|link=]]')
        end,
        order = 3000+i,
    })
end
 
 
-- ----------------------------------------------------------------------------
-- Invoke callables
-- ----------------------------------------------------------------------------
 
local p = {}
 
--
-- Template:Item table
--
 
function p.item_table(frame)
    --[[
    Creates a generic table for items.
 
    Examples
    --------
    = p.item_table{
         q_tables = 'items, vendor_rewards',
         q_join = 'items.name = vendor_rewards.reward',
        q_where= 'vendor_rewards.reward IS NOT NULL AND (items.class = "Active Skill Gems" OR items.class = "Support Skill Gems")',
        vendor=1,
    }
 
    ]]
 
    local t = os.clock()
    -- args
    local tpl_args = getArgs(frame, {
            parentFirst = true
         })
    frame = m_util.misc.get_frame(frame)
 
    tpl_args.q_where = m_cargo.replace_holds{string=tpl_args.q_where}
 
    local modes = {
        skill = {
             data = data_map.skill_gem_new,
             header = i18n.item_table.skill_gem,
         },
         },
         item = {
         sort_type = 'text',
            data = data_map.generic_item,
    },
            header = i18n.item_table.item,
    {
        order = 4000,
        args = {'crit'},
        header = i18n.item_table.skill_critical_strike_chance,
        fields = {'skill_levels.critical_strike_chance'},
        display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '%s%%',
                },
            },
        },
        field_options = {
            [1] = {
                skill_levels = true,
            },
         },
         },
     }
     },
 
    {
    if tpl_args.mode == nil then
        order = 4001,
         tpl_args.mode = 'item'
         args = {'cast_time'},
    end
        header = i18n.item_table.cast_time,
 
        fields = {'skill.cast_time'},
    if modes[tpl_args.mode] == nil then
         display = h.display_value_factory{},
         error(i18n.errors.invalid_item_table_mode)
     },
     end
    {
 
        order = 4002,
    local results2 = {
         args = {'aspd', 'attack_speed', 'attack_speed_multiplier'},
         stats = {},
         header = i18n.item_table.attack_speed_multiplier,
         skill_levels = {},
         fields = {'skill_levels.attack_speed_multiplier'},
         pageIDs = {},
        display = h.display_value_factory{
    }
            fmt_options = {
 
                [1] = {
    local row_infos = {}
                    fmt = '%s%%',
    for _, row_info in ipairs(modes[tpl_args.mode].data) do
                },
         local enabled = false
            },
         if row_info.arg == nil then
         },
            enabled = true
         field_options = {
         elseif type(row_info.arg) == 'string' and m_util.cast.boolean(tpl_args[row_info.arg]) then
            [1] = {
            enabled = true
                skill_levels = true,
         elseif type(row_info.arg) == 'table' then
            },
             for _, argument in ipairs(row_info.arg) do
         },
                 if m_util.cast.boolean(tpl_args[argument]) then
    },
                     enabled = true
    {
                    break
        order = 4003,
                 end
        args = {'dmgeff'},
             end
        header = i18n.item_table.damage_effectiveness,
         end
        fields = {'skill_levels.damage_effectiveness'},
 
         display = h.display_value_factory{
         if enabled then
             fmt_options = {
            row_info.options = row_info.options or {}
                 [1] = {
             row_infos[#row_infos+1] = row_info
                     fmt = '%s%%',
         end
                 },
     end
             },
 
         },
     -- Parse stat arguments
         field_options = {
    local stat_columns = {}
             [1] = {
    local query_stats = {}
                skill_levels = true,
    local i = 0
            },
    repeat
         },
         i = i + 1
     },
 
     {
         local prefix = string.format('stat_column%s_', i)
        order = 5000,
         local col_info = {
        args = {'mcm', 'cost_multiplier'},
             header = tpl_args[prefix .. 'header'] or tostring(i),
         header = i18n.item_table.cost_multiplier,
            format = tpl_args[prefix .. 'format'],
         fields = {'skill_levels.cost_multiplier'},
             stat_format = tpl_args[prefix .. 'stat_format'] or 'separate',
         display = h.display_value_factory{
             order = tonumber(tpl_args[prefix .. 'order']) or (10000000 + i),
             fmt_options = {
            stats = {},
                [1] = {
            options = {},
                    fmt = '%s%%',
         }
                },
 
             },
         local j = 0
        },
        repeat
        field_options = {
             j = j +1
            [1] = {
 
                skill_levels = true,
             local stat_info = {
             },
                id = tpl_args[string.format('%sstat%s_id', prefix, j)],
        },
             }
    },
 
    {
             if stat_info.id then
        order = 5001,
                col_info.stats[#col_info.stats+1] = stat_info
        args = {'mana'},
                query_stats[stat_info.id] = {}
        header = i18n.item_table.mana_cost,
            else
         fields = {'skill_levels.cost_amounts', 'skill_levels.mana_reservation_percent', 'skill_levels.mana_reservation_flat'},
                 -- Stop iteration entirely if this was the first index but no stat was supplied. We assume that we stop in this case.
         display = function (tr, data, fields, data2)
                if j == 1 then
             local appendix = ''
                    i = nil
             local cost_field = ''
                 end
            local sdata = data2.skill_levels[data['items._pageName']]
                -- stop iteration
            -- Percentage Mana reservation is in level 0 of most of the skill_levels at this point, but spellslinger and awakened blasphemy change it per-level
                 j = nil
             -- Flat Mana resservation is in real gem levels, but maybe there will be fixed flat mana reservations in the future.
             -- Per-use mana costs are stored in the real gem levels if they change, or in 0 if it's fixed.
            if(sdata['1']['skill_levels.mana_reservation_percent'] ~= nil or sdata['0']['skill_levels.mana_reservation_percent'] ~= nil) then
                cost_field = 'skill_levels.mana_reservation_percent'
                 appendix = appendix .. '%%'
            elseif(sdata['1']['skill_levels.mana_reservation_flat'] ~= nil or sdata['0']['skill_levels.mana_reservation_flat'] ~= nil) then
                cost_field = 'skill_levels.mana_reservation_flat'
                 appendix = appendix .. ' ' .. i18n.item_table.reserves_mana_suffix
            elseif(sdata['1']['skill_levels.cost_amounts'] ~= nil or sdata['0']['skill_levels.cost_amounts'] ~= nil) then
                 cost_field = 'skill_levels.cost_amounts'
             end
             end
        until j == nil
            h.display_value_factory{
 
                fmt_options = {
         -- Don't add this column if no stats were provided.
                    [1] = {
         if #col_info.stats > 0 then
                        fmt = '%d' .. appendix,
             stat_columns[#stat_columns+1] = col_info
                    },
        end
                },
    until i == nil
            }(tr, data, {cost_field}, data2)
 
        end,
     for _, col_info in ipairs(stat_columns) do
         -- Need one set of options per field.
         local row_info = {
         field_options = {
            --arg
             [1] = {
            header = col_info.header,
                skill_levels = true,
            fields = {},
            },
            display = function(tr, data, properties)
            [2] = {
                if col_info.stat_format == 'separate' then
                skill_levels = true,
                    local stat_texts = {}
            },
                    local num_stats = 0
            [3] = {
                    local vmax = 0
                skill_levels = true,
                    for _, stat_info in ipairs(col_info.stats) do
            },
                        num_stats = num_stats + 1
        },
                        -- stat results from outside body
    },
                        local stat = (results2.stats[data['items._pageName']] or {})[stat_info.id]
     {
                        if stat ~= nil then
        order = 6000,
                            stat_texts[#stat_texts+1] = m_util.html.format_value(tpl_args, frame, stat, {no_color=true})
         args = {'vaal'},
                            vmax = vmax + stat.max
        header = i18n.item_table.vaal_souls_requirement,
                        end
        fields = {'skill_levels.vaal_souls_requirement'},
                    end
        display = h.display_value_factory{},
 
        field_options = {
                    if num_stats ~= #stat_texts then
            [1] = {
                        tr:wikitext(m_util.html.td.na())
                skill_levels = true,
                    else
            },
                        local text
        },
                        if col_info.format then
    },
                            text = string.format(col_info.format, unpack(stat_texts))
    {
                        else
        order = 6001,
                            text = table.concat(stat_texts, ', ')
        args = {'vaal'},
                        end
        header = i18n.item_table.stored_uses,
 
        fields = {'skill_levels.vaal_stored_uses'},
                        tr:tag('td')
        display = h.display_value_factory{},
                            :attr('data-sort-value', vmax)
        field_options = {
                            :attr('class', 'tc -mod')
            [1] = {
                            :wikitext(text)
                skill_levels = true,
                    end
            },
                elseif col_info.stat_format == 'add' then
        },
                    local total_stat = {
    },
                        min = 0,
    {
                        max = 0,
        order = 7000,
                        avg = 0,
        args = {'radius'},
                    }
        header = i18n.item_table.primary_radius,
                    for _, stat_info in ipairs(col_info.stats) do
        fields = {'skill.radius', 'skill.radius_description'},
                        local stat = (results2.stats[data['items._pageName']] or {})[stat_info.id]
        field_options = {
                        if stat ~= nil then
            [2] = {
                            for k, v in pairs(total_stat) do
                optional = true,
                                total_stat[k] = v + stat[k]
            },
                            end
        },
                        end
        display = function (tr, data)
                    end
            tr
 
                :tag('td')
                    if col_info.format == nil then
                    :attr('data-sort-value', data['skill.radius'])
                        col_info.format = '%s'
                    :wikitext(h.display_descriptor_value_factory{tbl=data, key='skill.radius_description'}(data['skill.radius']))
                    end
        end,
 
    },
                    tr:tag('td')
    {
                        :attr('data-sort-value', total_stat.max)
        order = 7001,
                        :attr('class', 'tc -mod')
        args = {'radius'},
                        :wikitext(string.format(col_info.format, m_util.html.format_value(tpl_args, frame, total_stat, {no_color=true})))
        header = i18n.item_table.secondary_radius,
                else
        fields = {'skill.radius_secondary', 'skill.radius_secondary_description'},
                    error(string.format(i18n.errors.generic_argument_parameter, 'stat_format', col_info.stat_format))
        field_options = {
                end
            [2] = {
            end,
                optional = true,
            order = col_info.order,
            },
         }
        },
         table.insert(row_infos, row_info)
        display = function (tr, data)
     end
            tr
                :tag('td')
                    :attr('data-sort-value', data['skill.radius_secondary'])
                    :wikitext(h.display_descriptor_value_factory{tbl=data, key='skill.radius_secondary_description'}(data['skill.radius_secondary']))
        end,
    },
    {
        order = 7002,
        args = {'radius'},
        header = i18n.item_table.tertiary_radius,
        fields = {'skill.radius_tertiary', 'skill.radius_tertiary_description'},
        field_options = {
            [2] = {
                optional = true,
            },
        },
        display = function (tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', data['skill.radius_tertiary'])
                  :wikitext(h.display_descriptor_value_factory{tbl=data, key='skill.radius_tertiary_description'}(data['skill.radius_tertiary']))
        end,
    },
    {
        order = 8000,
        args = {'drop', 'drop_text'},
        header = i18n.item_table.drop_text,
        fields = {'items.drop_text'},
         display = h.display_value_factory{},
         sort_type = 'text',
     },
}
 
-- ----------------------------------------------------------------------------
-- Main functions
-- ----------------------------------------------------------------------------


     -- sort the rows
local function _item_table(args)
     table.sort(row_infos, function (a, b)
     --[[
        return (a.order or 0) < (b.order or 0)
     Creates a generic table for items.
    end)


     -- Parse query arguments
     Examples
     local tables_assoc = {items=true}
    --------
    local fields = {
     = p.item_table{
        'items._pageID',
        tables='vendor_rewards, quest_rewards, skill_gems',
         'items._pageName',
         join='items._pageID=vendor_rewards._pageID, items._pageID=quest_rewards._pageID, items._pageID=skill_gems._pageID',
        'items.name',
         where='(items.class="Active Skill Gems" OR items.class = "Support Skill Gems") AND (vendor_rewards.quest_id IS NOT NULL OR quest_rewards.quest_id IS NOT NULL)',
        'items.inventory_icon',
         vendor=1,
         'items.html',
        'items.size_x',
         'items.size_y',
     }
     }
    ]]


     --
     m_item_util = m_item_util or (use_sandbox and require('Module:Item util/sandbox') or require('Module:Item util'))
    local prepend = {
        q_groupBy=true,
        q_tables=true,
    }


     local query = {}
     local t = os.clock()
    for key, value in pairs(tpl_args) do
        if string.sub(key, 0, 2) == 'q_' then
            if prepend[key] then
                value = ',' .. value
            end


            query[string.sub(key, 3)] = value
    args.mode = args.mode or 'item'
        end
     local modes = {
    end
         skill = {
 
            data = data_map.skill_gem,
     local skill_levels = {}
             header = i18n.item_table.skill_gem,
    for _, rowinfo in ipairs(row_infos) do
         },
         if type(rowinfo.fields) == 'function' then
         item = {
             rowinfo.fields = rowinfo.fields()
             data = data_map.generic_item,
         end
             header = i18n.item_table.item,
         for index, field in ipairs(rowinfo.fields) do
        },
             rowinfo.options[index] = rowinfo.options[index] or {}
    }
             if rowinfo.options[index].skill_levels then
    if modes[args.mode] == nil then
                skill_levels[#skill_levels+1] = field
        error(i18n.errors.invalid_item_table_mode)
            else
                fields[#fields+1] = field
                tables_assoc[m_util.string.split(field, '%.')[1]] = true
            end
        end
     end
     end


     if #skill_levels > 0 then
    -- A where clause is required; there are far too many items to list in one table
         fields[#fields+1] = 'skill.max_level'
     if args.where == nil then
        tables_assoc.skill = true
         error(string.format(i18n.errors.generic_required_parameter, 'where'))
 
     end
     end


     -- Reformat the tables and fields so they can be retrieved correctly:
     local results2 = {
    local tables = {}
        stats = {},
    for table_name,_ in pairs(tables_assoc) do
         skill_levels = {},
         tables[#tables+1] = table_name
        pageIDs = {},
    end
     }
    local tbls = table.concat(tables,',') .. (query.tables or '')
     query.tables = m_util.string.split(tbls, ',')


     for index, field in ipairs(fields) do
    local row_infos = {}
         fields[index] = string.format('%s=%s', field, field)
     for _, row_info in ipairs(modes[args.mode].data) do
    end
         local enabled = false
    query.fields = fields
        if type(row_info.args) == 'table' then
 
            for _, a in ipairs(row_info.args) do
    -- Take care of the minimum required joins, joins from templates
                if m_util.cast.boolean(args[a]) then
    -- must still be userdefined:
                    enabled = true
    local joins = {}
                    break
    for index, table_name in ipairs(tables) do
                end
         if table_name ~= 'items' then
            end
             joins[#joins+1] = string.format('items._pageID=%s._pageID', table_name)
        else
            enabled = true
        end
         if enabled then
            row_info.field_options = row_info.field_options or {}
             row_infos[#row_infos+1] = row_info
         end
         end
    end
    if #joins > 0 and query.join then
        query.join = table.concat(joins, ',') .. ',' .. query.join
    elseif #joins > 0 and not query.join then
        query.join = table.concat(joins, ',')
    elseif #joins == 0 and query.join then
        -- leave query.join as is
     end
     end


     -- Needed to eliminate duplicates supplied via table joins:
     -- Parse stat arguments
     query.groupBy = 'items._pageID' .. (query.groupBy or '')
    local stat_columns = {}
 
    local query_stats = {}
    -- Query results:
     for i=1, math.huge do -- repeat until no more columns are found
    local results = m_cargo.query(query.tables, query.fields, query)
        local prefix = string.format('stat_column%s_', i)
 
        if args[prefix .. 'stat1_id'] == nil then
    if #results == 0 and tpl_args.default ~= nil then
            -- Each column requires at least one stat id
         return tpl_args.default
            break
        end
        local col_info = {
            header = args[prefix .. 'header'] or tostring(i),
            format = args[prefix .. 'format'],
            stat_format = args[prefix .. 'stat_format'] or 'separate',
            order = tonumber(args[prefix .. 'order']) or (10000000 + i),
            stats = {},
        }
        for j=1, math.huge do
            local stat_id = args[string.format('%sstat%s_id', prefix, j)]
            if stat_id == nil then
                break
            end
            table.insert(col_info.stats, {id=stat_id})
            query_stats[stat_id] = true
        end
         table.insert(stat_columns, col_info)
     end
     end


     if #results > 0 then
     for _, col_info in ipairs(stat_columns) do
        -- Create a list of found pageIDs for column specific queries:
        local row_info = {
        for _,v in ipairs(results) do
            header = col_info.header,
            results2.pageIDs[#results2.pageIDs+1] = v['items._pageID']
            fields = {},
        end
            display = function (tr, data)
                if col_info.stat_format == 'separate' then
                    local stat_texts = {}
                    local num_stats = 0
                    local vmax = 0
                    for _, stat_info in ipairs(col_info.stats) do
                        num_stats = num_stats + 1
                        -- stat results from outside body
                        local stat = (results2.stats[data['items._pageName']] or {})[stat_info.id]
                        if stat ~= nil then
                            stat_texts[#stat_texts+1] = m_util.html.format_value(args, stat, {color=false})
                            vmax = vmax + stat.max
                        end
                    end


        -- fetch skill level information
                    if num_stats ~= #stat_texts then
        if #skill_levels > 0 then
                        tr:node(m_util.html.table_cell('na'))
            skill_levels[#skill_levels+1] = 'skill_levels._pageName'
                    else
            skill_levels[#skill_levels+1] = 'skill_levels.level'
                        local text
            local pages = {}
                        if col_info.format then
            for _, row in ipairs(results) do
                            text = string.format(col_info.format, unpack(stat_texts))
                pages[#pages+1] = string.format('(skill_levels._pageID="%s" AND skill_levels.level IN (0, 1, %s))', row['items._pageID'], row['skill.max_level'])
                        else
            end
                            text = table.concat(stat_texts, ', ')
            local temp = m_cargo.query(
                        end
                {'skill_levels'},
                skill_levels,
                {
                    where=table.concat(pages, ' OR '),
                    groupBy='skill_levels._pageID, skill_levels.level',
                }
            )
            -- map to results
            for _, row in ipairs(temp) do
                if results2.skill_levels[row['skill_levels._pageName']] == nil then
                  results2.skill_levels[row['skill_levels._pageName']] = {}
                end
                -- TODO: convert to int?
                results2.skill_levels[row['skill_levels._pageName']][row['skill_levels.level']] = row
            end
        end


        if #stat_columns > 0 then
                        tr:tag('td')
            local pages = {}
                            :attr('data-sort-value', vmax)
            for _, row in ipairs(results) do
                            :attr('class', 'tc -mod')
                pages[#pages+1] = string.format('item_stats._pageID="%s"', row['items._pageID'])
                            :wikitext(text)
            end
                    end
                elseif col_info.stat_format == 'add' then
                    local total_stat = {
                        min = 0,
                        max = 0,
                        avg = 0,
                    }
                    for _, stat_info in ipairs(col_info.stats) do
                        local stat = (results2.stats[data['items._pageName']] or {})[stat_info.id]
                        if stat ~= nil then
                            for k, v in pairs(total_stat) do
                                total_stat[k] = v + stat[k]
                            end
                        end
                    end


            local query_stat_ids = {}
                    if col_info.format == nil then
            for stat_id, _ in pairs(query_stats) do
                        col_info.format = '%s'
                query_stat_ids[#query_stat_ids+1] = string.format('item_stats.id="%s"', stat_id)
                    end
            end


            if tpl_args.q_where then
                    tr:tag('td')
                tpl_args.q_where = string.format(' AND (%s)', tpl_args.q_where)
                        :attr('data-sort-value', total_stat.max)
            else
                        :attr('class', 'tc -mod')
                tpl_args.q_where = ''
                        :wikitext(string.format(col_info.format, m_util.html.format_value(args, total_stat, {no_color=true})))
            end
                else
 
                     error(string.format(i18n.errors.generic_argument_parameter, 'stat_format', col_info.stat_format))
            local temp = m_cargo.query(
                end
                {'items', 'item_stats'},
            end,
                {'item_stats._pageName', 'item_stats.id', 'item_stats.min', 'item_stats.max', 'item_stats.avg'},
            order = col_info.order,
                {
        }
                     where=string.format('item_stats.is_implicit IS NULL AND (%s) AND (%s)', table.concat(query_stat_ids, ' OR '), table.concat(pages, ' OR ')),
        table.insert(row_infos, row_info)
                    join='items._pageID=item_stats._pageID',
    end
                    -- Cargo workaround: avoid duplicates using groupBy
                    groupBy='items._pageID, item_stats.id',
                }
            )


            for _, row in ipairs(temp) do
    -- sort the rows
                local stat = {
    table.sort(row_infos, function (a, b)
                    min = tonumber(row['item_stats.min']),
        return (a.order or 0) < (b.order or 0)
                    max = tonumber(row['item_stats.max']),
    end)
                    avg = tonumber(row['item_stats.avg']),
                }


                if results2.stats[row['item_stats._pageName']] == nil then
    -- Build Cargo query
                    results2.stats[row['item_stats._pageName']] = {[row['item_stats.id']] = stat}
    local tables = {'items'}
                else
    local fields = {
                    results2.stats[row['item_stats._pageName']][row['item_stats.id']] = stat
        'items._pageID',
                end
        'items._pageName',
        'items.name',
        'items.inventory_icon',
        'items.html',
        'items.size_x',
        'items.size_y',
    }
    local query = {
        where = args.where,
        groupBy = table.concat({'items._pageID', args.groupBy}, ', '),
        having = args.having,
        orderBy = args.orderBy,
        limit = args.limit,
        offset = args.offset,
    }
 
    -- Namespace condition
    -- This is mainly to prevent items from user pages or other testing pages
    -- from being returned in the query results.
    if args.namespaces ~= 'any' then
        local namespaces = m_util.cast.table(args.namespaces, {callback=m_util.cast.number})
        if #namespaces > 0 then
            namespaces = table.concat(namespaces, ',')
        else
            namespaces = m_item_util.get_item_namespaces{format = 'list'}
        end
        query.where = string.format('(%s) AND items._pageNamespace IN (%s)', query.where, namespaces)
    end
 
    -- Minimum required tables and fields, based on display options
    local skill_levels = {}
    for _, rowinfo in ipairs(row_infos) do
        if type(rowinfo.fields) == 'function' then
            rowinfo.fields = rowinfo.fields()
        end
        for index, field in ipairs(rowinfo.fields) do
            rowinfo.field_options[index] = rowinfo.field_options[index] or {}
            if rowinfo.field_options[index].skill_levels then
                skill_levels[#skill_levels+1] = field
            else
                fields[#fields+1] = field
                tables[#tables+1] = m_util.string.split(field, '.', true)[1]
             end
             end
         end
         end
     end
     end
    if #skill_levels > 0 then
        fields[#fields+1] = 'skill.max_level'
        tables[#tables+1] = 'skill'
    end
    tables = m_util.table.remove_duplicates(tables)


    -- Minimum required joins, based on display options
    local joins = {}
    for _, table_name in ipairs(tables) do
        if table_name ~= 'items' then
            joins[#joins+1] = string.format('items._pageID=%s._pageID', table_name)
        end
    end


     --
     -- Append additional tables
     -- Display the table
     args.tables = m_util.cast.table(args.tables)
     --
     if type(args.tables) == 'table' and #args.tables > 0 then
        tables = m_util.table.merge(tables, args.tables)
    end


     local tbl = mw.html.create('table')
     -- Make join clause
     tbl:attr('class', 'wikitable sortable item-table')
     if #joins > 0 or args.join then
 
        -- m_util.table.merge rebuilds the table, which removes empty values
    -- Headers:
        query.join = table.concat(m_util.table.merge(joins, {args.join}), ', ')
    local tr = tbl:tag('tr')
    end
    tr
        :tag('th')
            :wikitext(modes[tpl_args.mode].header)
            :done()
    for _, row_info in ipairs(row_infos) do
        local th = tr:tag('th')


        if row_info.colspan then
    -- Query results
            th:attr('colspan', row_info.colspan)
    local results = m_cargo.query(tables, fields, query)
        end


        th
    if #results == 0 and args.default ~= nil then
            :attr('data-sort-type', row_info.sort_type or 'number')
        return args.default
            :wikitext(row_info.header)
     end
     end


     -- Rows:
     if #results > 0 then
    for _, row in ipairs(results) do
        -- Create a list of found pageIDs for column specific queries:
        tr = tbl:tag('tr')
        for _,v in ipairs(results) do
 
             results2.pageIDs[#results2.pageIDs+1] = v['items._pageID']
        local il_args = {
            skip_query=true,
            page=row['items._pageName'],
            name=row['items.name'],
            inventory_icon=row['items.inventory_icon'],
             width=row['items.size_x'],
            height=row['items.size_y'],
        }
 
        if tpl_args.no_html == nil then
            il_args.html = row['items.html']
         end
         end


         if tpl_args.large then
        -- fetch skill level information
             il_args.large = tpl_args.large
         if #skill_levels > 0 then
        end
             skill_levels[#skill_levels+1] = 'skill_levels._pageName'
 
            skill_levels[#skill_levels+1] = 'skill_levels.level'
        tr
            local pages = {}
             :tag('td')
             for _, row in ipairs(results) do
                 :wikitext(f_item_link(il_args))
                 pages[#pages+1] = string.format('(skill_levels._pageID="%s" AND skill_levels.level IN (0, 1, %s))', row['items._pageID'], row['skill.max_level'])
                 :done()
            end
 
            local temp = m_cargo.query(
        for _, rowinfo in ipairs(row_infos) do
                {'skill_levels'},
             -- this has been cast from a function in an earlier step
                skill_levels,
            local display = true
                 {
             for index, field in ipairs(rowinfo.fields) do
                    where=table.concat(pages, ' OR '),
                -- this will bet set to an empty value not nil confusingly
                    groupBy='skill_levels._pageID, skill_levels.level',
                 if row[field] == nil or row[field] == '' then
                }
                    local opts = rowinfo.options[index]
            )
                    if opts.optional ~= true and opts.skill_levels ~= true then
             -- map to results
                        display = false
             for _, row in ipairs(temp) do
                        break
                 if results2.skill_levels[row['skill_levels._pageName']] == nil then
                    else
                  results2.skill_levels[row['skill_levels._pageName']] = {}
                        row[field] = nil
                    end
                 end
                 end
                -- TODO: convert to int?
                results2.skill_levels[row['skill_levels._pageName']][row['skill_levels.level']] = row
             end
             end
            if display then
        end
                 rowinfo.display(tr, row, rowinfo.fields, results2)
 
             else
        if #stat_columns > 0 then
                 tr:wikitext(m_util.html.td.na())
            local stat_results = m_cargo.query(
                {'items', 'item_stats'},
                 {'item_stats._pageName', 'item_stats.id', 'item_stats.min', 'item_stats.max', 'item_stats.avg'},
                {
                    join = 'items._pageID=item_stats._pageID',
                    where = string.format(
                        'item_stats._pageID IN (%s) AND item_stats.id IN ("%s")',
                        table.concat(m_util.table.column(results, 'items._pageID'), ','),
                        table.concat(m_util.table.keys(query_stats), '","')
                    ),
                    groupBy = 'items._pageID, item_stats.id',
                }
            )
             for _, row in ipairs(stat_results) do
                 local stat = {
                    min = tonumber(row['item_stats.min']),
                    max = tonumber(row['item_stats.max']),
                    avg = tonumber(row['item_stats.avg']),
                }
 
                if results2.stats[row['item_stats._pageName']] == nil then
                    results2.stats[row['item_stats._pageName']] = {[row['item_stats.id']] = stat}
                else
                    results2.stats[row['item_stats._pageName']][row['item_stats.id']] = stat
                end
             end
             end
         end
         end
     end
     end


     cats = {}
     --
     if #results == query.limit then
    -- Display the table
         cats[#cats+1] = i18n.categories.query_limit
    --
 
    local tbl = mw.html.create('table')
    tbl:attr('class', 'wikitable sortable item-table')
     if m_util.cast.boolean(args.responsive) then
         tbl:addClass('responsive-table')
     end
     end


     if #results == 0 then
     -- Headers:
         cats[#cats+1] = i18n.categories.no_results
    local tr = tbl:tag('tr')
     end
    tr
         :tag('th')
            :wikitext(modes[args.mode].header)
            :done()
     for _, row_info in ipairs(row_infos) do
        local th = tr:tag('th')


    mw.logObject({os.clock() - t, query})
        if row_info.colspan then
            th:attr('colspan', row_info.colspan)
        end


    return tostring(tbl) .. m_util.misc.add_category(cats, {ingore_blacklist=tpl_args.debug})
        th
end
            :attr('data-sort-type', row_info.sort_type or 'number')
            :wikitext(row_info.header)
    end


    -- Rows:
    for _, row in ipairs(results) do
        tr = tbl:tag('tr')


-------------------------------------------------------------------------------
        local il_args = {
-- Map item drops
            skip_query=true,
-------------------------------------------------------------------------------
            page=row['items._pageName'],
            name=row['items.name'],
            inventory_icon=row['items.inventory_icon'],
            width=row['items.size_x'],
            height=row['items.size_y'],
        }


function p.map_item_drops(frame)
        if args.no_html == nil then
    --[[
            il_args.html = row['items.html']
    Gets the area id from the map item and activates
        end
    Template:Area_item_drops.


    Examples:
        if args.large then
    = p.map_item_drops{page='Underground River Map (War for the Atlas)'}
            il_args.large = args.large
    ]]
        end


    -- Get args
        tr
    local tpl_args = getArgs(frame, {
            :tag('td')
        parentFirst = true
                :wikitext(h.item_link(il_args))
    })
                :done()
    frame = m_util.misc.get_frame(frame)


    tpl_args.page = tpl_args.page or tostring(mw.title.getCurrentTitle())
        for _, rowinfo in ipairs(row_infos) do
 
            -- this has been cast from a function in an earlier step
    local results = m_cargo.query(
            local display = true
        {'maps'},
           
        {'maps.area_id'},
            for index, field in ipairs(rowinfo.fields) do
        {
                -- this will bet set to an empty value not nil confusingly
             where=string.format('maps._pageName="%s" AND maps.area_id IS NOT NULL', tpl_args.page),
                if row[field] == nil or row[field] == '' then
             -- Only need each page name once
                    local options = rowinfo.field_options[index]
            groupBy='maps._pageName',
                    if options.optional ~= true and options.skill_levels ~= true then
         }
                        display = false
     )
                        break
     local id = ''
                    else
     if #results > 0 then
                        row[field] = nil
         id = results[1]['maps.area_id']
                    end
                end
            end
             if display then
                rowinfo.display(tr, row, rowinfo.fields, results2)
             else
                tr:node(m_util.html.table_cell('na'))
            end
         end
     end
 
     local cats = {}
     if #results == query.limit then
         cats[#cats+1] = i18n.categories.query_limit
    end
    if #results == 0 then
        cats[#cats+1] = i18n.categories.no_results
     end
     end
     return frame:expandTemplate{ title = 'Area item drops', args = {area_id=id} }
 
     mw.logObject({os.clock() - t, {tables=tables, fields=fields, query=query}})
 
    return tostring(tbl) .. m_util.misc.add_category(cats, {ignore_blacklist=args.debug})
end
end


-------------------------------------------------------------------------------
-------------------------------------------------------------------------------
-- Prophecy description
-- Map item drops
-------------------------------------------------------------------------------
-------------------------------------------------------------------------------


function p.prophecy_description(frame)
local function _map_item_drops(args)
     -- Get args
    --[[
     local tpl_args = getArgs(frame, {
    Gets the area id from the map item and activates
         parentFirst = true
    Template:Area_item_drops.
     })
 
     frame = m_util.misc.get_frame(frame)
    Examples:
    = p.map_item_drops{page='Underground River Map (War for the Atlas)'}
    ]]
 
    local tables = {'maps'}
    local fields = {
        'maps.area_id',
     }
    local query = {
        -- Only need each page name once
        groupBy = 'maps._pageName',
    }
    if args.page then
        -- Join with _pageData in order to check for page redirect
        tables[#tables+1] = '_pageData'
        fields[#fields+1] = '_pageData._pageNameOrRedirect'
        query.where = string.format(
            '_pageData._pageName="%s"',
            args.page
        )
        query.join = 'maps._pageName = _pageData._pageNameOrRedirect'
     else
        query.where = string.format(
            'maps._pageName="%s"',
            tostring(mw.title.getCurrentTitle())
         )
    end
    query.where = query.where .. ' AND maps.area_id > ""' -- area_id must not be empty or null
     local results = m_cargo.query(tables, fields, query)
     local id = ''
    if #results > 0 then
        id = results[1]['maps.area_id']
    end
    return mw.getCurrentFrame():expandTemplate{ title = 'Area item drops', args = {area_id=id} }
end
 
-------------------------------------------------------------------------------
-- Prophecy description
-------------------------------------------------------------------------------


     tpl_args.page = tpl_args.page or tostring(mw.title.getCurrentTitle())
local function _prophecy_description(args)
     args.page = args.page or tostring(mw.title.getCurrentTitle())


     local results = m_cargo.query(
     local results = m_cargo.query(
Line 2,151: Line 2,328:
         {'prophecies.objective', 'prophecies.reward'},
         {'prophecies.objective', 'prophecies.reward'},
         {
         {
             where=string.format('prophecies._pageName="%s"', tpl_args.page),
             where=string.format('prophecies._pageName="%s"', args.page),
             -- Only need each page name once
             -- Only need each page name once
             groupBy='prophecies._pageName',
             groupBy='prophecies._pageName',
Line 2,174: Line 2,351:
end
end


-- ----------------------------------------------------------------------------
-- Item disambiguation
-- ----------------------------------------------------------------------------
function h.find_aliases(tpl_args)
  --[[
  This function queries items for an item name, then checks if it has
  had any name changes then queries for that name as well.
  ]]


     -- Get initial name:
local function _simple_item_list(args)
     tpl_args.name_list = {
     --[[
         tpl_args.name or m_util.string.split(
     Creates a simple list of items.
            tostring(mw.title.getCurrentTitle()),
 
            ' %('
    Examples
         )
    --------
    = p.simple_item_list{
         q_tables='maps',
        q_join='items._pageID=maps._pageID',
        q_where='maps.tier=1 AND items.drop_enabled=1 AND items.rarity_id="normal"',
        no_html=1,
         link_from_name=1,
     }
     }
    ]]


    -- Query for items with similar name, repeat until no new names are
     local query = {}
    -- found.
     for key, value in pairs(args) do
     local n
        if string.sub(key, 0, 2) == 'q_' then
    local results = {}
            query[string.sub(key, 3)] = value
     local hash = {}
    repeat
        local n_old = #tpl_args.name_list
 
        -- Multiple HOLDS doesn't work. Using __FULL and REGEXP instead.
        local where_tbl = {}
        for _, item_name in ipairs(tpl_args.name_list) do
            for _, prefix in ipairs({'', 'Shaped '}) do
                where_tbl[#where_tbl+1] = string.format(
                    '(items.name_list__FULL REGEXP "(�|^)%s%s(�|$)")',
                    prefix,
                    item_name
                )
            end
         end
         end
        local where_str = table.concat(where_tbl, ' OR ')
    end


        results = m_cargo.query(
    local fields = {
            {'items', 'maps'},
        'items._pageName',
            {
        'items.name',
                'items._pageName',
        'items.class',
                'items.name',
    }
                'items.name_list',
                'items.release_version',
                'items.removal_version',
                'items.drop_enabled',
            },
            {
                join='items._pageName=maps._pageName',
                where=where_str,
                groupBy='items._pageName',
                orderBy='items.release_version DESC, items.removal_version DESC, items.name ASC, maps.area_id ASC',
            }
        )


        -- Filter duplicates:
    if args.no_icon == nil then
         for i,v in ipairs(results) do
         fields[#fields+1] = 'items.inventory_icon'
            local r = m_util.string.split(v['items.name_list'], '�')
    end
            if type(r) == string then
                r = {r}
            end


            for j,m in ipairs(r) do
    if args.no_html == nil then
                if hash[m] == nil then
        fields[#fields+1] = 'items.html'
                    hash[m] = m
    end
                    tpl_args.name_list[#tpl_args.name_list+1] = m
                end
            end
        end
    until #tpl_args.name_list == n_old


     return results
     local tables = m_util.cast.table(args.q_tables)
end
    table.insert(tables, 1, 'items')


function p.item_disambiguation(frame)
    query.groupBy = query.groupBy or 'items._pageID'
    --[[
    This function finds that items with a name or has had that name.


     To do
     local results = m_cargo.query(
    -----
        tables,
    Should text imply which is the original map, even if it isn't (Original)?
        fields,
    How to properly sort drop disabled items, with removal version?
        query
    How to deal with names that have been used multiple times? Terrace Map
     )
 
     Examples
    --------
    = p.item_disambiguation{name='Abyss Map'}
    = p.item_disambiguation{name='Caldera Map'}
    = p.item_disambiguation{name='Crypt Map'}
    = p.item_disambiguation{name='Catacombs Map'}
    = p.item_disambiguation{name='Harbinger Map (High Tier)'}
    ]]


    -- Get template arguments.
    local tpl_args = getArgs(frame, {
        parentFirst = true
    })
    frame = m_util.misc.get_frame(frame)
    local current_title = tostring(mw.title.getCurrentTitle())
    -- Get the page name.
    tpl_args.name = tpl_args.name or m_util.string.split(
        current_title,
        ' %('
    )[1]
    -- Query for items with similar name.
    local results = h.find_aliases(tpl_args)
    -- Format the results:
     local out = {}
     local out = {}
    local container = mw.html.create('div')
     for _, row in ipairs(results) do
    local tbl = container:tag('ul')
        local page
     for i,v in ipairs(results) do
         if args.use_name_as_link ~= nil then
         if v['items._pageName'] ~= current_title then
            page = row['items.name']
            -- Get the content inside the last parentheses:
        else
             local known_release = string.reverse(
             page = row['items._pageName']
                string.match(
        end
                    string.reverse(v['items._pageName']),
                    '%)(.-)%('
                )
            )


            local drop_enabled
        local link = h.item_link{
             if known_release == 'Original' then
             page=page,
                drop_enabled = i18n.item_disambiguation.original
            name=row['items.name'],
             elseif m_util.cast.boolean(v['items.drop_enabled']) then
            inventory_icon=row['items.inventory_icon'] or '',
                drop_enabled = i18n.item_disambiguation.drop_enabled
             html=row['items.html'] or '',
             else
             skip_query=true
                drop_enabled = i18n.item_disambiguation.drop_disabled
        }
            end


 
        if args.format == nil then
            if known_release ~= 'Original' then
            out[#out+1] = string.format('* %s', link)
                known_release = string.format(
        elseif args.format == 'none' then
                    i18n.item_disambiguation.known_release,
             out[#out+1] = link
                    known_release,
        elseif args.format == 'li' then
                    known_release
             out[#out+1] = string.format('<li>%s</li>', link)
                )
        else
            else
            error(string.format(i18n.errors.generic_argument_parameter, 'format', args.format))
                known_release = ''
             end
 
             tbl
                :tag('li')
                    :wikitext(
                        string.format(
                            i18n.item_disambiguation.list_pattern,
                            f_item_link{page=v['items._pageName']},
                            drop_enabled,
                            known_release
                        )
                    )
         end
         end
     end
     end
    out[#out+1] = tostring(container)


     -- Add a category when the template uses old template inputs:
     if args.format == nil then
    local old_args = {
         return table.concat(out, '\n')
         'war',
    elseif args.format == 'none' then
        'atlas',
         return table.concat(out, '\n')
        'awakening',
    elseif args.format == 'li' then
         'original',
         return table.concat(out)
        'heading',
        'hide_heading',
         'show_current',
    }
    for _,v in ipairs(old_args) do
        if tpl_args[v] ~= nil then
            return table.concat(out, '') .. m_util.misc.add_category(
                {'Pages with old template arguments'}
            )
        end
     end
     end
    return table.concat(out, '')
end
end


-- ----------------------------------------------------------------------------
-- Exported functions
-- ----------------------------------------------------------------------------


function p.simple_item_list(frame)
local p = {}
    --[[
    Creates a simple list of items.


    Examples
--
    --------
-- Template:Item table
    = p.simple_item_list{
--
        q_tables='maps',
p.item_table = m_util.misc.invoker_factory(_item_table)
        q_join='items._pageID=maps._pageID',
        q_where='maps.tier=1 AND items.drop_enabled=1 AND items.rarity_id="normal"',
        no_html=1,
        link_from_name=1,
    }
    ]]


    -- Args
--
    local tpl_args = getArgs(frame, {
-- Template:Map item drops
        parentFirst = true
--
    })
p.map_item_drops = m_util.misc.invoker_factory(_map_item_drops, {
     frame = m_util.misc.get_frame(frame)
     wrappers = cfg.wrappers.map_item_drops,
})


    local query = {}
--
    for key, value in pairs(tpl_args) do
-- Template:Prophecy description
        if string.sub(key, 0, 2) == 'q_' then
--
            query[string.sub(key, 3)] = value
p.prophecy_description = m_util.misc.invoker_factory(_prophecy_description, {
        end
    wrappers = cfg.wrappers.prophecy_description,
    end
})


    local fields = {
--
        'items._pageName',
-- Template:Simple item list
        'items.name',
--
        'items.class',
p.simple_item_list = m_util.misc.invoker_factory(_simple_item_list, {
    }
     wrappers = cfg.wrappers.simple_item_list,
 
})
    if tpl_args.no_icon == nil then
        fields[#fields+1] = 'items.inventory_icon'
    end
 
    if tpl_args.no_html == nil then
        fields[#fields+1] = 'items.html'
    end
 
    local tables = m_util.string.split(tpl_args.q_tables or '', ',%s*')
    table.insert(tables, 'items')
 
    query.groupBy = query.groupBy or 'items._pageID'
 
    local results = m_cargo.query(
        tables,
        fields,
        query
    )
 
    local out = {}
     for _, row in ipairs(results) do
        local page
        if tpl_args.use_name_as_link ~= nil then
            page = row['items.name']
        else
            page = row['items._pageName']
        end
 
        local link = f_item_link{
            page=page,
            name=row['items.name'],
            inventory_icon=row['items.inventory_icon'] or '',
            html=row['items.html'] or '',
            skip_query=true
        }
 
        if tpl_args.format == nil then
            out[#out+1] = string.format('* %s', link)
        elseif tpl_args.format == 'none' then
            out[#out+1] = link
        elseif tpl_args.format == 'li' then
            out[#out+1] = string.format('<li>%s</li>', link)
        else
            error(string.format(i18n.errors.generic_argument_parameter, 'format', tpl_args.format))
        end
    end
 
    if tpl_args.format == nil then
        return table.concat(out, '\n')
    elseif tpl_args.format == 'none' then
        return table.concat(out, '\n')
    elseif tpl_args.format == 'li' then
        return table.concat(out)
    end
end


-- ----------------------------------------------------------------------------
-- ----------------------------------------------------------------------------
-- Debug stuff
-- Debug
-- ----------------------------------------------------------------------------
-- ----------------------------------------------------------------------------
p.debug = {}
p.debug = {}
Line 2,474: Line 2,498:


function p.debug.skill_gem_all()
function p.debug.skill_gem_all()
     return p.debug._tbl_data(data_map.skill_gem_new)
     return p.debug._tbl_data(data_map.skill_gem)
end
end
-- ----------------------------------------------------------------------------
-- Return
-- ----------------------------------------------------------------------------


return p
return p

Latest revision as of 08:34, 17 September 2025

Module documentation[view] [edit] [history] [purge]


The item module provides functionality for creating item tables.

Implemented templates

This module implements the following templates:

-------------------------------------------------------------------------------
-- 
--                             Module:Item table
-- 
-- This module implements [[Template:Item table]] and other templates that query
-- and display tables or lists of items.
-------------------------------------------------------------------------------

require('Module:No globals')
local m_util = require('Module:Util')

-- Should we use the sandbox version of our submodules?
local use_sandbox = m_util.misc.maybe_sandbox('Item table')

local m_cargo = use_sandbox and require('Module:Cargo/sandbox') or require('Module:Cargo')
local m_game = use_sandbox and mw.loadData('Module:Game/sandbox') or mw.loadData('Module:Game')

-- Lazy loading
local m_item_util -- require('Module:Item util')
local f_item_link -- require('Module:Item link').item_link
local f_skill_link -- require('Module:Skill link').skill_link

-- The cfg table contains all localisable strings and configuration, to make it
-- easier to port this module to another wiki.
local cfg = use_sandbox and mw.loadData('Module:Item table/config/sandbox') or mw.loadData('Module:Item table/config')

local i18n = cfg.i18n

-- ----------------------------------------------------------------------------
-- Helper functions
-- ----------------------------------------------------------------------------

local h = {}
h.string = {}

function h.string.format(str, vars)
    --[[
    Allow string replacement using named arguments.
    
    TODO: 
    * Support %d ?
    * Support 0.2f ?

    Parameters
    ----------
    str : String to replace. 
    vars : Table of arguments.

    Examples
    --------
    = h.string.format('{foo} is {bar}.', {foo='Dinner', bar='nice'})

    References
    ----------
    http://lua-users.org/wiki/StringInterpolation
    ]]

    if not vars then
        vars = str
        str = vars[1]
    end
    
    return (string.gsub(str, "({([^}]+)})",
        function(whole, i)
          return vars[i] or whole
        end))
end

-- Lazy loading for Module:Item link
function h.item_link(args)
    if not f_item_link then
        f_item_link = use_sandbox and require('Module:Item link/sandbox').item_link or require('Module:Item link').item_link
    end
    return f_item_link(args)
end

-- Lazy loading for Module:Skill link
function h.skill_link(args)
    if not f_skill_link then
        f_skill_link = use_sandbox and require('Module:Skill link/sandbox').skill_link or require('Module:Skill link').skill_link
    end
    return f_skill_link(args)
end

function h.range_fields_factory(args)
    -- Returns a function that gets the range fields for the given keys
    local suffixes = {'maximum', 'text', 'colour'}
    if args.full then
        suffixes[#suffixes+1] = 'minimum'
    end
    return function ()
        local fields = {}
        for _, field in ipairs(args.fields) do
            for _, suffix in ipairs(suffixes) do
                fields[#fields+1] = string.format('%s_range_%s', field, suffix)
            end
        end
        return fields
    end
end

function h.display_value_factory(args)
    args.fmt_options = args.fmt_options or {}
    return function(tr, data, fields, data2)
        local values = {}
        local fmt_values = {}
        for index, field in ipairs(fields) do
            local value = {
                min=data[field],
                max=data[field],
                base=data[field],
            }
            local sdata = data2 and data2.skill_levels[data['items._pageName']]
            if sdata then --For skill data
                -- Use the fixed value, or the first-level value.
                value.min = value.min or sdata['0'][field] or sdata['1'][field]
                -- Fall back to the fixed value, and then the max level value.
                value.max = value.max or sdata['0'][field] or sdata[data['skill.max_level']][field]
            end
            if value.min then
                values[#values+1] = value.max
                local options = args.fmt_options[index] or {}
                -- global color is set, no overrides
                if args.color ~= nil then
                    options.color = false
                end
                fmt_values[#fmt_values+1] = m_util.html.format_value(nil, value, options)
            end
        end
        if #values == 0 then
            tr:node(m_util.html.table_cell('na'))
        else
            local td = tr:tag('td')
            td
                :attr('data-sort-value', table.concat(values, ', '))
                :wikitext(table.concat(fmt_values, ', '))
            if args.color then
                td:attr('class', 'tc -' .. args.color)
            end
        end
    end
end

function h.display_range_factory(args)
    -- args: table
    --  property
    return function (tr, data, fields)
        tr
            :tag('td')
                :attr('data-sort-value', data[string.format('%s_range_maximum', args.field)] or '0')
                :attr('class', 'tc -' .. (data[string.format('%s_range_colour', args.field)] or 'default'))
                :wikitext(data[string.format('%s_range_text', args.field)])
                :done()
    end
end

function h.display_range_composite_factory(args)
    -- division by default
    if args.func == nil then
        args.func = function (a, b)
            if b == 0 then
                return 'fail'
            end
            return a / b
        end
    end
    
    return function(tr, data, fields)
        local field = {}
        for i=1, 2 do 
            local fieldn = args['field' .. i]
            field[i] = {
                min = tonumber(data[string.format('%s_range_minimum', fieldn)]) or 0,
                max = tonumber(data[string.format('%s_range_maximum', fieldn)]) or 0,
                color = data[string.format('%s_range_colour', fieldn)] or 'default',
            }
        end
        
        field.min = args.func(field[1].min, field[2].min)
        field.max = args.func(field[1].max, field[2].max)
        
        if field.min == 'fail' or field.max == 'fail' then
            field.text = ''
            field.color = 'default'
        else
            for i=1, 2 do
                if field[i].color ~= 'default' then
                    field.color = field[i].color
                    break
                end
            end
            if field.min == field.max then
                field.text = string.format('%.2f', field.min)
            else
                field.text = string.format('(%.2f-%.2f)', field.min, field.max)
            end
        end
    
        tr
            :tag('td')
                :attr('data-sort-value', field.max)
                :attr('class', 'tc -' .. field.color)
                :wikitext(field.text)
                :done()
    end
end

function h.display_descriptor_value_factory(args)
    -- Arguments:
    --  key
    --  tbl
    args = args or {}
    return function (value)
        if args.tbl[args.key] then
            value = m_util.html.tooltip(value, args.tbl[args.key])
        end
        return value
    end
end

function h.display_atlas_tier_factory(args)
    args = args or {}
    return function (tr, data)
        for i=0,4 do
            local t = tonumber(data['atlas_maps.map_tier' .. i])

            if t == 0 then
                tr
                    :tag('td')
                        :attr('table-sort-value', 0)
                        :attr('class', 'table-cell-xmark')
                        :wikitext('✗')
            else
                if args.is_level then
                    t = t + 67
                end
                tr
                    :tag('td')
                        :attr('class', 'tc -value')
                        :wikitext(t)
                        :done()
            end
        end
    end
end

function h.display_yesno_factory(args)
    -- args:
    --  field
    --  condition
    args = args or {}
    args.condition = args.condition or function (value)
        return m_util.cast.boolean(value, {cast_nil=false})
    end
    return function (tr, data)
        local type = args.condition(data[args.field]) and 'yes' or 'no'
        tr:node(m_util.html.table_cell(type))
    end
end

function h.value_greater_than_zero(value)
    value = m_util.cast.number(value, {default=0})
    if value > 0 then
        return true
    end
    return false
end

function h.na_or_val(tr, value)
    if value == nil or value == '' then
        tr:node(m_util.html.table_cell('na'))
    else
        tr
            :tag('td')
                :attr('data-sort-value', value)
                :wikitext(value)
    end
end

-- ----------------------------------------------------------------------------
-- Data mappings
-- ----------------------------------------------------------------------------

local data_map = {}

-- for sort type see:
-- https://meta.wikimedia.org/wiki/Help:Sorting
data_map.generic_item = {
    {
        order = 1000,
        args = {'base_item'},
        header = i18n.item_table.base_item,
        fields = {'items.base_item', 'items.base_item_page'},
        display = function(tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', data['items.base_item'])
                    :wikitext(string.format('[[%s|%s]]', data['items.base_item_page'], data['items.base_item']))
        end,
        sort_type = 'text',
    },
    {
        order = 1001,
        args = {'class'},
        header = i18n.item_table.item_class,
        fields = {'items.class'},
        display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '[[%s]]',
                },
            },
        },
        sort_type = 'text',
    },
    {
        order = 1002,
        args = {'rarity'},
        header = i18n.item_table.rarity,
        fields = {'items.rarity'},
        display = h.display_value_factory{},
    },
    {
        order = 1003,
        args = {'rarity_id'},
        header = i18n.item_table.rarity_id,
        fields = {'items.rarity_id'},
        display = h.display_value_factory{},
    },
    {
        order = 1004,
        args = {'metadata_id'},
        header = i18n.item_table.metadata_id,
        fields = {'items.metadata_id'},
        display = h.display_value_factory{},
    },
    {
        order = 1005,
        args = {'size'},
        header = i18n.item_table.inventory_size,
        fields = {'items.size_x', 'items.size_y'},
        display = function (tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', data['items.size_x'] * data['items.size_y'])
                    :wikitext(string.format('%s×%s', data['items.size_x'], data['items.size_y']))
        end,
    },
    {
        order = 1100,
        args = {'essence'},
        header = i18n.item_table.essence_level,
        fields = {'essences.level'},
        display = h.display_value_factory{},
    },
    {
        order = 1300,
        args = {'stack_size'},
        header = i18n.item_table.stack_size,
        fields = {'stackables.stack_size'},
        display = h.display_value_factory{},
    },
    {
        order = 1301,
        args = {'stack_size_currency_tab'},
        header = i18n.item_table.stack_size_currency_tab,
        fields = {'stackables.stack_size_currency_tab'},
        display = h.display_value_factory{},
    },
    -- Requirements
    {
        order = 1400,
        args = {'level'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.level_icon,
                i18n.item_table.required_level
            ),
            i18n.item_table.required_level
        ),
        fields = h.range_fields_factory{fields={'items.required_level'}},
        display = h.display_range_factory{field='items.required_level'},
    },
    {
        order = 1401,
        args = {'str'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.str_icon,
                i18n.item_table.required_str
            ),
            i18n.item_table.required_str
        ),
        fields = h.range_fields_factory{fields={'items.required_strength'}},
        display = h.display_range_factory{field='items.required_strength'},
    },
    {
        order = 1402,
        args = {'dex'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.dex_icon,
                i18n.item_table.required_dex
            ),
            i18n.item_table.required_dex
        ),
        fields = h.range_fields_factory{fields={'items.required_dexterity'}},
        display = h.display_range_factory{field='items.required_dexterity'},
    },
    {
        order = 1403,
        args = {'int'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.int_icon,
                i18n.item_table.required_int
            ),
            i18n.item_table.required_int
        ),
        fields = h.range_fields_factory{fields={'items.required_intelligence'}},
        display = h.display_range_factory{field='items.required_intelligence'},
    },
    -- Armour 15xx
    {
        order = 1500,
        args = {'ar'},
        header = i18n.item_table.armour,
        fields = h.range_fields_factory{fields={'armours.armour'}},
        display = h.display_range_factory{field='armours.armour'},
    },
    {
        order = 1501,
        args = {'ev'},
        header =i18n.item_table.evasion,
        fields = h.range_fields_factory{fields={'armours.evasion'}},
        display = h.display_range_factory{field='armours.evasion'},
    },
    {
        order = 1502,
        args = {'es'},
        header = i18n.item_table.energy_shield,
        fields = h.range_fields_factory{fields={'armours.energy_shield'}},
        display = h.display_range_factory{field='armours.energy_shield'},
    },
    {
        order = 1503,
        args = {'wd'},
        header = i18n.item_table.ward,
        fields = h.range_fields_factory{fields={'armours.ward'}},
        display = h.display_range_factory{field='armours.ward'},
    },
    {
        order = 1504,
        args = {'block'},
        header = i18n.item_table.block,
        fields = h.range_fields_factory{fields={'shields.block'}},
        display = h.display_range_factory{field='shields.block'},
    },
    -- Weapons 16xx
    {
        order = 1600,
        args = {'weapon', 'damage'},
        header = i18n.item_table.damage,
        fields = {'weapons.damage_html', 'weapons.damage_avg'},
        display = function (tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', data['weapons.damage_avg'])
                    :wikitext(data['weapons.damage_html'])
        end,
    },
    {
        order = 1601,
        args = {'weapon', 'aps'},
        header = i18n.item_table.attacks_per_second,
        fields = h.range_fields_factory{fields={'weapons.attack_speed'}},
        display = h.display_range_factory{field='weapons.attack_speed'},
    },
    {
        order = 1602,
        args = {'weapon', 'crit'},
        header = i18n.item_table.local_critical_strike_chance,
        fields = h.range_fields_factory{fields={'weapons.critical_strike_chance'}},
        display = h.display_range_factory{field='weapons.critical_strike_chance'},
    },
    {
        order = 1603,
        args = {'physical_dps'},
        header = i18n.item_table.physical_dps,
        fields = h.range_fields_factory{fields={'weapons.physical_dps'}},
        display = h.display_range_factory{field='weapons.physical_dps'},
    },
    {
        order = 1604,
        args = {'lightning_dps'},
        header = i18n.item_table.lightning_dps,
        fields = h.range_fields_factory{fields={'weapons.lightning_dps'}},
        display = h.display_range_factory{field='weapons.lightning_dps'},
    },
    {
        order = 1605,
        args = {'cold_dps'},
        header = i18n.item_table.cold_dps,
        fields = h.range_fields_factory{fields={'weapons.cold_dps'}},
        display = h.display_range_factory{field='weapons.cold_dps'},
    },
    {
        order = 1606,
        args = {'fire_dps'},
        header = i18n.item_table.fire_dps,
        fields = h.range_fields_factory{fields={'weapons.fire_dps'}},
        display = h.display_range_factory{field='weapons.fire_dps'},
    },
    {
        order = 1607,
        args = {'chaos_dps'},
        header = i18n.item_table.chaos_dps,
        fields = h.range_fields_factory{fields={'weapons.chaos_dps'}},
        display = h.display_range_factory{field='weapons.chaos_dps'},
    },
    {
        order = 1608,
        args = {'elemental_dps'},
        header = i18n.item_table.elemental_dps,
        fields = h.range_fields_factory{fields={'weapons.elemental_dps'}},
        display = h.display_range_factory{field='weapons.elemental_dps'},
    },
    {
        order = 1609,
        args = {'poison_dps'},
        header = i18n.item_table.poison_dps,
        fields = h.range_fields_factory{fields={'weapons.poison_dps'}},
        display = h.display_range_factory{field='weapons.poison_dps'},
    },
    {
        order = 1610,
        args = {'dps'},
        header = i18n.item_table.dps,
        fields = h.range_fields_factory{fields={'weapons.dps'}},
        display = h.display_range_factory{field='weapons.dps'},
    },
    {
        order = 1611,
        args = {'reload_time'},
        header = i18n.item_table.reload_time,
        fields = h.range_fields_factory{fields={'weapons.reload_time'}},
        display = h.display_range_factory{field='weapons.reload_time'},
    },
    -- Flasks 17xx
    {
        order = 1700,
        args = {'flask_life'},
        header = i18n.item_table.flask_life,
        fields = h.range_fields_factory{fields={'flasks.life'}},
        display = h.display_range_factory{field='flasks.life'},
    },
    {
        order = 1701,
        args = {'flask_life_per_second'},
        header = i18n.item_table.flask_life_per_second,
        fields = h.range_fields_factory{fields={'flasks.life', 'flasks.duration'}, full=true},
        display = h.display_range_composite_factory{field1='flasks.life', field2='flasks.duration'},
    },
    {
        order = 1702,
        args = {'flask_life_per_charge'},
        header = i18n.item_table.flask_life_per_charge,
        fields = h.range_fields_factory{fields={'flasks.life', 'flasks.charges_per_use'}, full=true},
        display = h.display_range_composite_factory{field1='flasks.life', field2='flasks.charges_per_use'},
    },
    {
        order = 1703,
        args = {'flask_mana'},
        header = i18n.item_table.flask_mana,
        fields = h.range_fields_factory{fields={'flasks.mana'}},
        display = h.display_range_factory{field='flasks.mana'},
    },
    {
        order = 1704,
        args = {'flask_mana_per_second'},
        header = i18n.item_table.flask_mana_per_second,
        fields = h.range_fields_factory{fields={'flasks.mana', 'flasks.duration'}, full=true},
        display = h.display_range_composite_factory{field1='flasks.mana', field2='flasks.duration'},
    },
    {
        order = 1705,
        args = {'flask_mana_per_charge'},
        header = i18n.item_table.flask_mana_per_charge,
        fields = h.range_fields_factory{fields={'flasks.mana', 'flasks.charges_per_use'}, full=true},
        display = h.display_range_composite_factory{field1='flasks.mana', field2='flasks.charges_per_use'},
    },
    {
        order = 1706,
        args = {'flask'},
        header = i18n.item_table.flask_duration,
        fields = h.range_fields_factory{fields={'flasks.duration'}},
        display = h.display_range_factory{field='flasks.duration'},
    },
    {
        order = 1707,
        args = {'flask'},
        header = i18n.item_table.flask_charges_per_use,
        fields = h.range_fields_factory{fields={'flasks.charges_per_use'}},
        display = h.display_range_factory{field='flasks.charges_per_use'},
    },
    {
        order = 1708,
        args = {'flask'},
        header = i18n.item_table.flask_maximum_charges,
        fields = h.range_fields_factory{fields={'flasks.charges_max'}},
        display = h.display_range_factory{field='flasks.charges_max'},
    },
    -- Jewels 18xx
    {
        order = 1800,
        args = {'jewel_limit'},
        header = i18n.item_table.jewel_limit,
        fields = {'jewels.jewel_limit'},
        display = h.display_value_factory{},
    },
    {
        order = 1801,
        args = {'jewel_radius'},
        header = i18n.item_table.jewel_radius,
        fields = {'jewels.radius_html'},
        display = function (tr, data)
            tr
                :tag('td')
                    :wikitext(data['jewels.radius_html'])
        end,
    },
    -- Maps 19xx
    {
        order = 1900,
        args = {'map_tier'},
        header = i18n.item_table.map_tier,
        fields = {'maps.tier'},
        display = h.display_value_factory{},
    },
    {
        order = 1901,
        args = {'map_level'},
        header = i18n.item_table.map_level,
        fields = {'maps.area_level'},
        display = h.display_value_factory{},
    },
    {
        order = 1902,
        args = {'map_guild_character'},
        header = i18n.item_table.map_guild_character,
        fields = {'maps.guild_character'},
        display = h.display_value_factory{},
        sort_type = 'text',
    },
    {
        order = 1903,
        args = {'map_series'},
        header = i18n.item_table.map_series,
        fields = {'maps.series'},
        display = h.display_value_factory{},
    },
    {
        order = 1904,
        args = {'atlas_tier'},
        header = i18n.item_table.atlas_tier,
        fields = {'atlas_maps.map_tier0', 'atlas_maps.map_tier1', 'atlas_maps.map_tier2', 'atlas_maps.map_tier3', 'atlas_maps.map_tier4'},
        display = h.display_atlas_tier_factory{is_level=false},
        colspan = 5,
    },
    {
        order = 1905,
        args = {'atlas_level'},
        header = i18n.item_table.atlas_level,
        fields = {'atlas_maps.map_tier0', 'atlas_maps.map_tier1', 'atlas_maps.map_tier2', 'atlas_maps.map_tier3', 'atlas_maps.map_tier4'},
        display = h.display_atlas_tier_factory{is_level=true},
        colspan = 5,
    },
    -- Map fragments 20xx
    {
        order = 2000,
        args = {'map_fragment', 'map_fragment_limit'},
        header = i18n.item_table.map_fragment_limit,
        fields = {'map_fragments.map_fragment_limit'},
        display = h.display_value_factory{},
    },
    -- Hideout decorations 21xx
    {
        order = 2100,
        args = {'doodad', 'variation_count'},
        header = i18n.item_table.variation_count,
        fields = {'hideout_doodads.variation_count'},
        display = h.display_value_factory{
            color = 'mod',
        },
    },
    -- Corpse items 22xx
    {
        order = 2200,
        args = {'corpse', 'monster_category'},
        header = i18n.item_table.monster_category,
        fields = {'corpse_items.monster_category', 'corpse_items.monster_category_html'},
        display = function (tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', m_game.constants.monster.categories[data['corpse_items.monster_category']].id)
                    :wikitext(data['corpse_items.monster_category_html'])
        end,
    },
    {
        order = 2201,
        args = {'corpse', 'monster_abilities'},
        header = i18n.item_table.monster_abilities,
        fields = {'corpse_items.monster_abilities'},
        display = h.display_value_factory{
            color = 'mod',
        },
        sort_type = 'text',
    },
    -- Seed data 91xx,
    {
        order = 9100,
        args = {'seed', 'seed_type'},
        header = i18n.item_table.seed_type,
        fields = {'harvest_seeds.type', 'harvest_seeds.type_id'},
        display = function (tr, data)
            tr
                :tag('td')
                    :attr('table-sort-value', data['harvest_seeds.type'])
                    :attr('class', 'tc -' .. data['harvest_seeds.type_id'])
                    :wikitext(data['harvest_seeds.type'])
        end,
    },
    {
        order = 9101,
        args = {'seed', 'seed_tier'},
        header = i18n.item_table.seed_tier,
        fields = {'harvest_seeds.tier'},
        display = h.display_value_factory{},
    },
    {
        order = 9102,
        args = {'seed', 'seed_growth_cycles'},
        header = i18n.item_table.seed_growth_cycles,
        fields = {'harvest_seeds.growth_cycles'},
        display = h.display_value_factory{},
    },
    {
        order = 9110,
        args = {'seed', 'seed_cosumed_lifeforce', 'seed_consumed_lifeforce_percentage', 'seed_consumed_primal_lifeforce_percentage'},
        header = i18n.item_table.seed_consumed_primal_lifeforce_percentage,
        fields = {'harvest_seeds.consumed_primal_lifeforce_percentage'},
        display = h.display_value_factory{
            color = 'primal',
        },
    },
    {
        order = 9110,
        args = {'seed', 'seed_cosumed_lifeforce', 'seed_consumed_lifeforce_percentage', 'seed_consumed_vivid_lifeforce_percentage'},
        header = i18n.item_table.seed_consumed_vivid_lifeforce_percentage,
        fields = {'harvest_seeds.consumed_vivid_lifeforce_percentage'},
        display = h.display_value_factory{
            color = 'vivid',
        },
    },
    {
        order = 9110,
        args = {'seed', 'seed_cosumed_lifeforce', 'seed_consumed_lifeforce_percentage', 'seed_consumed_wild_lifeforce_percentage'},
        header = i18n.item_table.seed_consumed_wild_lifeforce_percentage,
        fields = {'harvest_seeds.consumed_wild_lifeforce_percentage'},
        display = h.display_value_factory{
            color = 'wild',
        },
    },
    {
        order = 9113,
        args = {'seed', 'seed_required_nearby_seeds', 'seed_required_nearby_seed_amount'},
        header = i18n.item_table.seed_required_nearby_seed_amount,
        fields = {'harvest_seeds.required_nearby_seed_amount'},
        display = h.display_value_factory{},
    },
    {
        order = 9114,
        args = {'seed', 'seed_required_nearby_seeds', 'seed_required_nearby_seed_tier'},
        header = i18n.item_table.seed_required_nearby_seed_tier,
        fields = {'harvest_seeds.required_nearby_seed_tier'},
        display = h.display_value_factory{},
    },
    {
        order = 12000,
        args = {'buff'},
        header = i18n.item_table.buff_effects,
        fields = {'item_buffs.stat_text'},
        display = h.display_value_factory{
            color = 'mod',
        },
        sort_type = 'text',
    },
    {
        order = 12001,
        args = {'stat'},
        header = i18n.item_table.stats,
        fields = {'items.stat_text'},
        display = h.display_value_factory{
            color = 'mod',
        },
        sort_type = 'text',
    },
    {
        order = 12002,
        args = {'inherent_skills'},
        header = i18n.item_table.inherent_skills,
        fields = {'inherent_skills.inherent_skills_text'},
        display = h.display_value_factory{
            color = 'mod',
        },
        sort_type = 'text',
    },
    {
        order = 12003,
        args = {'description'},
        header = i18n.item_table.description,
        fields = {'items.description'},
        display = h.display_value_factory{
            color = 'mod',
        },
        sort_type = 'text',
    },
    {
        order = 12004,
        args = {'seed', 'seed_effect'},
        header = i18n.item_table.seed_effects,
        fields = {'harvest_seeds.effect'},
        display = h.display_value_factory{
            color = 'crafted',
        },
        sort_type = 'text',
    },
    {
        order = 12100,
        args = {'flavour_text'},
        header = i18n.item_table.flavour_text,
        fields = {'items.flavour_text'},
        display = h.display_value_factory{
            color = 'flavour',
        },
        sort_type = 'text',
    },
    {
        order = 12200,
        args = {'help_text'},
        header = i18n.item_table.help_text,
        fields = {'items.help_text'},
        display = h.display_value_factory{
            color = 'help',
        },
        sort_type = 'text',
    },
    {
        order = 12300,
        args = {'buff_icon'},
        header = i18n.item_table.buff_icon,
        fields = {'item_buffs.icon'},
        display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '[[%s]]',
                },
            },
        },
        sort_type = 'text',
    },
    {
        order = 13000,
        args = {'prophecy', 'objective'},
        header = i18n.item_table.objective,
        fields = {'prophecies.objective'},
        display = h.display_value_factory{},
    },
    {
        order = 13001,
        args = {'prophecy', 'reward'},
        header = i18n.item_table.reward,
        fields = {'prophecies.reward'},
        display = h.display_value_factory{},
    },
    {
        order = 13002,
        args = {'prophecy', 'seal_cost'},
        header = i18n.item_table.seal_cost,
        fields = {'prophecies.seal_cost'},
        display = h.display_value_factory{
            color = 'currency',
        },
    },
    {
        order = 13003,
        args = {'prediction_text'},
        header = i18n.item_table.prediction_text,
        fields = {'prophecies.prediction_text'},
        display = h.display_value_factory{
            color = 'value',
        },
        sort_type = 'text',
    },
    {
        order = 14000,
        args = {'version', 'release_version'},
        header = i18n.item_table.release_version,
        fields = {'items.release_version'},
        display = function(tr, data)
            tr
                :tag('td')
                    :wikitext(
                        string.format(
                            i18n.item_table.version_link,
                            data['items.release_version'],
                            data['items.release_version']
                        )
                    )
        end,
    },
    {
        order = 15000,
        args = {'drop', 'drop_level'},
        header = i18n.item_table.drop_level,
        fields = {'items.drop_level'},
        display = h.display_value_factory{},
    },
    {
        order = 15001,
        args = {'drop_level_maximum'},
        header = i18n.item_table.drop_level_maximum,
        fields = {'items.drop_level_maximum'},
        display = h.display_value_factory{},
    },
    {
        order = 15002,
        args = {'version', 'removal_version'},
        header = i18n.item_table.removal_version,
        fields = {'items.removal_version'},
        display = function(tr, data)
            tr
                :tag('td')
                    :wikitext(
                        string.format(
                            i18n.item_table.version_link,
                            data['items.removal_version'],
                            data['items.removal_version']
                        )
                    )
        end,
    },
    {
        order = 15003,
        args = {'drop', 'drop_enabled'},
        header = i18n.item_table.drop_enabled,
        fields = {'items.drop_enabled'},
        display = h.display_value_factory{},
        display = h.display_yesno_factory{
            field = 'items.drop_enabled',
        },
        sort_type = 'text',
    },
    {
        order = 15004,
        args = {'drop', 'drop_areas'},
        header = i18n.item_table.drop_areas,
        fields = {'items.drop_areas_html'},
        display = h.display_value_factory{},
        sort_type = 'text',
    },
    {
        order = 15005,
        args = {'drop', 'drop_monsters'},
        header = i18n.item_table.drop_monsters,
        fields = {'items.drop_monsters'},
        display = function(tr, data, na, results2)
            if results2['drop_monsters_query'] == nil then
                results2['drop_monsters_query'] = m_cargo.query(
                    {'items', 'monsters', 'main_pages'},
                    {
                        'items._pageName',
                        'monsters._pageName',
                        'monsters.name',
                        'main_pages._pageName',
                    },
                    {
                        join=[[
                            items.drop_monsters HOLDS monsters.metadata_id,
                            monsters.metadata_id=main_pages.id
                        ]],
                        where=string.format([[
                            items._pageID IN (%s)
                            AND monsters.metadata_id IS NOT NULL
                        ]],
                            table.concat(results2.pageIDs, ', ')
                        ),
                        orderBy='items.drop_monsters',
                    }
                )

                results2['drop_monsters_query'] = m_cargo.map_results_to_id{
                    results=results2['drop_monsters_query'],
                    field='items._pageName',
                }
            end
            local results = results2['drop_monsters_query'][data['items._pageName']] or {}
            local tbl = {}
            for _,v in ipairs(results) do
                local page = v['main_pages._pageName'] or v['monsters._pageName'] or ''
                local name = v['monsters.name'] or v['items.drop_monsters'] or ''
                tbl[#tbl+1] = string.format('[[%s|%s]]', page, name)
            end
            h.na_or_val(tr, table.concat(tbl, '<br>'))
        end,
        sort_type = 'text',
    },
    {
        order = 15006,
        args = {'drop', 'drop_text'},
        header = i18n.item_table.drop_text,
        fields = {'items.drop_text'},
        display = h.display_value_factory{},
        sort_type = 'text',
    },
    {
        order = 16000,
        args = {'quest'},
        header = i18n.item_table.quest_rewards,
        fields = {'items._pageName'},
        display = function(tr, data, na, results2)
            if results2['quest_query'] == nil then
                results2['quest_query'] = m_cargo.query(
                    {'items', 'quest_rewards'},
                    {
                        'quest_rewards._pageName',
                        'quest_rewards.classes',
                        'quest_rewards.act',
                        'quest_rewards.quest'
                    },
                    {
                        join='items._pageName=quest_rewards._pageName',
                        where=string.format(
                            'items._pageID IN (%s) AND quest_rewards._pageName IS NOT NULL',
                            table.concat(results2.pageIDs, ', ')
                        ),
                        orderBy='quest_rewards.act, quest_rewards.quest',
                    }
                )
                results2['quest_query'] = m_cargo.map_results_to_id{
                    results=results2['quest_query'],
                    field='quest_rewards._pageName',
                }
            end

            local results = results2['quest_query'][data['items._pageName']] or {}
            local tbl = {}
            for _, v in ipairs(results) do
                local classes = table.concat(m_util.string.split(v['quest_rewards.classes'] or '', ',%s*'), ', ')
                if classes == '' or classes == nil then
                    classes = i18n.item_table.quest_rewards_any_classes
                end

                tbl[#tbl+1] = string.format(
                    i18n.item_table.quest_rewards_row_format,
                    v['quest_rewards.act'],
                    v['quest_rewards.quest'],
                    classes
                )
            end

            local value = table.concat(tbl, '<br>')
            if value == nil or value == '' then
                tr:node(m_util.html.table_cell('na'))
            else
                tr
                    :tag('td')
                        :attr('style', 'text-align:left')
                        :wikitext(value)
            end
        end,
        sort_type = 'text',
    },
    {
        order = 17000,
        args = {'vendor'},
        header = i18n.item_table.vendor_rewards,
        fields = {'items._pageName'},
        display = function(tr, data, na, results2)
            if results2['vendor_query'] == nil then
                results2['vendor_query'] = m_cargo.query(
                    {'items', 'vendor_rewards'},
                    {
                        'vendor_rewards._pageName',
                        'vendor_rewards.classes',
                        'vendor_rewards.act',
                        'vendor_rewards.npc',
                        'vendor_rewards.quest',
                    },
                    {
                        join='items._pageName=vendor_rewards._pageName',
                        where=string.format(
                            'items._pageID IN (%s) AND vendor_rewards._pageName IS NOT NULL',
                            table.concat(results2.pageIDs, ', ')
                        ),
                        orderBy='vendor_rewards.act, vendor_rewards.quest',
                    }
                )
                results2['vendor_query'] = m_cargo.map_results_to_id{
                    results=results2['vendor_query'],
                    field='vendor_rewards._pageName',
                }
            end
            local results = results2['vendor_query'][data['items._pageName']] or {}

            local tbl = {}
            for _, v in ipairs(results) do
                local classes = table.concat(m_util.string.split(v['vendor_rewards.classes'] or '', ',%s*'), ', ')
                if classes == '' or classes == nil then
                    classes = i18n.item_table.vendor_rewards_any_classes
                end

                tbl[#tbl+1] = string.format(
                    i18n.item_table.vendor_rewards_row_format,
                    v['vendor_rewards.act'],
                    v['vendor_rewards.quest'],
                    v['vendor_rewards.npc'],
                    classes
                )
            end

            local value = table.concat(tbl, '<br>')
            if value == nil or value == '' then
                tr:node(m_util.html.table_cell('na'))
            else
                tr
                    :tag('td')
                        :attr('style', 'text-align:left')
                        :wikitext(value)
            end
        end,
        sort_type = 'text',
    },
    {
        order = 18000,
        args = {'price', 'purchase_cost'},
        header = i18n.item_table.purchase_costs,
        fields = {'item_purchase_costs.name', 'item_purchase_costs.amount'},
        display = function (tr, data)
            -- Can purchase costs have multiple currencies and rows?
            -- Just switch to the same method as in sell_price then.
            local tbl = {}
            if data['item_purchase_costs.name'] ~= nil then
                tbl[#tbl+1] = string.format(
                        '%sx %s',
                        data['item_purchase_costs.amount'],
                        h.item_link{data['item_purchase_costs.name']}
                    )
            end
            h.na_or_val(tr, table.concat(tbl, '<br>'))
        end,
        sort_type = 'text',
    },
    {
        order = 18001,
        args = {'price', 'sell_price'},
        header = i18n.item_table.sell_price,
        fields = {'item_sell_prices.name', 'item_sell_prices.amount'},
        display = function(tr, data, na, results2)
            if results2['sell_price_query'] == nil then
                results2['sell_price_query'] = m_cargo.query(
                    {'items', 'item_sell_prices'},
                    {
                        'item_sell_prices.name',
                        'item_sell_prices.amount',
                        'item_sell_prices._pageID'
                    },
                    {
                        join='items._pageID=item_sell_prices._pageID',
                        where=string.format(
                            'items._pageID IN (%s) AND item_sell_prices._pageID IS NOT NULL',
                            table.concat(results2.pageIDs, ', ')
                        ),
                        orderBy='item_sell_prices.name',
                    }
                )
                results2['sell_price_query'] = m_cargo.map_results_to_id{
                    results=results2['sell_price_query'],
                    field='item_sell_prices._pageID',
                }
            end
            local results = results2['sell_price_query'][data['items._pageID']] or {}

            local tbl = {}
            for _,v in ipairs(results) do
                tbl[#tbl+1] = string.format(
                    '%sx %s',
                    v['item_sell_prices.amount'],
                    h.item_link{v['item_sell_prices.name']}
                )
            end
            h.na_or_val(tr, table.concat(tbl, '<br>'))
        end,
        sort_type = 'text',
    },
    {
        order = 19000,
        args = {'boss', 'boss_name'},
        header = i18n.item_table.boss_name,
        fields = {'maps.area_id'},
        display = function(tr, data, na, results2)
            if results2['boss_query'] == nil then
                results2['boss_query'] = m_cargo.query(
                    {'items', 'maps', 'areas', 'monsters', 'main_pages'},
                    {
                        'items._pageName',
                        'maps.area_id',
                        'areas.id',
                        'areas.boss_monster_ids',
                        'monsters._pageName',
                        'monsters.name',
                        'main_pages._pageName',
                    },
                    {
                        join=[[
                            items._pageID=maps._pageID,
                            maps.area_id=areas.id,
                            areas.boss_monster_ids HOLDS monsters.metadata_id,
                            monsters.metadata_id=main_pages.id
                        ]],
                        where=string.format([[
                            items._pageID IN (%s)
                            AND maps.area_id IS NOT NULL
                            AND areas.boss_monster_ids HOLDS LIKE "%%"
                        ]],
                            table.concat(results2.pageIDs, ', ')
                        ),
                        orderBy='areas.boss_monster_ids',
                    }
                )

                results2['boss_query'] = m_cargo.map_results_to_id{
                    results=results2['boss_query'],
                    field='items._pageName',
                }
            end
            local results = results2['boss_query'][data['items._pageName']] or {}
            local tbl = {}
            for _,v in ipairs(results) do
                local page = v['main_pages._pageName'] or v['monsters._pageName'] or ''
                local name = v['monsters.name'] or v['areas.boss_monster_ids'] or ''
                tbl[#tbl+1] = string.format('[[%s|%s]]', page, name)
            end
            h.na_or_val(tr, table.concat(tbl, '<br>'))
        end,
        sort_type = 'text',
    },
    {
        order = 19001,
        args = {'boss', 'boss_number'},
        header = i18n.item_table.boss_number,
        fields = {'maps.area_id'},
        display = function(tr, data, na, results2)
            if results2['boss_query'] == nil then
                results2['boss_query'] = m_cargo.query(
                    {'items', 'maps', 'areas', 'monsters', 'main_pages'},
                    {
                        'items._pageName',
                        'maps.area_id',
                        'areas.id',
                        'areas.boss_monster_ids',
                        'monsters._pageName',
                        'monsters.name',
                        'main_pages._pageName',
                    },
                    {
                        join=[[
                            items._pageID=maps._pageID,
                            maps.area_id=areas.id,
                            areas.boss_monster_ids HOLDS monsters.metadata_id,
                            monsters.metadata_id=main_pages.id
                        ]],
                        where=string.format([[
                            items._pageID IN (%s)
                            AND maps.area_id IS NOT NULL
                            AND areas.boss_monster_ids HOLDS LIKE "%%"
                        ]],
                            table.concat(results2.pageIDs, ', ')
                        ),
                        orderBy='areas.boss_monster_ids',
                    }
                )

                results2['boss_query'] = m_cargo.map_results_to_id{
                    results=results2['boss_query'],
                    field='items._pageName',
                }
            end
            local results = results2['boss_query'][data['items._pageName']] or {}
            local tbl = {}
            for _,v in ipairs(results) do
                tbl[#tbl+1] = v['areas.boss_monster_ids']
            end
            tr
                :tag('td')
                    :attr('data-sort-value', #tbl)
                    :wikitext(#tbl)
        end,
    },
    {
        order = 20000,
        args = {'legacy'},
        header = i18n.item_table.legacy,
        fields = {'items.name'},
        display = function(tr, data, na, results2)
            if results2['legacy_query'] == nil then
                results2['legacy_query'] = m_cargo.query(
                    {'items', 'legacy_variants'},
                    {
                        'items._pageID',
                        'items._pageName',
                        'items.frame_type',
                        'legacy_variants.removal_version',
                        'legacy_variants.implicit_stat_text',
                        'legacy_variants.explicit_stat_text',
                        'legacy_variants.stat_text',
                        'legacy_variants.base_item',
                        'legacy_variants.required_level'
                    },
                    {
                        join='items._pageID=legacy_variants._pageID',
                        where='legacy_variants.removal_version IS NOT NULL',
                        where=string.format(
                            'items._pageID IN (%s) AND legacy_variants.removal_version IS NOT NULL',
                            table.concat(results2.pageIDs, ', ')
                        ),
                        orderBy='items._pageName',
                    }
                )

                results2['legacy_query'] = m_cargo.map_results_to_id{
                    results=results2['legacy_query'],
                    field='items._pageName',
                }
            end
            local results = results2['legacy_query'][data['items._pageName']] or {}

            local tbl = mw.html.create('table')
                :attr('width', '100%')
            for _, v in ipairs(results) do
                local cell = {}
                local l = {
                    'legacy_variants.base_item',
                    'legacy_variants.stat_text'
                }

                -- Clean up data:
                for _, k in ipairs(l) do
                    if v[k] ~= nil then
                        local s = m_util.string.split(v[k], '*')
                        local s_flt = {}
                        for _, sss in ipairs(s) do
                            if sss ~= nil and sss ~= '' then
                                s_flt[#s_flt+1] = string.gsub(sss, '\n', '')
                            end
                        end

                        cell[#cell+1] = table.concat(s_flt, '<br>')
                    end
                end

                local sep = string.format(
                    '<span class="item-stat-separator -%s"></span>',
                    v['items.frame_type']
                )
                tbl
                    :tag('tr')
                        :attr('class', 'upgraded-from-set')
                        :tag('td')
                            :wikitext(
                                v['legacy_variants.removal_version']
                            )
                            :done()
                        :tag('td')
                            :attr('class', 'group legacy-stats plainlist')
                            :wikitext(table.concat(cell, sep))
                            :done()
                    :done()
            end
            tr
                :tag('td')
                    :node(tbl)
        end,
        sort_type = 'text',
    },
    {
        order = 21000,
        args = {'granted_skills'},
        header = i18n.item_table.granted_skills,
        fields = {'items.name'},
        display = function(tr, data, na, results2)
            if results2['granted_skills_query'] == nil then
                results2['granted_skills_query'] = m_cargo.query(
                    {'items', 'item_mods', 'mods', 'skill', 'items=items2'},
                    {
                        'items._pageName',
                        'items.name',
                        'item_mods.id',
                        'mods._pageName',
                        'mods.id',
                        'mods.granted_skill',
                        'mods.stat_text_raw',
                        'skill._pageName',
                        'skill.skill_id',
                        'skill.active_skill_name',
                        'skill.skill_icon',
                        'skill.stat_text',
                        'items2.class',
                        'items2.name',
                        'items2.inventory_icon',
                        'items2.size_x',
                        'items2.size_y',
                        'items2.html',
                    },
                    {
                        join='items._pageID=item_mods._pageID, item_mods.id=mods.id, mods.granted_skill=skill.skill_id, skill._pageID=items2._pageID',
                        where=string.format(
                            'items._pageID IN (%s) AND mods.granted_skill IS NOT NULL',
                            table.concat(results2.pageIDs, ', ')
                        ),
                    }
                )

                results2['granted_skills_query'] = m_cargo.map_results_to_id{
                    results=results2['granted_skills_query'],
                    field='items._pageName',
                }
            end
            local results = results2['granted_skills_query'][data['items._pageName']] or {}

            local tbl = {}
            for _, v in ipairs(results) do

                -- Check if a level for the skill is specified in the
                -- mod stat text.
                -- Stat ids have unreliable naming convention so using
                -- the mod stat text instead.
                local level = ''
                local stat_text = v['mods.stat_text_raw'] or ''
                local level_number = string.match(
                    stat_text:lower(),
                    h.string.format(
                        i18n.item_table.granted_skills_level_pattern,
                        {
                            granted_skills_level_label = i18n.item_table.granted_skills_level_label:lower()
                        }
                    )
                )

                -- If a level number was specified in the stat text
                -- then add it to the cell:
                if level_number then
                    level = h.string.format(
                        i18n.item_table.granted_skills_level_format,
                        {
                            granted_skills_level_label = i18n.item_table.granted_skills_level_label,
                            level_number = level_number,
                        }
                    )
                end

                -- Use different formats depending on if it's a gem or
                -- not:
                if v['items2.class'] == nil  then
                    tbl[#tbl+1] = h.string.format(
                        i18n.item_table.granted_skills_skill_output_format,
                        {
                            level = level,
                            sl = h.skill_link{
                                skip_query=true,
                                page = v['skill.active_skill_name']
                                    or v['skill._pageName']
                                    or v['mods._pageName']
                                    or '',
                                name = v['skill.active_skill_name']
                                    or v['skill.stat_text']
                                    or v['mods.granted_skill'],
                                icon = v['skill.skill_icon'],
                            },
                        }
                    )
                else
                    local il_args = {
                        skip_query=true,
                        page=v['items2._pageName'],
                        name=v['items2.name'],
                        inventory_icon=v['items2.inventory_icon'],
                        width=v['items2.size_x'],
                        height=v['items2.size_y'],
                    }

                    -- TODO: add in tpl_args.
                    if no_html == nil then
                        il_args.html = v['items2.html']
                    end
                    tbl[#tbl+1] = h.string.format(
                        i18n.item_table.granted_skills_gem_output_format,
                        {
                            level = level,
                            il = h.item_link(il_args),
                        }
                    )
                end
            end
            h.na_or_val(tr, table.concat(tbl, '<br>'))
        end,
        sort_type = 'text',
    },
    {
        order = 23000,
        args = {'alternate_art'},
        header = i18n.item_table.alternate_art,
        fields = {'items.alternate_art_inventory_icons'},
        display = function (tr, data)
            local alt_art = m_util.string.split(
                data['items.alternate_art_inventory_icons'],
                ','
            )

            -- TODO: Use il instead to handle size?
            -- local size = 39
            local out = {}
            for i,v in ipairs(alt_art) do
                out[#out+1] = string.format(
                    '[[%s|link=|%s]]',
                    v,
                    v
                )
            end

            tr
                :tag('td')
                    :wikitext(table.concat(out, ''))
        end,
        sort_type = 'text',
    },
}

data_map.skill_gem = {
    {
        order = 1000,
        args = {'gem_tier'},
        header = i18n.item_table.tier,
        fields = {'skill_gems.gem_tier'},
        display = h.display_value_factory{},
    },
    {
        order = 1001,
        args = {'skill_icon'},
        header = i18n.item_table.skill_icon,
        fields = {'skill.skill_icon'},
        display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '[[%s]]',
                },
            },
        },
        sort_type = 'text',
    },
    {
        order = 2000,
        args = {'stat', 'stat_text'},
        header = i18n.item_table.stats,
        fields = {'skill.stat_text'},
        display = h.display_value_factory{},
        sort_type = 'text',
    },
    {
        order = 2001,
        args = {'quality', 'quality_stat_text'},
        header = i18n.item_table.quality_stats,
        fields = {'skill.quality_stat_text'},
        display = h.display_value_factory{},
        sort_type = 'text',
    },
    {
        order = 2100,
        args = {'description'},
        header = i18n.item_table.description,
        fields = {'skill.description'},
        display = h.display_value_factory{},
        sort_type = 'text',
    },
    {
        order = 3000,
        args = {'level'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.level_icon,
                i18n.item_table.gem_level_requirement
            ),
            i18n.item_table.gem_level_requirement
        ),
        fields = h.range_fields_factory{fields={'items.required_level'}},
        display = h.display_range_factory{field='items.required_level'},
    },
    {
        order = 3001,
        args = {'str'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.str_icon,
                i18n.item_table.str_gem
            ),
            i18n.item_table.str_gem
        ),
        fields = {'skill_gems.strength_percent'},
        display = h.display_yesno_factory{
            field = 'skill_gems.strength_percent',
            condition = h.value_greater_than_zero,
        },
        sort_type = 'text',
    },
    {
        order = 3002,
        args = {'dex'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.dex_icon,
                i18n.item_table.dex_gem
            ),
            i18n.item_table.dex_gem
        ),
        fields = {'skill_gems.dexterity_percent'},
        display = h.display_yesno_factory{
            field = 'skill_gems.dexterity_percent',
            condition = h.value_greater_than_zero,
        },
        sort_type = 'text',
    },
    {
        order = 3003,
        args = {'int'},
        header = m_util.html.tooltip(
            string.format(
                '[[%s|link=|alt=%s]]',
                i18n.item_table.int_icon,
                i18n.item_table.int_gem
            ),
            i18n.item_table.int_gem
        ),
        fields = {'skill_gems.intelligence_percent'},
        display = h.display_yesno_factory{
            field = 'skill_gems.intelligence_percent',
            condition = h.value_greater_than_zero,
        },
        sort_type = 'text',
    },
    {
        order = 4000,
        args = {'crit'},
        header = i18n.item_table.skill_critical_strike_chance,
        fields = {'skill_levels.critical_strike_chance'},
        display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '%s%%',
                },
            },
        },
        field_options = {
            [1] = {
                skill_levels = true,
            },
        },
    },
    {
        order = 4001,
        args = {'cast_time'},
        header = i18n.item_table.cast_time,
        fields = {'skill.cast_time'},
        display = h.display_value_factory{},
    },
    {
        order = 4002,
        args = {'aspd', 'attack_speed', 'attack_speed_multiplier'},
        header = i18n.item_table.attack_speed_multiplier,
        fields = {'skill_levels.attack_speed_multiplier'},
        display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '%s%%',
                },
            },
        },
        field_options = {
            [1] = {
                skill_levels = true,
            },
        },
    },
    {
        order = 4003,
        args = {'dmgeff'},
        header = i18n.item_table.damage_effectiveness,
        fields = {'skill_levels.damage_effectiveness'},
        display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '%s%%',
                },
            },
        },
        field_options = {
            [1] = {
                skill_levels = true,
            },
        },
    },
    {
        order = 5000,
        args = {'mcm', 'cost_multiplier'},
        header = i18n.item_table.cost_multiplier,
        fields = {'skill_levels.cost_multiplier'},
        display = h.display_value_factory{
            fmt_options = {
                [1] = {
                    fmt = '%s%%',
                },
            },
        },
        field_options = {
            [1] = {
                skill_levels = true,
            },
        },
    },
    {
        order = 5001,
        args = {'mana'},
        header = i18n.item_table.mana_cost,
        fields = {'skill_levels.cost_amounts', 'skill_levels.mana_reservation_percent', 'skill_levels.mana_reservation_flat'},
        display = function (tr, data, fields, data2)
            local appendix = ''
            local cost_field = ''
            local sdata = data2.skill_levels[data['items._pageName']]
            -- Percentage Mana reservation is in level 0 of most of the skill_levels at this point, but spellslinger and awakened blasphemy change it per-level
            -- Flat Mana resservation is in real gem levels, but maybe there will be fixed flat mana reservations in the future.
            -- Per-use mana costs are stored in the real gem levels if they change, or in 0 if it's fixed.
            if(sdata['1']['skill_levels.mana_reservation_percent'] ~= nil or sdata['0']['skill_levels.mana_reservation_percent'] ~= nil) then
                cost_field = 'skill_levels.mana_reservation_percent'
                appendix = appendix .. '%%'
            elseif(sdata['1']['skill_levels.mana_reservation_flat'] ~= nil or sdata['0']['skill_levels.mana_reservation_flat'] ~= nil) then
                cost_field = 'skill_levels.mana_reservation_flat'
                appendix = appendix .. ' ' .. i18n.item_table.reserves_mana_suffix
            elseif(sdata['1']['skill_levels.cost_amounts'] ~= nil or sdata['0']['skill_levels.cost_amounts'] ~= nil) then
                cost_field = 'skill_levels.cost_amounts'
            end
            h.display_value_factory{
                fmt_options = {
                    [1] = {
                        fmt = '%d' .. appendix,
                    },
                },
            }(tr, data, {cost_field}, data2)
        end,
        -- Need one set of options per field.
        field_options = {
            [1] = {
                skill_levels = true,
            },
            [2] = {
                skill_levels = true,
            },
            [3] = {
                skill_levels = true,
            },
        },
    },
    {
        order = 6000,
        args = {'vaal'},
        header = i18n.item_table.vaal_souls_requirement,
        fields = {'skill_levels.vaal_souls_requirement'},
        display = h.display_value_factory{},
        field_options = {
            [1] = {
                skill_levels = true,
            },
        },
    },
    {
        order = 6001,
        args = {'vaal'},
        header = i18n.item_table.stored_uses,
        fields = {'skill_levels.vaal_stored_uses'},
        display = h.display_value_factory{},
        field_options = {
            [1] = {
                skill_levels = true,
            },
        },
    },
    {
        order = 7000,
        args = {'radius'},
        header = i18n.item_table.primary_radius,
        fields = {'skill.radius', 'skill.radius_description'},
        field_options = {
            [2] = {
                optional = true,
            },
        },
        display = function (tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', data['skill.radius'])
                    :wikitext(h.display_descriptor_value_factory{tbl=data, key='skill.radius_description'}(data['skill.radius']))
        end,
    },
    {
        order = 7001,
        args = {'radius'},
        header = i18n.item_table.secondary_radius,
        fields = {'skill.radius_secondary', 'skill.radius_secondary_description'},
        field_options = {
            [2] = {
                optional = true,
            },
        },
        display = function (tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', data['skill.radius_secondary'])
                    :wikitext(h.display_descriptor_value_factory{tbl=data, key='skill.radius_secondary_description'}(data['skill.radius_secondary']))
        end,
    },
    {
        order = 7002,
        args = {'radius'},
        header = i18n.item_table.tertiary_radius,
        fields = {'skill.radius_tertiary', 'skill.radius_tertiary_description'},
        field_options = {
            [2] = {
                optional = true,
            },
        },
        display = function (tr, data)
            tr
                :tag('td')
                    :attr('data-sort-value', data['skill.radius_tertiary'])
                   :wikitext(h.display_descriptor_value_factory{tbl=data, key='skill.radius_tertiary_description'}(data['skill.radius_tertiary']))
        end,
    },
    {
        order = 8000,
        args = {'drop', 'drop_text'},
        header = i18n.item_table.drop_text,
        fields = {'items.drop_text'},
        display = h.display_value_factory{},
        sort_type = 'text',
    },
}

-- ----------------------------------------------------------------------------
-- Main functions
-- ----------------------------------------------------------------------------

local function _item_table(args)
    --[[
    Creates a generic table for items.

    Examples
    --------
    = p.item_table{
        tables='vendor_rewards, quest_rewards, skill_gems',
        join='items._pageID=vendor_rewards._pageID, items._pageID=quest_rewards._pageID, items._pageID=skill_gems._pageID',
        where='(items.class="Active Skill Gems" OR items.class = "Support Skill Gems") AND (vendor_rewards.quest_id IS NOT NULL OR quest_rewards.quest_id IS NOT NULL)',
        vendor=1,
    }
    ]]

    m_item_util = m_item_util or (use_sandbox and require('Module:Item util/sandbox') or require('Module:Item util'))

    local t = os.clock()

    args.mode = args.mode or 'item'
    local modes = {
        skill = {
            data = data_map.skill_gem,
            header = i18n.item_table.skill_gem,
        },
        item = {
            data = data_map.generic_item,
            header = i18n.item_table.item,
        },
    }
    if modes[args.mode] == nil then
        error(i18n.errors.invalid_item_table_mode)
    end

    -- A where clause is required; there are far too many items to list in one table
    if args.where == nil then
        error(string.format(i18n.errors.generic_required_parameter, 'where'))
    end

    local results2 = {
        stats = {},
        skill_levels = {},
        pageIDs = {},
    }

    local row_infos = {}
    for _, row_info in ipairs(modes[args.mode].data) do
        local enabled = false
        if type(row_info.args) == 'table' then
            for _, a in ipairs(row_info.args) do
                if m_util.cast.boolean(args[a]) then
                    enabled = true
                    break
                end
            end
        else
            enabled = true
        end
        if enabled then
            row_info.field_options = row_info.field_options or {}
            row_infos[#row_infos+1] = row_info
        end
    end

    -- Parse stat arguments
    local stat_columns = {}
    local query_stats = {}
    for i=1, math.huge do -- repeat until no more columns are found
        local prefix = string.format('stat_column%s_', i)
        if args[prefix .. 'stat1_id'] == nil then
            -- Each column requires at least one stat id
            break
        end
        local col_info = {
            header = args[prefix .. 'header'] or tostring(i),
            format = args[prefix .. 'format'],
            stat_format = args[prefix .. 'stat_format'] or 'separate',
            order = tonumber(args[prefix .. 'order']) or (10000000 + i),
            stats = {},
        }
        for j=1, math.huge do
            local stat_id = args[string.format('%sstat%s_id', prefix, j)]
            if stat_id == nil then
                break
            end
            table.insert(col_info.stats, {id=stat_id})
            query_stats[stat_id] = true
        end
        table.insert(stat_columns, col_info)
    end

    for _, col_info in ipairs(stat_columns) do
        local row_info = {
            header = col_info.header,
            fields = {},
            display = function (tr, data)
                if col_info.stat_format == 'separate' then
                    local stat_texts = {}
                    local num_stats = 0
                    local vmax = 0
                    for _, stat_info in ipairs(col_info.stats) do
                        num_stats = num_stats + 1
                        -- stat results from outside body
                        local stat = (results2.stats[data['items._pageName']] or {})[stat_info.id]
                        if stat ~= nil then
                            stat_texts[#stat_texts+1] = m_util.html.format_value(args, stat, {color=false})
                            vmax = vmax + stat.max
                        end
                    end

                    if num_stats ~= #stat_texts then
                        tr:node(m_util.html.table_cell('na'))
                    else
                        local text
                        if col_info.format then
                            text = string.format(col_info.format, unpack(stat_texts))
                        else
                            text = table.concat(stat_texts, ', ')
                        end

                        tr:tag('td')
                            :attr('data-sort-value', vmax)
                            :attr('class', 'tc -mod')
                            :wikitext(text)
                    end
                 elseif col_info.stat_format == 'add' then
                    local total_stat = {
                        min = 0,
                        max = 0,
                        avg = 0,
                    }
                    for _, stat_info in ipairs(col_info.stats) do
                        local stat = (results2.stats[data['items._pageName']] or {})[stat_info.id]
                        if stat ~= nil then
                            for k, v in pairs(total_stat) do
                                total_stat[k] = v + stat[k]
                            end
                        end
                    end

                    if col_info.format == nil then
                        col_info.format = '%s'
                    end

                    tr:tag('td')
                        :attr('data-sort-value', total_stat.max)
                        :attr('class', 'tc -mod')
                        :wikitext(string.format(col_info.format, m_util.html.format_value(args, total_stat, {no_color=true})))
                 else
                    error(string.format(i18n.errors.generic_argument_parameter, 'stat_format', col_info.stat_format))
                 end
            end,
            order = col_info.order,
        }
        table.insert(row_infos, row_info)
    end

    -- sort the rows
    table.sort(row_infos, function (a, b)
        return (a.order or 0) < (b.order or 0)
    end)

    -- Build Cargo query
    local tables = {'items'}
    local fields = {
        'items._pageID',
        'items._pageName',
        'items.name',
        'items.inventory_icon',
        'items.html',
        'items.size_x',
        'items.size_y',
    }
    local query = {
        where = args.where,
        groupBy = table.concat({'items._pageID', args.groupBy}, ', '),
        having = args.having,
        orderBy = args.orderBy,
        limit = args.limit,
        offset = args.offset,
    }

    -- Namespace condition
    -- This is mainly to prevent items from user pages or other testing pages 
    -- from being returned in the query results.
    if args.namespaces ~= 'any' then
        local namespaces = m_util.cast.table(args.namespaces, {callback=m_util.cast.number})
        if #namespaces > 0 then
            namespaces = table.concat(namespaces, ',')
        else
            namespaces = m_item_util.get_item_namespaces{format = 'list'}
        end
        query.where = string.format('(%s) AND items._pageNamespace IN (%s)', query.where, namespaces)
    end

    -- Minimum required tables and fields, based on display options
    local skill_levels = {}
    for _, rowinfo in ipairs(row_infos) do
        if type(rowinfo.fields) == 'function' then
            rowinfo.fields = rowinfo.fields()
        end
        for index, field in ipairs(rowinfo.fields) do
            rowinfo.field_options[index] = rowinfo.field_options[index] or {}
            if rowinfo.field_options[index].skill_levels then
                skill_levels[#skill_levels+1] = field
            else
                fields[#fields+1] = field
                tables[#tables+1] = m_util.string.split(field, '.', true)[1]
            end
        end
    end
    if #skill_levels > 0 then
        fields[#fields+1] = 'skill.max_level'
        tables[#tables+1] = 'skill'
    end
    tables = m_util.table.remove_duplicates(tables)

    -- Minimum required joins, based on display options
    local joins = {}
    for _, table_name in ipairs(tables) do
        if table_name ~= 'items' then
            joins[#joins+1] = string.format('items._pageID=%s._pageID', table_name)
        end
    end

    -- Append additional tables
    args.tables = m_util.cast.table(args.tables)
    if type(args.tables) == 'table' and #args.tables > 0 then
        tables = m_util.table.merge(tables, args.tables)
    end

    -- Make join clause
    if #joins > 0 or args.join then
        -- m_util.table.merge rebuilds the table, which removes empty values
        query.join = table.concat(m_util.table.merge(joins, {args.join}), ', ')
    end

    -- Query results
    local results = m_cargo.query(tables, fields, query)

    if #results == 0 and args.default ~= nil then
        return args.default
    end

    if #results > 0 then
        -- Create a list of found pageIDs for column specific queries:
        for _,v in ipairs(results) do
            results2.pageIDs[#results2.pageIDs+1] = v['items._pageID']
        end

        -- fetch skill level information
        if #skill_levels > 0 then
            skill_levels[#skill_levels+1] = 'skill_levels._pageName'
            skill_levels[#skill_levels+1] = 'skill_levels.level'
            local pages = {}
            for _, row in ipairs(results) do
                pages[#pages+1] = string.format('(skill_levels._pageID="%s" AND skill_levels.level IN (0, 1, %s))', row['items._pageID'], row['skill.max_level'])
            end
            local temp = m_cargo.query(
                {'skill_levels'},
                skill_levels,
                {
                    where=table.concat(pages, ' OR '),
                    groupBy='skill_levels._pageID, skill_levels.level',
                }
            )
            -- map to results
            for _, row in ipairs(temp) do
                if results2.skill_levels[row['skill_levels._pageName']] == nil then
                   results2.skill_levels[row['skill_levels._pageName']] = {}
                end
                -- TODO: convert to int?
                results2.skill_levels[row['skill_levels._pageName']][row['skill_levels.level']] = row
            end
        end

        if #stat_columns > 0 then
            local stat_results = m_cargo.query(
                {'items', 'item_stats'},
                {'item_stats._pageName', 'item_stats.id', 'item_stats.min', 'item_stats.max', 'item_stats.avg'},
                {
                    join = 'items._pageID=item_stats._pageID',
                    where = string.format(
                        'item_stats._pageID IN (%s) AND item_stats.id IN ("%s")',
                        table.concat(m_util.table.column(results, 'items._pageID'), ','),
                        table.concat(m_util.table.keys(query_stats), '","')
                    ),
                    groupBy = 'items._pageID, item_stats.id',
                }
            )
            for _, row in ipairs(stat_results) do
                local stat = {
                    min = tonumber(row['item_stats.min']),
                    max = tonumber(row['item_stats.max']),
                    avg = tonumber(row['item_stats.avg']),
                }

                if results2.stats[row['item_stats._pageName']] == nil then
                    results2.stats[row['item_stats._pageName']] = {[row['item_stats.id']] = stat}
                else
                    results2.stats[row['item_stats._pageName']][row['item_stats.id']] = stat
                end
            end
        end
    end

    --
    -- Display the table
    --

    local tbl = mw.html.create('table')
    tbl:attr('class', 'wikitable sortable item-table')
    if m_util.cast.boolean(args.responsive) then
        tbl:addClass('responsive-table')
    end

    -- Headers:
    local tr = tbl:tag('tr')
    tr
        :tag('th')
            :wikitext(modes[args.mode].header)
            :done()
    for _, row_info in ipairs(row_infos) do
        local th = tr:tag('th')

        if row_info.colspan then
            th:attr('colspan', row_info.colspan)
        end

        th
            :attr('data-sort-type', row_info.sort_type or 'number')
            :wikitext(row_info.header)
    end

    -- Rows:
    for _, row in ipairs(results) do
        tr = tbl:tag('tr')

        local il_args = {
            skip_query=true,
            page=row['items._pageName'],
            name=row['items.name'],
            inventory_icon=row['items.inventory_icon'],
            width=row['items.size_x'],
            height=row['items.size_y'],
        }

        if args.no_html == nil then
            il_args.html = row['items.html']
        end

        if args.large then
            il_args.large = args.large
        end

        tr
            :tag('td')
                :wikitext(h.item_link(il_args))
                :done()

        for _, rowinfo in ipairs(row_infos) do
            -- this has been cast from a function in an earlier step
            local display = true
            
            for index, field in ipairs(rowinfo.fields) do
                -- this will bet set to an empty value not nil confusingly
                if row[field] == nil or row[field] == '' then
                    local options = rowinfo.field_options[index]
                    if options.optional ~= true and options.skill_levels ~= true then
                        display = false
                        break
                    else
                        row[field] = nil
                    end
                end
            end
            if display then
                rowinfo.display(tr, row, rowinfo.fields, results2)
            else
                tr:node(m_util.html.table_cell('na'))
            end
        end
    end

    local cats = {}
    if #results == query.limit then
        cats[#cats+1] = i18n.categories.query_limit
    end
    if #results == 0 then
        cats[#cats+1] = i18n.categories.no_results
    end

    mw.logObject({os.clock() - t, {tables=tables, fields=fields, query=query}})

    return tostring(tbl) .. m_util.misc.add_category(cats, {ignore_blacklist=args.debug})
end


-------------------------------------------------------------------------------
-- Map item drops
-------------------------------------------------------------------------------

local function _map_item_drops(args)
    --[[
    Gets the area id from the map item and activates
    Template:Area_item_drops.

    Examples:
    = p.map_item_drops{page='Underground River Map (War for the Atlas)'}
    ]]

    local tables = {'maps'}
    local fields = {
        'maps.area_id',
    }
    local query = {
        -- Only need each page name once
        groupBy = 'maps._pageName',
    }
    if args.page then
        -- Join with _pageData in order to check for page redirect
        tables[#tables+1] = '_pageData'
        fields[#fields+1] = '_pageData._pageNameOrRedirect'
        query.where = string.format(
            '_pageData._pageName="%s"',
            args.page
        )
        query.join = 'maps._pageName = _pageData._pageNameOrRedirect'
    else
        query.where = string.format(
            'maps._pageName="%s"',
            tostring(mw.title.getCurrentTitle())
        )
    end
    query.where = query.where .. ' AND maps.area_id > ""' -- area_id must not be empty or null
    local results = m_cargo.query(tables, fields, query)
    local id = ''
    if #results > 0 then
        id = results[1]['maps.area_id']
    end
    return mw.getCurrentFrame():expandTemplate{ title = 'Area item drops', args = {area_id=id} }
end

-------------------------------------------------------------------------------
-- Prophecy description
-------------------------------------------------------------------------------

local function _prophecy_description(args)
    args.page = args.page or tostring(mw.title.getCurrentTitle())

    local results = m_cargo.query(
        {'prophecies'},
        {'prophecies.objective', 'prophecies.reward'},
        {
            where=string.format('prophecies._pageName="%s"', args.page),
            -- Only need each page name once
            groupBy='prophecies._pageName',
        }
    )

    results = results[1]

    local out = {}

    if results['prophecies.objective'] then
        out[#out+1] = string.format('<h2>%s</h2>', i18n.prophecy_description.objective)
        out[#out+1] = results['prophecies.objective']
    end

    if results['prophecies.reward'] then
        out[#out+1] = string.format('<h2>%s</h2>', i18n.prophecy_description.reward)
        out[#out+1] = results['prophecies.reward']
    end

    return table.concat(out, '\n')
end


local function _simple_item_list(args)
    --[[
    Creates a simple list of items.

    Examples
    --------
    = p.simple_item_list{
        q_tables='maps',
        q_join='items._pageID=maps._pageID',
        q_where='maps.tier=1 AND items.drop_enabled=1 AND items.rarity_id="normal"',
        no_html=1,
        link_from_name=1,
    }
    ]]

    local query = {}
    for key, value in pairs(args) do
        if string.sub(key, 0, 2) == 'q_' then
            query[string.sub(key, 3)] = value
        end
    end

    local fields = {
        'items._pageName',
        'items.name',
        'items.class',
    }

    if args.no_icon == nil then
        fields[#fields+1] = 'items.inventory_icon'
    end

    if args.no_html == nil then
        fields[#fields+1] = 'items.html'
    end

    local tables = m_util.cast.table(args.q_tables)
    table.insert(tables, 1, 'items')

    query.groupBy = query.groupBy or 'items._pageID'

    local results = m_cargo.query(
        tables,
        fields,
        query
    )

    local out = {}
    for _, row in ipairs(results) do
        local page
        if args.use_name_as_link ~= nil then
            page = row['items.name']
        else
            page = row['items._pageName']
        end

        local link = h.item_link{
            page=page,
            name=row['items.name'],
            inventory_icon=row['items.inventory_icon'] or '',
            html=row['items.html'] or '',
            skip_query=true
        }

        if args.format == nil then
            out[#out+1] = string.format('* %s', link)
        elseif args.format == 'none' then
            out[#out+1] = link
        elseif args.format == 'li' then
            out[#out+1] = string.format('<li>%s</li>', link)
        else
            error(string.format(i18n.errors.generic_argument_parameter, 'format', args.format))
        end
    end

    if args.format == nil then
        return table.concat(out, '\n')
    elseif args.format == 'none' then
        return table.concat(out, '\n')
    elseif args.format == 'li' then
        return table.concat(out)
    end
end

-- ----------------------------------------------------------------------------
-- Exported functions
-- ----------------------------------------------------------------------------

local p = {}

--
-- Template:Item table
--
p.item_table = m_util.misc.invoker_factory(_item_table)

--
-- Template:Map item drops
--
p.map_item_drops = m_util.misc.invoker_factory(_map_item_drops, {
    wrappers = cfg.wrappers.map_item_drops,
})

--
-- Template:Prophecy description
--
p.prophecy_description = m_util.misc.invoker_factory(_prophecy_description, {
    wrappers = cfg.wrappers.prophecy_description,
})

--
-- Template:Simple item list
--
p.simple_item_list = m_util.misc.invoker_factory(_simple_item_list, {
    wrappers = cfg.wrappers.simple_item_list,
})

-- ----------------------------------------------------------------------------
-- Debug
-- ----------------------------------------------------------------------------
p.debug = {}

function p.debug._tbl_data(tbl)
    keys = {}
    for _, data in ipairs(tbl) do
        if type(data.arg) == 'string' then
            keys[data.arg] = 1
        elseif type(data.arg) == 'table' then
            for _, arg in ipairs(data.arg) do
                keys[arg] = 1
            end
        end
    end

    local out = {}
    for key, _ in pairs(keys) do
        out[#out+1] = string.format("['%s'] = '1'", key)
    end

    return table.concat(out, ', ')
end

function p.debug.generic_item_all()
    return p.debug._tbl_data(data_map.generic_item)
end

function p.debug.skill_gem_all()
    return p.debug._tbl_data(data_map.skill_gem)
end

return p