Module:Sandbox/User:Alsang: Difference between revisions
Jump to navigation
Jump to search
Content added Content deleted
(copy of module:Products) |
No edit summary |
||
Line 1: | Line 1: | ||
require('strict') |
|||
require('Module:Mw.html extension') |
require('Module:Mw.html extension') |
||
local |
local editbutton = require('Module:Edit button') |
||
local edit = editbutton("'''?''' (edit)") |
|||
local currency = require('Module:Currency').parse |
|||
local |
local hc = require('Module:Param Parse').has_content |
||
local |
local yn = require('Module:Yesno') |
||
local qty = require('Module:Quantity box')._main |
|||
local pcp = require('Module:Profession clickpic')._main |
|||
local currency = require('Module:Currency') |
|||
local album_xp_data = mw.loadData('Module:Experience/data').album |
|||
local lang = mw.language.getContentLanguage() |
|||
local tooltip = require('Module:Tooltip') |
|||
local p = {} |
local p = {} |
||
function p.main(frame) |
function p.main(frame) |
||
local args = frame:getParent().args |
|||
local albumxp = album_xp_data[tonumber(args.level)] |
|||
end |
|||
if albumxp then |
|||
local function recipe_sort(recipe_a, recipe_b) |
|||
albumxp = lang:formatNum(albumxp)..' xp' |
|||
-- Sort unknown levels to the end |
|||
if (recipe_a.level == nil) ~= (recipe_b.level == nil) then |
|||
return recipe_b.level == nil |
|||
end |
end |
||
--Checks if the recipe has been flagged as passive, defaults to no. |
|||
if recipe_a.level ~= nil then |
|||
local passive = yn(args.passive or 'no', false) |
|||
return recipe_a.level < recipe_b.level |
|||
local quant = 1 |
|||
if passive then |
|||
quant = 0.005 |
|||
end |
end |
||
-- Sort by name if same level |
|||
local function exparg(xp) |
|||
return recipe_a.output[1].name < recipe_b.output[1].name |
|||
if hc(xp) then |
|||
end |
|||
return qty(xp) |
|||
else |
|||
function p._main(args) |
|||
return qty(0)..edit..'[[Category:Needs experience info]]' |
|||
args = args or {} |
|||
local item = args[1] or mw.title.getCurrentTitle().text |
|||
local showPrices = yesno(args.showPrices) |
|||
local showValues = yesno(args.showValues) |
|||
local limit = tonumber(args.limit or 0) or 0 |
|||
if limit <= 0 then |
|||
limit = 500 |
|||
end |
|||
-- Query for data |
|||
local smw_data = mw.smw.ask{ |
|||
'[[Uses item::' .. item .. ']]', |
|||
'?Uses item', |
|||
'?Recipe JSON', |
|||
showValues and '?Value', |
|||
limit = limit |
|||
} |
|||
if not smw_data then |
|||
return ":''No products found. To force an update, click " |
|||
..purge('dml-'..mw.uri.anchorEncode(item), 'here', 'span') |
|||
..".''[[Category:Empty products lists]]" |
|||
end |
|||
-- Create a list of all recipes, grouped by output item (to keep them together in the sort) |
|||
local produced_items = {} |
|||
for _, product in ipairs(smw_data) do |
|||
local jsons = product['Recipe JSON'] |
|||
if type(jsons) == 'string' then |
|||
jsons = { jsons } |
|||
end |
end |
||
local parsed = {} |
|||
for _, json in ipairs(jsons) do |
|||
local json = mw.text.jsonDecode(json) |
|||
json.Value = product.Value |
|||
-- Filter out when this item isn't actually used (Will happen on pages with multiple recipes) |
|||
if Array.any(json.materials, function(mat) |
|||
return mat.name == item |
|||
end) then |
|||
table.insert(parsed, json) |
|||
end |
|||
end |
|||
table.sort(parsed, recipe_sort) |
|||
table.insert(produced_items, parsed) |
|||
end |
end |
||
local function levelarg(profession, level) |
|||
-- Sort by the smallest recipe in the group |
|||
if hc(profession) and hc(level) then |
|||
table.sort(produced_items, function(item1, item2) |
|||
return pcp(profession, level) |
|||
local first1 = item1[1] |
|||
else |
|||
local first2 = item2[1] |
|||
return edit..'[[Category:Needs level info]]' |
|||
if (first1 == nil) ~= (first2 == nil) then |
|||
return first2 == nil |
|||
end |
|||
if first1 == nil then |
|||
return false -- Both empty, equivalent |
|||
end |
|||
return recipe_sort(first1, first2) |
|||
end) |
|||
-- Flatten into a single list |
|||
local recipes = {} |
|||
for _, product in ipairs(produced_items) do |
|||
for _, json in ipairs(product) do |
|||
table.insert(recipes, json) |
|||
end |
end |
||
end |
end |
||
local function durationarg(duration) |
|||
-- Calculate shop prices |
|||
if hc(duration) then |
|||
return duration..' seconds' |
|||
local price_cache = {} |
|||
else |
|||
for _, recipe in ipairs(recipes) do |
|||
return edit..'[[Category:Needs duration info]]' |
|||
for _, item in ipairs(recipe.materials) do |
|||
-- Extract item name and quantity from the "item,#" format |
|||
-- Query for the shop buy price of the item |
|||
local shopPriceResult = price_cache[item.name] or mw.smw.ask{ |
|||
'[[Sold item::' .. item.name .. ']]', |
|||
'?Shop buy price' |
|||
} or {} |
|||
price_cache[item.name] = shopPriceResult |
|||
local shopPrice |
|||
if shopPriceResult[1] then |
|||
shopPrice = tonumber(shopPriceResult[1]['Shop buy price'] or 0) or 0 |
|||
end |
|||
-- Update the product with the total price |
|||
if shopPrice ~= nil then |
|||
item.price = shopPrice * item.quantity |
|||
end |
|||
end |
|||
end |
end |
||
end |
end |
||
local smw_properties = { |
|||
-- Create table |
|||
['Skill node name'] = args.name, |
|||
['Uses profession'] = args.profession, |
|||
['Profession level'] = args.level, |
|||
['Activity XP'] = args.xp, |
|||
['Activity album XP'] = album_xp_data[tonumber(args.level)], |
|||
['Activity KP'] = args.kp, |
|||
['Activity duration'] = args.duration, |
|||
['Activity coins'] = args.coins, |
|||
['Activity input'] = args.input, |
|||
['Activity container'] = args.container, |
|||
-- straight copy from Infobox Recipe, its good to have a standard format |
|||
['Activity JSON'] = mw.text.jsonEncode({ |
|||
xp = args.xp and tonumber(args.xp), |
|||
kp = args.kp and tonumber(args.kp), |
|||
duration = args.duration, |
|||
tool = args.tool, |
|||
materials = { |
|||
{ name = args.input, quantity = quant }, |
|||
{ name = args.container, quantity = quant }, |
|||
}, |
|||
profession = args.profession, |
|||
level = args.level and tonumber(args.level), |
|||
-- Make sure to update this when multiple outputs are supported |
|||
output = { |
|||
{ name = args.name, quantity = quant }, |
|||
} |
|||
}), |
|||
} |
|||
mw.smw.set(smw_properties) |
|||
local out = mw.html.create('table') |
local out = mw.html.create('table') |
||
:addClass('wikitable |
:addClass('wikitable') |
||
:tag('tr') |
:tag('tr') |
||
:tag('th') |
:tag('th') |
||
:attr{ colspan = ' |
:attr{ colspan = '2' } |
||
:wikitext( |
:wikitext(args.name) |
||
:done() |
:done() |
||
:tag('th'):wikitext('Level'):done() |
|||
:IF(showValues) |
|||
:tag('th'):wikitext('Value'):done() |
|||
:END() |
|||
:tag('th'):wikitext('Ingredients'):done() |
|||
:IF(showPrices) |
|||
:tag('th'):wikitext('Price'):done() |
|||
:END() |
|||
:done() |
:done() |
||
:tag('tr') |
|||
:tag('th') |
|||
for _, recipe in ipairs(recipes) do |
|||
:wikitext('Level required') |
|||
local row = out:tag('tr') |
|||
:done() |
|||
:tag('td') |
:tag('td') |
||
:wikitext(levelarg(args.profession, args.level)) |
|||
:css{ ['border-right'] = '0', ['padding-right'] = '0' } |
|||
:done() |
|||
:attr{ ['data-sort-value'] = recipe.output[1].name } |
|||
:done() |
|||
:wikitext(recipe.output[1].quantity .. ' ×') |
|||
:tag('tr') |
|||
:tag('th') |
|||
:IF(passive) |
|||
:wikitext('[[File:Passive small icon.png|20x20px|link=Passive Activity]] [['..args.profession..'|'..args.profession..' XP]] ') |
|||
:node(tooltip._span{ 'passive' }) |
|||
:node(tooltip._div{ name = 'passive', content = 'Passive activities give reduced experience when a players level is high enough to perform a new passive activity for that profession. The number shown here is the full experience.' }) |
|||
:ELSE() |
|||
:wikitext('[['..args.profession..'|'..args.profession..' XP]]') |
|||
:END() |
|||
:done() |
:done() |
||
:tag('td') |
:tag('td') |
||
:wikitext(exparg(args.xp)) |
|||
:addClass('plinkt-image no-border') |
|||
:done() |
|||
:css{ ['border-left'] = '0', ['padding-left'] = '0' } |
|||
:done() |
|||
:wikitext('[[File:' .. recipe.output[1].name .. '.png|link=' .. recipe.output[1].name .. '|30px]]') |
|||
:tag('tr') |
|||
:tag('th') |
|||
:wikitext('[[Album|Album XP]]') |
|||
:done() |
:done() |
||
:tag('td') |
:tag('td') |
||
: |
:wikitext(albumxp or 'Unknown') |
||
:done() |
|||
:wikitext('[[' .. recipe.output[1].name .. ']]') |
|||
:done() |
|||
:tag('tr') |
|||
:tag('th') |
|||
:wikitext('Duration') |
|||
:done() |
:done() |
||
:tag('td') |
:tag('td') |
||
:wikitext(durationarg(args.duration)) |
|||
:IF(recipe.profession) |
|||
:wikitext(('[[File:%s small icon.png|15px|link=%s]] %s'):format(recipe.profession, recipe.profession, recipe.level or 'Unknown')) |
|||
:ELSE() |
|||
:wikitext(('[[FileUnknown profession small icon.png|15px|link=Professions]] %s'):format(recipe.level or 'Unknown')) |
|||
:END() |
|||
:done() |
:done() |
||
:done() |
|||
if showValues then |
|||
if hc(args.respawn) then |
|||
out |
|||
row:tag('td') |
|||
:tag('tr') |
|||
:wikitext(currency(recipe.Value)) |
|||
:tag('th') |
|||
:wikitext('Respawn time') |
|||
:done() |
:done() |
||
:tag('td') |
|||
:wikitext(durationarg(args.respawn)) |
|||
row:tag('td') |
|||
:attr{ ['data-sort-value'] = '' } |
|||
:wikitext('Unknown') |
|||
:done() |
:done() |
||
end |
|||
end |
|||
local ingredients = row:tag('td') |
|||
:attr{ ['data-sort-value'] = table.concat(Array.map(recipe.materials, function(item) return item.name end), '\0') } |
|||
:tag('ul') |
|||
:css{ ['list-style'] = 'none', ['margin'] = '0', ['padding-left'] = '0' } |
|||
for _, item in ipairs(recipe.materials) do |
|||
ingredients:tag('li') |
|||
:wikitext(('%s × [[File:%s.png|link=%s|18px]] [[%s]]'):format(item.quantity, item.name, item.name, item.name)) |
|||
:done() |
:done() |
||
end |
|||
if showPrices then |
|||
local prices = row:tag('td') |
|||
:tag('ul') |
|||
:css{ ['list-style'] = 'none', ['margin'] = '0', ['padding-left'] = '0' } |
|||
for _, item in ipairs(recipe.materials) do |
|||
if item.price then |
|||
prices:tag('li'):wikitext(currency(item.price)):done() |
|||
else |
|||
prices:tag('li'):wikitext('Unknown'):done() |
|||
end |
|||
end |
|||
end |
|||
end |
end |
||
if hc(args.tool) then |
|||
out |
|||
:tag('tr') |
|||
:tag('th') |
|||
:wikitext('Required tool') |
|||
:done() |
|||
:tag('td') |
|||
:wikitext(args.tool) |
|||
:done() |
|||
:done() |
|||
end |
|||
if hc(args.input) then |
|||
out |
|||
:tag('tr') |
|||
:tag('th') |
|||
:wikitext('Required input') |
|||
:done() |
|||
:tag('td') |
|||
:wikitext(args.input) |
|||
:done() |
|||
:done() |
|||
end |
|||
if hc(args.container) then |
|||
out |
|||
:tag('tr') |
|||
:tag('th') |
|||
:wikitext('Required container') |
|||
:done() |
|||
:tag('td') |
|||
:wikitext(args.container) |
|||
:done() |
|||
:done() |
|||
end |
|||
if hc(args.coins) then |
|||
out |
|||
:tag('tr') |
|||
:tag('th') |
|||
:wikitext('Coins') |
|||
:done() |
|||
:tag('td') |
|||
:wikitext(currency.parse(args.coins)) |
|||
:done() |
|||
:done() |
|||
end |
|||
return out |
return out |
||
end |
end |
Revision as of 23:52, 15 December 2024
Module documentation
This documentation is transcluded from Module:Sandbox/User:Alsang/doc. [edit] [history] [purge]
This module does not have any documentation. Please consider adding documentation at Module:Sandbox/User:Alsang/doc. [edit]
Module:Sandbox/User:Alsang's function main is invoked by Template:Sandbox/User:Alsang.
Module:Sandbox/User:Alsang requires
.Module:Sandbox/User:Alsang loads data from Module:Experience/data.
require('strict')
require('Module:Mw.html extension')
local editbutton = require('Module:Edit button')
local edit = editbutton("'''?''' (edit)")
local hc = require('Module:Param Parse').has_content
local yn = require('Module:Yesno')
local qty = require('Module:Quantity box')._main
local pcp = require('Module:Profession clickpic')._main
local currency = require('Module:Currency')
local album_xp_data = mw.loadData('Module:Experience/data').album
local lang = mw.language.getContentLanguage()
local tooltip = require('Module:Tooltip')
local p = {}
function p.main(frame)
local args = frame:getParent().args
local albumxp = album_xp_data[tonumber(args.level)]
if albumxp then
albumxp = lang:formatNum(albumxp)..' xp'
end
--Checks if the recipe has been flagged as passive, defaults to no.
local passive = yn(args.passive or 'no', false)
local quant = 1
if passive then
quant = 0.005
end
local function exparg(xp)
if hc(xp) then
return qty(xp)
else
return qty(0)..edit..'[[Category:Needs experience info]]'
end
end
local function levelarg(profession, level)
if hc(profession) and hc(level) then
return pcp(profession, level)
else
return edit..'[[Category:Needs level info]]'
end
end
local function durationarg(duration)
if hc(duration) then
return duration..' seconds'
else
return edit..'[[Category:Needs duration info]]'
end
end
local smw_properties = {
['Skill node name'] = args.name,
['Uses profession'] = args.profession,
['Profession level'] = args.level,
['Activity XP'] = args.xp,
['Activity album XP'] = album_xp_data[tonumber(args.level)],
['Activity KP'] = args.kp,
['Activity duration'] = args.duration,
['Activity coins'] = args.coins,
['Activity input'] = args.input,
['Activity container'] = args.container,
-- straight copy from Infobox Recipe, its good to have a standard format
['Activity JSON'] = mw.text.jsonEncode({
xp = args.xp and tonumber(args.xp),
kp = args.kp and tonumber(args.kp),
duration = args.duration,
tool = args.tool,
materials = {
{ name = args.input, quantity = quant },
{ name = args.container, quantity = quant },
},
profession = args.profession,
level = args.level and tonumber(args.level),
-- Make sure to update this when multiple outputs are supported
output = {
{ name = args.name, quantity = quant },
}
}),
}
mw.smw.set(smw_properties)
local out = mw.html.create('table')
:addClass('wikitable')
:tag('tr')
:tag('th')
:attr{ colspan = '2' }
:wikitext(args.name)
:done()
:done()
:tag('tr')
:tag('th')
:wikitext('Level required')
:done()
:tag('td')
:wikitext(levelarg(args.profession, args.level))
:done()
:done()
:tag('tr')
:tag('th')
:IF(passive)
:wikitext('[[File:Passive small icon.png|20x20px|link=Passive Activity]] [['..args.profession..'|'..args.profession..' XP]] ')
:node(tooltip._span{ 'passive' })
:node(tooltip._div{ name = 'passive', content = 'Passive activities give reduced experience when a players level is high enough to perform a new passive activity for that profession. The number shown here is the full experience.' })
:ELSE()
:wikitext('[['..args.profession..'|'..args.profession..' XP]]')
:END()
:done()
:tag('td')
:wikitext(exparg(args.xp))
:done()
:done()
:tag('tr')
:tag('th')
:wikitext('[[Album|Album XP]]')
:done()
:tag('td')
:wikitext(albumxp or 'Unknown')
:done()
:done()
:tag('tr')
:tag('th')
:wikitext('Duration')
:done()
:tag('td')
:wikitext(durationarg(args.duration))
:done()
:done()
if hc(args.respawn) then
out
:tag('tr')
:tag('th')
:wikitext('Respawn time')
:done()
:tag('td')
:wikitext(durationarg(args.respawn))
:done()
:done()
end
if hc(args.tool) then
out
:tag('tr')
:tag('th')
:wikitext('Required tool')
:done()
:tag('td')
:wikitext(args.tool)
:done()
:done()
end
if hc(args.input) then
out
:tag('tr')
:tag('th')
:wikitext('Required input')
:done()
:tag('td')
:wikitext(args.input)
:done()
:done()
end
if hc(args.container) then
out
:tag('tr')
:tag('th')
:wikitext('Required container')
:done()
:tag('td')
:wikitext(args.container)
:done()
:done()
end
if hc(args.coins) then
out
:tag('tr')
:tag('th')
:wikitext('Coins')
:done()
:tag('td')
:wikitext(currency.parse(args.coins))
:done()
:done()
end
return out
end
return p