--[[ 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", }