Module:Items/UseTables: Difference between revisions

From Melvor Idle
(_getItemUseTable: Support GP & SC in various artisan skills)
m (Change Source Type to 'Downgrade' if the upgrade is a downgrade)
 
(37 intermediate revisions by 6 users not shown)
Line 1: Line 1:
local p = {}
local p = {}
local ItemData = mw.loadData('Module:Items/data')
local SkillData = mw.loadData('Module:Skills/data')


local Constants = require('Module:Constants')
local Constants = require('Module:Constants')
local Common = require('Module:Common')
local Shared = require('Module:Shared')
local Shared = require('Module:Shared')
local GameData = require('Module:GameData')
local SkillData = GameData.skillData
local Modifiers = require('Module:Modifiers')
local Skills = require('Module:Skills')
local Magic = require('Module:Magic')
local Magic = require('Module:Magic')
local Areas = require('Module:CombatAreas')
local Areas = require('Module:CombatAreas')
Line 12: Line 14:
local Agility = require('Module:Skills/Agility')
local Agility = require('Module:Skills/Agility')
local Shop = require('Module:Shop')
local Shop = require('Module:Shop')
 
local Num = require('Module:Number')
local SkillEnum = mw.loadData('Module:Constants/data').skill


--Brute forcing some item uses to make things easier
--Brute forcing some item uses to make things easier
local itemUseArray = {
local itemUseArray = {
Agility = {},
Agility = {},
Astrology = {'Stardust', 'Golden Stardust'},
Astrology = {'melvorF:Stardust', 'melvorF:Golden_Stardust', 'melvorItA:Abyssal_Stardust', 'melvorItA:Eternal_Stardust'},
Archaeology = {},
Attack = {},
Attack = {},
Combat = {'Gold Emerald Ring', 'Obsidian Cape', 'Throwing Power Gloves'},
Cartography = {},
Cooking = {'Cooking Gloves', 'Crown of Rhaelyx'},
Combat = {'melvorF:Gold_Emerald_Ring', 'melvorD:Obsidian_Cape', 'melvorF:Throwing_Power_Gloves'},
Crafting = {'Crown of Rhaelyx'},
Cooking = {'melvorD:Cooking_Gloves'},
Crafting = {'melvorItA:Abyssal_Crafting_Gloves'},
Defence = {},
Defence = {},
Farming = {'Compost', 'Weird Gloop', 'Bob's Rake'},
Farming = {'melvorD:Compost', 'melvorD:Weird_Gloop', 'melvorItA:Abyssal_Compost', 'melvorD:Bobs_Rake'},
Firemaking = {'Crown of Rhaelyx'},
Firemaking = {'melvorItA:Abyssal_Firemaking_Gloves'},
Fishing = {'Amulet of Fishing', 'Message in a Bottle', 'Barbarian Gloves'},
Fishing = {'melvorD:Message_In_A_Bottle', 'melvorD:Barbarian_Gloves'},
Fletching = {'Crown of Rhaelyx'},
Fletching = {'melvorItA:Abyssal_Fletching_Gloves'},
Herblore = {'Crown of Rhaelyx'},
Harvesting = {'melvorItA:Abyssal_Harvesting_Gloves'},
Herblore = {'melvorItA:Abyssal_Herblore_Gloves'},
Hitpoints = {},
Hitpoints = {},
Magic = {},
Magic = {},
Mining = {'Mining Gloves', 'Gem Gloves'},
Mining = {'melvorD:Mining_Gloves', 'melvorD:Gem_Gloves', 'melvorItA:Abyssal_Mining_Gloves'},
Prayer = {},
Prayer = {},
Ranged = {},
Ranged = {},
Runecrafting = {'Crown of Rhaelyx'},
Runecrafting = {'melvorItA:Abyssal_Runecrafting_Gloves'},
Slayer = {},
Slayer = {},
Smithing = {'Smithing Gloves', 'Crown of Rhaelyx'},
Smithing = {'melvorD:Smithing_Gloves', 'melvorItA:Abyssal_Smithing_Gloves'},
Smithing = {},
Strength = {},
Strength = {},
Summoning = {'Crown of Rhaelyx'},
Summoning = {},
Thieving = {'Chapeau Noir', 'Thieving Gloves', 'Gloves of Silence'},
Thieving = {'melvorF:Thieving_Gloves'},
Township = {},
Woodcutting = {},
Woodcutting = {},
}
local potionUseArray = {
[0] = 'Combat',
[1] = 'Combat',
[2] = 'Combat',
[3] = 'Combat',
[4] = 'Combat',
[5] = 'Combat',
[6] = 'Combat',
[7] = 'Woodcutting',
[8] = 'Fishing',
[9] = 'Firemaking',
[10] = 'Cooking',
[11] = 'Mining',
[12] = 'Smithing',
[13] = 'Thieving',
[14] = 'Farming',
[15] = 'Fletching',
[16] = 'Crafting',
[17] = 'Runecrafting',
[18] = 'Herblore',
[19] = 'Combat',
[20] = 'Combat',
[21] = 'Combat',
[22] = 'Combat',
[23] = 'Combat',
[24] = 'Agility',
[25] = 'Summoning',
[26] = 'Combat',
[27] = 'Combat',
[28] = 'Combat',
[29] = 'Astrology'
}
}
-- List of modifier IDs which, if present on an item and not specific to any
-- particular skill, indicate that the item applies to all skills
local allSkillModIDs = {
-- Skill XP
'skillXP',
'nonCombatSkillXP',
'abyssalSkillXP',
-- Mastery
'flatMasteryTokens',
'xpFromMasteryTokens',
'masteryPoolCap',
'masteryXP',
'masteryPoolProgress',
-- Resource preservation
'bypassGlobalPreservationChance',
'skillPreservationChance',
'skillPreservationCap',
-- Other preservation (consumables, summons, potion charges)
'consumablePreservationChance',
'summoningChargePreservationChance',
'potionChargePreservationChance',
-- Item doubling
'globalItemDoublingChance',
-- Resource quantity
'flatBasePrimaryProductQuantity',
'basePrimaryProductQuantity',
'flatBaseRandomProductQuantity',
'flatAdditionalSkillItem',
'flatAdditionalPrimaryProductQuantity',
-- Cost reduction
'skillCostReduction',
-- Off item chance
'offItemChance'
}
local function allSkillModCriteriaFromIDs(modIDs)
local modIDsWithProps = {}
for i, modID in ipairs(allSkillModIDs) do
table.insert(modIDsWithProps, {
["id"] = modID,
["type"] = 'id',
["props"] = {
["skillID"] = 'nil',
["modType"] = 'pos'
}
})
end
return Modifiers.getMatchCriteriaFromIDs(modIDsWithProps)
end
local allSkillModCriteria = allSkillModCriteriaFromIDs(allSkillModIDs)


function p._getItemUses(item, asList, addCategories)
function p._getItemUses(item, asList, addCategories)
Line 84: Line 110:
["Upgrade"] = '[[Upgrading Items]]',
["Upgrade"] = '[[Upgrading Items]]',
["Food"] = '[[Food]]',
["Food"] = '[[Food]]',
["Chest"] = '[[Chest Drop Tables|Can Be Opened]]',
["Chest"] = '[[Chest|Can Be Opened]]',
["Mastery"] = Icons.Icon({'Mastery'}),
["Mastery"] = Icons.Icon({'Mastery'}),
["AllSkills"] = 'All skills',
["AllSkills"] = 'All skills',
["AltMagic"] = Icons.Icon({'Alt. Magic', type='skill'}),
["AltMagic"] = Icons.Icon({'Alt. Magic', type='skill'}),
["ChargeStone"] = 'Powering ' .. Icons.Icon({'Crown of Rhaelyx', type='item'}),
["ChargeStone"] = 'Powering ' .. Icons.Icon({'Crown of Rhaelyx', type='item'}),
["Shop"] = Icons.Icon({'Shop'})
["Shop"] = Icons.Icon({'Shop'}),
["TownshipTask"] = Icons.Icon({'Tasks', type='township'})
}
}


local addUse = function(useName)
local addUse = function(useName)
local skillID = (type(useName) == 'number' and useName) or SkillEnum[useName]
local skillID = Constants.getSkillID(useName)
if type(skillID) == 'number' then
if skillID == nil then
if skillUses[skillID] == nil then
-- May have been passed a skill ID instead
skillUses[skillID] = Constants.getSkillName(skillID)
local skillName = Constants.getSkillName(useName)
if skillName ~= nil then
skillID = useName
skillUses[skillID] = skillName
end
end
elseif not otherUses[useName] then
else
skillUses[skillID] = useName
end
 
if skillID == nil and not otherUses[useName] then
otherUses[useName] = true
otherUses[useName] = true
end
end
end
end
local hasUse = function(useName)
local hasUse = function(useName)
local skillID = (type(useName) == 'number' and useName) or SkillEnum[useName]
local skillID = Constants.getSkillID(useName)
if type(skillID) == 'number' then
if skillID == nil and Constants.getSkillName(useName) ~= nil then
skillID = useName
end
if skillID ~= nil then
return (skillUses[skillID] ~= nil) or false
return (skillUses[skillID] ~= nil) or false
else
else
Line 113: Line 150:
-- Check for any overrides within itemUseArray
-- Check for any overrides within itemUseArray
for useName, itemList in pairs(itemUseArray) do
for useName, itemList in pairs(itemUseArray) do
if Shared.contains(itemList, item.name) then
if Shared.contains(itemList, item.id) then
addUse(useName)
addUse(useName)
end
end
end
-- If this is a potion add it to the appropriate uses table
if type(item.masteryID) == 'table' and item.masteryID[1] == SkillEnum.Herblore then
-- Default to 'Combat' if unknown
local potionUse = potionUseArray[item.masteryID[2]] or 'Combat'
addUse(potionUseArray[item.masteryID[2]] or 'Combat')
end
end


-- If the item has any modifiers that affect a given skill, add it to those tables
-- If the item has any modifiers that affect a given skill, add it to those tables
-- Added an exception for Mastery Tokens since they were being incorrectly flagged as usable for all skills
-- Added special handling for Mastery Tokens since they were being incorrectly flagged as usable for all skills
if item.modifiers ~= nil and (item.isToken == nil or not item.isToken) then
if item.modifiers ~= nil then
local skillArray = Constants.getModifierSkills(item.modifiers)
if item.modifiers.masteryToken ~= nil then
for i, skillName in ipairs(skillArray) do
-- Mastery tokens
addUse(skillName)
addUse('Mastery')
else
local modsAllSkills = Modifiers.getMatchingModifiers(item.modifiers, allSkillModCriteria)
if not Shared.tableIsEmpty(modsAllSkills.matched) then
addUse('AllSkills')
else
local skillArray = Modifiers.getModifierSkills(item.modifiers)
for i, skillName in ipairs(skillArray) do
addUse(skillName)
end
end
end
if item.modifiers.thievingStealth ~= nil then
addUse('melvorD:Thieving')
end
end
end
end
Line 140: Line 183:


