moreblocks/stairsplus/api/recipe.lua
2022-06-27 14:58:00 -07:00

242 lines
6.0 KiB
Lua

-- for registering recipe schemas
local api = stairsplus.api
api.registered_recipe_schemas = {}
api.registered_on_register_craft_schemas = {}
local function is_valid_output(item, shapes)
local item_name = item:match("^([^ ]+)")
return item_name and (shapes[item_name] or item_name == "node" or item_name:match(":"))
end
local function is_valid_item(item, shapes)
return is_valid_output(item, shapes) or item == ""
end
local function verify_schema(schema)
local problems = {}
if not (schema.type == "shaped" or schema.type == "shapeless" or schema.type == nil) then
table.insert(problems, ("unimplemented schema type %q"):format(schema.type))
end
if not is_valid_output(schema.output, api.registered_shapes) then
table.insert(problems, ("don't know how to handle output %q"):format(schema.output))
end
if schema.replacements then
for _, replacement in ipairs(schema.replacements) do
for _, item in ipairs(replacement) do
if not is_valid_item(item, api.registered_shapes) then
table.insert(problems, ("don't know how to handle replacement item %q"):format(item))
end
end
end
end
if schema.type == "shapeless" then
for _, item in ipairs(schema.recipe) do
if not is_valid_item(item, api.registered_shapes) then
table.insert(problems, ("don't know how to handle craft item %q"):format(item))
end
end
else
for _, row in ipairs(schema.recipe) do
for _, item in ipairs(row) do
if not is_valid_item(item, api.registered_shapes) then
table.insert(problems, ("don't know how to handle craft item %q"):format(item))
end
end
end
end
if #problems > 0 then
return table.concat(problems, ", ")
end
end
function api.register_on_register_craft_schema(func)
table.insert(api.registered_on_register_craft_schemas, func)
end
function api.register_craft_schema(schema)
local problems = verify_schema(schema)
if problems then
error(problems)
end
stairsplus.log("info", "registering craft schema %s", minetest.write_json(schema))
table.insert(api.registered_recipe_schemas, schema)
for _, func in ipairs(api.registered_on_register_craft_schemas) do
func(schema)
end
end
local function has_the_right_shapes(schema, shapes)
if not is_valid_output(schema.output, shapes) then
return false
end
if schema.replacements then
for _, replacement in ipairs(schema.replacements) do
for _, item in ipairs(replacement) do
if not is_valid_item(item, shapes) then
return false
end
end
end
end
if schema.type == "shapeless" then
for _, item in ipairs(schema.recipe) do
if not is_valid_item(item, shapes) then
return false
end
end
elseif schema.type == "shaped" or schema.type == nil then
for _, row in ipairs(schema.recipe) do
for _, item in ipairs(row) do
if not is_valid_item(item, shapes) then
return false
end
end
end
end
return true
end
local function register_for_schema(node, schema)
local recipe = table.copy(schema)
recipe.output = api.get_schema_recipe_item(node, recipe.output)
if recipe.replacements then
for _, replacement in ipairs(recipe.replacements) do
for i, item in ipairs(replacement) do
replacement[i] = api.get_schema_recipe_item(node, item)
end
end
end
if recipe.type == "shapeless" then
for i, item in ipairs(recipe.recipe) do
recipe.recipe[i] = api.get_schema_recipe_item(node, item)
end
elseif recipe.type == "shaped" or recipe.type == nil then
for _, row in ipairs(recipe.recipe) do
for i, item in ipairs(row) do
row[i] = api.get_schema_recipe_item(node, item)
end
end
end
stairsplus.log("info", "registering recipe %s", minetest.write_json(recipe))
minetest.register_craft(recipe)
end
function api.register_schema_crafts_for_node(node)
stairsplus.log("info", "registering schema crafts for %q", node)
local shapes = api.get_shapes_hash(node)
for _, schema in ipairs(api.registered_recipe_schemas) do
if has_the_right_shapes(schema, shapes) then
stairsplus.log("verbose", "using schema %s", minetest.write_json(schema))
register_for_schema(node, schema)
end
end
end
local function shapes_match(a, b)
local a_shapes = api.get_shapes(a)
local b_shapes = api.get_shapes(b)
if #a_shapes ~= #b_shapes then
return false
end
for i = 1, #a_shapes do
if a_shapes[i] ~= b_shapes[i] then
return false
end
end
return true
end
local function register_cooking_for_shapes(recipe)
local mod, name = recipe.recipe:match("^([^:]+):(.*)$")
if mod ~= "group" and not shapes_match(recipe.output, recipe.recipe) then
error(("error: shapes of %s and %s do not match"):format(recipe.output, recipe.recipe))
end
local shapes = api.get_shapes(recipe.output)
for _, shape in ipairs(shapes) do
local shape_def = api.registered_shapes[shape]
local input
if mod == "group" then
input = "group:" .. shape_def.name_format:format(name)
else
input = api.get_schema_recipe_item(recipe.recipe, shape)
end
minetest.register_craft({
type = "cooking",
output = api.get_schema_recipe_item(recipe.output, shape),
recipe = input,
cooktime = math.max(1.0, (recipe.cooktime or 3) * (shape_def.eighths / 8)),
})
end
end
local function register_fuel_for_shapes(recipe)
local mod, name = recipe.recipe:match("^([^:]+):(.*)$")
local shapes
if mod == "group" then
shapes = api.registered_shapes
else
shapes = api.get_shapes_hash(recipe.recipe)
end
if not shapes then
error(("don't know how to handle fuel %s"):format(recipe.recipe))
end
for shape in pairs(shapes) do
local shape_def = api.registered_shapes[shape]
local input
if mod == "group" then
input = "group:" .. shape_def.name_format:format(name)
else
input = api.get_schema_recipe_item(recipe.recipe, shape)
end
minetest.register_craft({
type = "fuel",
recipe = input,
burntime = math.max(1.0, (recipe.burntime or 1) * (shape_def.eighths / 8)),
})
end
end
function api.register_crafts_for_shapes(recipe)
if recipe.type == "cooking" then
register_cooking_for_shapes(recipe)
elseif recipe.type == "fuel" then
register_fuel_for_shapes(recipe)
else
error(("unsupported recipe type %s"):format(recipe.type))
end
end