39fc1ea626
The YCTIWY mod is already keeping track of offline player positions, so allow admins to use those for teleportation. This is sort of like the szutil_offlinepos mod, but as a standard game feature now. It can be useful for admins to quickly locate players who are trapped and need rescue, or suspected of griefing and need to be confirmed.
488 lines
13 KiB
Lua
488 lines
13 KiB
Lua
-- LUALOCALS < ---------------------------------------------------------
|
|
local ItemStack, math, minetest, next, nodecore, pairs, string, table,
|
|
vector
|
|
= ItemStack, math, minetest, next, nodecore, pairs, string, table,
|
|
vector
|
|
local math_pi, math_random, string_format, table_concat
|
|
= math.pi, math.random, string.format, table.concat
|
|
-- LUALOCALS > ---------------------------------------------------------
|
|
|
|
nodecore.amcoremod()
|
|
|
|
local disabled = nodecore.setting_bool(
|
|
"nc_yctiwy_disable",
|
|
false,
|
|
"Disable offline player inventory database",
|
|
[[By default, players' offline position and inventory is saved
|
|
for displaying offline "ghost" entities, which can be hidden
|
|
by a different setting, but even when hidden the database is
|
|
still kept up to date. Enabling this setting will completely
|
|
disable that database and purge all data, saving server
|
|
resources, but making ghosts not display when reenabled until
|
|
each player has joined again at least once.]]
|
|
)
|
|
|
|
local hidden = nodecore.setting_bool(
|
|
"nc_yctiwy_hide",
|
|
false,
|
|
"Do not display offline player entities",
|
|
[[By default, players' offline "ghosts" and inventories are
|
|
displayed as entities. Enabling this option hides
|
|
those, reducing resource impact on both client and server,
|
|
but also preventing players from accessing offline player
|
|
inventories. The offline database is still maintained, in
|
|
case the entities are later reenabled.]]
|
|
) or disabled
|
|
|
|
------------------------------------------------------------------------
|
|
-- DATABASE
|
|
|
|
local modname = minetest.get_current_modname()
|
|
local modstore = minetest.get_mod_storage()
|
|
|
|
local db = modstore:get_string("db")
|
|
if db == "" then db = nil end
|
|
db = db and minetest.deserialize(db)
|
|
db = db or {}
|
|
|
|
local drop = modstore:get_string("drop")
|
|
if drop == "" then drop = nil end
|
|
drop = drop and minetest.deserialize(drop)
|
|
drop = drop or {}
|
|
|
|
local function savedb()
|
|
modstore:set_string("db", next(db) and minetest.serialize(db) or "")
|
|
return modstore:set_string("drop", next(drop) and minetest.serialize(drop) or "")
|
|
end
|
|
|
|
if disabled then
|
|
for k, v in pairs(db) do
|
|
if not v.taken then
|
|
db[k] = nil
|
|
end
|
|
end
|
|
savedb()
|
|
elseif not (next(db) or next(drop)) then
|
|
function nodecore.yctiwy_import(newdb, newdrop)
|
|
db = newdb
|
|
drop = newdrop
|
|
savedb()
|
|
nodecore.yctiwy_import = nil
|
|
end
|
|
end
|
|
|
|
local function savestate(player)
|
|
if disabled or not nodecore.player_visible(player) then
|
|
db[player:get_player_name()] = nil
|
|
return
|
|
end
|
|
local ent = {
|
|
pos = player:get_pos(),
|
|
inv = {}
|
|
}
|
|
ent.pos.y = ent.pos.y + 1
|
|
local inv = player:get_inventory()
|
|
for i = 1, inv:get_size("main") do
|
|
local stack = inv:get_stack("main", i)
|
|
ent.inv[i] = stack:to_string()
|
|
end
|
|
ent.seen = minetest.get_gametime()
|
|
db[player:get_player_name()] = ent
|
|
end
|
|
|
|
------------------------------------------------------------------------
|
|
-- TELEPORT COMMAND
|
|
|
|
local teleport = minetest.registered_chatcommands.teleport
|
|
if teleport and teleport.func then
|
|
local cannot = nodecore.translate("Cannot teleport an offline player.")
|
|
local oldfunc = teleport.func
|
|
teleport.func = function(caller, ...)
|
|
local oldget = minetest.get_player_by_name
|
|
local function helper(...)
|
|
minetest.get_player_by_name = oldget
|
|
return ...
|
|
end
|
|
function minetest.get_player_by_name(name, ...)
|
|
local player = oldget(name, ...)
|
|
if player then return player end
|
|
local s = db[name]
|
|
if s and s.pos then
|
|
return {
|
|
get_pos = function() return s.pos end,
|
|
set_pos = function()
|
|
minetest.after(0, function()
|
|
return minetest.chat_send_player(caller, cannot)
|
|
end)
|
|
end,
|
|
get_attach = function() end,
|
|
}
|
|
end
|
|
end
|
|
return helper(oldfunc(caller, ...))
|
|
end
|
|
end
|
|
|
|
------------------------------------------------------------------------
|
|
-- MISC/UTILITY/COMMON
|
|
|
|
local function areaunloaded(pos)
|
|
local basepos = vector.floor(pos)
|
|
if minetest.get_node(pos).name == "ignore" then return true end
|
|
for dx = -1, 1, 2 do
|
|
for dy = -1, 1, 2 do
|
|
for dz = -1, 1, 2 do
|
|
if minetest.get_node({
|
|
x = basepos.x + dx,
|
|
y = basepos.y + dy,
|
|
z = basepos.z + dz
|
|
}).name == "ignore" then return true end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local function box(n) return {-n, -n, -n, n, n, n} end
|
|
|
|
local desctitle = nodecore.translate("Offline Player")
|
|
local function getdesc(pname, stack)
|
|
local d = desctitle .. "\n" .. nodecore.notranslate(pname)
|
|
if stack then return d .. "\n" .. nodecore.touchtip_stack(stack) end
|
|
return d
|
|
end
|
|
|
|
local function spawnentity(pos, subname, values)
|
|
if areaunloaded(pos) then return end
|
|
local o = minetest.add_entity(pos, modname .. ":" .. subname)
|
|
o = o and o:get_luaentity()
|
|
if not o then return end
|
|
nodecore.log("info", modname .. ": spawned " .. subname .. " at "
|
|
.. minetest.pos_to_string(pos, 2) .. (values.pname
|
|
and (" for " .. values.pname) or ""))
|
|
for k, v in pairs(values) do o[k] = v end
|
|
return o:yctiwy_check()
|
|
end
|
|
|
|
local function entrystack(entry, slot)
|
|
if not entry then return end
|
|
if entry.taken and entry.taken[slot] then return end
|
|
local stack = entry.inv and entry.inv[slot]
|
|
if (not stack) or (stack == "") then return end
|
|
stack = ItemStack(stack)
|
|
local def = minetest.registered_items[stack:get_name()]
|
|
if (not def) or def.virtual_item then return end
|
|
return stack
|
|
end
|
|
|
|
------------------------------------------------------------------------
|
|
-- INVENTORY ENTITIES
|
|
|
|
local thiefdata = {}
|
|
|
|
minetest.register_entity(modname .. ":slotent", {
|
|
initial_properties = nodecore.stackentprops(),
|
|
is_yctiwy = true,
|
|
slot = 1,
|
|
loose = 0,
|
|
yctiwy_check = function(self)
|
|
local ent = db[self.pname]
|
|
if minetest.get_player_by_name(self.pname) then
|
|
return self.object:remove()
|
|
end
|
|
local stack = entrystack(ent, self.slot)
|
|
if not stack then return self.object:remove() end
|
|
local props = nodecore.stackentprops(stack)
|
|
props.visual_size.x = props.visual_size.x / 2
|
|
props.visual_size.y = props.visual_size.y / 2
|
|
if props.is_visible then
|
|
props.collisionbox = box(0.2)
|
|
end
|
|
if not self.yawed then
|
|
self.yawed = true
|
|
self.object:set_yaw(math_random() * math_pi * 2)
|
|
end
|
|
self.rotating = self.rotating or (math_random(1, 2) == 1) and 0.1 or -0.1
|
|
props.automatic_rotate = self.rotating
|
|
self.object:set_properties(props)
|
|
self.description = getdesc(self.pname, stack)
|
|
end,
|
|
on_punch = function(self, puncher)
|
|
local ent = db[self.pname]
|
|
|
|
local punchname = puncher and puncher:get_player_name()
|
|
if not punchname then return end
|
|
local state = thiefdata[punchname]
|
|
if state and (state.target ~= self.pname or state.slot ~= self.slot
|
|
or state.exp < nodecore.gametime) then state = nil end
|
|
state = state or {
|
|
target = self.pname,
|
|
slot = self.slot,
|
|
final = nodecore.gametime + 2
|
|
}
|
|
state.exp = nodecore.gametime + 2
|
|
thiefdata[punchname] = state
|
|
if nodecore.gametime < state.final then return end
|
|
|
|
local stack = entrystack(ent, self.slot)
|
|
if not stack then return end
|
|
|
|
local function steallog(desc)
|
|
nodecore.log("action", string_format(
|
|
"%s: %q %s %q slot %d item %q at %s",
|
|
modname, puncher:get_player_name(), desc,
|
|
self.pname, self.slot, stack:to_string(),
|
|
minetest.pos_to_string(ent.pos, 0)))
|
|
end
|
|
local rp = vector.round(ent.pos)
|
|
if minetest.is_protected(rp, punchname) and
|
|
not minetest.is_protected(rp, self.pname) then
|
|
steallog("attempts to steal")
|
|
minetest.record_protection_violation(rp, punchname)
|
|
return
|
|
end
|
|
steallog("steals")
|
|
|
|
ent.taken = ent.taken or {}
|
|
ent.taken[self.slot] = true
|
|
savedb()
|
|
stack = puncher:get_inventory():add_item("main", stack)
|
|
if not stack:is_empty() then
|
|
nodecore.item_eject(self.object:get_pos(), stack)
|
|
end
|
|
return self.object:remove()
|
|
end
|
|
})
|
|
|
|
local s = 0.25
|
|
local offsets = {
|
|
{x = -s, y = -s, z = -s},
|
|
{x = s, y = -s, z = -s},
|
|
{x = -s, y = s, z = -s},
|
|
{x = s, y = s, z = -s},
|
|
{x = -s, y = -s, z = s},
|
|
{x = s, y = s, z = s},
|
|
{x = -s, y = s, z = s},
|
|
{x = s, y = -s, z = s}
|
|
}
|
|
|
|
local function spawninv(name, entry, existdb)
|
|
entry = entry or db[name]
|
|
if not entry then return end
|
|
|
|
for i = 1, #offsets do
|
|
if not existdb[name .. ":" .. i] and entrystack(entry, i) then
|
|
local p = vector.add(offsets[i], entry.pos)
|
|
spawnentity(p, "slotent", {
|
|
pname = name,
|
|
slot = i
|
|
})
|
|
end
|
|
end
|
|
end
|
|
|
|
------------------------------------------------------------------------
|
|
-- MARKER ENTITY
|
|
|
|
local function markertexture(pname)
|
|
local colors = {nodecore.player_model_colors(pname)}
|
|
while #colors > 3 do colors[#colors] = nil end
|
|
for i = 1, #colors do
|
|
colors[i] = string_format("(%s_marker_%d.png^[multiply:%s)",
|
|
modname, i, colors[i])
|
|
end
|
|
return table_concat(colors, "^")
|
|
end
|
|
|
|
minetest.register_entity(modname .. ":marker", {
|
|
initial_properties = {
|
|
visual = "sprite",
|
|
textures = {"nc_player_wield_slot.png"},
|
|
collisionbox = box(0.1),
|
|
visual_size = {x = 0.2, y = 0.2},
|
|
is_visible = true,
|
|
static_save = false
|
|
},
|
|
is_yctiwy = true,
|
|
yctiwy_check = function(self)
|
|
if (not self.pname) or minetest.get_player_by_name(self.pname)
|
|
or (not minetest.player_exists(self.pname)) or (not db[self.pname]) then
|
|
return self.object:remove()
|
|
end
|
|
self.object:set_properties({textures = {markertexture(self.pname)}})
|
|
end,
|
|
on_punch = function(self)
|
|
self.object:set_properties({pointable = false})
|
|
self.pointtime = 2
|
|
self.on_step = function(me, dtime)
|
|
local pt = me.pointtime or 0
|
|
pt = pt - dtime
|
|
if pt > 0 then
|
|
me.pointtime = pt
|
|
else
|
|
me.object:set_properties({pointable = true})
|
|
me.pointtime = nil
|
|
me.on_step = nil
|
|
end
|
|
end
|
|
end
|
|
})
|
|
|
|
local function spawnmarker(name, entry, existdb)
|
|
if existdb[name .. ":"] then return end
|
|
|
|
entry = entry or db[name]
|
|
if not entry then return end
|
|
|
|
spawnentity(entry.pos, "marker", {
|
|
description = getdesc(name),
|
|
pname = name
|
|
})
|
|
end
|
|
|
|
------------------------------------------------------------------------
|
|
-- STOLEN ITEMS
|
|
|
|
local takenitem = modname .. ":taken"
|
|
nodecore.register_virtual_item(takenitem, {
|
|
description = "",
|
|
inventory_image = "[combine:1x1",
|
|
hotbar_type = "yctiwy_taken",
|
|
})
|
|
|
|
nodecore.register_aism({
|
|
itemnames = takenitem,
|
|
interval = 1,
|
|
chance = 1,
|
|
action = function(stack)
|
|
local exp = stack:get_meta():get_float("exp") or 0
|
|
if exp < nodecore.gametime then return "" end
|
|
end
|
|
})
|
|
|
|
------------------------------------------------------------------------
|
|
-- PLAYER HOOKS
|
|
|
|
local recheck_timer = 0
|
|
|
|
minetest.register_on_leaveplayer(function(player)
|
|
recheck_timer = 0
|
|
savestate(player)
|
|
return savedb()
|
|
end)
|
|
|
|
minetest.register_on_shutdown(function()
|
|
for _, pl in pairs(minetest.get_connected_players()) do
|
|
savestate(pl)
|
|
end
|
|
return savedb()
|
|
end)
|
|
|
|
minetest.register_on_joinplayer(function(player)
|
|
recheck_timer = 0
|
|
local name = player:get_player_name()
|
|
local ent = db[name]
|
|
if (not ent) or (not ent.taken) then return end
|
|
local inv = player:get_inventory()
|
|
local takestack = ItemStack(takenitem)
|
|
takestack:get_meta():set_float("exp", nodecore.gametime + 4)
|
|
for k in pairs(ent.taken) do
|
|
inv:set_stack("main", k, takestack)
|
|
end
|
|
for _, o in pairs(minetest.luaentities) do
|
|
if o and o.is_yctiwy and o.pname == name then
|
|
o.object:remove()
|
|
end
|
|
end
|
|
end)
|
|
|
|
------------------------------------------------------------------------
|
|
-- TIMER
|
|
|
|
local function dumpinv(ent)
|
|
if areaunloaded(ent.pos) then return end
|
|
for k, v in pairs(ent.inv) do
|
|
if not (ent.taken and ent.taken[k]) then
|
|
local stack = ItemStack(v)
|
|
if not stack:is_empty() then
|
|
if minetest.add_item(ent.pos, stack) then
|
|
minetest.log(string_format("%s: deleted player %q"
|
|
.. " drops slot %d item %q at %s",
|
|
modname, ent.pname, k, stack:to_string(), minetest.pos_to_string(ent.pos, 0)))
|
|
ent.inv[k] = ""
|
|
else
|
|
return
|
|
end
|
|
end
|
|
end
|
|
end
|
|
return true
|
|
end
|
|
|
|
local function dropplayer(name, ent, skipsave)
|
|
ent = ent or db[name]
|
|
if not ent then return end
|
|
ent.pname = name
|
|
drop[#drop + 1] = ent
|
|
db[name] = nil
|
|
return skipsave or savedb()
|
|
end
|
|
|
|
local oldremove = minetest.remove_player
|
|
function minetest.remove_player(name, ...)
|
|
local function helper(...)
|
|
dropplayer(name)
|
|
return ...
|
|
end
|
|
return helper(oldremove(name, ...))
|
|
end
|
|
|
|
minetest.register_globalstep(function(dtime)
|
|
recheck_timer = recheck_timer - dtime
|
|
if recheck_timer > 0 then return end
|
|
recheck_timer = 3 + math_random() * 2
|
|
|
|
if not hidden then
|
|
for _, ent in pairs(minetest.luaentities) do
|
|
if ent.yctiwy_check then
|
|
ent:yctiwy_check()
|
|
end
|
|
end
|
|
end
|
|
|
|
local rollcall = {}
|
|
for _, pl in pairs(minetest.get_connected_players()) do
|
|
rollcall[pl:get_player_name()] = true
|
|
savestate(pl)
|
|
end
|
|
|
|
local existdb = {}
|
|
if not hidden then
|
|
|
|
for _, ent in pairs(minetest.luaentities) do
|
|
if ent.is_yctiwy then
|
|
existdb[(ent.pname or "") .. ":" .. (ent.slot or "")] = true
|
|
end
|
|
end
|
|
end
|
|
for name, ent in pairs(db) do
|
|
if not minetest.player_exists(name) then
|
|
dropplayer(name, ent, true)
|
|
elseif not (hidden or rollcall[name]) then
|
|
spawnmarker(name, ent, existdb)
|
|
spawninv(name, ent, existdb)
|
|
end
|
|
end
|
|
|
|
local batch = drop
|
|
drop = {}
|
|
for _, ent in pairs(batch) do
|
|
if areaunloaded(ent.pos) then
|
|
drop[#drop + 1] = ent
|
|
else
|
|
if not dumpinv(ent) then drop[#drop + 1] = ent end
|
|
end
|
|
end
|
|
|
|
return savedb()
|
|
end)
|