-- Check if the item is an entry requirement for any Slayer area
-- Check if the item is an entry requirement for any Slayer area
if not hasUse(SkillEnum.Slayer) and item.isEquipment then
if not hasUse('Slayer') and (item.validSlots ~= nil or item.occupiesSlots ~= nil or item.equipmentStats ~= nil) then
local slayerAreas = Areas.getAreas(function(area) return area.type == 'slayer' and type(area.entryRequirements) == 'table' end)
local slayerAreas = Areas.getAreas(function(area) return area.type == 'slayerArea' and type(area.entryRequirements) == 'table' end)
for i, area in pairs(slayerAreas) do
for i, area in ipairs(slayerAreas) do
for j, req in pairs(area.entryRequirements) do
for j, req in ipairs(area.entryRequirements) do
if req.type == "SlayerItem" and req.itemID == item.id then
if req.type == "SlayerItem" and req.itemID == item.id then
addUse(SkillEnum.Slayer)
addUse('Slayer')
break
break
end
end
end
end
if hasUse(SkillEnum.Slayer) then
if hasUse('Slayer') then
break
break
end
end
Line 155: Line 198:
end
end


-- Can the item be upgraded, or is it part of an upgrade recipe?
-- Is the item a cost in an upgrade?
if item.canUpgrade then
for i, upgrade in ipairs(GameData.rawData.itemUpgrades) do
addUse('Upgrade')
for j, itemCost in ipairs(upgrade.itemCosts) do
else
if itemCost.id == item.id then
for i, item2 in pairs(ItemData.Items) do
addUse('Upgrade')
if item2.itemsRequired ~= nil then
table.insert(categoryArray, '[[Category:Upgradeable Items]]')
for j, req in ipairs(item2.itemsRequired) do
break
if req[1] == item.id then
addUse('Upgrade')
break
end
end
if hasUse('Upgrade') then
break
end
end
end
end
end
end
if hasUse('Upgrade') then
if hasUse('Upgrade') then
break
table.insert(categoryArray, '[[Category:Upgradeable Items]]')
end
end
end


Line 182: Line 217:
end
end


if item.canOpen then
if item.dropTable ~= nil then
table.insert(categoryArray, '[[Category:Openable Items]]')
table.insert(categoryArray, '[[Category:Openable Items]]')
addUse('Chest')
addUse('Chest')
Line 190: Line 225:
-- All have somewhat consistent recipe data structures
-- All have somewhat consistent recipe data structures
local recipeSkillIDs = {
local recipeSkillIDs = {
SkillEnum.Cooking,
'melvorD:Cooking',
SkillEnum.Smithing,
'melvorD:Smithing',
SkillEnum.Fletching,
'melvorD:Fletching',
SkillEnum.Crafting,
'melvorD:Crafting',
SkillEnum.Runecrafting,
'melvorD:Runecrafting',
SkillEnum.Herblore
'melvorD:Herblore'
}
}
for i, recipeSkillID in ipairs(recipeSkillIDs) do
for i, recipeSkillID in ipairs(recipeSkillIDs) do
if not hasUse(recipeSkillID) then
if not hasUse(recipeSkillID) then
local recipeKey = (recipeSkillID == SkillEnum.Herblore and 'Potions') or 'Recipes'
local _, localSkillID = GameData.getLocalID(recipeSkillID)
local skillName = Constants.getSkillName(recipeSkillID)
-- Iterate over all recipes for the current skill
-- Iterate over all recipes for the current skill
for j, recipe in ipairs(SkillData[skillName][recipeKey]) do
for j, recipe in ipairs(SkillData[localSkillID].recipes) do
for k, itemCost in ipairs(recipe.itemCosts) do
for k, itemCost in ipairs(recipe.itemCosts) do
if itemCost.id == item.id then
if itemCost.id == item.id then
Line 231: Line 265:


-- Firemaking
-- Firemaking
if not hasUse(SkillEnum.Firemaking) and type(item.masteryID) == 'table' and item.masteryID[1] == SkillEnum.Firemaking then
if not hasUse('melvorD:Firemaking') then
addUse(SkillEnum.Firemaking)
for i, recipe in ipairs(SkillData.Firemaking.logs) do
if recipe.logID == item.id then
addUse('melvorD:Firemaking')
break
end
end
end
end


-- Farming
-- Farming
if not hasUse(SkillEnum.Farming) and item.grownItemID ~= nil then
if not hasUse('melvorD:Farming') then
addUse(SkillEnum.Farming)
for i, recipe in ipairs(SkillData.Farming.recipes) do
if recipe.seedCost.id == item.id then
addUse('melvorD:Farming')
break
end
end
end
end


-- Agility
-- Agility
if not hasUse(SkillEnum.Agility) and Shared.tableCount(Agility.getObstaclesForItem(item.id)) > 0 then
if not hasUse('melvorD:Agility') and not Shared.tableIsEmpty(Agility.getObstaclesForItem(item.id)) then
addUse(SkillEnum.Agility)
addUse('melvorD:Agility')
end
end


-- Summoning
-- Summoning
if not hasUse(SkillEnum.Summoning) then
if not hasUse('melvorD:Summoning') then
for i, recipe in ipairs(SkillData.Summoning.Marks) do
for i, recipe in ipairs(SkillData.Summoning.recipes) do
-- Tablets & Non-shard items
-- Tablets & Non-shard items
if recipe.itemID == item.id or Shared.contains(recipe.nonShardItemCosts, item.id) then
if recipe.productID == item.id or Shared.contains(recipe.nonShardItemCosts, item.id) then
addUse(SkillEnum.Summoning)
addUse('melvorD:Summoning')
break
break
else
else
Line 256: Line 300:
for j, itemCost in ipairs(recipe.itemCosts) do
for j, itemCost in ipairs(recipe.itemCosts) do
if itemCost.id == item.id then
if itemCost.id == item.id then
addUse(SkillEnum.Summoning)
addUse('melvorD:Summoning')
break
break
end
end
end
if hasUse(SkillEnum.Summoning) then
break
end
end
end
end
Line 270: Line 311:
if item.prayerPoints ~= nil then
if item.prayerPoints ~= nil then
table.insert(categoryArray, '[[Category:Buriable Items]]')
table.insert(categoryArray, '[[Category:Buriable Items]]')
if not hasUse(SkillEnum.Prayer) then
if not hasUse('melvorD:Prayer') then
addUse(SkillEnum.Prayer)
addUse('melvorD:Prayer')
end
elseif item.soulPoints ~= nil then
table.insert(categoryArray, '[[Category:Releasable Items]]')
if not hasUse('melvorD:Prayer') then
addUse('melvorD:Prayer')
end
end
end
end
Line 278: Line 324:
if not (hasUse('Magic') and hasUse('AltMagic')) then
if not (hasUse('Magic') and hasUse('AltMagic')) then
-- First check if the item its self is used in any spells
-- First check if the item its self is used in any spells
local spellList = Magic.getSpellsForItem(item.id, true)
local spellList = Magic.getSpellsUsingItem(item.id, true)
for i, spell in ipairs(spellList) do
for i, spell in ipairs(spellList) do
local useKey = (spell.type == 'AltMagic' and 'AltMagic' or 'Magic')
local useKey = (spell.type == 'altMagic' and 'AltMagic' or 'Magic')
if not hasUse(useKey) then
if not hasUse(useKey) then
addUse(useKey)
addUse(useKey)
Line 286: Line 332:
end
end
-- Check if the item provides runes, if it does check where they are used also
-- Check if the item provides runes, if it does check where they are used also
if item.providesRune ~= nil then
if item.providedRunes ~= nil then
for i, runeID in ipairs(item.providesRune) do
for i, rune in ipairs(item.providedRunes) do
if hasUse('Magic') and hasUse('AltMagic') then
if hasUse('Magic') and hasUse('AltMagic') then
break
break
else
else
local spellList = Magic.getSpellsForItem(runeID, false)
local spellList = Magic.getSpellsUsingItem(rune.id, false)
for j, spell in ipairs(spellList) do
for j, spell in ipairs(spellList) do
local useKey = (spell.type == 'AltMagic' and 'AltMagic' or 'Magic')
local useKey = (spell.type == 'altMagic' and 'AltMagic' or 'Magic')
if not hasUse(useKey) then
if not hasUse(useKey) then
addUse(useKey)
addUse(useKey)
Line 304: Line 350:


-- Other odds and ends:
-- Other odds and ends:
-- Mastery Tokens are tied to 'Mastery'
if item.isToken and item.skill ~= nil then
addUse('Mastery')
end


-- Skillcapes are tied to the appropriate skill
-- Skillcapes are tied to the appropriate skill
-- Except Maximum Skillcape, which is tied to all skills. (And so is the Signet Ring)
-- Except Maximum Skillcape, which is tied to all skills. (And so is the Signet Ring)
-- And combat skillcapes, since combat skills don't get special treatment
-- And combat skillcapes, since combat skills don't get special treatment
local ignoreCapes = {'Ranged Skillcape', 'Attack Skillcape', 'Strength Skillcape', 'HP Skillcape', 'Defence Skillcape'}
if item.tier == 'Skillcape' then
if Shared.contains({'Maximum Skillcape', "Aorpheat's Signet Ring", 'Ring of Wealth', 'Cape of Completion'}, item.name) then
local ignoreCapes = {
addUse('AllSkills')
'melvorD:Attack_Skillcape',
elseif item.name == 'Magic Skillcape' then
'melvorD:Strength_Skillcape',
addUse(SkillEnum.Magic)
'melvorD:Defence_Skillcape',
addUse('AltMagic')
'melvorD:Hitpoints_Skillcape',
elseif Shared.contains(item.name, 'Skillcape') and not Shared.contains(ignoreCapes, item.name) then
'melvorF:Ranged_Skillcape',
local skillName = Shared.splitString(item.name, ' ')[1]
'melvorTotH:Superior_Attack_Skillcape',
addUse(skillName)
'melvorTotH:Superior_Strength_Skillcape',
end
'melvorTotH:Superior_Defence_Skillcape',
 
'melvorTotH:Superior_Hitpoints_Skillcape',
if Shared.contains(item.name, 'Skillcape') or item.name == 'Cape of Completion' then
'melvorTotH:Superior_Ranged_Skillcape',
}
local allCapes = {
'melvorF:Max_Skillcape',
'melvorF:Cape_of_Completion',
'melvorTotH:Superior_Max_Skillcape',
'melvorTotH:Superior_Cape_Of_Completion',
}
if Shared.contains(allCapes, item.id) then
addUse('AllSkills')
elseif Shared.contains({'melvorF:Magic_Skillcape', 'melvorTotH:Superior_Magic_Skillcape'}, item.id) then
addUse('melvorD:Magic')
addUse('AltMagic')
elseif not Shared.contains(ignoreCapes, item.id) then
local splitName = Shared.splitString(item.name, ' ')
local skillName = (splitName[1] == 'Superior' and splitName[2]) or splitName[1]
addUse(skillName)
end
table.insert(categoryArray, '[[Category:Skillcapes]]')
table.insert(categoryArray, '[[Category:Skillcapes]]')
end
end
 
