Compare commits

...

3 Commits

Author SHA1 Message Date
Gabriel Tofvesson
40cf649c16 Implement itemstack fetching 2024-10-12 07:35:45 +02:00
Gabriel Tofvesson
3a46e08f37 Allow lenient Container dimensions 2024-10-12 07:35:29 +02:00
Gabriel Tofvesson
9ac62319fd Implement stack group transfers 2024-10-12 07:34:57 +02:00
7 changed files with 171 additions and 4 deletions

View File

@ -4,6 +4,40 @@ local Element = require("gfx.element")
local Container = Prop.attach(Element:new(), Children)
function Container:getHeight()
if self:isStrict() then
return Element.getHeight(self)
end
local max = 0
for _,child in self:_iterateChildren() do
max = math.max(max, child:getY() + child:getHeight())
end
return max
end
function Container:getWidth()
if self:isStrict() then
return Element.getWidth(self)
end
local max = 0
for _,child in self:_iterateChildren() do
max = math.max(max, child:getX() + child:getWidth())
end
return max
end
function Container:isStrict()
return self.strict == true
end
function Container:setStrict(strict)
local needReload = (not not self.strict) ~= strict
self.strict = strict
if needReload then
self:_reload()
end
end
function Container:_reload()
Element._reload(self)

View File

@ -546,6 +546,21 @@ local PAGES = {
}
}
local function paddedButton(label, width, height, parentHeight, parentWidth, onClick, bgColor, x, y)
local labelElement = Text:new{ text = label, bgColor = bgColor }
return Padding:new{
x = (parentWidth + (x or 1)) % parentWidth,
y = (parentHeight + (y or 1)) % parentHeight,
top = math.ceil((height - labelElement:getHeight()) / 2),
bottom = math.floor((height - labelElement:getHeight()) / 2),
left = math.ceil((width - labelElement:getWidth()) / 2),
right = math.floor((width - labelElement:getWidth()) / 2),
element = labelElement,
bgColor = bgColor,
onClick = onClick
}
end
local stuffContainer = List:new{
[Orientation:getId()] = Orientation.VERTICAL,
[Children:getId()] = {
@ -558,6 +573,53 @@ local PAGES = {
end
}
local bottomBar = List:new{
[Orientation:getId()] = Orientation.HORIZONTAL,
[Children:getId()] = {
paddedButton(
"Fetch",
state.width / 2,
3,
state.height,
state.width,
function()
state:itemTransaction(function()
group:transferTo(state.node)
end)
state:setPage("Main")
return true
end,
colors.gray,
1,
-2
),
paddedButton(
"Back",
state.width / 2,
3,
state.height,
state.width,
function()
state:setPage("GROUP_DETAIL", group)
return true
end,
colors.gray,
1,
-2
)
}
}
-- Anchored to (1,1)
local windowWrapper = Container:new{
[Children:getId()] = {
stuffContainer,
}
}
-- Set up event management
local function updateDisplayState()
local dataRequestText = paddedRequestCount:findById("data_request")
local dataDividerPad = paddedRequestCount:findById("data_divider")

View File

@ -1,6 +1,7 @@
local ItemStack = require("storage.itemstack")
local Sentinel = require("storage.sentinel")
local Chest = {}
local Chest = Sentinel.tag({}, Sentinel.CHEST)
Chest.__index = Chest
-- Homogeneity allows chest scan to clone empty itemDetail slot to all empty slots in chest

View File

@ -1,5 +1,7 @@
local Chest = require("storage.chest")
local Storage = {}
local Sentinel = require("storage.sentinel")
local Storage = Sentinel.tag({}, Sentinel.STORAGE)
Storage.__index = Storage
function Storage.assumeHomogeneous(names)

View File

@ -1,4 +1,6 @@
local ItemGroup = {}
local Sentinel = require("storage.sentinel")
local ItemGroup = Sentinel.tag({}, Sentinel.ITEMGROUP)
ItemGroup.__index = ItemGroup
--- Create a group of managed itemstacks
@ -107,6 +109,52 @@ function ItemGroup:addStack(stack)
return false
end
function ItemGroup:transferTo(target, itemCount)
local targetGroup = nil
if Sentinel:is(target, Sentinel.CHEST) or Sentinel:is(target, Sentinel.STORAGE) then
local identity = self:_getIdentityStack()
local find = target:find(function(stack) return identity:canTransfer(stack) end)
if #find == 0 then
return itemCount == nil or itemCount == 0, 0
end
targetGroup = ItemGroup:from(find[1])
for i=2,#find do
targetGroup:_addStack(find[i])
end
elseif Sentinel:is(target, Sentinel.ITEMSTACK) then
targetGroup = ItemGroup:from(target)
elseif Sentinel:is(target, Sentinel.ITEMGROUP) then
targetGroup = target
end
if targetGroup == nil then
error("Unexpected transfer target for ItemGroup:transferTo")
end
local targetCap = 0
for _,stack in targetGroup:_iterateStacks() do
targetCap = targetCap + (stack:getMaxCount() - stack:getCount())
end
-- TODO: Not efficient
local transferMax = math.min(itemCount or targetCap, targetCap, self:getItemCount())
local transfer = 0
for _,stack in targetGroup:_iterateStacks() do
local targetCount = math.min(target:getCount() + transferMax - transfer, stack:getMaxCount())
for _,from in self:_iterateStacks() do
if stack:getCount() == targetCount then
goto continue
end
local _, xfer = from:transferTo(stack)
transfer = transfer + xfer
end
::continue::
end
return itemCount == nil or (itemCount == transfer), transfer
end
function ItemGroup:getItemCount()
if self:_getIdentityStack():isEmpty() then
return 0

View File

@ -1,6 +1,7 @@
local Inventory = require("storage.inventory")
local Sentinel = require("storage.sentinel")
local ItemStack = {}
local ItemStack = Sentinel.tag({}, Sentinel.ITEMSTACK)
ItemStack.__index = ItemStack
function ItemStack:fromDetail(inv, detail, slot)

19
storage/sentinel.lua Normal file
View File

@ -0,0 +1,19 @@
local TABLE_KEY = "!__SENTINEL"
local Sentinel = {
CHEST = "CHEST",
ITEMSTACK = "ITEMSTACK",
ITEMGROUP = "ITEMGROUP",
STORAGE = "STORAGE"
}
function Sentinel:tag(table, sentinel)
table[TABLE_KEY] = sentinel
return table
end
function Sentinel:is(table, sentinel)
return table[TABLE_KEY] == sentinel
end
return Sentinel