mirror of
https://repo.or.cz/minetest_playereffects.git
synced 2025-01-09 18:40:18 +01:00
554 lines
17 KiB
Lua
554 lines
17 KiB
Lua
--[=[ Main tables ]=]
|
|
|
|
playereffects = {}
|
|
|
|
--[[ table containing the groups (experimental) ]]
|
|
playereffects.groups = {}
|
|
|
|
--[[ table containing all the HUD info tables, indexed by player names.
|
|
A single HUD info table is formatted like this: { text_id = 1, icon_id=2, pos = 0 }
|
|
Where: text_id: HUD ID of the textual effect description
|
|
icon_id: HUD ID of the effect icon (optional)
|
|
pos: Y offset factor (starts with 0)
|
|
Example of full table:
|
|
{ ["player1"] = {{ text_id = 1, icon_id=4, pos = 0 }}, ["player2] = { { text_id = 5, icon_id=6, pos = 0 }, { text_id = 7, icon_id=8, pos = 1 } } }
|
|
]]
|
|
playereffects.hudinfos = {}
|
|
|
|
--[[ table containing all the effect types ]]
|
|
playereffects.effect_types = {}
|
|
|
|
--[[ table containing all the active effects ]]
|
|
playereffects.effects = {}
|
|
|
|
--[[ table containing all the inactive effects.
|
|
Effects become inactive if a player leaves an become active again if they join again. ]]
|
|
playereffects.inactive_effects = {}
|
|
|
|
-- Variable for counting the effect_id
|
|
playereffects.last_effect_id = 0
|
|
|
|
--[=[ Include settings ]=]
|
|
dofile(minetest.get_modpath("playereffects").."/settings.lua")
|
|
|
|
-- defaults
|
|
if(playereffects.use_hud == nil) then
|
|
playereffects.use_hud = true
|
|
end
|
|
if(playereffects.use_autosave == nil) then
|
|
playereffects.use_autosave = true
|
|
end
|
|
if(playereffects.autosave_time == nil) then
|
|
playereffects.autosave_time = 10
|
|
end
|
|
if(playereffects.use_examples == nil) then
|
|
playereffects.use_examples = false
|
|
end
|
|
|
|
--[=[ Load inactive_effects and last_effect_id from playereffects.mt, if this file exists ]=]
|
|
do
|
|
local filepath = minetest.get_worldpath().."/playereffects.mt"
|
|
local file = io.open(filepath, "r")
|
|
if file then
|
|
minetest.log("action", "[playereffects] playereffects.mt opened.")
|
|
local string = file:read()
|
|
io.close(file)
|
|
if(string ~= nil) then
|
|
local savetable = minetest.deserialize(string)
|
|
playereffects.inactive_effects = savetable.inactive_effects
|
|
minetest.debug("[playereffects] playereffects.mt successfully read.")
|
|
minetest.debug("[playereffects] inactive_effects = "..dump(playereffects.inactive_effects))
|
|
playereffects.last_effect_id = savetable.last_effect_id
|
|
minetest.debug("[playereffects] last_effect_id = "..dump(playereffects.last_effect_id))
|
|
|
|
end
|
|
end
|
|
end
|
|
|
|
function playereffects.next_effect_id()
|
|
playereffects.last_effect_id = playereffects.last_effect_id + 1
|
|
return playereffects.last_effect_id
|
|
end
|
|
|
|
--[=[ API functions ]=]
|
|
function playereffects.register_effect_type(effect_type_id, description, icon, groups, apply, cancel, hidden, cancel_on_death, repeat_interval)
|
|
local effect_type = {}
|
|
effect_type.description = description
|
|
effect_type.apply = apply
|
|
effect_type.groups = groups
|
|
effect_type.icon = icon
|
|
if cancel ~= nil then
|
|
effect_type.cancel = cancel
|
|
else
|
|
effect_type.cancel = function() end
|
|
end
|
|
if hidden ~= nil then
|
|
effect_type.hidden = hidden
|
|
else
|
|
effect_type.hidden = false
|
|
end
|
|
if cancel_on_death ~= nil then
|
|
effect_type.cancel_on_death = cancel_on_death
|
|
else
|
|
effect_type.cancel_on_death = true
|
|
end
|
|
effect_type.repeat_interval = repeat_interval
|
|
|
|
playereffects.effect_types[effect_type_id] = effect_type
|
|
minetest.log("action", "[playereffects] Effect type "..effect_type_id.." registered!")
|
|
end
|
|
|
|
function playereffects.apply_effect_type(effect_type_id, duration, entity, repeat_interval_time_left)
|
|
local start_time = os.time()
|
|
local is_player = false
|
|
if(type(entity)=="userdata") then
|
|
if(entity.is_player ~= nil) then
|
|
if(entity:is_player() == true) then
|
|
is_player = true
|
|
end
|
|
end
|
|
end
|
|
|
|
local playername = entity:get_player_name()
|
|
local groups = playereffects.effect_types[effect_type_id].groups
|
|
for k,v in pairs(groups) do
|
|
playereffects.cancel_effect_group(v, entity)
|
|
end
|
|
|
|
local metadata
|
|
if(playereffects.effect_types[effect_type_id].repeat_interval == nil) then
|
|
local status = playereffects.effect_types[effect_type_id].apply(entity)
|
|
if(status == false) then
|
|
if is_player then
|
|
minetest.log("action", "[playereffects] Attempt to apply effect type "..effect_type_id.." to player "..playername.." failed!")
|
|
end
|
|
return false
|
|
else
|
|
metadata = status
|
|
end
|
|
end
|
|
|
|
|
|
local effect_id = playereffects.next_effect_id()
|
|
|
|
-- repeat stuff
|
|
local repeat_interval = playereffects.effect_types[effect_type_id].repeat_interval
|
|
if(repeat_interval ~= nil) then
|
|
if(repeat_interval_time_left == nil) then
|
|
repeat_interval_time_left = repeat_interval
|
|
end
|
|
end
|
|
|
|
-- Handle HUD
|
|
if is_player then
|
|
local smallest_hudpos
|
|
local biggest_hudpos = -1
|
|
local free_hudpos
|
|
if(playereffects.hudinfos[playername] == nil) then
|
|
playereffects.hudinfos[playername] = {}
|
|
end
|
|
local hudinfos = playereffects.hudinfos[playername]
|
|
for effect_id, hudinfo in pairs(hudinfos) do
|
|
local hudpos = hudinfo.pos
|
|
if(hudpos > biggest_hudpos) then
|
|
biggest_hudpos = hudpos
|
|
end
|
|
if(smallest_hudpos == nil) then
|
|
smallest_hudpos = hudpos
|
|
elseif(hudpos < smallest_hudpos) then
|
|
smallest_hudpos = hudpos
|
|
end
|
|
end
|
|
if(smallest_hudpos == nil) then
|
|
free_hudpos = 0
|
|
elseif(smallest_hudpos >= 0) then
|
|
free_hudpos = smallest_hudpos - 1
|
|
else
|
|
free_hudpos = biggest_hudpos + 1
|
|
end
|
|
|
|
--[[ show no more than 20 effects on the screen, so that hud_update does not need to be called so often ]]
|
|
local text_id, icon_id
|
|
if(free_hudpos <= 20) then
|
|
text_id, icon_id = playereffects.hud_effect(effect_type_id, entity, free_hudpos, duration, repeat_interval_time_left)
|
|
local hudinfo = {
|
|
text_id = text_id,
|
|
icon_id = icon_id,
|
|
pos = free_hudpos,
|
|
}
|
|
playereffects.hudinfos[playername][effect_id] = hudinfo
|
|
else
|
|
text_id, icon_id = nil, nil
|
|
end
|
|
end
|
|
|
|
local effect = {
|
|
playername = playername,
|
|
effect_id = effect_id,
|
|
effect_type_id = effect_type_id,
|
|
start_time = start_time,
|
|
repeat_interval_start_time = start_time,
|
|
time_left = duration,
|
|
repeat_interval_time_left = repeat_interval_time_left,
|
|
metadata = metadata,
|
|
}
|
|
|
|
playereffects.effects[effect_id] = effect
|
|
|
|
if(repeat_interval ~= nil) then
|
|
minetest.after(repeat_interval_time_left, playereffects.repeater, effect_id, duration, entity, playereffects.effect_types[effect_type_id].apply)
|
|
else
|
|
minetest.after(duration, function(effect_id) playereffects.cancel_effect(effect_id) end, effect_id)
|
|
end
|
|
|
|
return effect_id
|
|
end
|
|
|
|
function playereffects.repeater(effect_id, repetitions, entity, apply)
|
|
local effect = playereffects.effects[effect_id]
|
|
if(effect ~= nil and entity ~= nil and entity:get_luaentity() ~= nil) then
|
|
local repetitions = effect.time_left
|
|
apply(entity)
|
|
repetitions = repetitions - 1
|
|
effect.time_left = repetitions
|
|
if(repetitions <= 0) then
|
|
playereffects.cancel_effect(effect_id)
|
|
else
|
|
local repeat_interval = playereffects.effect_types[effect.effect_type_id].repeat_interval
|
|
effect.repeat_interval_time_left = repeat_interval
|
|
effect.repeat_interval_start_time = os.time()
|
|
minetest.after(
|
|
repeat_interval,
|
|
playereffects.repeater,
|
|
effect_id,
|
|
repetitions,
|
|
entity,
|
|
apply
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
function playereffects.cancel_effect_type(effect_type_id, cancel_all, entity_or_playername)
|
|
local effects = playereffects.get_player_effects(entity_or_playername)
|
|
if(cancel_all==nil) then cancel_all = false end
|
|
for e=1, #effects do
|
|
if(effects[e].effect_type_id == effect_type_id) then
|
|
playereffects.cancel_effect(effects[e].effect_id)
|
|
if(cancel_all==false) then
|
|
return
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
function playereffects.cancel_effect_group(groupname, entity_or_playername)
|
|
local effects = playereffects.get_player_effects(entity_or_playername)
|
|
for e=1,#effects do
|
|
local effect = effects[e]
|
|
local thesegroups = playereffects.effect_types[effect.effect_type_id].groups
|
|
local delete = false
|
|
for g=1,#thesegroups do
|
|
if(thesegroups[g] == groupname) then
|
|
playereffects.cancel_effect(effect.effect_id)
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
function playereffects.get_remaining_effect_time(effect_id)
|
|
local now = os.time()
|
|
local effect = playereffects.effects[effect_id]
|
|
if(effect ~= nil) then
|
|
return (effect.time_left - os.difftime(now, effect.start_time))
|
|
else
|
|
return nil
|
|
end
|
|
end
|
|
|
|
function playereffects.cancel_effect(effect_id)
|
|
local effect = playereffects.effects[effect_id]
|
|
if(effect ~= nil) then
|
|
local player = minetest.get_player_by_name(effect.playername)
|
|
if player then
|
|
local hudinfo = playereffects.hudinfos[effect.playername][effect_id]
|
|
if(hudinfo ~= nil) then
|
|
if(hudinfo.text_id~=nil) then
|
|
player:hud_remove(hudinfo.text_id)
|
|
end
|
|
if(hudinfo.icon_id~=nil) then
|
|
player:hud_remove(hudinfo.icon_id)
|
|
end
|
|
playereffects.hudinfos[effect.playername][effect_id] = nil
|
|
end
|
|
|
|
-- TODO: Implement cancellation for non-players
|
|
playereffects.effect_types[effect.effect_type_id].cancel(effect, player)
|
|
end
|
|
playereffects.effects[effect_id] = nil
|
|
end
|
|
end
|
|
|
|
function playereffects.get_player_effects(entity_or_playername)
|
|
-- TODO: support entity
|
|
local playername = ""
|
|
if type(entity_or_playername) == "string" then
|
|
playername = entity_or_playername
|
|
elseif type(entity_or_playername) == "userdata" and entity_or_playername:is_player() then
|
|
playername = entity_or_playername:get_player_name()
|
|
else
|
|
return {}
|
|
end
|
|
if(minetest.get_player_by_name(playername) ~= nil) then
|
|
local effects = {}
|
|
for k,v in pairs(playereffects.effects) do
|
|
if(v.playername == playername) then
|
|
table.insert(effects, v)
|
|
end
|
|
end
|
|
return effects
|
|
else
|
|
return {}
|
|
end
|
|
end
|
|
|
|
function playereffects.has_effect_type(entity_or_playername, effect_type_id)
|
|
local pe = playereffects.get_player_effects(entity_or_playername)
|
|
for i=1,#pe do
|
|
if pe[i].effect_type_id == effect_type_id then
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
--[=[ Saving all data to file ]=]
|
|
function playereffects.save_to_file()
|
|
local save_time = os.time()
|
|
local savetable = {}
|
|
local inactive_effects = {}
|
|
for id,effecttable in pairs(playereffects.inactive_effects) do
|
|
local playername = id
|
|
if playername ~= "" then
|
|
if(inactive_effects[playername] == nil) then
|
|
inactive_effects[playername] = {}
|
|
end
|
|
for i=1,#effecttable do
|
|
table.insert(inactive_effects[playername], effecttable[i])
|
|
end
|
|
end
|
|
end
|
|
for id,effect in pairs(playereffects.effects) do
|
|
local new_duration, new_repeat_duration
|
|
if(playereffects.effect_types[effect.effect_type_id].repeat_interval ~= nil) then
|
|
new_duration = effect.time_left
|
|
new_repeat_duration = effect.repeat_interval_time_left - os.difftime(save_time, effect.repeat_interval_start_time)
|
|
else
|
|
new_duration = effect.time_left - os.difftime(save_time, effect.start_time)
|
|
end
|
|
local new_effect = {
|
|
effect_id = effect.effect_id,
|
|
effect_type_id = effect.effect_type_id,
|
|
time_left = new_duration,
|
|
repeat_interval_time_left = new_repeat_duration,
|
|
start_time = effect.start_time,
|
|
repeat_interval_start_time = effect.repeat_interval_start_time,
|
|
playername = effect.playername,
|
|
metadata = effect.metadata
|
|
}
|
|
if(inactive_effects[effect.playername] == nil) then
|
|
inactive_effects[effect.playername] = {}
|
|
end
|
|
if effect.playername ~= "" then
|
|
table.insert(inactive_effects[effect.playername], new_effect)
|
|
end
|
|
end
|
|
|
|
savetable.inactive_effects = inactive_effects
|
|
savetable.last_effect_id = playereffects.last_effect_id
|
|
|
|
local savestring = minetest.serialize(savetable)
|
|
|
|
local filepath = minetest.get_worldpath().."/playereffects.mt"
|
|
local file = io.open(filepath, "w")
|
|
if file then
|
|
file:write(savestring)
|
|
io.close(file)
|
|
minetest.log("action", "[playereffects] Wrote playereffects data into "..filepath..".")
|
|
else
|
|
minetest.log("error", "[playereffects] Failed to write playereffects data into "..filepath..".")
|
|
end
|
|
end
|
|
|
|
--[=[ Callbacks ]=]
|
|
--[[ Cancel all effects on player death ]]
|
|
minetest.register_on_dieplayer(function(player)
|
|
local effects = playereffects.get_player_effects(player:get_player_name())
|
|
for e=1,#effects do
|
|
if(playereffects.effect_types[effects[e].effect_type_id].cancel_on_death == true) then
|
|
playereffects.cancel_effect(effects[e].effect_id)
|
|
end
|
|
end
|
|
end)
|
|
|
|
|
|
minetest.register_on_leaveplayer(function(player)
|
|
local leave_time = os.time()
|
|
local playername = player:get_player_name()
|
|
local effects = playereffects.get_player_effects(playername)
|
|
|
|
playereffects.hud_clear(player)
|
|
|
|
if(playereffects.inactive_effects[playername] == nil) then
|
|
playereffects.inactive_effects[playername] = {}
|
|
end
|
|
for e=1,#effects do
|
|
local new_duration = effects[e].time_left - os.difftime(leave_time, effects[e].start_time)
|
|
local new_effect = effects[e]
|
|
new_effect.time_left = new_duration
|
|
table.insert(playereffects.inactive_effects[playername], new_effect)
|
|
playereffects.cancel_effect(effects[e].effect_id)
|
|
end
|
|
end)
|
|
|
|
minetest.register_on_shutdown(function()
|
|
minetest.log("action", "[playereffects] Server shuts down. Rescuing data into playereffects.mt")
|
|
playereffects.save_to_file()
|
|
end)
|
|
|
|
minetest.register_on_joinplayer(function(player)
|
|
local playername = player:get_player_name()
|
|
|
|
-- load all the effects again (if any)
|
|
if(playereffects.inactive_effects[playername] ~= nil) then
|
|
for i=1,#playereffects.inactive_effects[playername] do
|
|
local effect = playereffects.inactive_effects[playername][i]
|
|
playereffects.apply_effect_type(effect.effect_type_id, effect.time_left, player, effect.repeat_interval_time_left)
|
|
end
|
|
playereffects.inactive_effects[playername] = nil
|
|
end
|
|
end)
|
|
|
|
playereffects.globalstep_timer = 0
|
|
playereffects.autosave_timer = 0
|
|
minetest.register_globalstep(function(dtime)
|
|
playereffects.globalstep_timer = playereffects.globalstep_timer + dtime
|
|
playereffects.autosave_timer = playereffects.autosave_timer + dtime
|
|
-- Update HUDs of all players
|
|
if(playereffects.globalstep_timer >= 1) then
|
|
playereffects.globalstep_timer = 0
|
|
|
|
local players = minetest.get_connected_players()
|
|
for p=1,#players do
|
|
playereffects.hud_update(players[p])
|
|
end
|
|
end
|
|
-- Autosave into file
|
|
if(playereffects.use_autosave == true and playereffects.autosave_timer >= playereffects.autosave_time) then
|
|
playereffects.autosave_timer = 0
|
|
minetest.log("action", "[playereffects] Autosaving mod data to playereffects.mt ...")
|
|
playereffects.save_to_file()
|
|
end
|
|
end)
|
|
|
|
|
|
|
|
|
|
--[=[ HUD ]=]
|
|
function playereffects.hud_update(player)
|
|
if(playereffects.use_hud == true) then
|
|
local now = os.time()
|
|
local playername = player:get_player_name()
|
|
local hudinfos = playereffects.hudinfos[playername]
|
|
if(hudinfos ~= nil) then
|
|
for effect_id, hudinfo in pairs(hudinfos) do
|
|
local effect = playereffects.effects[effect_id]
|
|
if(effect ~= nil and hudinfo.text_id ~= nil) then
|
|
local description = playereffects.effect_types[effect.effect_type_id].description
|
|
local repeat_interval = playereffects.effect_types[effect.effect_type_id].repeat_interval
|
|
if(repeat_interval ~= nil) then
|
|
local repeat_interval_time_left = os.difftime(effect.repeat_interval_start_time + effect.repeat_interval_time_left, now)
|
|
player:hud_change(hudinfo.text_id, "text", description .. " ("..tostring(effect.time_left).."/"..tostring(repeat_interval_time_left) .. "s )")
|
|
else
|
|
local time_left = os.difftime(effect.start_time + effect.time_left, now)
|
|
player:hud_change(hudinfo.text_id, "text", description .. " ("..tostring(time_left).." s)")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
function playereffects.hud_clear(player)
|
|
if(playereffects.use_hud == true) then
|
|
local playername = player:get_player_name()
|
|
local hudinfos = playereffects.hudinfos[playername]
|
|
if(hudinfos ~= nil) then
|
|
for effect_id, hudinfo in pairs(hudinfos) do
|
|
local effect = playereffects.effects[effect_id]
|
|
if(hudinfo.text_id ~= nil) then
|
|
player:hud_remove(hudinfo.text_id)
|
|
end
|
|
if(hudinfo.icon_id ~= nil) then
|
|
player:hud_remove(hudinfo.icon_id)
|
|
end
|
|
playereffects.hudinfos[playername][effect_id] = nil
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
function playereffects.hud_effect(effect_type_id, player, pos, time_left, repeat_interval_time_left)
|
|
local text_id, icon_id
|
|
local effect_type = playereffects.effect_types[effect_type_id]
|
|
if(playereffects.use_hud == true and effect_type.hidden == false) then
|
|
local color
|
|
if(playereffects.effect_types[effect_type_id].cancel_on_death == true) then
|
|
color = 0xFFFFFF
|
|
else
|
|
color = 0xF0BAFF
|
|
end
|
|
local description = playereffects.effect_types[effect_type_id].description
|
|
local text
|
|
if(repeat_interval_time_left ~= nil) then
|
|
text = description .. " ("..tostring(time_left).."/"..tostring(repeat_interval_time_left) .. "s )"
|
|
else
|
|
text = description .. " ("..tostring(time_left).." s)"
|
|
end
|
|
text_id = player:hud_add({
|
|
hud_elem_type = "text",
|
|
position = { x = 1, y = 0.3 },
|
|
name = "effect_"..effect_type_id,
|
|
text = text,
|
|
scale = { x = 170, y = 20},
|
|
alignment = { x = -1, y = 0 },
|
|
direction = 1,
|
|
number = color,
|
|
offset = { x = -5, y = pos*20 }
|
|
})
|
|
if(playereffects.effect_types[effect_type_id].icon ~= nil) then
|
|
icon_id = player:hud_add({
|
|
hud_elem_type = "image",
|
|
scale = { x = 1, y = 1 },
|
|
position = { x = 1, y = 0.3 },
|
|
name = "effect_icon_"..effect_type_id,
|
|
text = playereffects.effect_types[effect_type_id].icon,
|
|
alignment = { x = -1, y=0 },
|
|
direction = 0,
|
|
offset = { x = -186, y = pos*20 },
|
|
})
|
|
end
|
|
else
|
|
text_id = nil
|
|
icon_id = nil
|
|
end
|
|
return text_id, icon_id
|
|
end
|
|
|
|
|
|
-- LOAD EXAMPLES
|
|
if(playereffects.use_examples == true) then
|
|
dofile(minetest.get_modpath(minetest.get_current_modname()).."/examples.lua")
|
|
end
|