techage/furnace/cooking.lua
Joachim Stolberg eaca058b2c recipe bugfix
2019-08-18 23:16:13 +02:00

180 lines
4.0 KiB
Lua

--[[
TechAge
=======
Copyright (C) 2019 Joachim Stolberg
LGPLv2.1+
See LICENSE.txt for more information
Cooking routines for furnace
]]--
-- for lazy programmers
local P = minetest.string_to_pos
local M = minetest.get_meta
local S = techage.S
local range = techage.range
local Recipes = {} -- registered recipes
local Ingredients = {}
local KeyList = {} -- index to Recipes key translation
techage.furnace = {}
local function all_ingredients_available(output, ingr)
if Recipes[output] then
for idx,recipe in ipairs(Recipes[output]) do
local not_in_list = false
for _,item in ipairs(recipe.input) do
if not techage.in_list(ingr, item) then
not_in_list = true
end
end
if not_in_list == false then
return idx -- list number of the recipe
end
end
end
end
-- Return a list with all outputs of the given list of ingredients
local function get_recipes(ingr)
if #ingr > 0 then
local tbl = {}
for _,item in ipairs(ingr) do
if Ingredients[item] then
for _,output in ipairs(Ingredients[item]) do
if all_ingredients_available(output, ingr) then
techage.add_to_set(tbl, output)
end
end
end
end
return tbl
else
return KeyList
end
end
function techage.furnace.get_ingredients(pos)
local inv = M(pos):get_inventory()
local tbl = {}
for _,stack in ipairs(inv:get_list('src')) do
if stack:get_name() ~= "" then
tbl[#tbl+1] = stack:get_name()
end
end
return tbl
end
-- move recipe src items to output inventory
local function process(inv, recipe, output)
-- check if all ingredients are available
for _,item in ipairs(recipe.input) do
if not inv:contains_item("src", item) then
return false
end
end
-- remove items
for _,item in ipairs(recipe.input) do
inv:remove_item("src", item)
end
-- add to dst
local stack = ItemStack(output)
stack:set_count(recipe.number)
inv:add_item("dst", stack)
return true
end
function techage.furnace.smelting(pos, mem, elapsed)
local inv = M(pos):get_inventory()
local state = techage.RUNNING
if inv and not inv:is_empty("src") then
if not mem.output or not mem.num_recipe then
return techage.FAULT
end
local recipe = Recipes[mem.output][mem.num_recipe]
if not recipe then
return techage.FAULT
end
-- check dst inv
local item = ItemStack(mem.output)
if not inv:room_for_item("dst", item) then
return techage.BLOCKED
end
elapsed = elapsed + (mem.leftover or 0)
while elapsed >= recipe.time do
if process(inv, recipe, mem.output) == false then
mem.leftover = 0
return techage.STANDBY
else
state = techage.RUNNING
end
elapsed = elapsed - recipe.time
end
mem.leftover = elapsed
return state
end
return techage.STANDBY
end
function techage.furnace.get_output(mem, ingr, idx)
local tbl = get_recipes(ingr)
idx = range(idx, 1, #tbl)
mem.output = tbl[idx] or tbl[1]
mem.num_recipe = all_ingredients_available(mem.output, ingr)
return mem.output
end
function techage.furnace.get_num_recipes(ingr)
return #get_recipes(ingr)
end
function techage.furnace.reset_cooking(mem)
mem.leftover = 0
end
if minetest.global_exists("unified_inventory") then
unified_inventory.register_craft_type("ta3_melting", {
description = S("TA3 Melting"),
icon = "techage_concrete.png^techage_appl_furnace.png^techage_frame_ta3.png",
width = 2,
height = 2,
})
end
function techage.furnace.register_recipe(recipe)
local words = string.split(recipe.output, " ")
local output = words[1]
local number = tonumber(words[2] or 1)
table.insert(KeyList, output)
--print(recipe.output, dump(recipe.recipe))
if not Recipes[output] then
Recipes[output] = {}
end
table.insert(Recipes[output], {
input = recipe.recipe,
number = number,
time = math.max((recipe.time or 3) * number, 2),
})
for _,item in ipairs(recipe.recipe) do
if Ingredients[item] then
techage.add_to_set(Ingredients[item], output)
else
Ingredients[item] = {output}
end
end
if minetest.global_exists("unified_inventory") then
recipe.items = recipe.recipe
recipe.type = "ta3_melting"
unified_inventory.register_craft(recipe)
end
end