From eb2d072cdb2b15b9dd0c61b684ccc96cca25317e Mon Sep 17 00:00:00 2001 From: Zenon Seth Date: Fri, 17 Nov 2023 14:28:33 +0000 Subject: [PATCH] Add Vaccuum Chest --- ROADMAP.md | 1 - api/api.lua | 1 + api/vaccuum_chest.lua | 155 ++++++++++++++++++++++++ logic/groups.lua | 5 + logic/logic.lua | 1 + logic/network_logic.lua | 3 + logic/supplier.lua | 2 + logic/vaccuum_chest.lua | 89 ++++++++++++++ registration/machines_api_reg.lua | 9 ++ textures/logistica_vaccuum_bottom.png | Bin 0 -> 2324 bytes textures/logistica_vaccuum_front.png | Bin 0 -> 2375 bytes textures/logistica_vaccuum_particle.png | Bin 0 -> 1863 bytes textures/logistica_vaccuum_side.png | Bin 0 -> 2355 bytes textures/logistica_vaccuum_top.png | Bin 0 -> 2376 bytes 14 files changed, 265 insertions(+), 1 deletion(-) create mode 100644 api/vaccuum_chest.lua create mode 100644 logic/vaccuum_chest.lua create mode 100644 textures/logistica_vaccuum_bottom.png create mode 100644 textures/logistica_vaccuum_front.png create mode 100644 textures/logistica_vaccuum_particle.png create mode 100644 textures/logistica_vaccuum_side.png create mode 100644 textures/logistica_vaccuum_top.png diff --git a/ROADMAP.md b/ROADMAP.md index b9be5bb..79da773 100644 --- a/ROADMAP.md +++ b/ROADMAP.md @@ -3,7 +3,6 @@ Missing Features: High Priority: - Vaccuum Supply Chest -- Trash Can - Togglable Cable - Autocrafter diff --git a/api/api.lua b/api/api.lua index 19ae8f9..468d292 100644 --- a/api/api.lua +++ b/api/api.lua @@ -11,3 +11,4 @@ dofile(path.."/access_point.lua") dofile(path.."/lava_furnace.lua") dofile(path.."/lava_furnace_recipe.lua") dofile(path.."/trashcan.lua") +dofile(path.."/vaccuum_chest.lua") diff --git a/api/vaccuum_chest.lua b/api/vaccuum_chest.lua new file mode 100644 index 0000000..b6a39a5 --- /dev/null +++ b/api/vaccuum_chest.lua @@ -0,0 +1,155 @@ + +local FORMSPEC_NAME = "logistica_vaccuum_chest" +local ON_OFF_BUTTON = "on_off_btn" + +local forms = {} + +local function get_vaccuum_formspec(pos) + local posForm = "nodemeta:"..pos.x..","..pos.y..","..pos.z + local isOn = logistica.is_machine_on(pos) + + return "formspec_version[4]" .. + "size[10.5,10]" .. + logistica.ui.background.. + logistica.ui.on_off_btn(isOn, 7.0, 0.5, ON_OFF_BUTTON, "Vaccuum items:").. + "label[0.6,1.0;Supplies collected items to the network.]".. + "list["..posForm..";main;0.4,1.4;8,2;0]".. + "list[current_player;main;0.4,4.5;8,4;0]".. + "listring[]" +end + +local function show_vaccuum_formspec(playerName, pos) + forms[playerName] = {position = pos} + minetest.show_formspec(playerName, FORMSPEC_NAME, get_vaccuum_formspec(pos)) +end + +local function on_player_receive_fields(player, formname, fields) + if not player or not player:is_player() then return false end + if formname ~= FORMSPEC_NAME then return false end + local playerName = player:get_player_name() + if not forms[playerName] then return false end + local pos = forms[playerName].position + if minetest.is_protected(pos, playerName) then return true end + + if fields.quit then + forms[playerName] = nil + elseif fields[ON_OFF_BUTTON] then + logistica.toggle_machine_on_off(pos) + show_vaccuum_formspec(player:get_player_name(), pos) + end + return true +end + +local function on_vaccuum_rightclick(pos, node, clicker, itemstack, pointed_thing) + if not clicker or not clicker:is_player() then return end + if minetest.is_protected(pos, clicker:get_player_name()) then return end + show_vaccuum_formspec(clicker:get_player_name(), pos) +end + +local function after_place_vaccuum(pos, placer, itemstack) + local meta = minetest.get_meta(pos) + local inv = meta:get_inventory() + inv:set_size("main", logistica.get_supplier_inv_size(pos)) + logistica.set_node_tooltip_from_state(pos) + logistica.on_supplier_change(pos) +end + +local function allow_vaccuum_storage_inv_put(pos, _, _, stack, player) + if minetest.is_protected(pos, player:get_player_name()) then return 0 end + return stack:get_count() +end + +local function allow_vaccuum_inv_take(pos, _, _, stack, player) + if minetest.is_protected(pos, player:get_player_name()) then return 0 end + return stack:get_count() +end + +local function allow_vaccuum_inv_move(pos, _, _, _, _, count, player) + if minetest.is_protected(pos, player:get_player_name()) then return 0 end + return count +end + +local function on_vaccuum_inventory_put(pos, _, _, _, _) + logistica.update_cache_at_pos(pos, LOG_CACHE_SUPPLIER) +end + +local function on_vaccuum_inventory_take(pos, _, _, _, _) + logistica.update_cache_at_pos(pos, LOG_CACHE_SUPPLIER) +end + +local function can_dig_vaccuum(pos, _) + local inv = minetest.get_meta(pos):get_inventory() + return inv:is_empty("main") +end + +local function on_vaccuum_power(pos, power) + logistica.vaccuum_chest_on_power(pos, power) +end + +---------------------------------------------------------------- +-- Minetest registration +---------------------------------------------------------------- + +minetest.register_on_player_receive_fields(on_player_receive_fields) + +minetest.register_on_leaveplayer(function(objRef, timed_out) + if objRef:is_player() then + forms[objRef:get_player_name()] = nil + end +end) + +---------------------------------------------------------------- +-- Public Registration API +---------------------------------------------------------------- +-- `simpleName` is used for the description and for the name (can contain spaces) +-- `inventorySize` should be 16 at max +function logistica.register_vaccuum_chest(desc, name, inventorySize, tiles) + local lname = string.lower(name:gsub(" ", "_")) + local vaccuum_name = "logistica:"..lname + logistica.vaccuum_suppliers[vaccuum_name] = true + local grps = {oddly_breakable_by_hand = 3, cracky = 3 } + grps[logistica.TIER_ALL] = 1 + local def = { + description = desc, + drawtype = "normal", + tiles = tiles, + paramtype = "light", + paramtype2 = "facedir", + is_ground_content = false, + groups = grps, + drop = vaccuum_name, + sounds = logistica.node_sound_metallic(), + after_place_node = after_place_vaccuum, + after_destruct = logistica.on_supplier_change, + on_rightclick = on_vaccuum_rightclick, + allow_metadata_inventory_put = allow_vaccuum_storage_inv_put, + allow_metadata_inventory_take = allow_vaccuum_inv_take, + allow_metadata_inventory_move = allow_vaccuum_inv_move, + on_metadata_inventory_put = on_vaccuum_inventory_put, + on_metadata_inventory_take = on_vaccuum_inventory_take, + on_timer = logistica.vaccuum_chest_on_timer, + can_dig = can_dig_vaccuum, + logistica = { + inventory_size = inventorySize, + on_power = on_vaccuum_power, + } + } + + minetest.register_node(vaccuum_name, def) + + local def_disabled = table.copy(def) + local tiles_disabled = {} + for k, v in pairs(def.tiles) do tiles_disabled[k] = v.."^logistica_disabled.png" end + + def_disabled.tiles = tiles_disabled + def_disabled.groups = { oddly_breakable_by_hand = 3, cracky = 3, choppy = 3, not_in_creative_inventory = 1 } + def_disabled.on_construct = nil + def_disabled.after_destruct = nil + def_disabled.on_punch = nil + def_disabled.on_rightclick = nil + def_disabled.on_timer = nil + def_disabled.logistica = nil + + minetest.register_node(vaccuum_name.."_disabled", def_disabled) + +end diff --git a/logic/groups.lua b/logic/groups.lua index 5c70d89..539a618 100644 --- a/logic/groups.lua +++ b/logic/groups.lua @@ -7,6 +7,7 @@ logistica.mass_storage = {} logistica.item_storage = {} logistica.misc_machines = {} logistica.trashcans = {} +logistica.vaccuum_suppliers = {} logistica.tiers = {} logistica.TIER_ALL = "logistica_all_tiers" logistica.GROUP_ALL = "group:" .. logistica.TIER_ALL @@ -62,6 +63,10 @@ function logistica.is_trashcan(name) return logistica.trashcans[name] ~= nil end +function logistica.is_vaccuum_supplier(name) + return logistica.vaccuum_suppliers[name] ~= nil +end + function logistica.get_item_tiers(name) local tiers = {} for tier,_ in pairs(logistica.tiers) do diff --git a/logic/logic.lua b/logic/logic.lua index ea9bad6..8424e06 100644 --- a/logic/logic.lua +++ b/logic/logic.lua @@ -15,3 +15,4 @@ dofile(path.."/access_point.lua") dofile(path.."/access_point_formspec.lua") dofile(path.."/lava_furnace_guide_formspec.lua") dofile(path.."/trashcan.lua") +dofile(path.."/vaccuum_chest.lua") diff --git a/logic/network_logic.lua b/logic/network_logic.lua index e2bc1cf..bd61cce 100644 --- a/logic/network_logic.lua +++ b/logic/network_logic.lua @@ -147,6 +147,9 @@ local function recursive_scan_for_nodes_for_controller(network, positionHashes, elseif logistica.is_supplier(otherName) then network.suppliers[otherHash] = true valid = true + elseif logistica.is_vaccuum_supplier(otherName) then + network.suppliers[otherHash] = true + valid = true elseif logistica.is_mass_storage(otherName) then network.mass_storage[otherHash] = true valid = true diff --git a/logic/supplier.lua b/logic/supplier.lua index 16bd8a1..b62ac0a 100644 --- a/logic/supplier.lua +++ b/logic/supplier.lua @@ -23,6 +23,8 @@ end -- tries to put the given item in this supplier, returns what's leftover function logistica.put_item_in_supplier(pos, stack) + local nodeName = minetest.get_node(pos).name + if not logistica.is_supplier(nodeName) then return stack end -- only insert if its enabled if not logistica.is_machine_on(pos) then return stack end local origCount = stack:get_count() diff --git a/logic/vaccuum_chest.lua b/logic/vaccuum_chest.lua new file mode 100644 index 0000000..7aadd9e --- /dev/null +++ b/logic/vaccuum_chest.lua @@ -0,0 +1,89 @@ + +local TIMER_INTERVAL = 0.8 +local TIMER_INTERVAL_LONG = 2.0 +local DEF_RADIUS = 3 +local INV_MAIN = "main" +local ITEM_TAKE_PER_CYCLE_LIMIT = 10 +local NUM_PARTICLES_PER_COLLECT = 3 + +local function random_offset() + return vector.new((math.random() - 0.5)/4, (math.random() - 0.5)/4, (math.random() - 0.5)/4) +end + +local function add_particle_effect_for_item_taken(itemPos, vaccuumPos) + for _ = 1, NUM_PARTICLES_PER_COLLECT do + local startPos = vector.add(itemPos, random_offset()) + local endPos = vector.add(vaccuumPos, vector.new(0, -0.45, 0)) + local vel = vector.normalize(vector.subtract(endPos, startPos)) * 2 + minetest.add_particle({ + pos = startPos, + velocity = vel, + expirationtime = 2, + size = 1, + collisiondetection = true, + collision_removal = true, + object_collision = false, + texture = "logistica_vaccuum_particle.png", + }) + end +end + +-- returns how many were inserted +local function collect_items_into(pos, distance) + local inserted = 0 + + local nodeName = minetest.get_node(pos).name + if not logistica.is_vaccuum_supplier(nodeName) then return inserted end + local nodeDef = minetest.registered_nodes[nodeName] + if not nodeDef or not nodeDef.logistica or not nodeDef.on_metadata_inventory_put then + return inserted + end + + distance = distance + 0.5 + local minPos = vector.subtract(pos, distance) + local maxPos = vector.add(pos, distance) + local inv = minetest.get_meta(pos):get_inventory() + for _, obj in pairs(minetest.get_objects_in_area(minPos, maxPos)) do + local entity = obj:get_luaentity() + if entity + and entity.name == "__builtin:item" + and entity.itemstring ~= "" then + local itemStack = ItemStack(entity.itemstring) + if inv:room_for_item(INV_MAIN, itemStack) then + add_particle_effect_for_item_taken(obj:get_pos(), pos) + inv:add_item(INV_MAIN, itemStack) + -- this look unsafe, but we only target our supplier nodes + entity.itemstring = "" + inserted = inserted + 1 + obj:remove() + if ITEM_TAKE_PER_CYCLE_LIMIT > 0 + and inserted >= ITEM_TAKE_PER_CYCLE_LIMIT then + nodeDef.on_metadata_inventory_put(pos, nil, nil, nil, nil) + return inserted + end + end + end + end + if inserted > 0 then + nodeDef.on_metadata_inventory_put(pos, nil, nil, nil, nil) + end + return inserted +end + +-- global functions + +function logistica.vaccuum_chest_on_timer(pos, elapsed) + if not logistica.is_machine_on(pos) then return false end + local inserted = collect_items_into(pos, DEF_RADIUS) + if inserted then + logistica.start_node_timer(pos, TIMER_INTERVAL) + else + logistica.start_node_timer(pos, TIMER_INTERVAL_LONG) + end + return false +end + +function logistica.vaccuum_chest_on_power(pos, power) + logistica.set_node_tooltip_from_state(pos, nil, power) + logistica.start_node_timer(pos, TIMER_INTERVAL) +end \ No newline at end of file diff --git a/registration/machines_api_reg.lua b/registration/machines_api_reg.lua index c9f5a58..3f8a2ad 100644 --- a/registration/machines_api_reg.lua +++ b/registration/machines_api_reg.lua @@ -172,3 +172,12 @@ logistica.register_trashcan("Trashcan", "trashcan", { "logistica_trashcan_side.png", "logistica_trashcan_side.png", }) + +logistica.register_vaccuum_chest("Vaccuum Supplier Chest", "vaccuum_chest", 16, { + "logistica_vaccuum_top.png", + "logistica_vaccuum_bottom.png", + "logistica_vaccuum_side.png", + "logistica_vaccuum_side.png", + "logistica_vaccuum_side.png", + "logistica_vaccuum_front.png", +}) diff --git a/textures/logistica_vaccuum_bottom.png b/textures/logistica_vaccuum_bottom.png new file mode 100644 index 0000000000000000000000000000000000000000..18dab40b9eff4a09baa3050ba483b7493ac7afb2 GIT binary patch literal 2324 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 IDMjo44@bj%p8x;= literal 0 HcmV?d00001 diff --git a/textures/logistica_vaccuum_particle.png b/textures/logistica_vaccuum_particle.png new file mode 100644 index 0000000000000000000000000000000000000000..58d600d28ccbb6d3ab16fd1d1af1748f9400d0bf GIT binary patch literal 1863 zcmaJ?eQXnD7{6^np<^L3U=tH^-pBxNE8T_}f3DcQb-aF|!`8;&s-Nv=A~-jNk)? zXh?)Of|!>@MT%|#3hDy^mbYMk9_zzUma$+B#ZIGBw1FU76_dQcoHR}frD{NGu{i|> zhE}8zjzJEa%_RwbmV=mFx4Z&%RN6|2vXTzfjZ{;FBg0S%^%zyP=TS35lM1VFK$DU)VC9sgloUBuZG4iW_<#kA>KWjt z!kmJYi2y= z_*;xdWyUAOoxJq8bph>?XK)R{eTJtN8Wd715%6#iN^s$Ie^m7$f@HkG!m>{&H-ao| z3%6ZZY=3Fi>19Q>{*AtFZ|&D>f-WCT;D7KTzvch literal 0 HcmV?d00001 diff --git a/textures/logistica_vaccuum_side.png b/textures/logistica_vaccuum_side.png new file mode 100644 index 0000000000000000000000000000000000000000..e1f756585ef63dd39857d8cc9e51e8fc56fb9254 GIT binary patch literal 2355 zcmaJ@3se(l7M_5v2&Jo{LT$0*VDUvVNsu7f1RE1-0)$2qBw#I0CX-|&$;8Zr1iD=c z_oet#v&;_0^()ThPiPii?j5wdk(GsUEBzOk)*w|4EQXwaqz`0%1f#7*; z4uC_DWU0r2;rRpu8wfLLmm)tMX+~huC`Gap)O@u=PFP4~k(1CCW$5std_2jBEKLrT zc*KCfMlcxcv03f3*ds;!ykhX3Jw_4OufpU@5xrUi%PA)TC-4$@d?Yy(mN<UAx*!b&LU5#J1i7K(H6=v64W>sBk)s(Al=V4@nDW;DXZ1LuWqxAy9Ujq6oAAH z7hzmXPaM#?LI;6n5;Wy<;zUXT$X<*;aco}18qz~p^$HRsl?K(2A_CE`z?|Oz<&>3j zYC#hcz{qdIW=^cKVP+zmG%}VEuF>;!!fyUGz(!_uv8|A0C&68Wqg?t?Oo}85(J^jv zo(YZALF|uCO&LLO@qWTlNF|qRoRo>Q0*uaFz6@5S$VGxAk%%kg1w4SFiGeL?DpX($ z&=T^6@mzs`D@fD{1mXm~n4ggAC-p0Wu`yx{_B*O<-@_6kE@nu^O8BL~9LTIpKnjRf zl5Kvm6|G2R26vQ>%%@7CL8%d#GP)q;InmwaTUmwud! z$JzH>*&R7s>xx+G^U{Wh{X5Enu2`r3 zJhyn=-N?@u$RoB?cm4AddhIXCKc{_v=hPc|L$zmK?3~hjclgJbaMr)DM%RbkkJ^>D zQrG5tct1;8Yyl_~vfQDn}R z?6liOLF%y1qVVvvPwF@OlA0zFtDA0w7Wh8SKAngb2dADeUlJV4j|4wslO4(&8iK+i z*oOnvZi|ATpkS3kren*ol!{m}rPb^k7<@=Om;9(n4qH$TBa>g5QMDWCzN-s4+nHHf zIVGAKz5S%hc=M(H({5iu^*1F8hZYZ57RNI~L;9?axHE%Mc+VjSYK^+wc(Y;W{Poy( zmoAaDv1e$`?OTHv)8`7~uRxsV<8Iwwo;5S%@42OQIot1cU0*e|s^GiI>Jzu_h2g@< zmuqU2HD&WlblMP^p+cBif7W_t{p72iPDdmPg%wAve{Pveb>KJ6wq)Bi0 zCaSd;U8yQX?WY_~7Ie}=Z zitgI8aPNQC>@{Tc%$j?w^J()aTiQjhVcpJN?{?MY_jVa1&%Ik-o?VjGW|;8(^tMyyI^s@1uSSPf^qsUE z%vII*=fHFRnhu*}@C)am38tM7@Cz@_^fl}w8sGG}>o&;WN-i!ed+*+(n}=ibyLs)x z(}PD7{~Ono{ML=fSBI`CH?M9P7~mdx@VA?F`^|+bW?a83yPo2@bK`io&F#*<-}H}l zL-z*eP5n~~X9)+gMFw&BnTN6T_Uj*HyqW! l4%3Hi*ZTTCjZ4dV7~BygiS(g)UiQC^DkVd)Z`qpC{{xf`ZlC}F literal 0 HcmV?d00001 diff --git a/textures/logistica_vaccuum_top.png b/textures/logistica_vaccuum_top.png new file mode 100644 index 0000000000000000000000000000000000000000..4cce5a7fdf99dfa30976a7ab6929de3743943516 GIT binary patch literal 2376 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 literal 0 HcmV?d00001