--Special note for Charge Stone of Rhaelyx
--Special note for Charge Stone of Rhaelyx
if item.name == 'Charge Stone of Rhaelyx' then
if item.id == 'melvorD:Charge_Stone_of_Rhaelyx' then
addUse('ChargeStone')
addUse('ChargeStone')
end
end
Line 335: Line 393:
--Some items are needed to make shop purchases
--Some items are needed to make shop purchases
local shopArray = Shop.getItemCostArray(item.id)
local shopArray = Shop.getItemCostArray(item.id)
if Shared.tableCount(shopArray) > 0 then
if not Shared.tableIsEmpty(shopArray) then
addUse('Shop')
addUse('Shop')
end
end


-- Township Tasks
for _, task in ipairs(SkillData.Township.tasks) do
if task.goals.items ~= nil then -- Skip tasks with no items
if GameData.getEntityByID(task.goals.items, item.id) then
addUse('TownshipTask')
break
end
end
end
-- Generate result text
-- Generate result text
local useArray = {}
local useArray = {}
local prefix, delim = asList and '* ' or '', asList and '\r\n' or '<br/>'
local prefix, delim = asList and '* ' or '', asList and '\r\n' or '<br/>'
-- Always place 'All skills' use before any skills
if hasUse('AllSkills') then
table.insert(useArray, prefix .. (otherUseText.AllSkills or 'AllSkills'))
otherUses.AllSkills = nil
end
for skillID, skillName in Shared.spairs(skillUses, function(t, a, b) return t[a] < t[b] end) do
for skillID, skillName in Shared.spairs(skillUses, function(t, a, b) return t[a] < t[b] end) do
table.insert(useArray, prefix .. Icons.Icon({skillName, type='skill'}))
table.insert(useArray, prefix .. Icons.Icon({skillName, type='skill'}))
Line 362: Line 435:
end
end
if item == nil then
if item == nil then
return "ERROR: No item named "..itemName.." exists in the data module"
return Shared.printError('No item named "' .. itemName .. '" exists in the data module')
end
end


Line 369: Line 442:


function p._getItemUseTable(item)
function p._getItemUseTable(item)
local useArray = {}
local tableData = {}


-- Loop through all items to find anything that can be upgraded using our source
-- Loop through all upgrades to find anything that can be upgraded using our source
for i, item2 in ipairs(ItemData.Items) do
for i, upgrade in ipairs(GameData.rawData.itemUpgrades) do
if item2.itemsRequired ~= nil then
for j, itemCost in ipairs(upgrade.itemCosts) do
for j, req in pairs(item2.itemsRequired) do
if itemCost.id == item.id then
if req[1] == item.id then
local reqs = nil
local mat = item2.itemsRequired
-- Potions do have upgrade requirements though
local xp = 'N/A'
local upgradeItem = Items.getItemByID(upgrade.upgradedItemID)
local rowReq = nil
if upgradeItem ~= nil and upgradeItem.charges ~= nil and upgradeItem.tier ~= nil then
--Potions do have upgrade requirements though
local levelUnlock = GameData.getEntityByProperty(SkillData.Herblore.masteryLevelUnlocks, 'descriptionID', upgradeItem.tier + 1)
if item2.potionTier ~= nil then
if levelUnlock ~= nil then
rowReq = Icons._MasteryReq(item2.name, SkillData.Herblore.TierMasteryLevels[item2.potionTier + 1])
reqs = Icons._MasteryReq(upgradeItem.name, levelUnlock.level)
end
end
table.insert(useArray, {item = {id = item2.id, name = item2.name}, qty = 1, mats = mat, skill = 'Upgrade', req = rowReq, xp = xp, gp = item2.trimmedGPCost})
break
end
end
table.insert(tableData, {
['skill'] = 'Upgrade',
['useType'] = '[[Upgrading Items|' .. (upgrade.isDowngrade and 'Downgrade' or 'Upgrade') .. ']]',
['reqs'] = reqs,
['costs'] = Common.getCostString({ ["items"] = upgrade.itemCosts, ["currencies"] = upgrade.currencyCosts }),
['productName'] = upgradeItem.name,
['qty'] = (upgrade.quantity or 1)
})
end
end
end
end
Line 393: Line 472:
-- All have somewhat consistent recipe data structures
-- All have somewhat consistent recipe data structures
local recipeSkillIDs = {
local recipeSkillIDs = {
SkillEnum.Cooking,
'melvorD:Cooking',
SkillEnum.Smithing,
'melvorD:Smithing',
SkillEnum.Fletching,
'melvorD:Fletching',
SkillEnum.Crafting,
'melvorD:Crafting',
SkillEnum.Runecrafting,
'melvorD:Runecrafting',
SkillEnum.Herblore
'melvorD:Herblore'
}
}
for i, recipeSkillID in ipairs(recipeSkillIDs) do
for i, recipeSkillID in ipairs(recipeSkillIDs) do
local skillName = Constants.getSkillName(recipeSkillID)
local skillName = Constants.getSkillName(recipeSkillID)
local recipeKey = (recipeSkillID == SkillEnum.Herblore and 'Potions') or 'Recipes'
local _, localSkillID = GameData.getLocalID(recipeSkillID)
-- Iterate over all recipes for the current skill
-- Iterate over all recipes for the current skill
for j, recipe in ipairs(SkillData[skillName][recipeKey]) do
for j, recipe in ipairs(SkillData[localSkillID].recipes) do
local costLists = {recipe.alternativeCosts or {}, {{["itemCosts"] = recipe.itemCosts}}}
local costLists = {recipe.alternativeCosts or {}, {{["itemCosts"] = recipe.itemCosts}}}
for k, costList in pairs(costLists) do
for k, costList in pairs(costLists) do
Line 411: Line 491:
if itemCost.id == item.id then
if itemCost.id == item.id then
local recipeItemIDs = nil
local recipeItemIDs = nil
if recipeSkillID == SkillEnum.Herblore then
if recipeSkillID == 'melvorD:Herblore' then
recipeItemIDs = recipe.potionIDs
recipeItemIDs = recipe.potionIDs
elseif recipeSkillID == SkillEnum.Cooking then
elseif recipeSkillID == 'melvorD:Cooking' then
recipeItemIDs = {recipe.itemID, recipe.perfectCookID}
recipeItemIDs = {recipe.productID, recipe.perfectCookID}
else
else
recipeItemIDs = {recipe.itemID}
recipeItemIDs = {recipe.productID}
end
end
for o, recipeItemID in ipairs(recipeItemIDs) do
for o, recipeItemID in ipairs(recipeItemIDs) do
local recipeItem = Items.getItemByID(recipeItemID)
local recipeItem = Items.getItemByID(recipeItemID)
if recipeItem ~= nil then
if recipeItem ~= nil then
local itemDef = {id = recipe.itemID, name = recipeItem.name}
local lvl, isAbyssal = Skills.getRecipeLevelRealm(recipeSkillID, recipe)
local qty = (recipe.baseQuantity or 1) * (costDef.quantityMultiplier or 1)
local reqs = nil
local rowReq = recipe.level
if recipeSkillID == 'melvorD:Cooking' then
local reqVal = nil
-- Cooking includes the required utility (fire, furnace, pot) as a special requirement
if recipeSkillID == SkillEnum.Herblore then
local cookingCatIcon = {
["melvorD:Fire"] = 'Normal Cooking Fire',
["melvorD:Furnace"] = 'Basic Furnace',
["melvorD:Pot"] = 'Basic Pot'
}
local categoryIconName, categoryName = cookingCatIcon[recipe.categoryID], nil
local recipeCategory = GameData.getEntityByID(SkillData.Cooking.categories, recipe.categoryID)
if recipeCategory ~= nil then
categoryName = recipeCategory.modifierName or recipeCategory.name
end
if categoryIconName ~= nil and categoryName ~= nil then
local utilReq = Icons.Icon({'Cooking', categoryName, section = 'Cooking Upgrades', img = categoryIconName, type = 'upgrade'})
reqs = Icons._SkillReq(skillName, lvl, false, (isAbyssal and "melvorItA:Abyssal" or nil)) .. '<br>' .. utilReq
end
elseif recipeSkillID == 'melvorD:Herblore' then
-- Herblore may also have a mastery requirement
-- Herblore may also have a mastery requirement
local masteryLvl = SkillData.Herblore.TierMasteryLevels[o]
local levelUnlock = GameData.getEntityByProperty(SkillData.Herblore.masteryLevelUnlocks, 'descriptionID', recipeItem.tier + 1)
if masteryLvl ~= nil and masteryLvl > 1 then
if levelUnlock ~= nil and levelUnlock.level > 1 then
local masteryReq = Icons._MasteryReq(recipeItem.name, masteryLvl)
local masteryReq = Icons._MasteryReq(recipeItem.name, levelUnlock.level)
reqVal = rowReq + masteryLvl * 0.01
reqs = Icons._SkillReq(skillName, lvl, false, (isAbyssal and "melvorItA:Abyssal" or nil)) .. '<br>' .. masteryReq
rowReq = Icons._SkillReq(skillName, rowReq) .. '<br/>' .. masteryReq
-- Slightly increase level per potion tier for proper ordering using data-sort-value
lvl = lvl + levelUnlock.level * 0.01
end
end
end
end
table.insert(useArray, {item = itemDef, qty = qty, mats = costDef.itemCosts, gp = recipe.gpCost, sc = recipe.scCost, skill = skillName, reqVal = reqVal, req = rowReq, xp = recipe.baseXP})
table.insert(tableData, {
['skill'] = skillName,
['lvl'] = lvl,
['isAbyssal'] = isAbyssal,
['useType'] = Icons.Icon({skillName, type='skill'}),
['reqs'] = (reqs or Icons._SkillReq(skillName, lvl, false, (isAbyssal and "melvorItA:Abyssal" or nil))),
['costs'] = Common.getCostString({ ["items"] = costDef.itemCosts, ["currencies"] = recipe.currencyCosts }),
['productName'] = recipeItem.name,
['qty'] = (recipe.baseQuantity or 1) * (costDef.quantityMultiplier or 1),
['xp'] = recipe.baseAbyssalExperience or recipe.baseExperience
})
end
end
end
end
Line 446: Line 551:


-- Farming
-- Farming
if item.grownItemID ~= nil then
for i, recipe in ipairs(SkillData.Farming.recipes) do
local item2 = Items.getItemByID(item.grownItemID)
if recipe.seedCost.id == item.id then
local mat = {{id = item.id, qty = item.seedsRequired}}
local category = GameData.getEntityByID(SkillData.Farming.categories, recipe.categoryID)
local xp = item.farmingXP
local lvl, isAbyssal = Skills.getRecipeLevelRealm('melvorD:Farming', recipe)
local rowReq = item.farmingLevel
local product = Items.getItemByID(recipe.productId)
local qty = (item.tier ~= nil and item.tier == 'Tree' and 35 or 15)
 
table.insert(useArray, {item = {id = item2.id, name = item2.name}, qty = qty, mats = mat, skill = 'Farming', req = rowReq, xp = xp})
table.insert(tableData, {
['skill'] = 'Farming',
['lvl'] = lvl,
['isAbyssal'] = isAbyssal,
['useType'] = Icons.Icon({'Farming', type='skill'}),
['reqs'] = Icons._SkillReq('Farming', lvl, false, (isAbyssal and 'melvorItA:Abyssal' or nil)),
['costs'] = { recipe.seedCost },
['productName'] = product.name,
['qty'] = (5 * category.harvestMultiplier),
['xp'] = recipe.baseAbyssalExperience or recipe.baseExperience
})
end
end
end


