Module:StoreLine: Difference between revisions

From Discord Dungeons Wiki
Jump to navigation Jump to search
No edit summary
No edit summary
Tag: Reverted
Line 1: Line 1:
-- <pre>
local p = {}
local p = {}
local lang = mw.language.getContentLanguage()
 
local var = mw.ext.VariablesLua
local commas = require("Module:Addcommas")._add
local params = require('Module:Paramtest')
local params = require('Module:Paramtest')
local yesno = require('Module:Yesno')
local yesNo = require("Module:Yesno")
local commas = require("Module:Addcommas")
local purge = require("Module:Purge")._purge
 
local p2pIcon = '[[File:Member icon.png|frameless|link=Pay-to-play]]'
local f2pIcon = '[[File:Free-to-play icon.png|frameless|link=Free-to-play]]'


local ptitle = mw.title.getCurrentTitle()
function p.main(frame)
local ns = ptitle.nsText
    return p._main(frame:getParent().args)
local title = ptitle.fullText
end
local pgTitle = ptitle.text


function dump(o)
function dump(o)
Line 24: Line 27:
  end
  end


function p.main(frame)
function p._main(args)
     local args = frame:getParent().args
     local item, limit = params.defaults{
     -- Params and defaults
                            {args[1], mw.title.getCurrentTitle().text},
     local name, buyvalue, sellvalue, currency, level = params.defaults{
                            {args.Limit, 500}
         {args.name or args.Name, ''},
                        }
         {args.buy or args.Buy, ''},
    mw.log(string.format('Searching for shops that sell: %s', item))
         {args.sell or args.Sell, ''},
 
{args.currency or args.Currency, 'Gold'},
     -- Get parsed smw data
{args.level or args.Level, '?'},
     local data = p.getData(item, limit)
 
mw.log("data", dump(data), '\n\n\n')
 
    -- Create the header of the output
    local headerText = "<div class=\"seealso\">This list was created dynamically. For help, see [[Template:Store locations list/FAQ|the FAQ]]. "
        .. "To force an update of this list, click " .. purge('res', 'here', 'span') .. ".</div>\n"
    local restbl = mw.html.create('table')
    restbl:addClass('wikitable sortable align-center-3 align-center-4 align-center-5 align-center-6 align-center-7')
        :tag('tr')
            :tag('th'):wikitext('Seller'):done()
            :tag('th'):wikitext('Location'):done()
            :tag('th'):wikitext('Price<br/>sold at'):attr('data-sort-type', 'number'):done()
            :tag('th'):wikitext('Price<br/>bought at'):attr('data-sort-type', 'number'):done()
            :tag('th'):wikitext('Members?'):done()
         :done()
 
    -- Create the rows for the output table
    for _, shop in ipairs(data) do
mw.log("shop", dump(shop), '\n\n\n')
 
    for i, location in ipairs(shop.location) do
        restbl:tag('tr')
            :tag('td'):wikitext(shop.seller):done()
            :tag('td'):wikitext(location):done()
            :tag('td')
                :attr({['data-sort-value']=shop.sellSortValue})
                :wikitext(shop.sellvalue)
            :done()
            :tag('td')
                :attr({['data-sort-value']=shop.buySortValue})
                :wikitext(shop.buyvalue)
            :done()
            :tag('td'):wikitext(shop.members):done()
        :done()
end
    end
 
mw.log("restbl", tostring(restbl))
 
    return headerText .. tostring(restbl)
end
 
function p.getData(itemName, limit)
    -- Query smw
    local q = {
         '[[Sold item::'..itemName..']]',
        '?Sold by',
         '?Sold by.Store Is members only = Store Is members only',
        '?Sold by.Store location = Store location',
        '?Sold by.Has subobject.Store location=Store location subobj',
        '?Sold by.Is members only = Is members only',
        '?Sold by.NPC location = NPC location',
        '?Sold by.Has subobject.NPC location=NPC location subobj',
        '?Sold item image',
        '?Sold item',
        '?Sold item text',
        '?Store sell price',
        '?Store buy price',
        '?Store currency',
        '?Store notes',
        offset = 0,
        limit = limit or 500,
     }
     }
     local itemvalue = ''
     local t1 = os.clock()
     local smwname = params.default_to(args.smwname,name)
     local smwdata = mw.smw.ask(q)
     local displayname = params.default_to(args.displayname or args.DisplayName,name)
     local t2 = os.clock()
     local image = 'File:' .. params.default_to(args.image or args.Image, name .. '.png')
     local data = {}


    -- Lookup SMW value
     if smwdata == nil then
     if itemvalue == ''  then
         error('The item "' .. itemName .. '" is not sold in any shop, please check for typos[[Category:Empty store lists]]', 0)
         itemvalue = getSMWInfo(smwname) or ''
     end
     end
      
     mw.log(string.format('SMW (store locations list): entries: %d, time elapsed: %.3f ms.', #smwdata, (t2 - t1) * 1000))
    -- Check precached Module:DropsLine/itemData - gets GE alch so inaccurate
 
    -- if itemvalue == '' then
     table.sort(smwdata, function(a, b) return b['Store sell price'] < a['Store sell price'] end)
    --    local droppeditem_data = mw.loadData('Module:DropsLine/itemData')
 
    --    local cached_dropdata = droppeditem_data[name]
     -- Iterate through each shop instance of item
    --    if type(cached_dropdata) == 'table' and type(cached_dropdata[2]) == 'number' then
     for _, item in ipairs(smwdata) do
    --        itemvalue = cached_dropdata[2]/.6
        local members, location, region = {}, {}, {}
    --    end
    -- end
   
    local buymultiplier = var.var('BuyMultiplier', 1)
    local sellmultiplier = var.var('SellMultiplier', 1)
    local currency = var.var('Currency', currency)
    local namenotes = var.var('NameNotes', '')
    local useSmw = yesno(var.var('smw','yes'), true)
    local hideimage = yesno(var.var('hideImage','no'), false)
     local hidesell = yesno(var.var('hideSell','no'), false)
    local hidebuy = yesno(var.var('hideBuy','no'), false)
    local hidelevel = yesno(var.var('hideLevel','no'), false)
   
     buyvalue = commas._strip(buyvalue)
     sellvalue = commas._strip(sellvalue)
    levelvalue = commas._strip(level)
    local buy_smw = tonumber(buyvalue)
    local sell_smw = tonumber(sellvalue) or 1e10
    local level_smw = tonumber(levelvalue) or 1e10
    local buyCurrency = ''
    local sellCurrency = ''


    if buyvalue ~= 'N/A' then
        if type(item['Store Is members only']) == 'boolean' then
         if buyvalue == '' and itemvalue ~= '' then
            table.insert(members, item['Store Is members only'])
             buyvalue = math.floor(math.max(itemvalue*buymultiplier/1000,itemvalue*0.1))
        elseif type(item['Is members only']) == 'boolean' then
             buy_smw = buyvalue
            table.insert(members, item['Is members only'])
        end
       
         if type(item['Store location subobj']) == 'table' then
            for _, v in ipairs(item['Store location subobj']) do
                table.insert(location, v)
            end
        elseif type(item['Store location']) == 'string' then
             table.insert(location, item['Store location'])
        elseif type(item['NPC location subobj']) == 'table' then
            for _, v in ipairs(item['NPC location subobj']) do
                table.insert(location, v)
            end
        elseif type(item['NPC location']) == 'string' then
             table.insert(location, item['NPC location'])
         end
         end
         buyvalue = showCurrencyAmount(currency, buyvalue)
 
         item['Location'] = location
        -- item['Region'] = region
       
        if #members == 1 then
            item['Members'] = members[1]
        else
            item['Members'] = members
        end
 
mw.log(dump(item), '\n\n\n')
 
        -- Loop over smw return items
        -- item['Store location'] = 'x'
        -- item['Store Is members only'] = 'Yes'
        -- Process the item and add it to the final data table
        local dataline = p.processData(item, editbtn)
 
mw.log("dataline", dump(dataline), '\n\n\n')
 
 
        table.insert(data, dataline)
     end
     end
    if sellvalue ~= 'N/A' then
        if sellvalue == '' and itemvalue ~= '' then
            sellvalue = math.floor(math.max(itemvalue*sellmultiplier/1000,1))
            sell_smw = sellvalue
        end
        sellvalue = showCurrencyAmount(currency, sellvalue)


mw.log("sellvalue", sellvalue)
    return data
mw.log("sell_smw", sell_smw)
end
 
function p.processData(item, editbtn)
    local seller = item['Sold by'] or ""
    local namenotes = item['Store notes']
    if namenotes then
        seller = seller .. ' ' .. namenotes
     end
     end
      
      
     local ret = mw.html.create('tr'):css('text-align','center')
     local location = item['Location']
     if not hideimage then
    --local region = item['Region']
         ret:tag('td'):wikitext(mw.ustring.format('[[%s|link=%s]]', image, name))
 
     if type(location) == 'table' and #location == 0 then
         location = {editbtn}
    end
 
    if type(region) == 'table' and #region == 0 then
        region = {editbtn}
     end
     end
     -- no hideXXX parameter for the column that lists the actual items.
      
     ret:tag('td'):css('text-align','left'):wikitext(mw.ustring.format('[[%s|%s]]', name, displayname))
     local currency = item['Store currency']
     if not hidesell then
 
         if sellvalue == 'N/A' then
    local sellvalue = item['Store sell price'] or ''
             ret:tag('td'):attr('data-sort-value', 0):attr('class','table-na'):wikitext('<small>N/A</small>'):done()
    local sellSortValue = 0
        elseif sell_smw == 0 then
mw.log("sellvalue 1", sellvalue)
            ret:tag('td')
 
                :attr('data-sort-value', 0)
 
                 :wikitext('<small>Free</small>')
     if not(sellvalue == 1e10) then
             :done()
        sellvalue = tonumber(sellvalue)
         if sellvalue then
            sellSortValue = sellvalue
             sellvalue = commas(sellvalue)
mw.log("sellvalue", sellvalue)
            local currencyImg =  ''
            if(params.has_content(currencyImg)) then
                currencyImg = string.format('[[File:%s|link=%s]] ', currencyImg, currency)
                 sellvalue = currencyImg .. sellvalue
             end
         else
         else
             ret:tag('td')
             sellvalue = editbtn    -- If sellvalue can't be converted to a number it will default to the edit button
                :attr('data-sort-value', sell_smw)
                :wikitext(sellvalue)
            :done()
         end
         end
    else
        sellSortValue = sellvalue
        sellvalue = 'N/A'
     end
     end
     if not hidebuy then
   
        if buyvalue=='N/A' then
    local buyvalue = item['Store buy price'] or ''
             ret:tag('td'):attr('data-sort-value', 0):attr('class','table-na'):wikitext('<small>N/A</small>'):done()
    local buySortValue = 0
     if not(buyvalue == 'N/A') then
        buyvalue = tonumber(buyvalue)
        if buyvalue then
             buySortValue = buyvalue
            buyvalue = commas(buyvalue)
            local currencyImg = currencyImage(currency, buySortValue) or ''
            if(params.has_content(currencyImg)) then
                currencyImg = string.format('[[File:%s|link=%s]] ', currencyImg, currency)
                buyvalue = currencyImg .. buyvalue
            end
         else
         else
             ret:tag('td')
             buyvalue = editbtn -- If sellvalue can't be converted to a number it will default to the edit button
                    :attr('data-sort-value', buy_smw or 0)
                    :wikitext(buyvalue)
              :done()
         end
         end
     end
     end
 
   
if not hidelevel then
    -- members only?
        if levelvalue=='N/A' then
    local members = item['Members']
            ret:tag('td'):attr('data-sort-value', 0):attr('class','table-na'):wikitext('<small>N/A</small>'):done()
    if type(members) == 'table' then
        -- contains yes and/or no
        local hasYes, hasNo = false, false
        for _, value in ipairs(members) do
            if yesNo(value) then
                hasYes = true
            elseif not(yesNo(value)) then
                hasNo = true
            end
        end
        if hasYes and hasNo then
            members = f2pIcon.."/"..p2pIcon
        elseif hasYes then
            members = p2pIcon
        elseif hasNo then
            members = f2pIcon
         else
         else
             ret:tag('td')
             members = editbtn -- Unsupported type for yesNo, default to editbtn
                    :attr('data-sort-value', level_smw or 0)
                    :wikitext(levelvalue)
              :done()
         end
         end
    elseif yesNo(members) then
        members = p2pIcon
    elseif not(yesNo(members)) then
        members = f2pIcon
    else -- Unsupported type for yesNo, default to editbtn
        members = editbtn
     end
     end


     local i = 1
     return {
    while args['column' .. i] do
         seller = seller,
         if args['column' .. i]=='N/A' then
         location = location,
            ret:tag('td'):attr('data-sort-value', 0):attr('class','table-na'):wikitext('<small>N/A</small>'):done()
         region = region,
         else
         sellvalue = sellvalue,
        ret:tag('td'):wikitext(args['column' .. i]):done()
         sellSortValue = sellSortValue,
        end
         buyvalue = buyvalue,
        i = i+1
         buySortValue = buySortValue,
    end
         members = members
          
     }
   
    local onMain = ns == '' or ns == '(Main)'
    if onMain and useSmw then
         local smw = {}
         local smw_sub = {}
         local source = pgTitle
         smw['Sells item'] = name
         smw_sub['Sold by'] = source
        smw_sub['Sold item image'] = image
        smw_sub['Sold item'] = name
        smw_sub['Sold item text'] = displayname
        smw_sub['Store sell price'] = sell_smw --type = number for sorting purposes
        smw_sub['Store buy price'] = buy_smw or 'N/A'
        smw_sub['Store currency'] = currency
        smw_sub['Store notes'] = namenotes
        mw.smw.subobject(smw_sub) -- add item subobject to page
       
        mw.log("smw", dump(smw_sub))
       
        mw.smw.set(smw) -- add data to page
     end
    return tostring(ret)
end
end


local smwData = nil
function editbutton(title)
function getSMWInfo(item)
     local link = string.gsub(mw.title.getCurrentTitle():fullUrl("action=edit"), mw.title.getCurrentTitle().fullText, title)
    if smwData ~= nil then
    link = string.gsub(link, ' ', '_')
        return smwData
     link = string.format("<span class='plainlinks'>[%s '''?''' (edit)]</span>", link)
    end
     return link
     local smw = mw.smw.ask({
        '[['..item..']]',
        '?Value'
    })
     if smw and smw[1] then
        smwData = smw[1]['Value']
    else
        smwData = ''
    end
     return smwData
end
end


function showCurrencyAmount(currency, amount)
return string.format('%s [[%s]]', amount, currency)
    -- local image = currency and currency_image(currency, amount) or ''
    -- if image ~= '' and currency and tonumber(amount) then
    --    return string.format('[[File:%s|link=%s]] %s', image, currency, commas._add(amount))
    -- else
    -- end
end


return p
return p

Revision as of 13:50, 3 April 2022

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

-- <pre>
local p = {}

local commas = require("Module:Addcommas")._add
local params = require('Module:Paramtest')
local yesNo = require("Module:Yesno")
local purge = require("Module:Purge")._purge

local p2pIcon = '[[File:Member icon.png|frameless|link=Pay-to-play]]'
local f2pIcon = '[[File:Free-to-play icon.png|frameless|link=Free-to-play]]'

function p.main(frame)
    return p._main(frame:getParent().args)
end

function dump(o)
	if type(o) == 'table' then
	   local s = '{ '
	   for k,v in pairs(o) do
		  if type(k) ~= 'number' then k = '"'..k..'"' end
		  s = s .. '['..k..'] = ' .. dump(v) .. ','
	   end
	   return s .. '} '
	else
	   return tostring(o)
	end
 end

function p._main(args)
    local item, limit = params.defaults{
                            {args[1], mw.title.getCurrentTitle().text},
                            {args.Limit, 500}
                        }
    mw.log(string.format('Searching for shops that sell: %s', item))

    -- Get parsed smw data
    local data = p.getData(item, limit)

	mw.log("data", dump(data), '\n\n\n')

    -- Create the header of the output
    local headerText = "<div class=\"seealso\">This list was created dynamically. For help, see [[Template:Store locations list/FAQ|the FAQ]]. "
        .. "To force an update of this list, click " .. purge('res', 'here', 'span') .. ".</div>\n"
    local restbl = mw.html.create('table')
    restbl:addClass('wikitable sortable align-center-3 align-center-4 align-center-5 align-center-6 align-center-7')
        :tag('tr')
            :tag('th'):wikitext('Seller'):done()
            :tag('th'):wikitext('Location'):done()
            :tag('th'):wikitext('Price<br/>sold at'):attr('data-sort-type', 'number'):done()
            :tag('th'):wikitext('Price<br/>bought at'):attr('data-sort-type', 'number'):done()
            :tag('th'):wikitext('Members?'):done()
        :done()

    -- Create the rows for the output table
    for _, shop in ipairs(data) do
		mw.log("shop", dump(shop), '\n\n\n')

    	for i, location in ipairs(shop.location) do
	        restbl:tag('tr')
	            :tag('td'):wikitext(shop.seller):done()
	            :tag('td'):wikitext(location):done()
	            :tag('td')
	                :attr({['data-sort-value']=shop.sellSortValue})
	                :wikitext(shop.sellvalue)
	            :done()
	            :tag('td')
	                :attr({['data-sort-value']=shop.buySortValue})
	                :wikitext(shop.buyvalue)
	            :done()
	            :tag('td'):wikitext(shop.members):done()
	        :done()
		end
    end

	mw.log("restbl", tostring(restbl))

    return headerText .. tostring(restbl)
end

function p.getData(itemName, limit)
    -- Query smw
    local q = {
        '[[Sold item::'..itemName..']]',
        '?Sold by',
        '?Sold by.Store Is members only = Store Is members only',
        '?Sold by.Store location = Store location',
        '?Sold by.Has subobject.Store location=Store location subobj',
        '?Sold by.Is members only = Is members only',
        '?Sold by.NPC location = NPC location',
        '?Sold by.Has subobject.NPC location=NPC location subobj',
        '?Sold item image',
        '?Sold item',
        '?Sold item text',
        '?Store sell price',
        '?Store buy price',
        '?Store currency',
        '?Store notes',
        offset = 0,
        limit = limit or 500,
    }
    local t1 = os.clock()
    local smwdata = mw.smw.ask(q)
    local t2 = os.clock()
    local data = {}

    if smwdata == nil then
        error('The item "' .. itemName .. '" is not sold in any shop, please check for typos[[Category:Empty store lists]]', 0)
    end
    mw.log(string.format('SMW (store locations list): entries: %d, time elapsed: %.3f ms.', #smwdata, (t2 - t1) * 1000))

    table.sort(smwdata, function(a, b) return b['Store sell price'] < a['Store sell price'] end)

    -- Iterate through each shop instance of item
    for _, item in ipairs(smwdata) do
        local members, location, region = {}, {}, {}

        if type(item['Store Is members only']) == 'boolean' then
            table.insert(members, item['Store Is members only'])
        elseif type(item['Is members only']) == 'boolean' then
            table.insert(members, item['Is members only'])
        end
        
        if type(item['Store location subobj']) == 'table' then
            for _, v in ipairs(item['Store location subobj']) do
                table.insert(location, v)
            end
        elseif type(item['Store location']) == 'string' then
            table.insert(location, item['Store location'])
        elseif type(item['NPC location subobj']) == 'table' then
            for _, v in ipairs(item['NPC location subobj']) do
                table.insert(location, v)
            end
        elseif type(item['NPC location']) == 'string' then
            table.insert(location, item['NPC location'])
        end

        item['Location'] = location
        -- item['Region'] = region
        
        if #members == 1 then
            item['Members'] = members[1]
        else
            item['Members'] = members
        end

		mw.log(dump(item), '\n\n\n')

        -- Loop over smw return items
        -- item['Store location'] = 'x'
        -- item['Store Is members only'] = 'Yes'
        -- Process the item and add it to the final data table
        local dataline = p.processData(item, editbtn)

		mw.log("dataline", dump(dataline), '\n\n\n')


        table.insert(data, dataline)
    end

    return data
end

function p.processData(item, editbtn)
    local seller = item['Sold by'] or ""
    local namenotes = item['Store notes']
    if namenotes then
        seller = seller .. ' ' .. namenotes
    end
    
    local location = item['Location']
    --local region = item['Region']

    if type(location) == 'table' and #location == 0 then
        location = {editbtn}
    end

    if type(region) == 'table' and #region == 0 then
        region = {editbtn}
    end
    
    local currency = item['Store currency']

    local sellvalue = item['Store sell price'] or ''
    local sellSortValue = 0
	mw.log("sellvalue 1", sellvalue)


    if not(sellvalue == 1e10) then
        sellvalue = tonumber(sellvalue)
        if sellvalue then
            sellSortValue = sellvalue
            sellvalue = commas(sellvalue)
			mw.log("sellvalue", sellvalue)
            local currencyImg =  ''
            if(params.has_content(currencyImg)) then
                currencyImg = string.format('[[File:%s|link=%s]] ', currencyImg, currency)
                sellvalue = currencyImg .. sellvalue
            end
        else
            sellvalue = editbtn    -- If sellvalue can't be converted to a number it will default to the edit button
        end
    else
        sellSortValue = sellvalue
        sellvalue = 'N/A'
    end
    
    local buyvalue = item['Store buy price'] or ''
    local buySortValue = 0
    if not(buyvalue == 'N/A') then
        buyvalue = tonumber(buyvalue)
        if buyvalue then
            buySortValue = buyvalue
            buyvalue = commas(buyvalue)
            local currencyImg = currencyImage(currency, buySortValue) or ''
            if(params.has_content(currencyImg)) then
                currencyImg = string.format('[[File:%s|link=%s]] ', currencyImg, currency)
                buyvalue = currencyImg .. buyvalue
            end
        else
            buyvalue = editbtn -- If sellvalue can't be converted to a number it will default to the edit button
        end
    end
    
    -- members only?
    local members = item['Members']
    if type(members) == 'table' then
        -- contains yes and/or no
        local hasYes, hasNo = false, false
        for _, value in ipairs(members) do
            if yesNo(value) then
                hasYes = true
            elseif not(yesNo(value)) then
                hasNo = true
            end
        end
        if hasYes and hasNo then
            members = f2pIcon.."/"..p2pIcon
        elseif hasYes then
            members = p2pIcon
        elseif hasNo then
            members = f2pIcon
        else
            members = editbtn -- Unsupported type for yesNo, default to editbtn
        end
    elseif yesNo(members) then
        members = p2pIcon
    elseif not(yesNo(members)) then
        members = f2pIcon
    else -- Unsupported type for yesNo, default to editbtn
        members = editbtn
    end

    return {
        seller = seller,
        location = location,
        region = region,
        sellvalue = sellvalue,
        sellSortValue = sellSortValue,
        buyvalue = buyvalue,
        buySortValue = buySortValue,
        members = members
    }
end

function editbutton(title)
    local link = string.gsub(mw.title.getCurrentTitle():fullUrl("action=edit"), mw.title.getCurrentTitle().fullText, title)
    link = string.gsub(link, ' ', '_')
    link = string.format("<span class='plainlinks'>[%s '''?''' (edit)]</span>", link)
    return link
end


return p