2023-12-02 13:00:25 +01:00
|
|
|
---
|
|
|
|
--- Copyright 2023, Michieal.
|
|
|
|
--- License: GPL3. (Default Mineclone2 License)
|
|
|
|
--- Created by michieal.
|
|
|
|
--- DateTime: 12/2/23 5:47 AM
|
|
|
|
---
|
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
-- Locals (and cached)
|
|
|
|
local DEBUG = false -- debug constant for troubleshooting.
|
|
|
|
local pairs = pairs
|
|
|
|
|
|
|
|
-- Globals
|
2023-12-02 13:00:25 +01:00
|
|
|
mcl_fovapi = {}
|
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
mcl_fovapi.default_fov = {} -- Handles default fov for players
|
2023-12-02 13:00:25 +01:00
|
|
|
mcl_fovapi.registered_modifiers = {}
|
|
|
|
mcl_fovapi.applied_modifiers = {}
|
|
|
|
|
|
|
|
minetest.register_on_joinplayer(function(player)
|
|
|
|
local name = player:get_player_name()
|
|
|
|
-- Assign default FOV
|
|
|
|
mcl_fovapi.default_fov[name] = player:get_fov()
|
2023-12-03 12:47:28 +01:00
|
|
|
|
|
|
|
if DEBUG then
|
|
|
|
minetest.log("FOV::Player: " .. name .. "\nFOV: " .. player:get_fov())
|
|
|
|
end
|
|
|
|
|
2023-12-02 13:00:25 +01:00
|
|
|
end)
|
|
|
|
|
|
|
|
minetest.register_on_leaveplayer(function(player)
|
|
|
|
local name = player:get_player_name()
|
2023-12-03 14:10:41 +01:00
|
|
|
|
|
|
|
-- handle clean up
|
2023-12-02 13:00:25 +01:00
|
|
|
mcl_fovapi.default_fov[name] = nil
|
2023-12-03 14:10:41 +01:00
|
|
|
mcl_fovapi.applied_modifiers[name] = nil
|
2023-12-02 13:00:25 +01:00
|
|
|
end)
|
|
|
|
|
2023-12-09 23:22:36 +01:00
|
|
|
function mcl_fovapi.register_modifier(def)
|
|
|
|
if type(def.name) ~= "string" then
|
|
|
|
error("Modifier name must be a string")
|
|
|
|
end
|
|
|
|
if type(def.fov_factor) ~= "number" then
|
|
|
|
error("FOV factor must be a number")
|
|
|
|
end
|
|
|
|
if type(def.time) ~= "number" then
|
|
|
|
error("Transition time must be a number")
|
|
|
|
end
|
|
|
|
|
|
|
|
if def.on_start ~= nil and type(def.on_start) ~= "function" then
|
|
|
|
error("Callback on_start must be a function")
|
|
|
|
end
|
|
|
|
if def.on_end ~= nil and type(def.on_end) ~= "function" then
|
|
|
|
error("Callback on_end must be a function")
|
|
|
|
end
|
|
|
|
|
|
|
|
local mdef = {}
|
|
|
|
|
|
|
|
mdef.fov_factor = def.fov_factor
|
|
|
|
mdef.time = def.time
|
|
|
|
|
|
|
|
if def.is_multiplier == false then mdef.is_multiplier = false
|
|
|
|
else mdef.is_multiplier = true end
|
|
|
|
if def.exclusive == true then mdef.exclusive = true
|
|
|
|
else mdef.exclusive = false end
|
|
|
|
|
|
|
|
mdef.on_start = def.on_start
|
|
|
|
mdef.on_end = def.on_end
|
2023-12-02 13:00:25 +01:00
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
if DEBUG then
|
|
|
|
minetest.log("FOV::Modifier Definition Registered:\n" .. dump(def))
|
|
|
|
end
|
|
|
|
|
2023-12-09 23:22:36 +01:00
|
|
|
mcl_fovapi.registered_modifiers[def.name] = mdef
|
2023-12-02 13:00:25 +01:00
|
|
|
|
|
|
|
end
|
|
|
|
|
2023-12-03 14:24:21 +01:00
|
|
|
minetest.register_on_respawnplayer(function(player)
|
|
|
|
mcl_fovapi.remove_all_modifiers(player:get_player_name())
|
|
|
|
end)
|
|
|
|
|
2023-12-02 13:00:25 +01:00
|
|
|
function mcl_fovapi.apply_modifier(player, modifier_name)
|
2023-12-03 12:47:28 +01:00
|
|
|
if player == nil then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
if modifier_name == nil then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
if mcl_fovapi.registered_modifiers[modifier_name] == nil then
|
|
|
|
return
|
|
|
|
end
|
2023-12-09 03:00:11 +01:00
|
|
|
if mcl_fovapi.applied_modifiers and mcl_fovapi.applied_modifiers[player] and mcl_fovapi.applied_modifiers[player][modifier_name] then
|
2023-12-09 02:38:07 +01:00
|
|
|
if mcl_fovapi.applied_modifiers[player][modifier_name] and mcl_fovapi.applied_modifiers[player][modifier_name] == true then
|
|
|
|
return
|
|
|
|
end
|
2023-12-03 15:03:01 +01:00
|
|
|
end
|
2023-12-02 13:00:25 +01:00
|
|
|
|
2023-12-09 03:00:11 +01:00
|
|
|
if mcl_fovapi.applied_modifiers[player] == nil then
|
|
|
|
mcl_fovapi.applied_modifiers[player] = {}
|
|
|
|
end
|
|
|
|
|
2023-12-02 13:00:25 +01:00
|
|
|
local modifier = mcl_fovapi.registered_modifiers[modifier_name]
|
|
|
|
if modifier.on_start ~= nil then
|
|
|
|
modifier.on_start(player)
|
|
|
|
end
|
|
|
|
|
|
|
|
mcl_fovapi.applied_modifiers[player][modifier_name] = true -- set the applied to be true.
|
2023-12-09 03:00:11 +01:00
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
if DEBUG then
|
|
|
|
minetest.log("FOV::Player Applied Modifiers :" .. dump(mcl_fovapi.applied_modifiers[player]))
|
|
|
|
end
|
|
|
|
local pname = player:get_player_name()
|
|
|
|
|
|
|
|
if DEBUG then
|
|
|
|
minetest.log("FOV::Modifier applied to player:" .. pname .. " modifier: " .. modifier_name)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- modifier apply code.
|
|
|
|
if modifier.exclusive == true then
|
|
|
|
-- if exclusive, reset the player's fov, and apply the new fov.
|
|
|
|
if modifier.is_multiplier then
|
|
|
|
player:set_fov(0, false, 0)
|
|
|
|
end
|
|
|
|
player:set_fov(modifier.fov_factor, modifier.is_multiplier, modifier.time)
|
|
|
|
else
|
|
|
|
-- not exclusive? let's apply it in the mix.
|
|
|
|
-- assume is_multiplier is true.
|
|
|
|
player:set_fov(modifier.fov_factor, true, modifier.time)
|
|
|
|
end
|
2023-12-02 13:00:25 +01:00
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
end
|
2023-12-02 13:00:25 +01:00
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
function mcl_fovapi.remove_modifier(player, modifier_name)
|
|
|
|
if player == nil then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2023-12-09 02:38:07 +01:00
|
|
|
if mcl_fovapi.applied_modifiers[player][modifier_name] == nil then
|
|
|
|
return
|
|
|
|
end
|
2023-12-03 15:03:01 +01:00
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
if DEBUG then
|
|
|
|
local name = player:get_player_name()
|
|
|
|
minetest.log("FOV::Player: " .. name .. " modifier: " .. modifier_name .. "removed.")
|
|
|
|
end
|
2023-12-02 13:00:25 +01:00
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
mcl_fovapi.applied_modifiers[player][modifier_name] = nil
|
2023-12-09 23:22:36 +01:00
|
|
|
local modifier = mcl_fovapi.registered_modifiers[modifier_name]
|
2023-12-03 12:47:28 +01:00
|
|
|
|
|
|
|
-- check for other fov modifiers, and set them up, or reset to default.
|
|
|
|
|
|
|
|
local applied = {}
|
|
|
|
for k, _ in pairs(mcl_fovapi.applied_modifiers[player]) do
|
|
|
|
applied[k] = mcl_fovapi.registered_modifiers[k]
|
|
|
|
end
|
|
|
|
|
|
|
|
if #applied == 0 then
|
2023-12-09 23:22:36 +01:00
|
|
|
player:set_fov(0, false, modifier.time)
|
2023-12-03 12:47:28 +01:00
|
|
|
return
|
|
|
|
end
|
|
|
|
local exc = false
|
|
|
|
for k in applied do
|
|
|
|
if applied[k].exclusive == true then
|
|
|
|
exc = applied[k]
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- handle exclusives.
|
|
|
|
if exc ~= false then
|
|
|
|
player:set_fov(exc.fov_factor, exc.is_multiplier, 0) -- we want this to be immediate.
|
|
|
|
else
|
|
|
|
-- handle normal fov modifiers.
|
2023-12-09 23:22:36 +01:00
|
|
|
local fov_factor = 1
|
2023-12-03 12:47:28 +01:00
|
|
|
for x in applied do
|
2023-12-09 23:22:36 +01:00
|
|
|
fov_factor = fov_factor * x.fov_factor
|
2023-12-03 12:47:28 +01:00
|
|
|
end
|
2023-12-09 23:22:36 +01:00
|
|
|
player:set_fov(fov_factor, true, modifier.time)
|
2023-12-03 12:47:28 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
if mcl_fovapi.registered_modifiers[modifier_name].on_end ~= nil then
|
|
|
|
mcl_fovapi.registered_modifiers[modifier_name].on_end(player)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function mcl_fovapi.remove_all_modifiers(player)
|
|
|
|
if player == nil then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
if DEBUG then
|
|
|
|
local name = player:get_player_name()
|
|
|
|
minetest.log("FOV::Player: " .. name .. " modifiers have been reset.")
|
|
|
|
end
|
|
|
|
|
|
|
|
for x in mcl_fovapi.applied_modifiers[player] do
|
|
|
|
x = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
player:set_fov(0, false, 0)
|
2023-12-03 14:24:21 +01:00
|
|
|
if mcl_fovapi.registered_modifiers[modifier_name].on_end ~= nil then
|
|
|
|
mcl_fovapi.registered_modifiers[modifier_name].on_end(player)
|
|
|
|
end
|
2023-12-02 13:00:25 +01:00
|
|
|
|
|
|
|
end
|
|
|
|
|
2023-12-03 12:47:28 +01:00
|
|
|
--[[
|
|
|
|
Notes:
|
|
|
|
set_fov(fov, is_multiplier, transition_time): Sets player's FOV
|
|
|
|
|
|
|
|
fov: FOV value.
|
|
|
|
is_multiplier: Set to true if the FOV value is a multiplier. Defaults to false.
|
|
|
|
transition_time: If defined, enables smooth FOV transition. Interpreted as the time (in seconds) to reach target FOV.
|
|
|
|
If set to 0, FOV change is instantaneous. Defaults to 0.
|
|
|
|
Set fov to 0 to clear FOV override.
|
|
|
|
|
|
|
|
--]]
|