Line 458: Line 574:
local obstacles = Agility.getObstaclesForItem(item.id)
local obstacles = Agility.getObstaclesForItem(item.id)
for i, obstacle in ipairs(obstacles) do
for i, obstacle in ipairs(obstacles) do
local itemCosts = {}
table.insert(tableData, {
for j, itemDef in ipairs(obstacle.cost.items) do
['skill'] = 'Agility',
table.insert(itemCosts, {id = itemDef[1], qty = itemDef[2]})
['lvl'] = Skills.getRecipeLevelRealm('Agility', obstacle),
end
['isAbyssal'] = obstacle.realm == 'melvorItA:Abyssal',
local req = Agility._getObstacleRequirements(obstacle)
['useType'] = Icons.Icon({'Agility', type='skill'}),
--local objType = (obstacle.category == nil and 'Pillar') or 'Obstacle'
['reqs'] = Agility._getObstacleRequirements(obstacle),
table.insert(useArray, {item = {id = obstacle.id, name = obstacle.name}, qty = 1, mats = itemCosts, gp = obstacle.cost.gp, sc = obstacle.cost.slayerCoins, skill = 'Agility', req = req, type = 'skill'})
['costs'] = Common.getCostString({ ["items"] = obstacle.itemCosts, ["currencies"] = obstacle.currencyCosts }),
['productName'] = obstacle.name,
['iconType'] = 'agility',
['qty'] = 1
})
end
end


-- Summoning
-- Summoning
for i, recipe in ipairs(SkillData.Summoning.Marks) do
for i, recipe in ipairs(SkillData.Summoning.recipes) do
local recipeGPCost = SkillData.Summoning.RecipeGPCost
local recipeCost = 0
if recipe.abyssalLevel ~= nil then
recipeCost = SkillData.Summoning.recipeAPCost
else
recipeCost = SkillData.Summoning.recipeGPCost
end
local useShards = false
local useShards = false
local recipeItem = nil
local recipeItem = nil
Line 479: Line 604:
end
end
-- Non-shard items
-- Non-shard items
for j, nonShardItemID in ipairs(recipe.nonShardItemCosts) do
-- Familiar recipes may also have a currency cost without any non-shard
-- items, so account for this with a dummy ID such that one iteration
-- of the below loop always occurs
local nonShardItemIDs = (Shared.tableIsEmpty(recipe.nonShardItemCosts) and {''} or recipe.nonShardItemCosts)
for j, nonShardItemID in ipairs(nonShardItemIDs) do
if useShards or nonShardItemID == item.id then
if useShards or nonShardItemID == item.id then
-- Item is used in this particular synergy recipe
-- Item is used in this particular synergy recipe
if recipeItem == nil then
if recipeItem == nil then
recipeItem = Items.getItemByID(recipe.itemID)
recipeItem = Items.getItemByID(recipe.productID)
end
end
local nonShardItem = Items.getItemByID(nonShardItemID)
local nonShardItem = Items.getItemByID(nonShardItemID)
local itemValue = math.max(nonShardItem.sellsFor, 20)
local nonShardQty = math.max(1, math.floor(recipeGPCost / itemValue))
local recipeCosts = Shared.clone(recipe.itemCosts)
local recipeCosts = Shared.clone(recipe.itemCosts)
table.insert(recipeCosts, {id = nonShardItemID, qty = nonShardQty})
local xp = recipe.baseAbyssalExperience or recipe.baseExperience
table.insert(useArray, {item = {id = recipe.itemID, name = recipeItem.name}, qty = recipe.baseQuantity, mats = recipeCosts, gp = recipe.gpCost, sc = recipe.scCost, skill = 'Summoning', req = recipe.level, xp = recipe.baseXP})
local lvl, isAbyssal = Skills.getRecipeLevelRealm('melvorD:Summoning', recipe)
local recipeCosts = {}
for k, itemCost in ipairs(recipe.itemCosts) do
table.insert(recipeCosts, {id = itemCost.id, quantity = itemCost.quantity})
end
if nonShardItem ~= nil then
-- Item ID can be nil for recipes such as Leprechaun or Cyclops
local itemValue = math.max(nonShardItem.sellsFor, 20)
local nonShardQty = math.max(1, math.ceil(recipeCost / itemValue))
table.insert(recipeCosts, { id = nonShardItemID, quantity = nonShardQty })
end
table.insert(tableData, {
['skill'] = 'Summoning',
['lvl'] = lvl,
['isAbyssal'] = isAbyssal,
['useType'] = Icons.Icon({ 'Summoning', type='skill' }),
['reqs'] = Icons._SkillReq('Summoning', lvl, false, (isAbyssal and 'melvorItA:Abyssal' or nil)),
['costs'] = Common.getCostString({ ["items"] = recipeCosts, ["currencies"] = recipe.currencyCosts }),
['productName'] = recipeItem.name,
['qty'] = recipe.baseQuantity,
['xp'] = xp
})
end
end
end
end
Line 497: Line 645:
--Handle shop purchases using Module:Shop
--Handle shop purchases using Module:Shop
local shopUses = Shop.getItemCostArray(item.id)
local shopUses = Shop.getItemCostArray(item.id)
for i, purchase in ipairs(shopUses) do
for i, shopUse in ipairs(shopUses) do
local rowReq = Shop.getRequirementString(purchase.unlockRequirements)
local purchase = shopUse.purchase
local iconType = (purchase.contains.items ~= nil and Shared.tableCount(purchase.contains.items) > 0) and 'item' or 'upgrade'
 
table.insert(useArray, {item = {name = purchase.name}, qty = 1, mats = purchase.cost.items, skill = 'Shop', req = rowReq, xp = 'N/A', gp = purchase.cost.gp, type = iconType})
table.insert(tableData, {
['skill'] = 'Shop',
['lvl'] = (not Shared.tableIsEmpty(purchase.purchaseRequirements) and purchase.purchaseRequirements[1].level or nil),
['useType'] = Icons.Icon({'Shop'}),
['reqs'] = Common.getRequirementString(purchase.purchaseRequirements),
['costs'] = Common.getCostString(purchase.cost),
['productName'] = Common.getPurchaseName(purchase),
['iconType'] = Common.getPurchaseIconType(purchase),
['qty'] = (purchase.qty or 1)
})
end
end


--Finally build the table using what we've learned
--Finally build the table using what we've learned
table.sort(useArray, function(a, b)
table.sort(tableData, function(a, b)
local aReqVal = a.reqVal ~= nil and a.reqVal or a.req
local aLvl = a.lvl
local bReqVal = b.reqVal ~= nil and b.reqVal or b.req
local bLvl = b.lvl
if a.skill ~= b.skill then
if a.skill ~= b.skill then
return a.skill < b.skill
return a.skill < b.skill
elseif type(aReqVal) ~= type(bReqVal) then
elseif type(aLvl) ~= type(bLvl) then
return tostring(aReqVal) < tostring(bReqVal)
return tostring(aLvl) < tostring(bLvl)
elseif aReqVal ~= bReqVal then
elseif aLvl ~= bLvl then
return aReqVal < bReqVal
return aLvl < bLvl
else
else
return a.item.name < b.item.name
return a.productName < b.productName
end
end
end)
end)


local useTable = p.buildUseTable(item, tableData)
local spellUseTable = p._getSpellUseTable(item)
if spellUseTable ~= nil and spellUseTable ~= '' then
useTable = useTable .. '\r\n===' .. Icons.Icon({'Magic', type='skill', size=30}) .. '===\r\n' .. spellUseTable
end
if useTable == '' then
return ''
else
return '==Uses==\r\n' .. useTable
end
end
function p.buildUseTable(item, tableData)
if Shared.tableIsEmpty(tableData) then return '' end
local showRequirements = false
local showXP = false
for i, data in ipairs(tableData) do
if not showRequirements and tableData[i].reqs ~= nil then showRequirements = true end
if not showXP and tableData[i].xp ~= nil then showXP = true end
if showRequirements and showXP then break end
end


local resultPart = {}
local resultTable = mw.html.create('table')
if Shared.tableCount(useArray) > 0 then
resultTable:addClass('wikitable stickyHeader sortable col-1-img')
local typeTextList = {
local tableHeader = resultTable:tag('tr'):addClass('headerRow-0')
["Shop"] = Icons.Icon({'Shop'}),
:tag('th'):wikitext('Product'):attr('colspan', 2)
["Upgrade"] = '[[Upgrading Items|Upgrade]]'
:tag('th'):wikitext('Used In')
}


-- Header
if showRequirements then tableHeader:tag('th'):wikitext('Requires') end
table.insert(resultPart, '{| class="wikitable stickyHeader sortable"')
if showXP then tableHeader:tag('th'):wikitext('Exp') end
table.insert(resultPart, '\r\n|- class="headerRow-0"')
tableHeader:tag('th'):wikitext('Costs')
table.insert(resultPart, '\r\n!colspan=2|Item Created!!Type!!Requirements!!XP!!Ingredients')


