techage/basis/guide.lua

558 lines
15 KiB
Lua
Raw Normal View History

--[[
TechAge
=======
Copyright (C) 2019 Joachim Stolberg
LGPLv2.1+
See LICENSE.txt for more information
Construction Guide
]]--
-- Load support for intllib.
local MP = minetest.get_modpath("techage")
local S, NS = dofile(MP.."/intllib.lua")
local SMELTING_TIME = 2
local Tabs = S("Manufacture,Construction")
local Recipes = {} -- registered recipes
local KeyList = {} -- index to Recipes key translation
local NumRecipes = 0
-- formspec images
local function plan(images)
local tbl = {}
for y=0,9 do
for x=0,9 do
local img = images[y+1][x+1] or false
if img ~= false then
tbl[#tbl+1] = "item_image["..(x*0.8)..","..(y*0.8)..";0.8,0.8;"..img..".png]"
end
end
end
return table.concat(tbl)
end
local function formspec_manufacture(idx)
idx = math.min(idx, #KeyList)
local key = KeyList[idx]
local input1 = Recipes[key].input[1] or ""
local input2 = Recipes[key].input[2] or ""
local input3 = Recipes[key].input[3] or ""
local input4 = Recipes[key].input[4] or ""
local num = Recipes[key].number
local output = Recipes[key].output
if num > 1 then
output = output.." "..num
end
return "size[11,8]"..
default.gui_bg..
default.gui_bg_img..
default.gui_slots..
"tabheader[0,0;tab;"..Tabs..";2;;true]"..
"label[1,0.2;"..S("Manufacture Manual").."]"..
"container[1,1]"..
"item_image_button[0,0;1,1;"..input1..";b1;]"..
"item_image_button[1,0;1,1;"..input2..";b2;]"..
"item_image_button[0,1;1,1;"..input3..";b3;]"..
"item_image_button[1,1;1,1;"..input4..";b4;]"..
"item_image[2.6,0;0.8,0.8;"..Recipes[key].icon.."]"..
"image[2.3,0.6;1.6,1;gui_furnace_arrow_bg.png^[transformR270]"..
"item_image_button[4,0.5;1,1;"..output..";b5;]"..
"label[2,2.2;"..Recipes[key].hints.."]"..
"label[2,4;Recipe "..idx.." of "..NumRecipes.."]"..
"button[2,5.5;0.8,0.8;priv;<<]"..
"button[3,5.5;0.8,0.8;next;>>]"..
"container_end[]"
end
local function formspec_construction(recipe)
return "size[11,8]"..
default.gui_bg..
default.gui_bg_img..
default.gui_slots..
"tabheader[0,0;tab;"..Tabs..";3;;true]"..
"label[0,0;"..BurnerHelp.."]"..
"label[1,5;"..S("Cross-section")..":]"..
"container[4,4]"..
draw(BurnerImages)..
"container_end[]"
end
local function on_receive_fields(pos, formname, fields, sender)
local meta = minetest.get_meta(pos)
local recipe_idx = meta:get_int("recipe_idx")
if recipe_idx == 0 then recipe_idx = 1 end
if fields.tab == "1" then
meta:set_string("formspec", formspec1)
elseif fields.tab == "2" then
meta:set_string("formspec", formspec2(recipe_idx))
elseif fields.next == ">>" then
recipe_idx = math.min(recipe_idx + 1, NumRecipes)
meta:set_int("recipe_idx", recipe_idx)
meta:set_string("formspec", formspec2(recipe_idx))
elseif fields.priv == "<<" then
recipe_idx = math.max(recipe_idx - 1, 1)
meta:set_int("recipe_idx", recipe_idx)
meta:set_string("formspec", formspec2(recipe_idx))
end
end
local function can_dig(pos, player)
local meta = minetest.get_meta(pos);
local inv = meta:get_inventory()
return inv:is_empty("dst") and inv:is_empty("src")
end
local function allow_metadata_inventory_put(pos, listname, index, stack, player)
if minetest.is_protected(pos, player:get_player_name()) then
return 0
end
if listname == "src" then
return stack:get_count()
elseif listname == "dst" then
return 0
end
end
local function allow_metadata_inventory_move(pos, from_list, from_index, to_list, to_index, count, player)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
local stack = inv:get_stack(from_list, from_index)
return allow_metadata_inventory_put(pos, to_list, to_index, stack, player)
end
local function allow_metadata_inventory_take(pos, listname, index, stack, player)
if minetest.is_protected(pos, player:get_player_name()) then
return 0
end
return stack:get_count()
end
-- generate an unique key based on the unsorted and
-- variable number of inventory items
local function recipe_key(items)
local tbl = {}
-- remove items which exist more than once
for _,item in ipairs(items) do
tbl[item] = true
end
local names = {}
for key,_ in pairs(tbl) do
names[#names + 1] = key
end
-- bring in a sorted order
table.sort(names)
return table.concat(names, "-")
end
-- determine recipe based on inventory items
local function get_recipe(inv)
-- collect items
local stacks = {}
local names = {}
for _,stack in ipairs(inv:get_list("src")) do
if not stack:is_empty() then
table.insert(names, stack:get_name())
table.insert(stacks, stack)
else
table.insert(stacks, ItemStack(""))
end
end
local key = recipe_key(names)
local recipe = Recipes[key]
if recipe then
return {
input = recipe.input,
stacks = stacks,
output = ItemStack(recipe.output.." "..recipe.number),
heat = recipe.heat,
time = recipe.time,
}
end
return nil
end
-- prepare recipe and store in cache table for faster access
local function store_recipe_in_cache(pos)
local hash = minetest.hash_node_position(pos)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
local recipe = get_recipe(inv)
Cache[hash] = recipe
return recipe
end
-- read value from the node below
local function get_heat(pos)
local heat = 0
pos.y = pos.y - 1
local node = minetest.get_node(pos)
local meta = minetest.get_meta(pos)
pos.y = pos.y + 1
if minetest.get_item_group(node.name, "techage_flame") > 0 then
heat = meta:get_int("heat")
end
return heat
end
-- Start melting if heat is ok AND source items available
function techage.switch_to_active(pos)
local meta = minetest.get_meta(pos)
local heat = get_heat(pos)
local recipe = store_recipe_in_cache(pos)
if recipe and heat >= recipe.heat then
minetest.swap_node(pos, {name = "techage:meltingpot_active"})
minetest.registered_nodes["techage:meltingpot_active"].on_construct(pos)
meta:set_string("infotext", S("Melting Pot active (heat=")..heat..")")
minetest.get_node_timer(pos):start(2)
return true
end
meta:set_string("infotext", S("Melting Pot inactive (heat=")..heat..")")
return false
end
function techage.update_heat(pos)
local meta = minetest.get_meta(pos)
local heat = get_heat(pos)
meta:set_string("infotext", S("Melting Pot inactive (heat=")..heat..")")
end
local function set_inactive(meta, pos, heat)
minetest.get_node_timer(pos):stop()
minetest.swap_node(pos, {name = "techage:meltingpot"})
minetest.registered_nodes["techage:meltingpot"].on_construct(pos)
meta:set_string("infotext", S("Melting Pot inactive (heat=")..heat..")")
end
-- Stop melting if heat to low OR no source items available
local function switch_to_inactive(pos)
local meta = minetest.get_meta(pos)
local heat = get_heat(pos)
local hash = minetest.hash_node_position(pos)
local recipe = Cache[hash] or store_recipe_in_cache(pos)
if not recipe or heat < recipe.heat then
set_inactive(meta, pos, heat)
return true
end
meta:set_string("infotext", S("Melting Pot active (heat=")..heat..")")
return false
end
local function index(list, x)
for idx, v in pairs(list) do
if v == x then return idx end
end
return nil
end
-- move recipe src items to output inventory
local function process(inv, recipe, heat)
if heat < recipe.heat then
return false
end
local res = false
if inv:room_for_item("dst", recipe.output) then
for _,item in ipairs(recipe.input) do
res = false
for _, stack in ipairs(recipe.stacks) do
if stack:get_count() > 0 and stack:get_name() == item then
stack:take_item(1)
res = true
break
end
end
if res == false then
return false
end
end
inv:add_item("dst", recipe.output)
inv:set_list("src", recipe.stacks)
return true
end
return false
end
local function smelting(pos, recipe, heat, elapsed)
local meta = minetest.get_meta(pos)
local inv = meta:get_inventory()
elapsed = elapsed + meta:get_int("leftover")
while elapsed >= recipe.time do
if process(inv, recipe, heat) == false then
meta:set_int("leftover", 0)
set_inactive(meta, pos, heat)
return false
end
elapsed = elapsed - recipe.time
end
meta:set_int("leftover", elapsed)
return true
end
local function pot_node_timer(pos, elapsed)
if switch_to_inactive(pos) == false then
local hash = minetest.hash_node_position(pos)
local heat = get_heat(pos)
local recipe = Cache[hash] or store_recipe_in_cache(pos)
if recipe then
return smelting(pos, recipe, heat, elapsed)
end
end
return false
end
minetest.register_node("techage:meltingpot_active", {
description = S("Melting Pot"),
tiles = {
{
image = "techage_meltingpot_top_active.png",
backface_culling = false,
animation = {
type = "vertical_frames",
aspect_w = 16,
aspect_h = 16,
length = 1,
},
},
"default_cobble.png^techage_meltingpot.png",
},
drawtype = "nodebox",
node_box = {
type = "fixed",
fixed = {
{-10/16, -8/16, -10/16, 10/16, 9/16, -6/16},
{-10/16, -8/16, 6/16, 10/16, 9/16, 10/16},
{-10/16, -8/16, -10/16, -6/16, 9/16, 10/16},
{ 6/16, -8/16, -10/16, 10/16, 9/16, 10/16},
{ -6/16, -8/16, -6/16, 6/16, 5/16, 6/16},
},
},
selection_box = {
type = "fixed",
fixed = {-10/16, -8/16, -10/16, 10/16, 9/16, 10/16},
},
on_construct = function(pos)
local meta = minetest.get_meta(pos)
meta:set_string("formspec", formspec1)
local inv = meta:get_inventory()
inv:set_size('src', 4)
inv:set_size('dst', 4)
end,
on_timer = function(pos, elapsed)
return pot_node_timer(pos, elapsed)
end,
on_receive_fields = function(pos, formname, fields, sender)
on_receive_fields(pos, formname, fields, sender)
end,
on_metadata_inventory_move = function(pos)
store_recipe_in_cache(pos)
switch_to_inactive(pos)
end,
on_metadata_inventory_put = function(pos)
store_recipe_in_cache(pos)
switch_to_inactive(pos)
end,
on_metadata_inventory_take = function(pos)
store_recipe_in_cache(pos)
switch_to_inactive(pos)
end,
can_dig = can_dig,
drop = "techage:meltingpot",
is_ground_content = false,
groups = {cracky = 3, not_in_creative_inventory=1},
sounds = default.node_sound_metal_defaults(),
allow_metadata_inventory_put = allow_metadata_inventory_put,
allow_metadata_inventory_move = allow_metadata_inventory_move,
allow_metadata_inventory_take = allow_metadata_inventory_take,
})
minetest.register_node("techage:meltingpot", {
description = S("Melting Pot"),
tiles = {
"default_cobble.png",
"default_cobble.png^techage_meltingpot.png",
},
drawtype = "nodebox",
node_box = {
type = "fixed",
fixed = {
{-10/16, -8/16, -10/16, 10/16, 9/16, -6/16},
{-10/16, -8/16, 6/16, 10/16, 9/16, 10/16},
{-10/16, -8/16, -10/16, -6/16, 9/16, 10/16},
{ 6/16, -8/16, -10/16, 10/16, 9/16, 10/16},
{ -6/16, -8/16, -6/16, 6/16, -4/16, 6/16},
},
},
selection_box = {
type = "fixed",
fixed = {-10/16, -8/16, -10/16, 10/16, 9/16, 10/16},
},
on_construct = function(pos)
local meta = minetest.get_meta(pos)
meta:set_string("formspec", formspec1)
meta:set_string("infotext", S("Melting Pot inactive (heat=0)"))
local inv = meta:get_inventory()
inv:set_size('src', 4)
inv:set_size('dst', 4)
end,
on_metadata_inventory_move = function(pos)
store_recipe_in_cache(pos)
techage.switch_to_active(pos)
end,
on_metadata_inventory_put = function(pos)
store_recipe_in_cache(pos)
techage.switch_to_active(pos)
end,
on_metadata_inventory_take = function(pos)
store_recipe_in_cache(pos)
techage.switch_to_active(pos)
end,
on_receive_fields = function(pos, formname, fields, sender)
on_receive_fields(pos, formname, fields, sender)
end,
can_dig = can_dig,
is_ground_content = false,
groups = {cracky = 3},
sounds = default.node_sound_metal_defaults(),
allow_metadata_inventory_put = allow_metadata_inventory_put,
allow_metadata_inventory_move = allow_metadata_inventory_move,
allow_metadata_inventory_take = allow_metadata_inventory_take,
})
minetest.register_craft({
output = "techage:meltingpot",
recipe = {
{"default:cobble", "default:copper_ingot", "default:cobble"},
{"default:cobble", "", "default:cobble"},
{"default:cobble", "default:cobble", "default:cobble"},
},
})
if minetest.global_exists("unified_inventory") then
unified_inventory.register_craft_type("melting", {
description = S("Melting"),
icon = "default_cobble.png^techage_meltingpot.png",
width = 2,
height = 2,
})
unified_inventory.register_craft_type("burning", {
description = S("Burning"),
icon = "techage_smoke.png",
width = 1,
height = 1,
})
unified_inventory.register_craft({
output = "techage:charcoal",
items = {"group:wood"},
type = "burning",
})
end
function techage.ironage_register_recipe(recipe)
local key = recipe_key(recipe.recipe)
local output = string.split(recipe.output, " ")
local number = tonumber(output[2] or 1)
table.insert(KeyList, key)
Recipes[key] = {
input = recipe.recipe,
output = output[1],
number = number,
heat = math.max(recipe.heat or 3, 2),
time = math.max(recipe.time or 2, 2*number),
}
NumRecipes = NumRecipes + 1
if minetest.global_exists("unified_inventory") then
recipe.items = recipe.recipe
recipe.type = "melting"
unified_inventory.register_craft(recipe)
end
end
techage.ironage_register_recipe({
output = "default:obsidian",
recipe = {"default:cobble"},
heat = 5,
time = 4,
})
techage.ironage_register_recipe({
output = "default:coral_skeleton",
recipe = {"gravelsieve:compressed_gravel"},
heat = 4,
time = 4,
})
techage.ironage_register_recipe({
output = "default:bronze_ingot 4",
recipe = {"default:copper_ingot", "default:copper_ingot", "default:copper_ingot", "default:tin_ingot"},
heat = 4,
time = 8,
})
local PileHelp = S([[Coal Pile to produce charcoal:
- build a 5x5 block dirt base
- place a lighter in the centre
- build a 3x3x3 wood cube around
- cover all with dirt to a 5x5x5 cube
- keep a hole to the lighter
- ignite the lighter and immediately
- close the pile with one wood and one dirt
- open the pile after the smoke disappeared]])
local BurnerHelp = S([[Coal Burner to heat the melting pot:
- build a 3x3xN cobble tower
- more height means more flame heat
- keep a hole open on one side
- put a lighter in
- fill the tower from the top with charcoal
- ignite the lighter
- place the pot in the flame]])
local PileImages = {
{"default_dirt", "default_dirt", "default_dirt", "default_dirt", "default_dirt"},
{"default_dirt", "default_wood", "default_wood", "default_wood", "default_dirt"},
{"default_dirt", "default_wood", "default_wood", "default_wood", "default_dirt"},
{"default_dirt", "default_wood", "techage_lighter", "default_wood", "default_dirt"},
{"default_dirt", "default_dirt", "default_dirt", "default_dirt", "default_dirt"},
}
local BurnerImages = {
false, false, "default_cobble", "techage_charcoal", "default_cobble",
false, false, "default_cobble", "techage_charcoal", "default_cobble",
false, false, "default_cobble", "techage_charcoal", "default_cobble",
false, false, false, "techage_lighter", "default_cobble",
false, false, "default_cobble", "default_cobble", "default_cobble",
}