milo: item rewrite + ability to remove nbts from recipes

This commit is contained in:
kepler155c@gmail.com
2018-12-31 11:39:24 -05:00
parent b726e5011f
commit 9cd480f188
5 changed files with 209 additions and 211 deletions

View File

@@ -131,16 +131,28 @@ local function turtleCraft(recipe, storage, request, count)
if storage:export(storage.turtleInventory, k, count, item) ~= count then
request.status = 'unknown error'
request.statusCode = Craft.STATUS_ERROR
_debug(item)
return
end
end
turtle.select(1)
if turtle.craft() then
request.crafted = request.crafted + count * recipe.count
request.status = 'crafted'
request.statusCode = Craft.STATUS_SUCCESS
local l = storage.turtleInventory.adapter.list()
local crafted = l[1]
if recipe.result ~= itemDB:makeKey(crafted) then
_debug('expected: ' .. recipe.result)
_debug('got: ' .. itemDB:makeKey(crafted))
request.aborted = true
request.status = 'Failed to craft: ' .. recipe.result
request.statusCode = Craft.STATUS_ERROR
else
request.crafted = request.crafted + count * recipe.count
request.status = 'crafted'
request.statusCode = Craft.STATUS_SUCCESS
end
else
_debug('just failed')
request.status = 'Failed to craft'
request.statusCode = Craft.STATUS_ERROR
end
@@ -174,20 +186,35 @@ function Craft.craftRecipe(recipe, count, storage, origItem)
return Craft.craftRecipeInternal(recipe, count, storage, origItem)
end
local function adjustCounts(recipe, count, ingredients)
local function adjustCounts(recipe, count, ingredients, storage)
-- decrement ingredients used
for key,icount in pairs(Craft.sumIngredients(recipe)) do
ingredients[key].count = ingredients[key].count - (icount * count)
if ingredients[key].count == 0 and not storage.cache[key] then
--
elseif not storage.cache[key] then
_debug({ key, ingredients[key].count, 'nocache' })
elseif storage.cache[key].count ~= ingredients[key].count then
_debug({ key, ingredients[key].count, storage.cache[key].count })
end
end
-- increment crafted
local result = ingredients[recipe.result]
result.count = result.count + (count * recipe.count)
end
function Craft.craftRecipeInternal(recipe, count, storage, origItem)
local request = origItem.ingredients[recipe.result]
if request.aborted then
_debug('aborted')
return 0
end
if origItem.pending[recipe.result] then
request.status = 'processing'
request.statusCode = Craft.STATUS_INFO
@@ -234,7 +261,7 @@ function Craft.craftRecipeInternal(recipe, count, storage, origItem)
end
end
end
_G._p = origItem.ingredients
local crafted = 0
while canCraft > 0 do
local batch = math.min(canCraft, maxCount)
@@ -248,7 +275,7 @@ function Craft.craftRecipeInternal(recipe, count, storage, origItem)
break
end
adjustCounts(recipe, batch, origItem.ingredients)
adjustCounts(recipe, batch, origItem.ingredients, storage)
crafted = crafted + batch
canCraft = canCraft - maxCount

View File

@@ -234,6 +234,17 @@ function Milo:eject(item, count)
return count
end
function Milo:updateRecipe(result, recipe)
-- save the recipe
if recipe then
recipe = Util.shallowCopy(recipe)
recipe.result = nil
end
self.context.userRecipes[result] = recipe
Util.writeTable(Craft.USER_RECIPES, self.context.userRecipes)
Craft.loadRecipes()
end
function Milo:saveMachineRecipe(recipe, result, machine)
local key = Milo:uniqueKey(result)

View File