-- Rows
for i, data in ipairs(tableData) do
for i, row in ipairs(useArray) do
local recipeRow = resultTable:tag('tr')
local qty = row.qty ~= nil and row.qty or 1
:tag('td'):wikitext(Icons.Icon({ data.productName, type=(data.iconType or 'item'), notext=true })):attr('data-sort-value', data.productName) -- Icon Image
local iconType = row.type ~= nil and row.type or 'item'
:tag('td')
local iconName = row.item.name
:wikitext((data.qty ~= nil and data.qty > 1 and ("'''" .. data.qty .. "x''' ") or '')) -- Icon Qty
if row.skill == 'Agility' then
:wikitext(Icons.Icon({ data.productName, type=(data.iconType or 'item'), noicon=true })) -- Icon Text
iconName = 'Agility'
:tag('td'):wikitext(data.useType):attr('data-sort-value', (data.skill or nil)) -- Used In
-- Requirements
if showRequirements then
if data.reqs ~= nil then
recipeRow:tag('td'):wikitext(data.reqs):attr('data-sort-value', (data.lvl or 0))
else
recipeRow:tag('td'):wikitext('N/A'):addClass('table-na')
end
end
local typeName = row.skill ~= nil and row.skill or ''
end
local typeText = typeTextList[typeName] or Icons.Icon({typeName, type='skill'}) or ''
-- Exp
local reqVal, reqText = row.reqVal, 'None'
if showXP then
if type(row.req) == 'number' then
if data.skill ~= nil and data.xp ~= nil then
reqVal = row.req
local iconClass = (data.isAbyssal and 'abyss-icon' or nil)
reqText = Icons._SkillReq(typeName, row.req)
local xpText = (data.isAbyssal and ' AXP' or ' XP')
elseif type(row.req) == 'string' then
recipeRow:tag('td'):attr('data-sort-value', data.xp)
reqText = row.req
:wikitext(Icons.Icon({ data.skill, notext=true, type='skill', class=iconClass }))
:wikitext(' ' .. Num.formatnum(data.xp) .. xpText)
else
recipeRow:tag('td'):wikitext('N/A'):addClass('table-na'):attr('data-sort-value', 0)
end
end
local xpVal, xpText = 0, 'N/A'
end
if type(row.xp) == 'string' then
-- Costs
xpText = row.xp
local costsRow = recipeRow:tag('td')
elseif type(row.xp) == 'number' then
if type(data.costs) == 'table' then
xpVal = row.xp
for i, mat in ipairs(data.costs) do
xpText = Shared.formatnum(row.xp) .. ' ' .. Icons.Icon({typeName, type='skill', notext=true}) .. ' XP'
if i > 1 then costsRow:tag('br') end
end
local matItem = Items.getItemByID(mat.id)
local matRow = {}
if matItem == nil then
if type(row.mats) == 'table' then
costsRow:wikitext(mat.quantity .. 'x ?????')
for j, itemCost in ipairs(row.mats) do
else
local matItemID = itemCost.id or itemCost[1] or -1
costsRow:wikitext(Icons.Icon({ matItem.name, type='item', qty=mat.quantity }))
local matItem = Items.getItemByID(matItemID)
local matQty = itemCost.qty or itemCost[2] or 1
if matItem == nil then
table.insert(matRow, 'ERROR: Failed to find item with ID ' .. itemCost.id .. '[[Category:Pages with script errors]]')
elseif type(matQty) == 'number' then
table.insert(matRow, Icons.Icon({matItem.name, type='item', qty=matQty}))
else
table.insert(matRow, Icons.Icon({matItem.name, type='item'}))
end
end
end
end
end
if row.gp ~= nil and row.gp > 0 then
else
table.insert(matRow, Icons.GP(row.gp))
costsRow:wikitext(data.costs)
end
if row.sc ~= nil and row.sc > 0 then
table.insert(matRow, Icons.SC(row.sc))
end
-- Item created
table.insert(resultPart, '\r\n|-\r\n|data-sort-value="' .. row.item.name .. '"| ')
table.insert(resultPart, Icons.Icon({iconName, row.item.name, type=iconType, notext=true, size=50}))
table.insert(resultPart, '\r\n| ')
if qty > 1 then
table.insert(resultPart, "'''" .. Shared.formatnum(qty) .. "x''' ")
end
table.insert(resultPart, Icons.Icon({iconName, row.item.name, type=iconType, noicon=true}))
-- Type
table.insert(resultPart, '\r\n|data-sort-value="' .. typeName .. '"| ' .. typeText)
-- Requirements
table.insert(resultPart, '\r\n|style="text-align:right;"')
if row.reqVal ~= nil then
table.insert(resultPart, ' data-sort-value="' .. reqVal .. '"')
end
table.insert(resultPart, '| ' .. reqText)
-- XP
table.insert(resultPart, '\r\n|style="text-align:right;" data-sort-value="' .. xpVal .. '"| ' .. xpText)
-- Ingredients
table.insert(resultPart, '\r\n| ' .. table.concat(matRow, '<br/>'))
end
end
table.insert(resultPart, '\r\n|}')
end
local spellUseTable = p._getSpellUseTable(item)
if spellUseTable ~= nil and spellUseTable ~= '' then
table.insert(resultPart, '\r\n===' .. Icons.Icon({'Magic', type='skill', size=30}) .. '===\r\n' .. spellUseTable)
end
if Shared.tableCount(resultPart) == 0 then
return ''
else
return '==Uses==\r\n' .. table.concat(resultPart)
end
end
return tostring(resultTable)
end
end


Line 614: Line 761:
local item = Items.getItem(itemName)
local item = Items.getItem(itemName)
if item == nil then
if item == nil then
return "ERROR: No item named "..itemName.." exists in the data module"
return Shared.printError('No item named "' .. itemName .. '" exists in the data module')
end
end


Line 621: Line 768:


function p._getSpellUseTable(item)
function p._getSpellUseTable(item)
local spellList = Magic.getSpellsForItem(item.id, true)
local spellList = Magic.getSpellsUsingItem(item.id, true)
--Bail immediately if no spells are found
--Bail immediately if no spells are found
if Shared.tableCount(spellList) == 0 then
if Shared.tableIsEmpty(spellList) then
return ''
return ''
end
--Adding a check for if the Items column is needed
local hasItems = false
for i, spell in pairs(spellList) do
if Magic._getSpellItems(spell) ~= '' then
hasItems = true
break
end
end
end


local resultPart = {}
local resultPart = {}
table.insert(resultPart, '{|class="wikitable sortable"\r\n!colspan="2"|Spell')
table.insert(resultPart, '{|class="wikitable sortable"\r\n!colspan="2"|Spell')
table.insert(resultPart, '!!Requirements')
table.insert(resultPart, '!!Requires')
table.insert(resultPart, '!!Type!!style="width:275px"|Description')
table.insert(resultPart, '!!Type!!style="width:275px"|Description')
table.insert(resultPart, '!!Runes')
table.insert(resultPart, '!!Runes')
if hasItems then
table.insert(resultPart, '!!Item Cost')
end
for i, spell in pairs(spellList) do
for i, spell in pairs(spellList) do
local spellBook = Magic.getSpellBookFromSpell(spell)
local rowPart = {}
local rowPart = {}
table.insert(rowPart, '\r\n|-\r\n|data-sort-value="'..spell.name..'"|')
table.insert(rowPart, '\r\n|-\r\n|data-sort-value="'..spell.name..'"|')
local iconType = (spell.type == 'Auroras' and 'aurora') or (spell.type == 'Curses' and 'curse') or 'spell'
local iconType = Magic._getSpellIconType(spell)
table.insert(rowPart, Icons.Icon({spell.name, type=iconType, notext=true, size=50}))
table.insert(rowPart, Icons.Icon({spell.name, type=iconType, notext=true}))
table.insert(rowPart, '||'..Icons.Icon({spell.name, type=iconType, noicon=true}))
table.insert(rowPart, '||'..Icons.Icon({spell.name, type=iconType, noicon=true}))
table.insert(rowPart, '||data-sort-value="'..spell.level..'"|'..Icons._SkillReq('Magic', spell.level))
table.insert(rowPart, '||data-sort-value="'..Skills.getRecipeLevel('melvorD:Summoning', spell)..'"|'..Magic._getSpellRequirements(spell))
--Handle required items/dungeon clears
table.insert(rowPart, '||data-sort-value="'.. spellBook.id ..'"|')
if spell.requiredItem ~= nil and spell.requiredItem >= 0 then
table.insert(rowPart, Magic.getSpellTypeLink(spellBook.id))
local reqItem = Items.getItemByID(spell.requiredItem)
table.insert(rowPart, '<br/>'..Icons.Icon({reqItem.name, type='item', notext=true})..' equipped')
end
if spell.requiredDungeonCompletion ~= nil then
local dung = Areas.getAreaByID('dungeon', spell.requiredDungeonCompletion[1])
table.insert(rowPart, '<br/>'..Icons.Icon({dung.name, type='dungeon', notext=true, qty=spell.requiredDungeonCompletion[2]})..' Clears')
end
table.insert(rowPart, '||data-sort-value="'..Magic.getSpellTypeIndex(spell.type)..'"|')
table.insert(rowPart, Magic.getSpellTypeLink(spell.type))
table.insert(rowPart, '||'..Magic._getSpellStat(spell, 'description'))
table.insert(rowPart, '||'..Magic._getSpellStat(spell, 'description'))
table.insert(rowPart, '||style="text-align:center"|')
table.insert(rowPart, '||style="text-align:center"|')
table.insert(rowPart, Magic._getSpellRunes(spell))
table.insert(rowPart, Magic._getSpellRunes(spell))
if hasItems then
table.insert(rowPart, '||')
table.insert(rowPart, Magic._getSpellItems(spell))
end
table.insert(resultPart, table.concat(rowPart))
table.insert(resultPart, table.concat(rowPart))
end
end
Line 664: Line 820:
local item = Items.getItem(itemName)
local item = Items.getItem(itemName)
if item == nil then
if item == nil then
return "ERROR: No item named "..itemName.." exists in the data module"
return Shared.printError('No item named "' .. itemName .. '" exists in the data module')
end
end


Line 670: Line 826:
end
end


--[==[
--[[
-- Uncomment this block and execute 'p.test()' within the debug console
-- Uncomment this block and execute 'p.test()' within the debug console
-- to test after making changes
-- to test after making changes
Line 702: Line 858:
'Ancient Ring of Mastery',
'Ancient Ring of Mastery',
'Mysterious Stone',
'Mysterious Stone',
'Charge Stone of Rhaelyx',
'Mastery Token (Cooking)',
'Mastery Token (Cooking)',
'Gem Gloves',
'Gem Gloves',
'Basic Bag'
'Basic Bag',
'Bird Nest',
'Abyssal Stone',
'Withered Ash',
'Wrath Rune',
'Magic Bones',
'Bowstring',
'Superior Max Skillcape',
'Abyssal Coin Contract II',
'Dark Summon Consumable II',
'Abyssal Slayer Gear Upgrade Kit',
'Topaz Bolts (Enchanted)',
'Summoning Shard (Black)',
'Dragon Javelin',
}
}
local checkFuncs = {
local checkFuncs = {
p.getItemUses,
--p.getItemUses,
p.getItemUseTable
p.getItemUseTable
}
}
Line 730: Line 900:
end
end
end
end
--]==]
--]]


return p
return p

Latest revision as of 05:21, 11 December 2024

Documentation for this module may be created at Module:Items/UseTables/doc

local p = {}

local Constants = require('Module:Constants')
local Common = require('Module:Common')
local Shared = require('Module:Shared')
local GameData = require('Module:GameData')
local SkillData = GameData.skillData
local Modifiers = require('Module:Modifiers')
local Skills = require('Module:Skills')
local Magic = require('Module:Magic')
local Areas = require('Module:CombatAreas')
local Items = require('Module:Items')
local Icons = require('Module:Icons')
local Agility = require('Module:Skills/Agility')
local Shop = require('Module:Shop')
local Num = require('Module:Number')

--Brute forcing some item uses to make things easier
local itemUseArray = {
	Agility = {},
	Astrology = {'melvorF:Stardust', 'melvorF:Golden_Stardust', 'melvorItA:Abyssal_Stardust', 'melvorItA:Eternal_Stardust'},
	Archaeology = {},
	Attack = {},
	Cartography = {},
	Combat = {'melvorF:Gold_Emerald_Ring', 'melvorD:Obsidian_Cape', 'melvorF:Throwing_Power_Gloves'},
	Cooking = {'melvorD:Cooking_Gloves'},
	Crafting = {'melvorItA:Abyssal_Crafting_Gloves'},
	Defence = {},
	Farming = {'melvorD:Compost', 'melvorD:Weird_Gloop', 'melvorItA:Abyssal_Compost', 'melvorD:Bobs_Rake'},
	Firemaking = {'melvorItA:Abyssal_Firemaking_Gloves'},
	Fishing = {'melvorD:Message_In_A_Bottle', 'melvorD:Barbarian_Gloves'},
	Fletching = {'melvorItA:Abyssal_Fletching_Gloves'},
	Harvesting = {'melvorItA:Abyssal_Harvesting_Gloves'},
	Herblore = {'melvorItA:Abyssal_Herblore_Gloves'},
	Hitpoints = {},
	Magic = {},
	Mining = {'melvorD:Mining_Gloves', 'melvorD:Gem_Gloves', 'melvorItA:Abyssal_Mining_Gloves'},
	Prayer = {},
	Ranged = {},
	Runecrafting = {'melvorItA:Abyssal_Runecrafting_Gloves'},
	Slayer = {},
	Smithing = {'melvorD:Smithing_Gloves', 'melvorItA:Abyssal_Smithing_Gloves'},
	Smithing = {},
	Strength = {},
	Summoning = {},
	Thieving = {'melvorF:Thieving_Gloves'},
	Township = {},
	Woodcutting = {},
}

