From 406c397a3e2657733ec68f35f191e4aac292f5a1 Mon Sep 17 00:00:00 2001 From: Sokomine Date: Tue, 17 Dec 2013 05:18:50 +0100 Subject: [PATCH] added the actual code --- depends.txt | 2 + init.lua | 30 +++ locked_doors.lua | 225 ++++++++++++++++++ locked_elevator.lua | 182 ++++++++++++++ locked_travelnet.lua | 132 ++++++++++ textures/locked_travelnet_lock_inv.png | Bin 0 -> 4360 bytes .../locked_travelnet_travelnet_back_lock.png | Bin 0 -> 323 bytes .../locked_travelnet_travelnet_front_lock.png | Bin 0 -> 277 bytes .../locked_travelnet_travelnet_side_lock.png | Bin 0 -> 553 bytes 9 files changed, 571 insertions(+) create mode 100644 depends.txt create mode 100644 init.lua create mode 100644 locked_doors.lua create mode 100644 locked_elevator.lua create mode 100644 locked_travelnet.lua create mode 100644 textures/locked_travelnet_lock_inv.png create mode 100644 textures/locked_travelnet_travelnet_back_lock.png create mode 100644 textures/locked_travelnet_travelnet_front_lock.png create mode 100644 textures/locked_travelnet_travelnet_side_lock.png diff --git a/depends.txt b/depends.txt new file mode 100644 index 0000000..2a8174a --- /dev/null +++ b/depends.txt @@ -0,0 +1,2 @@ +locks +travelnet diff --git a/init.lua b/init.lua new file mode 100644 index 0000000..8da1ec9 --- /dev/null +++ b/init.lua @@ -0,0 +1,30 @@ + + +--[[ + Locked version of my travelnet that allow players to choose a destination out of a list + Copyright (C) 2013 Sokomine + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + + Version: 1.0 + + Changelog: +--]] + + +dofile(minetest.get_modpath("locked_travelnet").."/locked_travelnet.lua"); +-- does not work very satisfying yet +--dofile(minetest.get_modpath("locked_travelnet").."/locked_elevator.lua"); +-- not yet useful; right-click opens config menu + door... +--dofile(minetest.get_modpath("locked_travelnet").."/locked_doors.lua"); diff --git a/locked_doors.lua b/locked_doors.lua new file mode 100644 index 0000000..74ef97a --- /dev/null +++ b/locked_doors.lua @@ -0,0 +1,225 @@ +-- Doors that are especially useful for travelnet elevators but can also be used in other situations. +-- All doors (not only these here) in front of a travelnet or elevator are opened automaticly when a player arrives +-- and are closed when a player departs from the travelnet or elevator. +-- Autor: Sokomine + +locked_travelnet_doors_transform = function( pos, node, puncher, transform_into ) + + if( not( locks:lock_allow_use( pos, puncher ))) then + minetest.chat_send_player( puncher:get_player_name(), "This door is locked. It can only be opened by its owner or people with a key that fits."); + return; + end + + local olddata = locks:get_lockdata( pos ); + + minetest.env:add_node(pos, {name = transform_into, param2 = node.param2}) + locks:set_lockdata( pos, olddata ); +end + + + +minetest.register_node("locked_travelnet:elevator_door_steel_open", { + description = "elevator door (open)", + drawtype = "nodebox", + -- top, bottom, side1, side2, inner, outer + tiles = {"default_stone.png"}, + paramtype = "light", + paramtype2 = "facedir", + is_ground_content = true, + groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2,not_in_creative_inventory=1}, + -- larger than one node but slightly smaller than a half node so that wallmounted torches pose no problem + node_box = { + type = "fixed", + fixed = { + {-0.90, -0.5, 0.4, -0.49, 1.5, 0.5}, + { 0.49, -0.5, 0.4, 0.9, 1.5, 0.5}, + }, + }, + selection_box = { + type = "fixed", + fixed = { + {-0.9, -0.5, 0.4, 0.9, 1.5, 0.5}, + }, + }, + drop = "locked_travelnet:elevator_door_steel_closed", + on_rightclick = function(pos, node, puncher) + + locked_travelnet_doors_transform( pos, node, puncher, "locked_travelnet:elevator_door_steel_closed" ); + end, + + on_construct = function(pos) + locks:lock_init( pos, + "size[8,2]".. + "field[0.3,0.6;6,0.7;locks_sent_lock_command;Locked door. Type /help for help:;]".. + "button_exit[6.3,1.2;1.7,0.7;locks_sent_input;Proceed]" ); + end, + + after_place_node = function(pos, placer, itemstack) + locks:lock_set_owner( pos, placer, "Shared locked door" ); + end, + + on_receive_fields = function(pos, formname, fields, sender) + locks:lock_handle_input( pos, formname, fields, sender ); + end, + + can_dig = function(pos,player) + return locks:lock_allow_dig( pos, player ); + end, +}) + +minetest.register_node("locked_travelnet:elevator_door_steel_closed", { + description = "elevator door (closed)", + drawtype = "nodebox", + -- top, bottom, side1, side2, inner, outer + tiles = {"default_stone.png"}, + paramtype = "light", + paramtype2 = "facedir", + is_ground_content = true, + groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2}, + node_box = { + type = "fixed", + fixed = { + {-0.5, -0.5, 0.4, -0.01, 1.5, 0.5}, + { 0.01, -0.5, 0.4, 0.5, 1.5, 0.5}, + }, + }, + selection_box = { + type = "fixed", + fixed = { + {-0.5, -0.5, 0.4, 0.5, 1.5, 0.5}, + }, + }, + on_rightclick = function(pos, node, puncher) + locked_travelnet_doors_transform( pos, node, puncher, "locked_travelnet:elevator_door_steel_open" ); + end, + on_construct = function(pos) + locks:lock_init( pos, + "size[8,2]".. + "field[0.3,0.6;6,0.7;locks_sent_lock_command;Locked door. Type /help for help:;]".. + "button_exit[6.3,1.2;1.7,0.7;locks_sent_input;Proceed]" ); + end, + + after_place_node = function(pos, placer, itemstack) + locks:lock_set_owner( pos, placer, "Shared locked door" ); + end, + + on_receive_fields = function(pos, formname, fields, sender) + locks:lock_handle_input( pos, formname, fields, sender ); + end, + + can_dig = function(pos,player) + return locks:lock_allow_dig( pos, player ); + end, +}) + + + + +minetest.register_node("locked_travelnet:elevator_door_glass_open", { + description = "elevator door (open)", + drawtype = "nodebox", + -- top, bottom, side1, side2, inner, outer + tiles = {"travelnet_elevator_door_glass.png"}, + paramtype = "light", + paramtype2 = "facedir", + is_ground_content = true, + groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2,not_in_creative_inventory=1}, + -- larger than one node but slightly smaller than a half node so that wallmounted torches pose no problem + node_box = { + type = "fixed", + fixed = { + {-0.99, -0.5, 0.4, -0.49, 1.5, 0.5}, + { 0.49, -0.5, 0.4, 0.99, 1.5, 0.5}, + }, + }, + selection_box = { + type = "fixed", + fixed = { + {-0.9, -0.5, 0.4, 0.9, 1.5, 0.5}, + }, + }, + drop = "locked_travelnet:elevator_door_glass_closed", + on_rightclick = function(pos, node, puncher) + locked_travelnet_doors_transform( pos, node, puncher, "locked_travelnet:elevator_door_glass_closed" ); + end, + on_construct = function(pos) + locks:lock_init( pos, + "size[8,2]".. + "field[0.3,0.6;6,0.7;locks_sent_lock_command;Locked door. Type /help for help:;]".. + "button_exit[6.3,1.2;1.7,0.7;locks_sent_input;Proceed]" ); + end, + + after_place_node = function(pos, placer, itemstack) + locks:lock_set_owner( pos, placer, "Shared locked door" ); + end, + + on_receive_fields = function(pos, formname, fields, sender) + locks:lock_handle_input( pos, formname, fields, sender ); + end, + + can_dig = function(pos,player) + return locks:lock_allow_dig( pos, player ); + end, +}) + +minetest.register_node("locked_travelnet:elevator_door_glass_closed", { + description = "elevator door (closed)", + drawtype = "nodebox", + -- top, bottom, side1, side2, inner, outer + tiles = {"travelnet_elevator_door_glass.png"}, + paramtype = "light", + paramtype2 = "facedir", + is_ground_content = true, + groups = {snappy=2,choppy=2,oddly_breakable_by_hand=2}, + node_box = { + type = "fixed", + fixed = { + {-0.5, -0.5, 0.4, -0.01, 1.5, 0.5}, + { 0.01, -0.5, 0.4, 0.5, 1.5, 0.5}, + }, + }, + selection_box = { + type = "fixed", + fixed = { + {-0.5, -0.5, 0.4, 0.5, 1.5, 0.5}, + }, + }, + on_rightclick = function(pos, node, puncher) + locked_travelnet_doors_transform( pos, node, puncher, "locked_travelnet:elevator_door_glass_open" ); + end, + on_construct = function(pos) + locks:lock_init( pos, + "size[8,2]".. + "field[0.3,0.6;6,0.7;locks_sent_lock_command;Locked door. Type /help for help:;]".. + "button_exit[6.3,1.2;1.7,0.7;locks_sent_input;Proceed]" ); + end, + + after_place_node = function(pos, placer, itemstack) + locks:lock_set_owner( pos, placer, "Shared locked door" ); + end, + + on_receive_fields = function(pos, formname, fields, sender) + locks:lock_handle_input( pos, formname, fields, sender ); + end, + + can_dig = function(pos,player) + return locks:lock_allow_dig( pos, player ); + end, +}) + +minetest.register_craft({ + output = "locked_travelnet:elevator_door_glass_closed", + recipe = {{'travelnet:elevator_door_glass_closed','locks:lock'}, + } + }) + +minetest.register_craft({ + output = "locked_travelnet:elevator_door_steel_closed", + recipe = { + {'travelnet:elevator_door_steel_closed','locks:lock'}, + } + }) + + +print( "[Mod] locked_travelnet: loading locked_travelnet:elevator_door_xxx_closed (steel and glass)"); + diff --git a/locked_elevator.lua b/locked_elevator.lua new file mode 100644 index 0000000..1dcdcc8 --- /dev/null +++ b/locked_elevator.lua @@ -0,0 +1,182 @@ +-- This version of the travelnet box allows to move up or down only. +-- The network name is determined automaticly from the position (x/z coordinates). +-- Autor: Sokomine + +minetest.register_node("locked_travelnet:elevator", { + description = "Shared locked elevator", + + drawtype = "nodebox", + sunlight_propagates = true, + paramtype = 'light', + paramtype2 = "facedir", + + selection_box = { + type = "fixed", + fixed = { -0.5, -0.5, -0.5, 0.5, 1.5, 0.5 } + }, + + node_box = { + type = "fixed", + fixed = { + + { 0.48, -0.5,-0.5, 0.5, 0.5, 0.5}, + {-0.5 , -0.5, 0.48, 0.48, 0.5, 0.5}, + {-0.5, -0.5,-0.5 ,-0.48, 0.5, 0.5}, + + --groundplate to stand on + { -0.5,-0.5,-0.5,0.5,-0.48, 0.5}, + }, + }, + + + tiles = { + + "travelnet_elevator_inside_floor.png", -- view from top + "default_stone.png", -- view from bottom + "travelnet_elevator_inside_bottom.png", -- left side + "travelnet_elevator_inside_bottom.png", -- right side + "travelnet_elevator_inside_bottom.png", -- front view + "travelnet_elevator_inside_bottom.png", -- backward view + }, + inventory_image = "travelnet_elevator_inv.png", + wield_image = "travelnet_elevator_wield.png", + + groups = {cracky=1,choppy=1,snappy=1}, + + + light_source = 10, + + on_construct = function(pos) + local meta = minetest.env:get_meta(pos) + --- prepare the lock of the travelnet + locks:lock_init( pos, + "size[12,10]".. + "field[0.3,5.6;6,0.7;station_name;Name of this station:;]".. +-- "button_exit[6.3,6.2;1.7,0.7;station_set;Store]".. + "field[0.3,3.0;6,0.7;locks_sent_lock_command;Locked travelnet. Type /help for help:;]".. + "button_exit[6.3,3.2;1.7,0.7;locks_sent_input;Store]" ); + end, + + after_place_node = function(pos, placer, itemstack) + local meta = minetest.env:get_meta(pos); + meta:set_string("infotext", "Elevator (unconfigured)"); + meta:set_string("station_name", ""); + meta:set_string("station_network",""); + meta:set_string("owner", placer:get_player_name() ); + -- request initinal data + + local p = {x=pos.x, y=pos.y+1, z=pos.z} + local p2 = minetest.dir_to_facedir(placer:get_look_dir()) + minetest.env:add_node(p, {name="locked_travelnet:elevator_top", paramtype2="facedir", param2=p2}) + + locks:lock_set_owner( pos, placer, "Shared locked elevator" ); + end, + + on_receive_fields = function(pos, formname, fields, sender) + + -- abort if no input has been sent + if( fields.quit ) then + return; + end + + -- if the user already has the right to use this and did input text + if( (not(fields.locks_sent_lock_command) + or fields.locks_sent_lock_command=="") + and locks:lock_allow_use( pos, sender )) then + + travelnet.on_receive_fields( pos, formname, fields, sender ); + + -- a command for the lock? + else + locks:lock_handle_input( pos, formname, fields, sender ); + end + end, + + on_punch = function(pos, node, puncher) + travelnet.update_formspec(pos, puncher:get_player_name()) + end, + + can_dig = function( pos, player ) + if( not(locks:lock_allow_dig( pos, player ))) then + return false; + end + return travelnet.can_dig( pos, player, 'elevator' ) + end, + + after_dig_node = function(pos, oldnode, oldmetadata, digger) + travelnet.remove_box( pos, oldnode, oldmetadata, digger ) + end, + + -- taken from VanessaEs homedecor fridge + on_place = function(itemstack, placer, pointed_thing) + local pos = pointed_thing.above; + local node = minetest.env:get_node({x=pos.x, y=pos.y+1, z=pos.z}); + -- leftover elevator_top nodes can be removed by placing a new elevator underneath + if( node ~= nil and node.name ~= "air" and node.name ~= 'locked_travelnet:elevator_top') then + minetest.chat_send_player( placer:get_player_name(), 'Not enough vertical space to place the travelnet box!' ) + return; + end + return minetest.item_place(itemstack, placer, pointed_thing); + end, + + on_destruct = function(pos) + local p = {x=pos.x, y=pos.y+1, z=pos.z} + minetest.env:remove_node(p) + end +}) + +minetest.register_node("locked_travelnet:elevator_top", { + description = "Elevator Top", + + drawtype = "nodebox", + sunlight_propagates = true, + paramtype = 'light', + paramtype2 = "facedir", + + selection_box = { + type = "fixed", + fixed = { 0, 0, 0, 0, 0, 0 } +-- fixed = { -0.5, -0.5, -0.5, 0.5, 0.5, 0.5 } + }, + + node_box = { + type = "fixed", + fixed = { + + { 0.48, -0.5,-0.5, 0.5, 0.5, 0.5}, + {-0.5 , -0.5, 0.48, 0.48, 0.5, 0.5}, + {-0.5, -0.5,-0.5 ,-0.48, 0.5, 0.5}, + + --top ceiling + { -0.5, 0.48,-0.5,0.5, 0.5, 0.5}, + }, + }, + + + tiles = { + + "default_stone.png", -- view from top + "travelnet_elevator_inside_ceiling.png", -- view from bottom + "travelnet_elevator_inside_top_control.png", -- left side + "travelnet_elevator_inside_top.png", -- right side + "travelnet_elevator_inside_top.png", -- front view + "travelnet_elevator_inside_top.png", -- backward view + }, + inventory_image = "travelnet_elevator_inv.png", + wield_image = "travelnet_elevator_wield.png", + + light_source = 10, + + groups = {cracky=1,choppy=1,snappy=1,not_in_creative_inventory=1}, +}) + + + +minetest.register_craft({ + output = 'locked_travelnet:elevator', + recipe = { + { 'travelnet:elevator', 'locks:lock' }, + }, +}) + +print( "[Mod] locked_travelnet: loading locked_travelnet:elevator"); diff --git a/locked_travelnet.lua b/locked_travelnet.lua new file mode 100644 index 0000000..bffc6bb --- /dev/null +++ b/locked_travelnet.lua @@ -0,0 +1,132 @@ +-- contains the node definition for a general travelnet that can be used by anyone +-- further travelnets can only be installed by the owner or by people with the travelnet_attach priv +-- digging of such a travelnet is limited to the owner and to people with the travelnet_remove priv (useful for admins to clean up) +-- (this can be overrided in config.lua) +-- Autor: Sokomine +minetest.register_node("locked_travelnet:travelnet", { + + description = "Shared locked travelnet box", + + drawtype = "nodebox", + sunlight_propagates = true, + paramtype = 'light', + paramtype2 = "facedir", + + selection_box = { + type = "fixed", + fixed = { -0.5, -0.5, -0.5, 0.5, 1.5, 0.5 } + }, + + node_box = { + type = "fixed", + fixed = { + + { 0.45, -0.5,-0.5, 0.5, 1.45, 0.5}, + {-0.5 , -0.5, 0.45, 0.45, 1.45, 0.5}, + {-0.5, -0.5,-0.5 ,-0.45, 1.45, 0.5}, + + --groundplate to stand on + { -0.5,-0.5,-0.5,0.5,-0.45, 0.5}, + --roof + { -0.5, 1.45,-0.5,0.5, 1.5, 0.5}, + + }, + }, + + tiles = { + "default_clay.png", -- view from top + "default_clay.png", -- view from bottom + "locked_travelnet_travelnet_side_lock.png", -- left side + "locked_travelnet_travelnet_side_lock.png", -- right side + "locked_travelnet_travelnet_back_lock.png", -- front view + "locked_travelnet_travelnet_front_lock.png", -- backward view + }, + inventory_image = "locked_travelnet_lock_inv.png", + + groups = {cracky=1,choppy=1,snappy=1}, + + light_source = 10, + + on_construct = function(pos) + local meta = minetest.env:get_meta(pos) + --- prepare the lock of the travelnet + locks:lock_init( pos, + "size[12,10]".. + "field[0.3,5.6;6,0.7;station_name;Name of this station:;]".. + "field[0.3,6.6;6,0.7;station_network;Assign to Network:;]".. + "field[0.3,7.6;6,0.7;owner_name;(optional) owned by:;]".. +-- "button_exit[6.3,6.2;1.7,0.7;station_set;Store]".. + "field[0.3,3.0;6,0.7;locks_sent_lock_command;Locked travelnet. Type /help for help:;]".. + "button_exit[6.3,3.2;1.7,0.7;locks_sent_input;Store]" ); + end, + + after_place_node = function(pos, placer, itemstack) + local meta = minetest.env:get_meta(pos); + meta:set_string("infotext", "Travelnet-box (unconfigured)"); + meta:set_string("station_name", ""); + meta:set_string("station_network",""); + meta:set_string("owner", placer:get_player_name() ); + -- request initinal data + locks:lock_set_owner( pos, placer, "Shared locked travelnet" ); + end, + + on_receive_fields = function(pos, formname, fields, sender) + + -- abort if no input has been sent + if( fields.quit ) then + return; + end + + -- if the user already has the right to use this and did input text + if( (not(fields.locks_sent_lock_command) + or fields.locks_sent_lock_command=="") + and locks:lock_allow_use( pos, sender )) then + + travelnet.on_receive_fields( pos, formname, fields, sender ); + + -- a command for the lock? + else + locks:lock_handle_input( pos, formname, fields, sender ); + end + end, + + on_punch = function(pos, node, puncher) + travelnet.update_formspec(pos, puncher:get_player_name()) + end, + + can_dig = function( pos, player ) + + if( not(locks:lock_allow_dig( pos, player ))) then + return false; + end + + return travelnet.can_dig( pos, player, 'travelnet box' ) + end, + + after_dig_node = function(pos, oldnode, oldmetadata, digger) + travelnet.remove_box( pos, oldnode, oldmetadata, digger ) + end, + + -- taken from VanessaEs homedecor fridge + on_place = function(itemstack, placer, pointed_thing) + + local pos = pointed_thing.above; + if( minetest.env:get_node({x=pos.x, y=pos.y+1, z=pos.z}).name ~= "air" ) then + + minetest.chat_send_player( placer:get_player_name(), 'Not enough vertical space to place the travelnet box!' ) + return; + end + return minetest.item_place(itemstack, placer, pointed_thing); + end, + +}) + + +minetest.register_craft({ + output = 'locked_travelnet:travelnet', + recipe = { + { 'travelnet:travelnet', 'locks:lock' }, + }, +}) + +print( "[Mod] locked_travelnet: loading locked_travelnet:travelnet"); diff --git a/textures/locked_travelnet_lock_inv.png b/textures/locked_travelnet_lock_inv.png new file mode 100644 index 0000000000000000000000000000000000000000..4c28b68be058920c21aae87d21e3ccef3efc033f GIT binary patch literal 4360 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 literal 0 HcmV?d00001 diff --git a/textures/locked_travelnet_travelnet_side_lock.png b/textures/locked_travelnet_travelnet_side_lock.png new file mode 100644 index 0000000000000000000000000000000000000000..cc3faacccc574fcaab8b1f0e73ca12f6c0b5726f GIT binary patch literal 553 zcmV+^0@nSBP)=}X=`Z}R@0=My{2Nt+84)FSLRs$=I86LgUoKp8%o zkKdu|Ism<1k3=Fd9MypEX=CX1^2r4NH$T780An|K%immoE2}nQ&Vvjziyd*e;IrL; zR!Zk8<4zFerPc84R=y$M%T?z*)PX8sEck2-=mL8u7zDKl0H$dIi204RN-3Qi#NR)a zfB|3wF2D~6;Bb3;`ko*H!8c zgYOOlPas?906)%+5rD?#-%|PU(x(6Ft709PSUZN~y&pA=p-0zsLjNuZY7xHZI&tlW z)yg2z#sFA}^-Txj@6z3uR_l_j0NyiA+;3EI?ou5?8#r>P%jI;|ihp<}mzP$rIrFhq zz6^Ma9U%c8Ao9)NN`E~Q`MG#Uo8MTg0d`CaoX?