@@ -26,9 +26,14 @@ local craftPage = UI.Page {
x = 6, y = 4,
value = 'Eject',
},
eject = UI.Checkbox {
eject = UI.Chooser {
x = 15, y = 4, width = 7,
value = true,
nochoice = 'No',
choices = {
{ name = 'Yes', value = true },
{ name = 'No', value = false },
},
},
},
resources = UI.Window {

View File

@@ -6,18 +6,14 @@ local UI = require('ui')
local Util = require('util')
local colors = _G.colors
local device = _G.device
local context = Milo:getContext()
local page = UI.Page {
titleBar = UI.TitleBar {
title = 'Limit Resource',
previousPage = true,
event = 'form_cancel',
},
local manageTab = UI.Window {
tabTitle = 'Manage',
form = UI.Form {
x = 1, y = 2, height = 10, ex = -1,
x = 1, ex = -1, ey = -1,
--manualControls = true,
[1] = UI.TextEntry {
formLabel = 'Name', formKey = 'displayName', help = 'Override display name',
shadowText = 'Display name',
@@ -42,100 +38,80 @@ local page = UI.Page {
formLabel = 'Ignore NBT', formKey = 'ignoreNbtHash',
help = 'Ignore NBT of item',
},
machineButton = UI.Button {
x = 2, y = -2, width = 10,
formLabel = 'Machine',
event = 'select_machine',
text = 'Assign',
},
infoButton = UI.Button {
x = 2, y = -2,
event = 'show_info',
text = 'Info',
},
resetButton = UI.Button {
x = 9, y = -2,
event = 'reset',
text = 'Reset',
help = 'Clear recipe and all settings',
},
},
rsControl = UI.SlideOut {
backgroundColor = colors.cyan,
titleBar = UI.TitleBar {
title = "Redstone Control",
},
form = UI.Form {
y = 2,
[1] = UI.Chooser {
width = 7,
formLabel = 'RS Control', formKey = 'rsControl',
nochoice = 'No',
choices = {
{ name = 'Yes', value = true },
{ name = 'No', value = false },
},
help = 'Control via redstone'
},
[2] = UI.Chooser {
width = 25,
formLabel = 'RS Device', formKey = 'rsDevice',
--choices = devices,
help = 'Redstone Device'
},
[3] = UI.Chooser {
width = 10,
formLabel = 'RS Side', formKey = 'rsSide',
--nochoice = 'No',
choices = {
{ name = 'up', value = 'up' },
{ name = 'down', value = 'down' },
{ name = 'east', value = 'east' },
{ name = 'north', value = 'north' },
{ name = 'west', value = 'west' },
{ name = 'south', value = 'south' },
},
help = 'Output side'
},
}
local machinesTab = UI.Window {
tabTitle = 'Machine',
backgroundColor = colors.cyan,
grid = UI.ScrollingGrid {
x = 2, ex = -2, y = 2, ey = -2,
disableHeader = true,
columns = {
{ heading = 'Name', key = 'displayName'},
},
sortColumn = 'displayName',
help = 'Double-click to set machine',
},
machines = UI.SlideOut {
backgroundColor = colors.cyan,
titleBar = UI.TitleBar {
title = 'Select Machine',
event = 'cancel_machine',
}
local recipeTab = UI.Window {
tabTitle = 'Recipe',
backgroundColor = colors.cyan,
grid = UI.ScrollingGrid {
x = 2, ex = -2, y = 2, ey = -4,
disableHeader = true,
columns = {
{ heading = 'Slot', key = 'slot', width = 2 },
{ heading = 'Key', key = 'key' },
},
grid = UI.ScrollingGrid {
y = 2, ey = -5,
disableHeader = true,
columns = {
{ heading = 'Name', key = 'displayName'},
},
sortColumn = 'displayName',
},
button1 = UI.Button {
x = -14, y = -3,
text = 'Ok', event = 'set_machine',
},
button2 = UI.Button {
x = -9, y = -3,
text = 'Cancel', event = 'cancel_machine',
},
statusBar = UI.StatusBar { values = 'Enter or double click to select' },
sortColumn = 'slot',
},
info = UI.SlideOut {
titleBar = UI.TitleBar {
title = "Information",
},
textArea = UI.TextArea {
x = 2, ex = -2, y = 3, ey = -4,
backgroundColor = colors.black,
},
cancel = UI.Button {
ex = -2, y = -2, width = 6,
text = 'Okay',
event = 'hide_info',
},
ignoreNBT = UI.Button {
x = -13, y = -2,
text = 'Ignore NBT', event = 'ignore_nbt',
},
}
local infoTab = UI.Window {
tabTitle = 'Info',
backgroundColor = colors.cyan,
textArea = UI.TextArea {
x = 2, ex = -2, y = 2,
},
}
local resetTab = UI.Window {
tabTitle = 'Reset',
backgroundColor = colors.cyan,
textArea = UI.TextArea {
y = 2, ey = 6,
textColor = colors.yellow,
value = [[ Warning!
This will clear all setting,
recipe, and machine for this item.]]
},
resetButton = UI.Button {
x = 17, y = 7,
event = 'reset',
text = 'Reset',
help = 'Clear recipe and all settings',
},
}
local page = UI.Page {
titleBar = UI.TitleBar {
title = 'Item settings',
previousPage = true,
},
tabs = UI.Tabs {
y = 2, ey = -2,
[1] = manageTab,
[2] = recipeTab,
[3] = machinesTab,
[4] = infoTab,
[5] = resetTab,
},
statusBar = UI.StatusBar { },
notification = UI.Notification { },
@@ -146,17 +122,19 @@ function page:enable(item)
self.item = Util.shallowCopy(item)
self.res = item.resource or { }
self.res.displayName = self.item.displayName
self.form:setValues(self.res)
self.titleBar.title = item.displayName or item.name
manageTab.form:setValues(self.res)
local machine = Craft.machineLookup[self.item.key]
self.form.machineButton.inactive = not machine
if machine then
self:filterMachines(machine)
end
self.tabs:selectTab(manageTab)
self.tabs:setActive(machinesTab, machine)
self.tabs:setActive(recipeTab, Craft.findRecipe(item))
UI.Page.enable(self)
self:focusFirst()
end
function page:filterMachines(machine)
@@ -165,65 +143,105 @@ function page:filterMachines(machine)
return node.adapter and node.adapter.online and node.adapter.pushItems
end
end)
self.machines.grid:setValues(t)
self.machines.grid:setSelected('name', machine)
machinesTab.grid:setValues(t)
machinesTab.grid:setSelected('name', machine)
end
function page.machines.grid:getDisplayValues(row)
function machinesTab.grid:getDisplayValues(row)
row = Util.shallowCopy(row)
row.displayName = row.displayName or row.name
return row
end
function page.machines.grid:getRowTextColor(row, selected)
function machinesTab.grid:getRowTextColor(row, selected)
if row.name == Craft.machineLookup[page.item.key] then
return colors.yellow
end
return UI.Grid:getRowTextColor(row, selected)
end
function page.rsControl:enable()
local devices = self.form[2].choices
Util.clear(devices)
for _,dev in pairs(device) do
if dev.setOutput then
table.insert(devices, { name = dev.name, value = dev.name })
end
end
function machinesTab:eventHandler(event)
if event.type == 'grid_select' then
Craft.machineLookup[page.item.key] = event.selected.name
Util.writeTable(Craft.MACHINE_LOOKUP, Craft.machineLookup)
if Util.size(devices) == 0 then
table.insert(devices, { name = 'None found', values = '' })
end
self.grid:draw()
page.notification:info('Machine saved')
UI.SlideOut.enable(self)
return true
end
end
function page.rsControl:eventHandler(event)
if event.type == 'form_cancel' then
self:hide()
elseif event.type == 'form_complete' then
self:hide()
else
return UI.SlideOut.eventHandler(self, event)
function recipeTab:enable()
self.recipe = Craft.findRecipe(page.item)
local t = { }
if self.recipe then
for k, v in pairs(self.recipe.ingredients) do
table.insert(t, {
slot = k,
key = v,
})
end
end
return true
self.grid:setValues(t)
UI.Window.enable(self)
end
function recipeTab:eventHandler(event)
if event.type == 'ignore_nbt' then
local selected = self.grid:getSelected()
local item = itemDB:splitKey(selected.key)
item.nbtHash = nil
selected.key = itemDB:makeKey(item)
self.grid:draw()
self.recipe.ingredients = { }
for _, v in pairs(self.grid.values) do
self.recipe.ingredients[v.slot] = v.key
end
Milo:updateRecipe(self.recipe.result, self.recipe)
page.notification:info('Recipe updated')
return true
end
end
function infoTab:draw()
local item = page.item
local value =
string.format('%s%s%s\n%s\n',
Ansi.orange, item.displayName, Ansi.reset,
item.name)
if item.nbtHash then
value = value .. item.nbtHash .. '\n'
end
value = value .. string.format('\n%sDamage:%s %s',
Ansi.yellow, Ansi.reset, item.damage)
if item.maxDamage and item.maxDamage > 0 then
value = value .. string.format(' (max: %s)', item.maxDamage)
end
if item.maxCount then
value = value .. string.format('\n%sStack Size: %s%s',
Ansi.yellow, Ansi.reset, item.maxCount)
end
self.textArea.value = value
UI.Window.draw(self)
end
function page:eventHandler(event)
if event.type == 'form_cancel' then
UI:setPreviousPage()
elseif event.type == 'show_rs' then
self.rsControl:show()
elseif event.type == 'select_machine' then
self.machines:show()
elseif event.type == 'reset' then
if context.userRecipes[self.item.key] then
context.userRecipes[self.item.key] = nil
Util.writeTable(Craft.USER_RECIPES, context.userRecipes)
Craft.loadRecipes()
Milo:updateRecipe(self.item.key, nil)
end
if context.resources[self.item.key] then
@@ -238,47 +256,8 @@ function page:eventHandler(event)
UI:setPreviousPage()
elseif event.type == 'grid_select' then
Craft.machineLookup[self.item.key] = event.selected.name
self.machines.grid:draw()
elseif event.type == 'set_machine' then
local machine = self.machines.grid:getSelected()
if machine then
Util.writeTable(Craft.MACHINE_LOOKUP, Craft.machineLookup)
end
self.machines:hide()
elseif event.type == 'cancel_machine' then
self.machines:hide()
elseif event.type == 'show_info' then
local value =
string.format('%s%s%s\n%s\n',
Ansi.orange, self.item.displayName, Ansi.reset,
self.item.name)
if self.item.nbtHash then
value = value .. self.item.nbtHash .. '\n'
end
value = value .. string.format('\n%sDamage:%s %s',
Ansi.yellow, Ansi.reset, self.item.damage)
if self.item.maxDamage and self.item.maxDamage > 0 then
value = value .. string.format(' (max: %s)', self.item.maxDamage)
end
if self.item.maxCount then
value = value .. string.format('\n%sStack Size: %s%s',
Ansi.yellow, Ansi.reset, self.item.maxCount)
end
self.info.textArea.value = value
self.info:show()
elseif event.type == 'hide_info' then
self.info:hide()
elseif event.type == 'tab_activate' then
event.activated:focusFirst()
elseif event.type == 'form_invalid' then
self.notification:error(event.message)
@@ -290,7 +269,7 @@ function page:eventHandler(event)
elseif event.type == 'form_complete' then
local item = self.item
if self.form:save() then
if manageTab.form:save() then
if self.res.displayName ~= self.origItem.displayName then
self.origItem.displayName = self.res.displayName
itemDB:add(self.origItem)

View File

@@ -9,7 +9,7 @@ local turtle = _G.turtle
local context = Milo:getContext()
local function learnRecipe(ignoreNBT)
local function learnRecipe()
local ingredients = Milo:getTurtleInventory()
if not ingredients then
@@ -83,10 +83,6 @@ local function learnRecipe(ignoreNBT)
newRecipe.count = recipe.count
if ignoreNBT then
recipe.nbtHash = nil
end
local key = Milo:uniqueKey(recipe)
if recipe.maxCount ~= 64 then
newRecipe.maxCount = recipe.maxCount
@@ -95,16 +91,10 @@ local function learnRecipe(ignoreNBT)
if ingredient.maxDamage > 0 then
-- ingredient.damage = '*' -- I don't think this is right
end
if ignoreNBT then
ingredient.nbtHash = nil
end
ingredients[k] = Milo:uniqueKey(ingredient)
end
context.userRecipes[key] = newRecipe
Util.writeTable(Craft.USER_RECIPES, context.userRecipes)
Craft.loadRecipes()
Milo:updateRecipe(key, newRecipe)
turtle.emptyInventory()
return recipe
@@ -115,29 +105,15 @@ local pages = {
index = 2,
validFor = 'Turtle Crafting',
notice = UI.Text {
x = 2, y = 2,
x = 3, y = 2,
textColor = colors.yellow,
value = 'Place recipe in turtle',
},
ignoreNBT = UI.Checkbox {
x = 3, y = 4,
help = 'Ignore damage of item',
value = true,
},
text = UI.Text {
x = 7, y = 4,
value = 'Ignore NBT (recommended)',
},
ignoreInfo = UI.TextArea {
x = 2, ex = -2, y = 6, ey = -2,
textColor = colors.yellow,
value = 'Some items contain unique NBT information. This information can be ignored for most items.',
},
},
}
function pages.turtleCraft:validate()
local recipe, msg = learnRecipe(self, self.ignoreNBT.value)
local recipe, msg = learnRecipe()
if recipe then
local displayName = itemDB:getName(recipe)