-- List of modifier IDs which, if present on an item and not specific to any
-- particular skill, indicate that the item applies to all skills
local allSkillModIDs = {
	-- Skill XP
	'skillXP',
	'nonCombatSkillXP',
	'abyssalSkillXP',
	-- Mastery
	'flatMasteryTokens',
	'xpFromMasteryTokens',
	'masteryPoolCap',
	'masteryXP',
	'masteryPoolProgress',
	-- Resource preservation
	'bypassGlobalPreservationChance',
	'skillPreservationChance',
	'skillPreservationCap',
	-- Other preservation (consumables, summons, potion charges)
	'consumablePreservationChance',
	'summoningChargePreservationChance',
	'potionChargePreservationChance',
	-- Item doubling
	'globalItemDoublingChance',
	-- Resource quantity
	'flatBasePrimaryProductQuantity',
	'basePrimaryProductQuantity',
	'flatBaseRandomProductQuantity',
	'flatAdditionalSkillItem',
	'flatAdditionalPrimaryProductQuantity',
	-- Cost reduction
	'skillCostReduction',
	-- Off item chance
	'offItemChance'
}

local function allSkillModCriteriaFromIDs(modIDs)
	local modIDsWithProps = {}
	for i, modID in ipairs(allSkillModIDs) do
		table.insert(modIDsWithProps, {
			["id"] = modID,
			["type"] = 'id',
			["props"] = {
				["skillID"] = 'nil',
				["modType"] = 'pos'
			}
		})
	end
	return Modifiers.getMatchCriteriaFromIDs(modIDsWithProps)
end

local allSkillModCriteria = allSkillModCriteriaFromIDs(allSkillModIDs)

function p._getItemUses(item, asList, addCategories)
	-- Another fun one. This time getting all the different possible ways an item can be used
	local categoryArray = {}
	local skillUses = {}
	local otherUses = {}
	local otherUseText = {
		["Combat"] = Icons.Icon({'Combat'}),
		["Upgrade"] = '[[Upgrading Items]]',
		["Food"] = '[[Food]]',
		["Chest"] = '[[Chest|Can Be Opened]]',
		["Mastery"] = Icons.Icon({'Mastery'}),
		["AllSkills"] = 'All skills',
		["AltMagic"] = Icons.Icon({'Alt. Magic', type='skill'}),
		["ChargeStone"] = 'Powering ' .. Icons.Icon({'Crown of Rhaelyx', type='item'}),
		["Shop"] = Icons.Icon({'Shop'}),
		["TownshipTask"] = Icons.Icon({'Tasks', type='township'})
	}

	local addUse = function(useName)
		local skillID = Constants.getSkillID(useName)
		if skillID == nil then
			-- May have been passed a skill ID instead
			local skillName = Constants.getSkillName(useName)
			if skillName ~= nil then
				skillID = useName
				skillUses[skillID] = skillName
			end
		else
			skillUses[skillID] = useName
		end

		if skillID == nil and not otherUses[useName] then
			otherUses[useName] = true
		end
	end
	local hasUse = function(useName)
		local skillID = Constants.getSkillID(useName)
		if skillID == nil and Constants.getSkillName(useName) ~= nil then
			skillID = useName
		end
		if skillID ~= nil then
			return (skillUses[skillID] ~= nil) or false
		else
			return otherUses[useName] or false
		end
	end

	-- Check for any overrides within itemUseArray
	for useName, itemList in pairs(itemUseArray) do
		if Shared.contains(itemList, item.id) then
			addUse(useName)
		end
	end

	-- If the item has any modifiers that affect a given skill, add it to those tables
	-- Added special handling for Mastery Tokens since they were being incorrectly flagged as usable for all skills
	if item.modifiers ~= nil then
		if item.modifiers.masteryToken ~= nil then
			-- Mastery tokens
			addUse('Mastery')
		else
			local modsAllSkills = Modifiers.getMatchingModifiers(item.modifiers, allSkillModCriteria)
			if not Shared.tableIsEmpty(modsAllSkills.matched) then
				addUse('AllSkills')
			else
				local skillArray = Modifiers.getModifierSkills(item.modifiers)
				for i, skillName in ipairs(skillArray) do
					addUse(skillName)
				end
			end
		end
		if item.modifiers.thievingStealth ~= nil then
			addUse('melvorD:Thieving')
		end
	end

	--First things added to the list are non-skill things that are easy to check
	if not hasUse('Combat') and (Items.hasCombatStats(item) or item.specialAttacks ~= nil) then
		addUse('Combat')
	end

	-- Check if the item is an entry requirement for any Slayer area
	if not hasUse('Slayer') and (item.validSlots ~= nil or item.occupiesSlots ~= nil or item.equipmentStats ~= nil) then
		local slayerAreas = Areas.getAreas(function(area) return area.type == 'slayerArea' and type(area.entryRequirements) == 'table' end)
		for i, area in ipairs(slayerAreas) do
			for j, req in ipairs(area.entryRequirements) do
				if req.type == "SlayerItem" and req.itemID == item.id then
					addUse('Slayer')
					break
				end
			end
			if hasUse('Slayer') then
				break
			end
		end
	end

	-- Is the item a cost in an upgrade?
	for i, upgrade in ipairs(GameData.rawData.itemUpgrades) do
		for j, itemCost in ipairs(upgrade.itemCosts) do
			if itemCost.id == item.id then
				addUse('Upgrade')
				table.insert(categoryArray, '[[Category:Upgradeable Items]]')
				break
			end
		end
		if hasUse('Upgrade') then
			break
		end
	end

	if item.healsFor ~= nil then
		table.insert(categoryArray, '[[Category:Food Items]]')
		addUse('Food')
	end

	if item.dropTable ~= nil then
		table.insert(categoryArray, '[[Category:Openable Items]]')
		addUse('Chest')
	end

	-- Cooking, Smithing, Fletching, Crafting, Runecrafting, Herblore
	-- All have somewhat consistent recipe data structures
	local recipeSkillIDs = {
		'melvorD:Cooking',
		'melvorD:Smithing',
		'melvorD:Fletching',
		'melvorD:Crafting',
		'melvorD:Runecrafting',
		'melvorD:Herblore'
	}
	for i, recipeSkillID in ipairs(recipeSkillIDs) do
		if not hasUse(recipeSkillID) then
			local _, localSkillID = GameData.getLocalID(recipeSkillID)
			-- Iterate over all recipes for the current skill
			for j, recipe in ipairs(SkillData[localSkillID].recipes) do
				for k, itemCost in ipairs(recipe.itemCosts) do
					if itemCost.id == item.id then
						addUse(recipeSkillID)
						break
					end
				end
				-- Some items (such as Arrow shafts) have multiple recipes
				if not hasUse(recipeSkillID) and type(recipe.alternativeCosts) == 'table' then
					for k, altCost in ipairs(recipe.alternativeCosts) do
						for m, itemCost in ipairs(altCost.itemCosts) do
							if itemCost.id == item.id then
								addUse(recipeSkillID)
								break
							end
						end
						if hasUse(recipeSkillID) then
							break
						end
					end
				end
				if hasUse(recipeSkillID) then
					break
				end
			end
		end
	end

	-- Firemaking
	if not hasUse('melvorD:Firemaking') then
		for i, recipe in ipairs(SkillData.Firemaking.logs) do
			if recipe.logID == item.id then
				addUse('melvorD:Firemaking')
				break
			end
		end
	end

	-- Farming
	if not hasUse('melvorD:Farming') then
		for i, recipe in ipairs(SkillData.Farming.recipes) do
			if recipe.seedCost.id == item.id then
				addUse('melvorD:Farming')
				break
			end
		end
	end

	-- Agility
	if not hasUse('melvorD:Agility') and not Shared.tableIsEmpty(Agility.getObstaclesForItem(item.id)) then
		addUse('melvorD:Agility')
	end

	-- Summoning
	if not hasUse('melvorD:Summoning') then
		for i, recipe in ipairs(SkillData.Summoning.recipes) do
			-- Tablets & Non-shard items
			if recipe.productID == item.id or Shared.contains(recipe.nonShardItemCosts, item.id) then
				addUse('melvorD:Summoning')
				break
			else
				-- Shards
				for j, itemCost in ipairs(recipe.itemCosts) do
					if itemCost.id == item.id then
						addUse('melvorD:Summoning')
						break
					end
				end
			end
		end
	end

	-- Prayer
	if item.prayerPoints ~= nil then
		table.insert(categoryArray, '[[Category:Buriable Items]]')
		if not hasUse('melvorD:Prayer') then
			addUse('melvorD:Prayer')
		end
	elseif item.soulPoints ~= nil then
		table.insert(categoryArray, '[[Category:Releasable Items]]')
		if not hasUse('melvorD:Prayer') then
			addUse('melvorD:Prayer')
		end
	end

	-- Magic
	if not (hasUse('Magic') and hasUse('AltMagic')) then
		-- First check if the item its self is used in any spells
		local spellList = Magic.getSpellsUsingItem(item.id, true)
		for i, spell in ipairs(spellList) do
			local useKey = (spell.type == 'altMagic' and 'AltMagic' or 'Magic')
			if not hasUse(useKey) then
				addUse(useKey)
			end
		end
		-- Check if the item provides runes, if it does check where they are used also
		if item.providedRunes ~= nil then
			for i, rune in ipairs(item.providedRunes) do
				if hasUse('Magic') and hasUse('AltMagic') then
					break
				else
					local spellList = Magic.getSpellsUsingItem(rune.id, false)
					for j, spell in ipairs(spellList) do
						local useKey = (spell.type == 'altMagic' and 'AltMagic' or 'Magic')
						if not hasUse(useKey) then
							addUse(useKey)
						end
					end
				end
			end
		end
	end

	-- Other odds and ends:

	-- Skillcapes are tied to the appropriate skill
	-- Except Maximum Skillcape, which is tied to all skills. (And so is the Signet Ring)
	-- And combat skillcapes, since combat skills don't get special treatment
	if item.tier == 'Skillcape' then
		local ignoreCapes = {
			'melvorD:Attack_Skillcape',
			'melvorD:Strength_Skillcape',
			'melvorD:Defence_Skillcape',
			'melvorD:Hitpoints_Skillcape',
			'melvorF:Ranged_Skillcape',
			'melvorTotH:Superior_Attack_Skillcape',
			'melvorTotH:Superior_Strength_Skillcape',
			'melvorTotH:Superior_Defence_Skillcape',
			'melvorTotH:Superior_Hitpoints_Skillcape',
			'melvorTotH:Superior_Ranged_Skillcape',
		}
		local allCapes = {
			'melvorF:Max_Skillcape',
			'melvorF:Cape_of_Completion',
			'melvorTotH:Superior_Max_Skillcape',
			'melvorTotH:Superior_Cape_Of_Completion',
		}
		if Shared.contains(allCapes, item.id) then
			addUse('AllSkills')
		elseif Shared.contains({'melvorF:Magic_Skillcape', 'melvorTotH:Superior_Magic_Skillcape'}, item.id) then
			addUse('melvorD:Magic')
			addUse('AltMagic')
		elseif not Shared.contains(ignoreCapes, item.id) then
			local splitName = Shared.splitString(item.name, ' ')
			local skillName = (splitName[1] == 'Superior' and splitName[2]) or splitName[1]
			addUse(skillName)
		end
		table.insert(categoryArray, '[[Category:Skillcapes]]')
	end
	
	--Special note for Charge Stone of Rhaelyx
	if item.id == 'melvorD:Charge_Stone_of_Rhaelyx' then
		addUse('ChargeStone')
	end

	--Some items are needed to make shop purchases
	local shopArray = Shop.getItemCostArray(item.id)
	if not Shared.tableIsEmpty(shopArray) then
		addUse('Shop')
	end

	-- Township Tasks
	for _, task in ipairs(SkillData.Township.tasks) do
		if task.goals.items ~= nil then -- Skip tasks with no items
			if GameData.getEntityByID(task.goals.items, item.id) then
				addUse('TownshipTask')
				break
			end
		end
	end
	
	-- Generate result text
	local useArray = {}
	local prefix, delim = asList and '* ' or '', asList and '\r\n' or '<br/>'
	-- Always place 'All skills' use before any skills
	if hasUse('AllSkills') then
		table.insert(useArray, prefix .. (otherUseText.AllSkills or 'AllSkills'))
		otherUses.AllSkills = nil
	end
	for skillID, skillName in Shared.spairs(skillUses, function(t, a, b) return t[a] < t[b] end) do
		table.insert(useArray, prefix .. Icons.Icon({skillName, type='skill'}))
	end
	for use, _ in Shared.skpairs(otherUses) do
		table.insert(useArray, prefix .. (otherUseText[use] or use))
	end

	return table.concat(useArray, delim) .. (addCategories and table.concat(categoryArray, '') or '')
