123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670 |
- tpad = {}
- tpad.version = "1.2"
- tpad.mod_name = minetest.get_current_modname()
- tpad.texture = "tpad-texture.png"
- tpad.mesh = "tpad-mesh.obj"
- tpad.nodename = "tpad:tpad"
- tpad.mod_path = minetest.get_modpath(tpad.mod_name)
- local PRIVATE_PAD_STRING = "Private (only owner)"
- local PUBLIC_PAD_STRING = "Public (only owner's network)"
- local GLOBAL_PAD_STRING = "Global (any network)"
- local PRIVATE_PAD = 1
- local PUBLIC_PAD = 2
- local GLOBAL_PAD = 4
- local RED_ESCAPE = minetest.get_color_escape_sequence("#FF0000")
- local GREEN_ESCAPE = minetest.get_color_escape_sequence("#00FF00")
- local BLUE_ESCAPE = minetest.get_color_escape_sequence("#0000FF")
- local YELLOW_ESCAPE = minetest.get_color_escape_sequence("#FFFF00")
- local CYAN_ESCAPE = minetest.get_color_escape_sequence("#00FFFF")
- local MAGENTA_ESCAPE = minetest.get_color_escape_sequence("#FF00FF")
- local WHITE_ESCAPE = minetest.get_color_escape_sequence("#FFFFFF")
- local OWNER_ESCAPE_COLOR = CYAN_ESCAPE
- local padtype_flag_to_string = {
- [PRIVATE_PAD] = PRIVATE_PAD_STRING,
- [PUBLIC_PAD] = PUBLIC_PAD_STRING,
- [GLOBAL_PAD] = GLOBAL_PAD_STRING,
- }
- local padtype_string_to_flag = {
- [PRIVATE_PAD_STRING] = PRIVATE_PAD,
- [PUBLIC_PAD_STRING] = PUBLIC_PAD,
- [GLOBAL_PAD_STRING] = GLOBAL_PAD,
- }
- local short_padtype_string = {
- [PRIVATE_PAD] = "private",
- [PUBLIC_PAD] = "public",
- [GLOBAL_PAD] = "global",
- }
- local smartfs = dofile(tpad.mod_path .. "/lib/smartfs.lua")
- local notify = dofile(tpad.mod_path .. "/notify.lua")
- -- workaround storage to tell the main dialog about the last clicked pad
- local last_clicked_pos = {}
- -- workaround storage to tell the main dialog about last selected pad in the lists
- local last_selected_index = {}
- local last_selected_global_index = {}
- -- memory of shown waypoints
- local waypoint_hud_ids = {}
- minetest.register_privilege("tpad_admin", {
- description = "Can edit and destroy any tpad",
- give_to_singleplayer = true,
- })
- -- ========================================================================
- -- local helpers
- -- ========================================================================
- local function copy_file(source, dest)
- local src_file = io.open(source, "rb")
- if not src_file then
- return false, "copy_file() unable to open source for reading"
- end
- local src_data = src_file:read("*all")
- src_file:close()
- local dest_file = io.open(dest, "wb")
- if not dest_file then
- return false, "copy_file() unable to open dest for writing"
- end
- dest_file:write(src_data)
- dest_file:close()
- return true, "files copied successfully"
- end
- -- alias to make copy_file() available to storage.lua
- tpad._copy_file = copy_file
- local function custom_or_default(modname, path, filename)
- local default_filename = "default/" .. filename
- local full_filename = path .. "/custom." .. filename
- local full_default_filename = path .. "/" .. default_filename
- os.rename(path .. "/" .. filename, full_filename)
- local file = io.open(full_filename, "rb")
- if not file then
- minetest.debug("[" .. modname .. "] Copying " .. default_filename .. " to " .. filename .. " (path: " .. path .. ")")
- local success, err = copy_file(full_default_filename, full_filename)
- if not success then
- minetest.debug("[" .. modname .. "] " .. err)
- return false
- end
- file = io.open(full_filename, "rb")
- if not file then
- minetest.debug("[" .. modname .. "] Unable to load " .. filename .. " file from path " .. path)
- return false
- end
- end
- file:close()
- return full_filename
- end
- -- load storage facilities and verify it
- dofile(tpad.mod_path .. "/storage.lua")
- -- ========================================================================
- -- load custom recipe
- -- ========================================================================
- local recipes_filename = custom_or_default(tpad.mod_name, tpad.mod_path, "recipes.lua")
- if recipes_filename then
- local recipes = dofile(recipes_filename)
- if type(recipes) == "table" and recipes[tpad.nodename] then
- minetest.register_craft({
- output = tpad.nodename,
- recipe = recipes[tpad.nodename],
- })
- end
- end
- -- ========================================================================
- -- callback bound in register_chatcommand("tpad")
- -- ========================================================================
- function tpad.command(playername, param)
- tpad.hud_off(playername)
- if(param == "off") then return end
- local player = minetest.get_player_by_name(playername)
- local pads = tpad._get_stored_pads(playername)
- local shortest_distance = nil
- local closest_pad = nil
- local playerpos = player:getpos()
- for strpos, pad in pairs(pads) do
- local pos = minetest.string_to_pos(strpos)
- local distance = vector.distance(pos, playerpos)
- if not shortest_distance or distance < shortest_distance then
- closest_pad = {
- pos = pos,
- name = pad.name .. " " .. strpos,
- }
- shortest_distance = distance
- end
- end
- if closest_pad then
- waypoint_hud_ids[playername] = player:hud_add({
- hud_elem_type = "waypoint",
- name = closest_pad.name,
- world_pos = closest_pad.pos,
- number = 0xFF0000,
- })
- notify(playername, "Waypoint to " .. closest_pad.name .. " displayed")
- end
- end
- function tpad.hud_off(playername)
- local player = minetest.get_player_by_name(playername)
- local hud_id = waypoint_hud_ids[playername]
- if hud_id then
- player:hud_remove(hud_id)
- end
- end
- -- ========================================================================
- -- callbacks bound in register_node()
- -- ========================================================================
- function tpad.get_pos_from_pointed(pointed)
- local node_above = minetest.get_node_or_nil(pointed.above)
- local node_under = minetest.get_node_or_nil(pointed.under)
- if not node_above or not node_under then return end
- local def_above = minetest.registered_nodes[node_above.name]
- or minetest.nodedef_default
- local def_under = minetest.registered_nodes[node_under.name]
- or minetest.nodedef_default
- if not def_above.buildable_to and not def_under.buildable_to then return end
- if def_under.buildable_to then
- return pointed.under
- end
- return pointed.above
- end
- function tpad.on_place(itemstack, placer, pointed_thing)
- if tpad.max_total_pads_reached(placer) then
- notify.warn(placer, "You can't place any more pads")
- return itemstack
- end
- local pos = tpad.get_pos_from_pointed(pointed_thing) or {}
- itemstack = minetest.rotate_node(itemstack, placer, pointed_thing)
- local placed = minetest.get_node_or_nil(pos)
- if placed and placed.name == tpad.nodename then
- local meta = minetest.get_meta(pos)
- local playername = placer:get_player_name()
- meta:set_string("owner", playername)
- meta:set_string("infotext", "TPAD Station by " .. playername .. " - right click to interact")
- tpad.set_pad_data(pos, "", PRIVATE_PAD_STRING)
- end
- return itemstack
- end
- local submit = {}
- function tpad.max_total_pads_reached(placer)
- local placername = placer:get_player_name()
- if minetest.get_player_privs(placername).tpad_admin then
- return false
- end
- local localnet = submit.local_helper(placername)
- return #localnet.by_index >= tpad.get_max_total_pads()
- end
- function tpad.max_global_pads_reached(playername)
- if minetest.get_player_privs(playername).tpad_admin then
- return false
- end
- local localnet = submit.local_helper(playername)
- local count = 0
- for _, pad in pairs(localnet.by_name) do
- if pad.type == GLOBAL_PAD then
- count = count + 1
- end
- end
- return count >= tpad.get_max_global_pads()
- end
- function submit.global_helper()
- local allpads = tpad._get_all_pads()
- local result = {
- by_name = {},
- by_index = {},
- }
- for ownername, pads in pairs(allpads) do
- for strpos, pad in pairs(pads) do
- if pad.type == GLOBAL_PAD then
- pad = tpad.decorate_pad_data(strpos, pad, ownername)
- table.insert(result.by_index, pad.global_fullname)
- result.by_name[pad.global_fullname] = pad
- end
- end
- end
- table.sort(result.by_index)
- return result
- end
- function submit.local_helper(ownername, omit_private_pads)
- local pads = tpad._get_stored_pads(ownername)
- local result = {
- by_name = {},
- by_index = {},
- }
- for strpos, pad in pairs(pads) do
- local skip = omit_private_pads and pad.type == PRIVATE_PAD
- if not skip then
- pad = tpad.decorate_pad_data(strpos, pad, ownername)
- table.insert(result.by_index, pad.local_fullname)
- result.by_name[pad.local_fullname] = pad
- end
- end
- table.sort(result.by_index)
- return result
- end
- function submit.save(form)
- if form.playername ~= form.ownername and not minetest.get_player_privs(form.playername).tpad_admin then
- notify.warn(form.playername, "The selected pad doesn't belong to you")
- return
- end
- local padname = form.state:get("padname_field"):getText()
- local strpadtype = form.state:get("padtype_dropdown"):getSelectedItem()
- if strpadtype == GLOBAL_PAD_STRING and tpad.max_global_pads_reached(form.playername) then
- notify.warn(form.playername, "Can't add more pads to the Global Network, set to 'Public' instead")
- strpadtype = PUBLIC_PAD_STRING
- end
- tpad.set_pad_data(form.clicked_pos, padname, strpadtype)
- end
- function submit.teleport(form)
- local pads_listbox = form.state:get("pads_listbox")
- local selected_item = pads_listbox:getSelectedItem()
- local pad
- if form.globalnet then
- pad = form.globalnet.by_name[selected_item]
- else
- pad = form.localnet.by_name[selected_item]
- end
- if not pad then
- notify.err(form.playername, "Error! Missing pad data!")
- return
- end
- local player = minetest.get_player_by_name(form.playername)
- player:move_to(pad.pos, false)
- local padname = form.globalnet and pad.global_fullname or pad.local_fullname
- notify(form.playername, "Teleported to " .. padname)
- tpad.hud_off(form.playername)
- minetest.after(0, function()
- minetest.close_formspec(form.playername, form.formname)
- end)
- end
- function submit.admin(form)
- form.state = tpad.forms.admin:show(form.playername)
- form.formname = "tpad.forms.admin"
- local max_total_field = form.state:get("max_total_field")
- max_total_field:setText(tpad.get_max_total_pads())
- local max_global_field = form.state:get("max_global_field")
- max_global_field:setText(tpad.get_max_global_pads())
- local function admin_save()
- local max_total = tonumber(max_total_field:getText())
- local max_global = tonumber(max_global_field:getText())
- tpad.set_max_total_pads(max_total)
- tpad.set_max_global_pads(max_global)
- minetest.after(0, function()
- minetest.close_formspec(form.playername, form.formname)
- end)
- end
- max_total_field:onKeyEnter(admin_save)
- max_total_field:onKeyEnter(admin_save)
- form.state:get("save_button"):onClick(admin_save)
- end
- function submit.global(form)
- if minetest.get_player_privs(form.playername).tpad_admin then
- form.state = tpad.forms.global_network_admin:show(form.playername)
- form.formname = "tpad.forms.global_network_admin"
- form.state:get("admin_button"):onClick(function()
- minetest.after(0, function()
- submit.admin(form)
- end)
- end)
- else
- form.state = tpad.forms.global_network:show(form.playername)
- form.formname = "tpad.forms.global_network"
- end
- form.globalnet = submit.global_helper()
- local last_index = last_selected_global_index[form.playername]
- local pads_listbox = form.state:get("pads_listbox")
- pads_listbox:clearItems()
- for _, pad_item in ipairs(form.globalnet.by_index) do
- pads_listbox:addItem(pad_item)
- end
- pads_listbox:setSelected(last_index)
- pads_listbox:onClick(function()
- last_selected_global_index[form.playername] = pads_listbox:getSelected()
- end)
- pads_listbox:onDoubleClick(function() submit.teleport(form) end)
- form.state:get("teleport_button"):onClick(function() submit.teleport(form) end)
- form.state:get("local_button"):onClick(function()
- minetest.after(0, function()
- tpad.on_rightclick(form.clicked_pos, form.node, form.clicker)
- end)
- end)
- end
- function submit.delete(form)
- minetest.after(0, function()
- local pads_listbox = form.state:get("pads_listbox")
- local selected_item = pads_listbox:getSelectedItem()
- local delete_pad = form.localnet.by_name[selected_item]
- if not delete_pad then
- notify.warn(form.playername, "Please select a pad first")
- return
- end
- if form.playername ~= form.ownername and not minetest.get_player_privs(form.playername).tpad_admin then
- notify.warn(form.playername, "The selected pad doesn't belong to you")
- return
- end
- if minetest.pos_to_string(delete_pad.pos) == minetest.pos_to_string(form.clicked_pos) then
- notify.warn(form.playername, "You can't delete the current pad, destroy it manually")
- return
- end
- local function reshow_main()
- minetest.after(0, function()
- tpad.on_rightclick(form.clicked_pos, form.node, minetest.get_player_by_name(form.playername))
- end)
- end
- local delete_state = tpad.forms.confirm_pad_deletion:show(form.playername)
- delete_state:get("padname_label"):setText(
- YELLOW_ESCAPE .. delete_pad.local_fullname ..
- WHITE_ESCAPE .. " by " ..
- OWNER_ESCAPE_COLOR .. form.ownername
- )
- local confirm_button = delete_state:get("confirm_button")
- confirm_button:onClick(function()
- last_selected_index[form.playername .. ":" .. form.ownername] = nil
- tpad.del_pad(form.ownername, delete_pad.pos)
- minetest.remove_node(delete_pad.pos)
- notify(form.playername, "Pad " .. delete_pad.local_fullname .. " destroyed")
- reshow_main()
- end)
- local deny_button = delete_state:get("deny_button")
- deny_button:onClick(reshow_main)
- end)
- end
- function tpad.on_rightclick(clicked_pos, node, clicker)
- local playername = clicker:get_player_name()
- local clicked_meta = minetest.get_meta(clicked_pos)
- local ownername = clicked_meta:get_string("owner")
- local pad = tpad.get_pad_data(clicked_pos)
- if not pad or not ownername then
- notify.err(playername, "Error! Missing pad data!")
- return
- end
- local form = {}
- form.playername = playername
- form.clicker = clicker
- form.ownername = ownername
- form.clicked_pos = clicked_pos
- form.node = node
- form.omit_private_pads = false
- last_clicked_pos[playername] = clicked_pos;
- if ownername == playername or minetest.get_player_privs(playername).tpad_admin then
- form.formname = "tpad.forms.main_owner"
- form.state = tpad.forms.main_owner:show(playername)
- local padname_field = form.state:get("padname_field")
- padname_field:setLabel("This pad name (owned by " .. OWNER_ESCAPE_COLOR .. ownername .. WHITE_ESCAPE .. ")")
- padname_field:setText(pad.name)
- padname_field:onKeyEnter(function() submit.save(form) end)
- form.state:get("save_button"):onClick(function() submit.save(form) end)
- form.state:get("delete_button"):onClick(function() submit.delete(form) end)
- form.state:get("padtype_dropdown"):setSelectedItem(padtype_flag_to_string[pad.type])
- elseif pad.type == PRIVATE_PAD then
- notify.warn(playername, "This pad is private")
- return
- else
- form.omit_private_pads = true
- form.formname = "tpad.forms.main_visitor"
- form.state = tpad.forms.main_visitor:show(playername)
- form.state:get("visitor_label"):setText("Pad \"" .. pad.name .. "\", owned by " .. OWNER_ESCAPE_COLOR .. ownername)
- end
- form.localnet = submit.local_helper(ownername, form.omit_private_pads)
- local last_click_key = playername .. ":" .. ownername
- local last_index = last_selected_index[last_click_key]
- local pads_listbox = form.state:get("pads_listbox")
- pads_listbox:clearItems()
- for _, pad_item in ipairs(form.localnet.by_index) do
- pads_listbox:addItem(pad_item)
- end
- pads_listbox:setSelected(last_index)
- pads_listbox:onClick(function()
- last_selected_index[last_click_key] = pads_listbox:getSelected()
- end)
- pads_listbox:onDoubleClick(function() submit.teleport(form) end)
- form.state:get("teleport_button"):onClick(function() submit.teleport(form) end)
- form.state:get("global_button"):onClick(function()
- minetest.after(0, function()
- submit.global(form)
- end)
- end)
- end
- function tpad.can_dig(pos, player)
- local meta = minetest.get_meta(pos)
- local ownername = meta:get_string("owner")
- local playername = player:get_player_name()
- if ownername == "" or ownername == nil or playername == ownername
- or minetest.get_player_privs(playername).tpad_admin then
- return true
- end
- notify.warn(playername, "This pad doesn't belong to you")
- return false
- end
- function tpad.on_destruct(pos)
- local meta = minetest.get_meta(pos)
- local ownername = meta:get_string("owner")
- tpad.del_pad(ownername, pos)
- end
- -- ========================================================================
- -- forms
- -- ========================================================================
- tpad.forms = {}
- local function forms_add_padlist(state, is_global)
- local pads_listbox = state:listbox(0.2, 2.4, 7.6, 4, "pads_listbox", {})
- local teleport_button = state:button(0.2, 7, 1.5, 0, "teleport_button", "Teleport")
- local close_button = state:button(6.5, 7, 1.5, 0, "close_button", "Close")
- close_button:setClose(true)
- if is_global then
- local local_button = state:button(1.8, 7, 2.5, 0, "local_button", "Local Network")
- local_button:setClose(true)
- else
- local global_button = state:button(1.8, 7, 2.5, 0, "global_button", "Global Network")
- global_button:setClose(true)
- end
- state:label(0.2, 7.5, "teleport_label", "(you can doubleclick on a pad to teleport)")
- end
- tpad.forms.main_owner = smartfs.create("tpad.forms.main_owner", function(state)
- state:size(8, 8);
- state:field(0.5, 1, 6, 0, "padname_field", "", "")
- local save_button = state:button(6.5, 0.7, 1.5, 0, "save_button", "Save")
- save_button:setClose(true)
- local padtype_dropdown = state:dropdown(0.2, 1.2, 6.4, 0, "padtype_dropdown")
- padtype_dropdown:addItem(PRIVATE_PAD_STRING)
- padtype_dropdown:addItem(PUBLIC_PAD_STRING)
- padtype_dropdown:addItem(GLOBAL_PAD_STRING)
- local delete_button = state:button(4.4, 7, 1.5, 0, "delete_button", "Delete")
- forms_add_padlist(state)
- end)
- tpad.forms.main_visitor = smartfs.create("tpad.forms.main_visitor", function(state)
- state:size(8, 8)
- state:label(0.2, 1, "visitor_label", "")
- forms_add_padlist(state)
- end)
- tpad.forms.confirm_pad_deletion = smartfs.create("tpad.forms.confirm_pad_deletion", function(state)
- state:size(8, 2.5)
- state:label(0, 0, "intro_label", "Are you sure you want to destroy pad")
- state:label(0, 0.5, "padname_label", "")
- state:label(0, 1, "outro_label", "(you will not get the pad back)")
- state:button(0, 2.2, 2, 0, "confirm_button", "Yes, delete it")
- state:button(6, 2.2, 2, 0, "deny_button", "No, keep it")
- end)
- tpad.forms.global_network = smartfs.create("tpad.forms.global_network", function(state)
- state:size(8, 8)
- state:label(0.2, 1, "visitor_label", "Pick a pad from the Global Pads Network")
- local is_global = true
- forms_add_padlist(state, is_global)
- end)
- tpad.forms.global_network_admin = smartfs.create("tpad.forms.global_network_admin", function(state)
- state:size(8, 8)
- state:label(0.2, 1, "visitor_label", "Pick a pad from the Global Pads Network")
- local admin_button = state:button(4.4, 7, 1.5, 0, "admin_button", "Admin")
- admin_button:setClose(true)
- local is_global = true
- forms_add_padlist(state, is_global)
- end)
- tpad.forms.admin = smartfs.create("tpad.forms.admin", function(state)
- state:size(8, 8)
- state:label(0.2, 0.2, "admin_label", "TPAD Settings")
- state:field(0.5, 2, 6, 0, "max_total_field", "Max total pads per player")
- state:field(0.5, 3.5, 6, 0, "max_global_field", "Max global pads per player")
- local save_button = state:button(6.5, 0.7, 1.5, 0, "save_button", "Save")
- save_button:setClose(true)
- local close_button = state:button(6.5, 7, 1.5, 0, "close_button", "Close")
- close_button:setClose(true)
- end)
- -- ========================================================================
- -- helper functions
- -- ========================================================================
- function tpad.decorate_pad_data(pos, pad, ownername)
- pad = table.copy(pad)
- if type(pos) == "string" then
- pad.strpos = pos
- pad.pos = minetest.string_to_pos(pos)
- else
- pad.pos = pos
- pad.strpos = minetest.pos_to_string(pos)
- end
- pad.owner = ownername
- pad.name = pad.name or ""
- pad.type = pad.type or PUBLIC_PAD
- pad.local_fullname = pad.name .. " " .. pad.strpos .. " " .. short_padtype_string[pad.type]
- pad.global_fullname = "[" .. ownername .. "] " .. pad.name .. " " .. pad.strpos
- return pad
- end
- function tpad.get_pad_data(pos)
- local meta = minetest.get_meta(pos)
- local ownername = meta:get_string("owner")
- local pads = tpad._get_stored_pads(ownername)
- local strpos = minetest.pos_to_string(pos)
- local pad = pads[strpos]
- if not pad then return end
- return tpad.decorate_pad_data(pos, pad, ownername)
- end
- function tpad.set_pad_data(pos, padname, padtype)
- local meta = minetest.get_meta(pos)
- local ownername = meta:get_string("owner")
- local pads = tpad._get_stored_pads(ownername)
- local strpos = minetest.pos_to_string(pos)
- local pad = pads[strpos]
- if not pad then
- pad = {}
- end
- pad.name = padname
- pad.type = padtype_string_to_flag[padtype]
- pads[strpos] = pad
- tpad._set_stored_pads(ownername, pads)
- end
- function tpad.del_pad(ownername, pos)
- local pads = tpad._get_stored_pads(ownername)
- pads[minetest.pos_to_string(pos)] = nil
- tpad._set_stored_pads(ownername, pads)
- end
- -- ========================================================================
- -- register node and bind callbacks
- -- ========================================================================
- local collision_box = {
- type = "fixed",
- fixed = {
- { -0.5, -0.5, -0.5, 0.5, -0.3, 0.5 },
- }
- }
- minetest.register_node(tpad.nodename, {
- drawtype = "mesh",
- tiles = { tpad.texture },
- mesh = tpad.mesh,
- paramtype = "light",
- paramtype2 = "facedir",
- on_place = tpad.on_place,
- collision_box = collision_box,
- selection_box = collision_box,
- description = "Teleporter Pad",
- groups = {choppy = 2, dig_immediate = 2},
- on_rightclick = tpad.on_rightclick,
- can_dig = tpad.can_dig,
- on_destruct = tpad.on_destruct,
- })
- minetest.register_chatcommand("tpad", {func = tpad.command})
|