end

function p.getItemUses(frame)
	local itemName = frame.args ~= nil and frame.args[1] or frame
	local item = Items.getItem(itemName)
	local addCategories = false
	local asList = true
	if frame.args ~= nil then
		addCategories = frame.args.addCategories ~= nil and frame.args.addCategories ~= '' and frame.args.addCategories ~= 'false'
		asList = frame.args.addCategories == nil or frame.args.addCategories == '' or frame.args.addCategories == 'true'
	end
	if item == nil then
		return Shared.printError('No item named "' .. itemName .. '" exists in the data module')
	end

	return p._getItemUses(item, asList, addCategories)
end

function p._getItemUseTable(item)
	local tableData = {}

	-- Loop through all upgrades to find anything that can be upgraded using our source
	for i, upgrade in ipairs(GameData.rawData.itemUpgrades) do
		for j, itemCost in ipairs(upgrade.itemCosts) do
			if itemCost.id == item.id then
				local reqs = nil
				-- Potions do have upgrade requirements though
				local upgradeItem = Items.getItemByID(upgrade.upgradedItemID)
				if upgradeItem ~= nil and upgradeItem.charges ~= nil and upgradeItem.tier ~= nil then
					local levelUnlock = GameData.getEntityByProperty(SkillData.Herblore.masteryLevelUnlocks, 'descriptionID', upgradeItem.tier + 1)
					if levelUnlock ~= nil then
						reqs = Icons._MasteryReq(upgradeItem.name, levelUnlock.level)
					end
				end
				table.insert(tableData, {
					['skill'] = 'Upgrade',
					['useType'] = '[[Upgrading Items|' .. (upgrade.isDowngrade and 'Downgrade' or 'Upgrade') .. ']]',
					['reqs'] = reqs,
					['costs'] = Common.getCostString({ ["items"] = upgrade.itemCosts, ["currencies"] = upgrade.currencyCosts }),
					['productName'] = upgradeItem.name,
					['qty'] = (upgrade.quantity or 1)
				})
			end
		end
	end

	-- Cooking, Smithing, Fletching, Crafting, Runecrafting, Herblore
	-- All have somewhat consistent recipe data structures
	local recipeSkillIDs = {
		'melvorD:Cooking',
		'melvorD:Smithing',
		'melvorD:Fletching',
		'melvorD:Crafting',
		'melvorD:Runecrafting',
		'melvorD:Herblore'
	}
	for i, recipeSkillID in ipairs(recipeSkillIDs) do
		local skillName = Constants.getSkillName(recipeSkillID)
		local _, localSkillID = GameData.getLocalID(recipeSkillID)
			
		-- Iterate over all recipes for the current skill
		for j, recipe in ipairs(SkillData[localSkillID].recipes) do
			local costLists = {recipe.alternativeCosts or {}, {{["itemCosts"] = recipe.itemCosts}}}
			for k, costList in pairs(costLists) do
				for m, costDef in pairs(costList) do
					for n, itemCost in ipairs(costDef.itemCosts) do
						if itemCost.id == item.id then
							local recipeItemIDs = nil
							if recipeSkillID == 'melvorD:Herblore' then
								recipeItemIDs = recipe.potionIDs
							elseif recipeSkillID == 'melvorD:Cooking' then
								recipeItemIDs = {recipe.productID, recipe.perfectCookID}
							else
								recipeItemIDs = {recipe.productID}
							end
							for o, recipeItemID in ipairs(recipeItemIDs) do
								local recipeItem = Items.getItemByID(recipeItemID)
								if recipeItem ~= nil then
									local lvl, isAbyssal = Skills.getRecipeLevelRealm(recipeSkillID, recipe)
									local reqs = nil
									if recipeSkillID == 'melvorD:Cooking' then
										-- Cooking includes the required utility (fire, furnace, pot) as a special requirement
										local cookingCatIcon = {
											["melvorD:Fire"] = 'Normal Cooking Fire',
											["melvorD:Furnace"] = 'Basic Furnace',
											["melvorD:Pot"] = 'Basic Pot'
										}
										local categoryIconName, categoryName = cookingCatIcon[recipe.categoryID], nil
										local recipeCategory = GameData.getEntityByID(SkillData.Cooking.categories, recipe.categoryID)
										if recipeCategory ~= nil then
											categoryName = recipeCategory.modifierName or recipeCategory.name
										end
										if categoryIconName ~= nil and categoryName ~= nil then
											local utilReq = Icons.Icon({'Cooking', categoryName, section = 'Cooking Upgrades', img = categoryIconName, type = 'upgrade'})
											reqs = Icons._SkillReq(skillName, lvl, false, (isAbyssal and "melvorItA:Abyssal" or nil)) .. '<br>' .. utilReq
										end
									elseif recipeSkillID == 'melvorD:Herblore' then
										-- Herblore may also have a mastery requirement
										local levelUnlock = GameData.getEntityByProperty(SkillData.Herblore.masteryLevelUnlocks, 'descriptionID', recipeItem.tier + 1)
										if levelUnlock ~= nil and levelUnlock.level > 1 then
											local masteryReq = Icons._MasteryReq(recipeItem.name, levelUnlock.level)
											reqs = Icons._SkillReq(skillName, lvl, false, (isAbyssal and "melvorItA:Abyssal" or nil)) .. '<br>' .. masteryReq
											-- Slightly increase level per potion tier for proper ordering using data-sort-value
											lvl = lvl + levelUnlock.level * 0.01
										end
									end
									table.insert(tableData, {
										['skill'] = skillName,
										['lvl'] = lvl,
										['isAbyssal'] = isAbyssal,
										['useType'] = Icons.Icon({skillName, type='skill'}),
										['reqs'] = (reqs or Icons._SkillReq(skillName, lvl, false, (isAbyssal and "melvorItA:Abyssal" or nil))),
										['costs'] = Common.getCostString({ ["items"] = costDef.itemCosts, ["currencies"] = recipe.currencyCosts }),
										['productName'] = recipeItem.name,
										['qty'] = (recipe.baseQuantity or 1) * (costDef.quantityMultiplier or 1),
										['xp'] = recipe.baseAbyssalExperience or recipe.baseExperience
									})
								end
							end
							break
						end
					end
				end
			end
		end
	end

	-- Farming
	for i, recipe in ipairs(SkillData.Farming.recipes) do
		if recipe.seedCost.id == item.id then
			local category = GameData.getEntityByID(SkillData.Farming.categories, recipe.categoryID)
			local lvl, isAbyssal = Skills.getRecipeLevelRealm('melvorD:Farming', recipe)
			local product = Items.getItemByID(recipe.productId)

			table.insert(tableData, {
				['skill'] = 'Farming',
				['lvl'] = lvl,
				['isAbyssal'] = isAbyssal,
				['useType'] = Icons.Icon({'Farming', type='skill'}),
				['reqs'] = Icons._SkillReq('Farming', lvl, false, (isAbyssal and 'melvorItA:Abyssal' or nil)),
				['costs'] = { recipe.seedCost },
				['productName'] = product.name,
				['qty'] = (5 * category.harvestMultiplier),
				['xp'] = recipe.baseAbyssalExperience or recipe.baseExperience
			})
		end
	end

	-- Agility
	local obstacles = Agility.getObstaclesForItem(item.id)
	for i, obstacle in ipairs(obstacles) do
		table.insert(tableData, {
			['skill'] = 'Agility',
			['lvl'] = Skills.getRecipeLevelRealm('Agility', obstacle),
			['isAbyssal'] = obstacle.realm == 'melvorItA:Abyssal',
			['useType'] = Icons.Icon({'Agility', type='skill'}),
			['reqs'] = Agility._getObstacleRequirements(obstacle),
			['costs'] = Common.getCostString({ ["items"] = obstacle.itemCosts, ["currencies"] = obstacle.currencyCosts }),
			['productName'] = obstacle.name,
			['iconType'] = 'agility',
			['qty'] = 1
		})
	end

	-- Summoning
	for i, recipe in ipairs(SkillData.Summoning.recipes) do
		local recipeCost = 0
		if recipe.abyssalLevel ~= nil then
			recipeCost = SkillData.Summoning.recipeAPCost
		else
			recipeCost = SkillData.Summoning.recipeGPCost
		end
		local useShards = false
		local recipeItem = nil
		for j, itemCost in ipairs(recipe.itemCosts) do
			if itemCost.id == item.id then
				useShards = true
				break
			end
		end
		-- Non-shard items
		-- Familiar recipes may also have a currency cost without any non-shard
		-- items, so account for this with a dummy ID such that one iteration
		-- of the below loop always occurs
		local nonShardItemIDs = (Shared.tableIsEmpty(recipe.nonShardItemCosts) and {''} or recipe.nonShardItemCosts)
		for j, nonShardItemID in ipairs(nonShardItemIDs) do
			if useShards or nonShardItemID == item.id then
				-- Item is used in this particular synergy recipe
				if recipeItem == nil then
					recipeItem = Items.getItemByID(recipe.productID)
				end
				local nonShardItem = Items.getItemByID(nonShardItemID)
				local recipeCosts = Shared.clone(recipe.itemCosts)
				local xp = recipe.baseAbyssalExperience or recipe.baseExperience
				local lvl, isAbyssal = Skills.getRecipeLevelRealm('melvorD:Summoning', recipe)
				local recipeCosts = {}
				for k, itemCost in ipairs(recipe.itemCosts) do
					table.insert(recipeCosts, {id = itemCost.id, quantity = itemCost.quantity})
				end
				if nonShardItem ~= nil then
					-- Item ID can be nil for recipes such as Leprechaun or Cyclops
					local itemValue = math.max(nonShardItem.sellsFor, 20)
					local nonShardQty = math.max(1, math.ceil(recipeCost / itemValue))
					table.insert(recipeCosts, { id = nonShardItemID, quantity = nonShardQty })
				end
				table.insert(tableData, {
					['skill'] = 'Summoning',
					['lvl'] = lvl,
					['isAbyssal'] = isAbyssal,
					['useType'] = Icons.Icon({ 'Summoning', type='skill' }),
					['reqs'] = Icons._SkillReq('Summoning', lvl, false, (isAbyssal and 'melvorItA:Abyssal' or nil)),
					['costs'] = Common.getCostString({ ["items"] = recipeCosts, ["currencies"] = recipe.currencyCosts }),
					['productName'] = recipeItem.name,
					['qty'] = recipe.baseQuantity,
					['xp'] = xp
				})
			end
		end
	end

	--Handle shop purchases using Module:Shop
	local shopUses = Shop.getItemCostArray(item.id)
	for i, shopUse in ipairs(shopUses) do
		local purchase = shopUse.purchase

		table.insert(tableData, {
			['skill'] = 'Shop',
			['lvl'] = (not Shared.tableIsEmpty(purchase.purchaseRequirements) and purchase.purchaseRequirements[1].level or nil),
			['useType'] = Icons.Icon({'Shop'}),
			['reqs'] = Common.getRequirementString(purchase.purchaseRequirements),
			['costs'] = Common.getCostString(purchase.cost),
			['productName'] = Common.getPurchaseName(purchase),
			['iconType'] = Common.getPurchaseIconType(purchase),
			['qty'] = (purchase.qty or 1)
		})
	end

	--Finally build the table using what we've learned
	table.sort(tableData, function(a, b)
		local aLvl = a.lvl
		local bLvl = b.lvl
		if a.skill ~= b.skill then
			return a.skill < b.skill
		elseif type(aLvl) ~= type(bLvl) then
			return tostring(aLvl) < tostring(bLvl)
		elseif aLvl ~= bLvl then
			return aLvl < bLvl
		else
			return a.productName < b.productName
		end
	end)

	local useTable = p.buildUseTable(item, tableData)

	local spellUseTable = p._getSpellUseTable(item)
	if spellUseTable ~= nil and spellUseTable ~= '' then
		useTable = useTable .. '\r\n===' .. Icons.Icon({'Magic', type='skill', size=30}) .. '===\r\n' .. spellUseTable
	end
	if useTable == '' then
		return ''
	else
		return '==Uses==\r\n' .. useTable
	end
end

function p.buildUseTable(item, tableData)
	if Shared.tableIsEmpty(tableData) then return '' end

	local showRequirements = false
	local showXP = false

	for i, data in ipairs(tableData) do
		if not showRequirements and tableData[i].reqs ~= nil then showRequirements = true end
		if not showXP and tableData[i].xp ~= nil then showXP = true end
		if showRequirements and showXP then break end
	end

	local resultTable = mw.html.create('table')
	resultTable:addClass('wikitable stickyHeader sortable col-1-img')
	local tableHeader = resultTable:tag('tr'):addClass('headerRow-0')
		:tag('th'):wikitext('Product'):attr('colspan', 2)
		:tag('th'):wikitext('Used In')

	if showRequirements then tableHeader:tag('th'):wikitext('Requires') end
	if showXP then tableHeader:tag('th'):wikitext('Exp') end
	tableHeader:tag('th'):wikitext('Costs')

	for i, data in ipairs(tableData) do
		local recipeRow = resultTable:tag('tr')
			:tag('td'):wikitext(Icons.Icon({ data.productName, type=(data.iconType or 'item'), notext=true })):attr('data-sort-value', data.productName) -- Icon Image
			:tag('td')
				:wikitext((data.qty ~= nil and data.qty > 1 and ("'''" .. data.qty .. "x''' ") or '')) -- Icon Qty
				:wikitext(Icons.Icon({ data.productName, type=(data.iconType or 'item'), noicon=true })) -- Icon Text
			:tag('td'):wikitext(data.useType):attr('data-sort-value', (data.skill or nil)) -- Used In
		-- Requirements
		if showRequirements then
			if data.reqs ~= nil then
				recipeRow:tag('td'):wikitext(data.reqs):attr('data-sort-value', (data.lvl or 0))
			else
				recipeRow:tag('td'):wikitext('N/A'):addClass('table-na')
			end
		end
		-- Exp
		if showXP then
			if data.skill ~= nil and data.xp ~= nil then
				local iconClass = (data.isAbyssal and 'abyss-icon' or nil)
				local xpText = (data.isAbyssal and ' AXP' or ' XP')
				recipeRow:tag('td'):attr('data-sort-value', data.xp)
					:wikitext(Icons.Icon({ data.skill, notext=true, type='skill', class=iconClass }))
					:wikitext(' ' .. Num.formatnum(data.xp) .. xpText)
			else
				recipeRow:tag('td'):wikitext('N/A'):addClass('table-na'):attr('data-sort-value', 0)
			end
		end
		-- Costs
		local costsRow = recipeRow:tag('td')
		if type(data.costs) == 'table' then
			for i, mat in ipairs(data.costs) do
				if i > 1 then costsRow:tag('br') end
				local matItem = Items.getItemByID(mat.id)
				if matItem == nil then
					costsRow:wikitext(mat.quantity .. 'x ?????')
				else
					costsRow:wikitext(Icons.Icon({ matItem.name, type='item', qty=mat.quantity }))
				end
			end
		else
			costsRow:wikitext(data.costs)
		end
	end

	return tostring(resultTable)
end

function p.getItemUseTable(frame)
	local itemName = frame.args ~= nil and frame.args[1] or frame
	local item = Items.getItem(itemName)
	if item == nil then
		return Shared.printError('No item named "' .. itemName .. '" exists in the data module')
	end

	return p._getItemUseTable(item)
end

function p._getSpellUseTable(item)
	local spellList = Magic.getSpellsUsingItem(item.id, true)
	--Bail immediately if no spells are found
	if Shared.tableIsEmpty(spellList) then
		return ''
	end

	--Adding a check for if the Items column is needed
	local hasItems = false
	for i, spell in pairs(spellList) do
		if Magic._getSpellItems(spell) ~= '' then
			hasItems = true
			break
		end
	end

	local resultPart = {}
	table.insert(resultPart, '{|class="wikitable sortable"\r\n!colspan="2"|Spell')
	table.insert(resultPart, '!!Requires')
	table.insert(resultPart, '!!Type!!style="width:275px"|Description')
	table.insert(resultPart, '!!Runes')
	if hasItems then
		table.insert(resultPart, '!!Item Cost')
	end

	for i, spell in pairs(spellList) do
		local spellBook = Magic.getSpellBookFromSpell(spell)
		local rowPart = {}
		table.insert(rowPart, '\r\n|-\r\n|data-sort-value="'..spell.name..'"|')
		local iconType = Magic._getSpellIconType(spell)
		table.insert(rowPart, Icons.Icon({spell.name, type=iconType, notext=true}))
		table.insert(rowPart, '||'..Icons.Icon({spell.name, type=iconType, noicon=true}))
		table.insert(rowPart, '||data-sort-value="'..Skills.getRecipeLevel('melvorD:Summoning', spell)..'"|'..Magic._getSpellRequirements(spell))
		table.insert(rowPart, '||data-sort-value="'.. spellBook.id ..'"|')
		table.insert(rowPart, Magic.getSpellTypeLink(spellBook.id))
		table.insert(rowPart, '||'..Magic._getSpellStat(spell, 'description'))
		table.insert(rowPart, '||style="text-align:center"|')
		table.insert(rowPart, Magic._getSpellRunes(spell))
		if hasItems then
			table.insert(rowPart, '||')
			table.insert(rowPart, Magic._getSpellItems(spell))
		end
		table.insert(resultPart, table.concat(rowPart))
	end
	 --Add the table end and add the table to the result string
	table.insert(resultPart, '\r\n|}')
	return table.concat(resultPart)
end

function p.getSpellUseTable(frame)
	local itemName = frame.args ~= nil and frame.args[1] or frame
	local item = Items.getItem(itemName)
	if item == nil then
		return Shared.printError('No item named "' .. itemName .. '" exists in the data module')
	end

	return p._getSpellUseTable(item)
end

--[[
-- Uncomment this block and execute 'p.test()' within the debug console
-- to test after making changes
function p.test()
	local checkItems = {
		'Gold Bar',
		'Raw Shrimp',
		'Coal Ore',
		'Rune Platebody',
		'Arrow Shafts',
		'Garum Seeds',
		'Rune Essence',
		'Runite Bar',
		'Water Rune',
		'Steam Rune',
		'Controlled Heat Potion II',
		'Wolf',
		'Cyclops',
		'Leprechaun',
		'Redwood Logs',
		'Carrot Cake',
		'Carrot Cake (Perfect)',
		'Mantalyme Herb',
		'Carrot',
		'Topaz',
		'Rune Essence',
		'Infernal Claw',
		'Chapeau Noir',
		'Stardust',
		'Rope',
		'Ancient Ring of Mastery',
		'Mysterious Stone',
		'Charge Stone of Rhaelyx',
		'Mastery Token (Cooking)',
		'Gem Gloves',
		'Basic Bag',
		'Bird Nest',
		'Abyssal Stone',
		'Withered Ash',
		'Wrath Rune',
		'Magic Bones',
		'Bowstring',
		'Superior Max Skillcape',
		'Abyssal Coin Contract II',
		'Dark Summon Consumable II',
		'Abyssal Slayer Gear Upgrade Kit',
		'Topaz Bolts (Enchanted)',
		'Summoning Shard (Black)',
		'Dragon Javelin',
	}
	local checkFuncs = {
		--p.getItemUses,
		p.getItemUseTable
	}
	local errCount = 0
	for i, item in ipairs(checkItems) do
		local param = {args={item}}
		mw.log('=' .. item .. '=')
		for j, func in ipairs(checkFuncs) do
			local callSuccess, retVal = pcall(func, param)
			if not callSuccess then
				errCount = errCount + 1
				mw.log('Error with item "' .. item .. '": ' .. retVal)
			else
				mw.log(retVal)
			end
		end
	end
	if errCount == 0 then
		mw.log('Test successful')
	else
		mw.log('Test failed with ' .. errCount .. ' failures')
	end
end
--]]

return p