forked from mtcontrib/factions
1278 lines
37 KiB
Lua
1278 lines
37 KiB
Lua
--! @class factions
|
|
--! @brief main class for factions
|
|
factions = {}
|
|
|
|
-- Create cold databases.
|
|
factions.factions = colddb.Colddb(minetest.get_worldpath() .. "/factions/factions")
|
|
factions.parcels = colddb.Colddb(minetest.get_worldpath() .. "/factions/parcels")
|
|
factions.players = colddb.Colddb(minetest.get_worldpath() .. "/factions/players")
|
|
factions.player_ips = colddb.Colddb(minetest.get_worldpath() .. "/factions/ips")
|
|
|
|
-- Memory only storage.
|
|
factions.onlineplayers = {}
|
|
|
|
---------------------
|
|
--! @brief returns whether a faction can be created or not (allows for implementation of blacklists and the like)
|
|
--! @param name String containing the faction's name
|
|
factions.can_create_faction = function(name)
|
|
if #name > factions_config.faction_name_max_length then
|
|
return false
|
|
elseif factions.factions.get(name) ~= nil then
|
|
return false
|
|
else
|
|
return true
|
|
end
|
|
end
|
|
|
|
|
|
util = {
|
|
coords3D_string = function(coords)
|
|
return coords.x..", "..coords.y..", "..coords.z
|
|
end
|
|
}
|
|
|
|
starting_ranks = {["leader"] = {"build", "door", "container", "name", "description", "motd", "invite", "kick"
|
|
, "player_title", "spawn", "with_draw", "territory", "claim", "access", "disband", "flags", "ranks", "promote"},
|
|
["moderator"] = {"claim", "door", "build", "spawn", "invite", "kick", "promote"},
|
|
["member"] = {"build", "container", "door"}
|
|
}
|
|
|
|
-- Faction permissions:
|
|
--
|
|
-- build: dig and place nodes
|
|
-- pain_build: dig and place nodes but take damage doing so
|
|
-- door: open/close or dig doors
|
|
-- container: be able to use containers like chest
|
|
-- name: set the faction's name
|
|
-- description: Set the faction description
|
|
-- motd: set the faction's message of the day
|
|
-- invite: (un)invite players to join the faction
|
|
-- kick: kick players off the faction
|
|
-- player_title: set player titles
|
|
-- spawn: set the faction's spawn
|
|
-- with_draw: withdraw money from the faction's bank
|
|
-- territory: claim or unclaim territory
|
|
-- claim: (un)claim parcels of land
|
|
-- access: manage access to territory and parcels of land to players or factions
|
|
-- disband: disband the faction
|
|
-- flags: manage faction's flags
|
|
-- ranks: create, edit, and delete ranks
|
|
-- promote: set a player's rank
|
|
-- diplomacy: be able to control the faction's diplomacy
|
|
|
|
factions.permissions = {"build", "pain_build", "door", "container", "name", "description", "motd", "invite", "kick"
|
|
, "player_title", "spawn", "with_draw", "territory", "claim", "access", "disband", "flags", "ranks", "promote"}
|
|
factions.permissions_desc = {"dig and place nodes", "dig and place nodes but take damage doing so", "open/close or dig faction doors", "be able to use containers like chest", "set the faction's name"
|
|
, "Set the faction description", "set the faction's message of the day", "(un)invite players to join the faction", "kick players off the faction", "set player titles", "set the faction's spawn"
|
|
, "withdraw money from the faction's bank", "claim or unclaim territory", "(un)claim parcels of land", "manage access to territory and parcels of land to players or factions"
|
|
, "disband the faction", "manage faction's flags", "create, edit, and delete ranks", "set a player's rank"}
|
|
|
|
-- open: can the faction be joined without an invite?
|
|
-- monsters: can monsters spawn on your land?
|
|
-- tax_kick: will players be kicked for not paying tax?
|
|
-- animals: can animals spawn on your land?
|
|
factions.flags = {"open", "monsters", "tax_kick", "animals"}
|
|
factions.flags_desc = {"can the faction be joined without an invite?", "can monsters spawn on your land?(unused)", "will players be kicked for not paying tax?(unused)", "can animals spawn on your land?(unused)"}
|
|
|
|
if factions_config.faction_diplomacy == true then
|
|
table.insert(factions.permissions, "diplomacy")
|
|
|
|
table.insert(factions.permissions_desc, "be able to control the faction's diplomacy")
|
|
|
|
local lt = starting_ranks["leader"]
|
|
table.insert(lt, "diplomacy")
|
|
starting_ranks["leader"] = lt
|
|
end
|
|
|
|
function factions.new()
|
|
return {
|
|
name = "",
|
|
--! @brief power of a faction (needed for parcel claiming)
|
|
power = factions_config.power,
|
|
--! @brief maximum power of a faction
|
|
maxpower = factions_config.maxpower,
|
|
--! @brief power currently in use
|
|
usedpower = 0.,
|
|
--! @brief list of player names
|
|
players = {},
|
|
--! @brief table of ranks/permissions
|
|
ranks = starting_ranks,
|
|
--! @brief name of the leader
|
|
leader = nil,
|
|
--! @brief spawn of the faction
|
|
spawn = {x=0, y=0, z=0},
|
|
--! @brief default joining rank for new members
|
|
default_rank = "member",
|
|
--! @brief default rank assigned to the leader
|
|
default_leader_rank = "leader",
|
|
--! @brief faction's description string
|
|
description = "Default faction description.",
|
|
--! @brief faction's message of the day.
|
|
message_of_the_day = "",
|
|
--! @brief list of players currently invited (can join with /f join)
|
|
invited_players = {},
|
|
--! @brief table of claimed parcels (keys are parcelpos strings)
|
|
land = {},
|
|
--! @brief table of allies
|
|
allies = {},
|
|
--
|
|
request_inbox = {},
|
|
--! @brief table of enemies
|
|
enemies = {},
|
|
--!
|
|
neutral = {},
|
|
--! @brief table of parcels/factions that are under attack
|
|
attacked_parcels = {},
|
|
--! @brief whether faction is closed or open (boolean)
|
|
join_free = false,
|
|
--! @brief gives certain privileges
|
|
is_admin = false,
|
|
--! @brief last time anyone logged on
|
|
last_logon = os.time(),
|
|
--! @brief how long this has been without parcels
|
|
no_parcel = os.time(),
|
|
}
|
|
end
|
|
|
|
|
|
--! @brief create a new empty faction
|
|
function factions.new_faction(name)
|
|
local faction = factions.new()
|
|
|
|
faction.name = name
|
|
factions.factions.set(name, faction)
|
|
factions.on_create(name)
|
|
minetest.after(1,
|
|
function(name)
|
|
factions.on_no_parcel(name)
|
|
end, name)
|
|
|
|
factions.onlineplayers[name] = {}
|
|
|
|
return faction
|
|
end
|
|
|
|
function factions.start_diplomacy(name, faction)
|
|
for l, i in pairs(factions.get_faction_list()) do
|
|
local fac = factions.factions.get(i)
|
|
if i ~= name and not (faction.neutral[i] or faction.allies[i] or faction.enemies[i]) then
|
|
if factions_config.faction_diplomacy == true then
|
|
factions.new_neutral(name, i)
|
|
factions.new_neutral(i, name)
|
|
else
|
|
factions.new_enemy(name, i)
|
|
factions.new_enemy(i, name)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
function factions.set_name(oldname, name)
|
|
local faction = factions.factions.get(oldname)
|
|
faction.name = name
|
|
|
|
for i, v in pairs(factions.get_faction_list()) do
|
|
if v ~= oldname then
|
|
local fac = factions.factions.get(v)
|
|
|
|
if fac.neutral[oldname] then
|
|
fac.neutral[oldname] = nil
|
|
fac.neutral[name] = true
|
|
end
|
|
|
|
if fac.allies[oldname] then
|
|
fac.allies[oldname] = nil
|
|
fac.allies[name] = true
|
|
end
|
|
|
|
if fac.enemies[oldname] then
|
|
fac.enemies[oldname] = nil
|
|
fac.enemies[name] = true
|
|
end
|
|
|
|
if fac.request_inbox[oldname] then
|
|
local value = fac.request_inbox[oldname]
|
|
fac.request_inbox[oldname] = nil
|
|
fac.request_inbox[name] = value
|
|
end
|
|
|
|
factions.factions.set(v, fac)
|
|
end
|
|
end
|
|
|
|
for parcel in pairs(faction.land) do
|
|
factions.parcels.set(parcel, name)
|
|
end
|
|
|
|
for playername in pairs(faction.players) do
|
|
factions.players.set(playername, name)
|
|
end
|
|
|
|
for playername in pairs(factions.onlineplayers[oldname]) do
|
|
updateFactionName(playername, name)
|
|
end
|
|
|
|
factions.onlineplayers[name] = factions.onlineplayers[oldname]
|
|
factions.onlineplayers[oldname] = nil
|
|
|
|
factions.factions.remove(oldname)
|
|
|
|
factions.factions.set(name, faction)
|
|
factions.on_set_name(name, oldname)
|
|
|
|
end
|
|
|
|
function factions.increase_power(name, power)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.power = faction.power + power
|
|
|
|
if faction.power > faction.maxpower - faction.usedpower then
|
|
faction.power = faction.maxpower - faction.usedpower
|
|
end
|
|
|
|
for i in pairs(factions.onlineplayers[name]) do
|
|
updateHudPower(minetest.get_player_by_name(i), faction)
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.decrease_power(name, power)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.power = faction.power - power
|
|
|
|
for i in pairs(factions.onlineplayers[name]) do
|
|
updateHudPower(minetest.get_player_by_name(i), faction)
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.increase_maxpower(name, power)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.maxpower = faction.maxpower + power
|
|
|
|
for i in pairs(factions.onlineplayers[name]) do
|
|
updateHudPower(minetest.get_player_by_name(i), faction)
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.decrease_maxpower(name, power)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.maxpower = faction.maxpower - power
|
|
|
|
if faction.maxpower < 0. then -- should not happen
|
|
faction.maxpower = 0.
|
|
end
|
|
|
|
for i in pairs(factions.onlineplayers[name]) do
|
|
updateHudPower(minetest.get_player_by_name(i), faction)
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.increase_usedpower(name, power)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.usedpower = faction.usedpower + power
|
|
|
|
for i in pairs(factions.onlineplayers[name]) do
|
|
updateHudPower(minetest.get_player_by_name(i), faction)
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.decrease_usedpower(name, power)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.usedpower = faction.usedpower - power
|
|
if faction.usedpower < 0. then
|
|
faction.usedpower = 0.
|
|
end
|
|
for i in pairs(factions.onlineplayers[name]) do
|
|
updateHudPower(minetest.get_player_by_name(i), faction)
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.count_land(name)
|
|
local count = 0.
|
|
for k, v in pairs(factions.factions.get(name).land) do
|
|
count = count + 1
|
|
end
|
|
return count
|
|
end
|
|
|
|
minetest.register_on_prejoinplayer(function(name, ip)
|
|
factions.player_ips.set(name, ip)
|
|
end)
|
|
|
|
function factions.add_player(name, player, rank)
|
|
local faction = factions.factions.get(name)
|
|
|
|
if factions.onlineplayers[name] == nil then
|
|
factions.onlineplayers[name] = {}
|
|
end
|
|
|
|
factions.onlineplayers[name][player] = true
|
|
|
|
factions.on_player_join(name, player)
|
|
|
|
if factions_config.enable_power_per_player then
|
|
local ip = factions.player_ips.get(player)
|
|
local notsame = true
|
|
for i, k in pairs(faction.players) do
|
|
local other_ip = factions.player_ips.get(i)
|
|
if other_ip == ip then
|
|
notsame = false
|
|
break
|
|
end
|
|
end
|
|
if notsame then
|
|
factions.increase_maxpower(name, factions_config.powermax_per_player)
|
|
end
|
|
end
|
|
|
|
faction.players[player] = rank or faction.default_rank
|
|
factions.players.set(player, name)
|
|
faction.invited_players[player] = nil
|
|
local pdata = minetest.get_player_by_name(player)
|
|
if pdata then
|
|
local ipc = pdata:is_player_connected(player)
|
|
if ipc then
|
|
createHudFactionName(pdata, name)
|
|
createHudPower(pdata, faction)
|
|
end
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.check_players_in_faction(name)
|
|
for i, k in pairs(factions.factions.get(name).players) do
|
|
return true
|
|
end
|
|
factions.disband(name, "Zero players on faction.")
|
|
return false
|
|
end
|
|
|
|
function factions.remove_player(name, player)
|
|
local faction = factions.factions.get(name)
|
|
|
|
if factions.onlineplayers[name] == nil then
|
|
factions.onlineplayers[name] = {}
|
|
end
|
|
|
|
factions.onlineplayers[name][player] = nil
|
|
|
|
faction.players[player] = nil
|
|
|
|
factions.factions.set(name, faction)
|
|
|
|
factions.players.remove(player)
|
|
factions.on_player_leave(name, player)
|
|
|
|
if factions_config.enable_power_per_player then
|
|
local ip = factions.player_ips.get(player)
|
|
local notsame = true
|
|
for i,k in pairs(faction.players) do
|
|
local other_ip = factions.player_ips.get(i)
|
|
if other_ip == ip then
|
|
notsame = false
|
|
break
|
|
end
|
|
end
|
|
if notsame then
|
|
factions.decrease_maxpower(name, factions_config.powermax_per_player)
|
|
end
|
|
end
|
|
|
|
local pdata = minetest.get_player_by_name(player)
|
|
if pdata then
|
|
local ipc = pdata:is_player_connected(player)
|
|
|
|
if ipc then
|
|
removeHud(pdata,"factionName")
|
|
removeHud(pdata,"powerWatch")
|
|
end
|
|
end
|
|
|
|
factions.check_players_in_faction(name)
|
|
end
|
|
|
|
local parcel_size = factions_config.parcel_size
|
|
|
|
--! @param parcelpos position of the wanted parcel
|
|
--! @return whether this faction can claim a parcelpos
|
|
function factions.can_claim_parcel(name, parcelpos)
|
|
local fn = factions.parcels.get(parcelpos)
|
|
|
|
if fn == nil then
|
|
return true
|
|
end
|
|
|
|
local faction = factions.factions.get(name)
|
|
|
|
if fn then
|
|
local fac = factions.factions.get(fn)
|
|
|
|
if fac.power < 0. and faction.power >= factions_config.power_per_parcel and not faction.allies[fn] and not faction.neutral[fn] then
|
|
return true
|
|
else
|
|
return false
|
|
end
|
|
elseif faction.power < factions_config.power_per_parcel then
|
|
return false
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
--! @brief claim a parcel, update power and update global parcels table
|
|
function factions.claim_parcel(name, parcelpos)
|
|
-- check if claiming over other faction's territory
|
|
local otherfac = factions.parcels.get(parcelpos)
|
|
if otherfac then
|
|
factions.unclaim_parcel(otherfac, parcelpos)
|
|
factions.parcelless_check(otherfac)
|
|
end
|
|
factions.parcels.set(parcelpos, name)
|
|
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.land[parcelpos] = true
|
|
|
|
factions.factions.set(name, faction)
|
|
|
|
factions.decrease_power(name, factions_config.power_per_parcel)
|
|
factions.increase_usedpower(name, factions_config.power_per_parcel)
|
|
factions.on_claim_parcel(name, parcelpos)
|
|
factions.parcelless_check(name)
|
|
end
|
|
|
|
--! @brief claim a parcel, update power and update global parcels table
|
|
function factions.unclaim_parcel(name, parcelpos)
|
|
factions.parcels.remove(parcelpos)
|
|
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.land[parcelpos] = nil
|
|
|
|
factions.factions.set(name, faction)
|
|
|
|
factions.increase_power(name, factions_config.power_per_parcel)
|
|
factions.decrease_usedpower(name, factions_config.power_per_parcel)
|
|
factions.on_unclaim_parcel(name, parcelpos)
|
|
factions.parcelless_check(name)
|
|
end
|
|
|
|
function factions.parcelless_check(name)
|
|
local faction = factions.factions.get(name)
|
|
|
|
if faction.land then
|
|
local count = 0
|
|
for index, value in pairs(faction.land) do
|
|
count = count + 1
|
|
break
|
|
end
|
|
if count > 0 then
|
|
if faction.no_parcel ~= -1 then
|
|
factions.broadcast(name, "Faction " .. name .. " will not be disbanded because it now has parcels.")
|
|
end
|
|
faction.no_parcel = -1
|
|
else
|
|
faction.no_parcel = os.time()
|
|
factions.on_no_parcel(name)
|
|
end
|
|
factions.factions.set(name, faction)
|
|
end
|
|
end
|
|
|
|
--! @brief disband faction, updates global players and parcels table
|
|
function factions.disband(name, reason)
|
|
local faction = factions.factions.get(name)
|
|
|
|
if not faction.is_admin then
|
|
for i, v in pairs(factions.get_faction_list()) do
|
|
local fac = factions.factions.get(v)
|
|
if fac ~= nil and fac.name ~= name then
|
|
if fac.enemies[name] then
|
|
factions.end_enemy(fac.name, name)
|
|
end
|
|
|
|
if fac.allies[name] then
|
|
factions.end_alliance(fac.name, name)
|
|
end
|
|
|
|
if fac.neutral[name] then
|
|
factions.end_neutral(fac.name, name)
|
|
end
|
|
|
|
if fac.request_inbox[name] then
|
|
fac.request_inbox[name] = nil
|
|
end
|
|
end
|
|
factions.factions.set(v, fac)
|
|
end
|
|
|
|
for k, _ in pairs(faction.players) do -- remove players affiliation
|
|
factions.players.remove(k)
|
|
end
|
|
|
|
for k, v in pairs(faction.land) do -- remove parcel claims
|
|
factions.parcels.remove(k)
|
|
end
|
|
|
|
factions.on_disband(name, reason)
|
|
|
|
for i, l in pairs(factions.onlineplayers[name]) do
|
|
removeHud(i, "factionName")
|
|
removeHud(i, "powerWatch")
|
|
end
|
|
|
|
factions.onlineplayers[name] = nil
|
|
|
|
factions.factions.remove(name)
|
|
end
|
|
end
|
|
|
|
--! @brief change the faction leader
|
|
function factions.set_leader(name, player)
|
|
local faction = factions.factions.get(name)
|
|
|
|
if faction.leader then
|
|
faction.players[faction.leader] = faction.default_rank
|
|
end
|
|
faction.leader = player
|
|
faction.players[player] = faction.default_leader_rank
|
|
factions.on_new_leader()
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.set_message_of_the_day(name, text)
|
|
local faction = factions.factions.get(name)
|
|
faction.message_of_the_day = text
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
--! @brief check permissions for a given player
|
|
--! @return boolean indicating permissions. Players not in faction always receive false
|
|
function factions.has_permission(name, player, permission)
|
|
local faction = factions.factions.get(name)
|
|
|
|
local p = faction.players[player]
|
|
if not p then
|
|
return false
|
|
end
|
|
local perms = faction.ranks[p]
|
|
if perms then
|
|
for i in ipairs(perms) do
|
|
if perms[i] == permission then
|
|
return true
|
|
end
|
|
end
|
|
else
|
|
return false
|
|
end
|
|
end
|
|
|
|
function factions.set_description(name, new)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.description = new
|
|
factions.on_change_description(name)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
--! @brief places player in invite list
|
|
function factions.invite_player(name, player)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.invited_players[player] = true
|
|
factions.on_player_invited(name, player)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
--! @brief removes player from invite list (can no longer join via /f join)
|
|
function factions.revoke_invite(name, player)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.invited_players[player] = nil
|
|
factions.on_revoke_invite(name, player)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
--! @brief set faction openness
|
|
function factions.toggle_join_free(name, bool)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.join_free = bool
|
|
factions.on_toggle_join_free(name)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
--! @return true if a player can use /f join, false otherwise
|
|
function factions.can_join(name, player)
|
|
local faction = factions.factions.get(name)
|
|
return faction.join_free or faction.invited_players[player]
|
|
end
|
|
|
|
function factions.new_alliance(name, faction)
|
|
local bfaction = factions.factions.get(name)
|
|
|
|
bfaction.allies[faction] = true
|
|
factions.on_new_alliance(name, faction)
|
|
if bfaction.enemies[faction] then
|
|
factions.end_enemy(name, faction)
|
|
end
|
|
if bfaction.neutral[faction] then
|
|
factions.end_neutral(name, faction)
|
|
end
|
|
|
|
factions.factions.set(name, bfaction)
|
|
end
|
|
|
|
function factions.end_alliance(name, faction)
|
|
local bfaction = factions.factions.get(name)
|
|
|
|
bfaction.allies[faction] = nil
|
|
factions.on_end_alliance(name, faction)
|
|
|
|
factions.factions.set(name, bfaction)
|
|
end
|
|
|
|
function factions.new_neutral(name, faction)
|
|
local bfaction = factions.factions.get(name)
|
|
|
|
bfaction.neutral[faction] = true
|
|
|
|
factions.on_new_neutral(name, faction)
|
|
if bfaction.allies[faction] then
|
|
factions.end_alliance(name, faction)
|
|
end
|
|
if bfaction.enemies[faction] then
|
|
factions.end_enemy(name, faction)
|
|
end
|
|
|
|
factions.factions.set(name, bfaction)
|
|
end
|
|
|
|
function factions.end_neutral(name, faction)
|
|
local bfaction = factions.factions.get(name)
|
|
|
|
bfaction.neutral[faction] = nil
|
|
factions.on_end_neutral(name, faction)
|
|
|
|
factions.factions.set(name, bfaction)
|
|
end
|
|
|
|
function factions.new_enemy(name, faction)
|
|
local bfaction = factions.factions.get(name)
|
|
|
|
bfaction.enemies[faction] = true
|
|
factions.on_new_enemy(name, faction)
|
|
|
|
if bfaction.allies[faction] then
|
|
factions.end_alliance(name, faction)
|
|
end
|
|
|
|
if bfaction.neutral[faction] then
|
|
factions.end_neutral(name, faction)
|
|
end
|
|
|
|
factions.factions.set(name, bfaction)
|
|
end
|
|
|
|
function factions.end_enemy(name, faction)
|
|
local bfaction = factions.factions.get(name)
|
|
|
|
bfaction.enemies[faction] = nil
|
|
factions.on_end_enemy(name, faction)
|
|
|
|
factions.factions.set(name, bfaction)
|
|
end
|
|
|
|
--! @brief faction's member will now spawn in a new place
|
|
function factions.set_spawn(name, pos)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.spawn = {x = pos.x, y = pos.y, z = pos.z}
|
|
factions.on_set_spawn(name)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.tp_spawn(name, playername)
|
|
local faction = factions.factions.get(name)
|
|
|
|
player = minetest.get_player_by_name(playername)
|
|
|
|
if player then
|
|
player:set_pos(faction.spawn)
|
|
minetest.sound_play("whoosh", {pos = faction.spawn, gain = 0.5, max_hear_distance = 10})
|
|
end
|
|
end
|
|
|
|
--! @brief create a new rank with permissions
|
|
--! @param rank the name of the new rank
|
|
--! @param rank a list with the permissions of the new rank
|
|
function factions.add_rank(name, rank, perms)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.ranks[rank] = perms
|
|
factions.on_add_rank(name, rank)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
--! @brief replace an rank's permissions
|
|
--! @param rank the name of the rank to edit
|
|
--! @param add or remove permissions to the rank
|
|
function factions.replace_privs(name, rank, perms)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.ranks[rank] = perms
|
|
factions.on_replace_privs(name, rank)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.remove_privs(name, rank, perms)
|
|
local faction = factions.factions.get(name)
|
|
|
|
local revoked = false
|
|
local p = faction.ranks[rank]
|
|
|
|
for index, perm in pairs(p) do
|
|
if table_Contains(perms, perm) then
|
|
revoked = true
|
|
table.remove(p, index)
|
|
end
|
|
end
|
|
|
|
faction.ranks[rank] = p
|
|
|
|
if revoked then
|
|
factions.on_remove_privs(name, rank, perms)
|
|
else
|
|
factions.broadcast(name, "No privilege was revoked from rank " .. rank .. ".")
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.add_privs(name, rank, perms)
|
|
local faction = factions.factions.get(name)
|
|
|
|
local added = false
|
|
local p = faction.ranks[rank]
|
|
|
|
for index, perm in pairs(perms) do
|
|
if not table_Contains(p, perm) then
|
|
added = true
|
|
table.insert(p, perm)
|
|
end
|
|
end
|
|
|
|
faction.ranks[rank] = p
|
|
|
|
if added then
|
|
factions.on_add_privs(name, rank, perms)
|
|
else
|
|
factions.broadcast(name, "The rank " .. rank .. " already has these privileges.")
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.set_rank_name(name, oldrank, newrank)
|
|
local faction = factions.factions.get(name)
|
|
|
|
local copyrank = faction.ranks[oldrank]
|
|
|
|
faction.ranks[newrank] = copyrank
|
|
faction.ranks[oldrank] = nil
|
|
|
|
for player, r in pairs(faction.players) do
|
|
if r == oldrank then
|
|
faction.players[player] = newrank
|
|
end
|
|
end
|
|
|
|
if oldrank == faction.default_leader_rank then
|
|
faction.default_leader_rank = newrank
|
|
factions.broadcast(name, "The default leader rank has been set to " .. newrank)
|
|
end
|
|
|
|
if oldrank == faction.default_rank then
|
|
faction.default_rank = newrank
|
|
factions.broadcast(name, "The default rank given to new players is set to " .. newrank)
|
|
end
|
|
|
|
factions.on_set_rank_name(name, oldrank, newrank)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.set_def_rank(name, rank)
|
|
local faction = factions.factions.get(name)
|
|
|
|
for player, r in pairs(faction.players) do
|
|
if r == rank or r == nil or not faction.ranks[r] then
|
|
faction.players[player] = rank
|
|
end
|
|
end
|
|
|
|
faction.default_rank = rank
|
|
factions.on_set_def_rank(name, rank)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
function factions.reset_ranks(name)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.ranks = starting_ranks
|
|
faction.default_rank = "member"
|
|
faction.default_leader_rank_rank = "leader"
|
|
for player, r in pairs(faction.players) do
|
|
if not player == leader and (r == nil or not faction.ranks[r]) then
|
|
faction.players[player] = faction.default_rank
|
|
elseif player == leader then
|
|
faction.players[player] = faction.default_leader_rank_rank
|
|
end
|
|
end
|
|
factions.on_reset_ranks(name)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
--! @brief delete a rank and replace it
|
|
--! @param rank the name of the rank to be deleted
|
|
--! @param newrank the rank given to players who were previously "rank"
|
|
function factions.delete_rank(name, rank, newrank)
|
|
local faction = factions.factions.get(name)
|
|
|
|
for player, r in pairs(faction.players) do
|
|
if r == rank then
|
|
faction.players[player] = newrank
|
|
end
|
|
end
|
|
faction.ranks[rank] = nil
|
|
factions.on_delete_rank(name, rank, newrank)
|
|
if rank == faction.default_leader_rank then
|
|
faction.default_leader_rank = newrank
|
|
factions.broadcast(name, "The default leader rank has been set to "..newrank)
|
|
end
|
|
if rank == faction.default_rank then
|
|
faction.default_rank = newrank
|
|
factions.broadcast(name, "The default rank given to new players is set to "..newrank)
|
|
end
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
--! @brief set a player's rank
|
|
function factions.promote(name, member, rank)
|
|
local faction = factions.factions.get(name)
|
|
|
|
faction.players[member] = rank
|
|
factions.on_promote(name, member)
|
|
|
|
factions.factions.set(name, faction)
|
|
end
|
|
|
|
--! @brief send a message to all members
|
|
function factions.broadcast(name, msg, sender)
|
|
if factions.onlineplayers[name] == nil then
|
|
factions.onlineplayers[name] = {}
|
|
end
|
|
|
|
local message = name .. "> ".. msg
|
|
|
|
if sender then
|
|
message = sender .. "@" .. message
|
|
end
|
|
|
|
message = "Faction<" .. message
|
|
for k, _ in pairs(factions.onlineplayers[name]) do
|
|
minetest.chat_send_player(k, message)
|
|
end
|
|
end
|
|
|
|
--! @brief checks whether a faction has at least one connected player
|
|
function factions.is_online(name)
|
|
if factions.onlineplayers[name] == nil then
|
|
factions.onlineplayers[name] = {}
|
|
end
|
|
for playername, _ in pairs(factions.onlineplayers[name]) do
|
|
return true
|
|
end
|
|
return false
|
|
end
|
|
|
|
--------------------------
|
|
-- callbacks for events --
|
|
|
|
function factions.on_create(name) --! @brief called when the faction is added to the global faction list
|
|
minetest.chat_send_all("Faction " .. name .. " has been created.")
|
|
end
|
|
|
|
function factions.on_set_name(name, oldname)
|
|
minetest.chat_send_all("Faction " .. oldname .. " has been changed its name to ".. name ..".")
|
|
end
|
|
|
|
function factions.on_no_parcel(name)
|
|
local faction = factions.factions.get(name)
|
|
|
|
local now = os.time() - faction.no_parcel
|
|
local l = factions_config.maximum_parcelless_faction_time
|
|
|
|
factions.broadcast(name, "This faction will disband in " .. l - now .. " seconds, because it has no parcels.")
|
|
end
|
|
|
|
function factions.on_player_leave(name, player)
|
|
factions.broadcast(name, player .. " has left this faction")
|
|
end
|
|
|
|
function factions.on_player_join(name, player)
|
|
factions.broadcast(name, player .. " has joined this faction")
|
|
end
|
|
|
|
function factions.on_claim_parcel(name, pos)
|
|
factions.broadcast(name, "Parcel (" .. pos .. ") has been claimed.")
|
|
end
|
|
|
|
function factions.on_unclaim_parcel(name, pos)
|
|
factions.broadcast(name, "Parcel ("..pos..") has been unclaimed.")
|
|
end
|
|
|
|
function factions.on_disband(name, reason)
|
|
local msg = "Faction " .. name .. " has been disbanded."
|
|
if reason then
|
|
msg = msg .. " (" .. reason .. ")"
|
|
end
|
|
minetest.chat_send_all(msg)
|
|
end
|
|
|
|
function factions.on_new_leader(name)
|
|
local faction = factions.factions.get(name)
|
|
factions.broadcast(name, faction.leader .. " is now the leader of this faction")
|
|
end
|
|
|
|
function factions.on_change_description(name)
|
|
local faction = factions.factions.get(name)
|
|
factions.broadcast(name, "Faction description has been modified to: " .. faction.description)
|
|
end
|
|
|
|
function factions.on_player_invited(name, player)
|
|
local faction = factions.factions.get(name)
|
|
minetest.chat_send_player(player, "You have been invited to faction " .. faction.name)
|
|
end
|
|
|
|
function factions.on_toggle_join_free(name, player)
|
|
local faction = factions.factions.get(name)
|
|
if faction.join_free then
|
|
factions.broadcast(name, "This faction is now invite-free.")
|
|
else
|
|
factions.broadcast(name, "This faction is no longer invite-free.")
|
|
end
|
|
end
|
|
|
|
function factions.on_new_alliance(name, faction)
|
|
factions.broadcast(name, "This faction is now allied with " .. faction)
|
|
end
|
|
|
|
function factions.on_end_alliance(name, faction)
|
|
factions.broadcast(name, "This faction is no longer allied with " .. faction .. "!")
|
|
end
|
|
|
|
function factions.on_new_neutral(name, faction)
|
|
factions.broadcast(name, "This faction is now neutral with ".. faction)
|
|
end
|
|
|
|
function factions.on_end_neutral(name, faction)
|
|
factions.broadcast(name, "This faction is no longer neutral with " .. faction .. "!")
|
|
end
|
|
|
|
function factions.on_new_enemy(name, faction)
|
|
factions.broadcast(name, "This faction is now at war with " .. faction)
|
|
end
|
|
|
|
function factions.on_end_enemy(name, faction)
|
|
factions.broadcast(name, "This faction is no longer at war with " .. faction .. "!")
|
|
end
|
|
|
|
function factions.on_set_spawn(name)
|
|
local faction = factions.factions.get(name)
|
|
factions.broadcast(name, "The faction spawn has been set to (" .. util.coords3D_string(faction.spawn) .. ").")
|
|
end
|
|
|
|
function factions.on_add_rank(name, rank)
|
|
local faction = factions.factions.get(name)
|
|
factions.broadcast(name, "The rank " .. rank .. " has been created with privileges: " .. table.concat(faction.ranks[rank], ", "))
|
|
end
|
|
|
|
function factions.on_replace_privs(name, rank)
|
|
local faction = factions.factions.get(name)
|
|
factions.broadcast(name, "The privileges in rank " .. rank .. " have been delete and changed to: " .. table.concat(faction.ranks[rank], ", "))
|
|
end
|
|
|
|
function factions.on_remove_privs(name, rank, privs)
|
|
factions.broadcast(name, "The privileges in rank " .. rank .. " have been revoked: " .. table.concat(privs, ", "))
|
|
end
|
|
|
|
function factions.on_add_privs(name, rank, privs)
|
|
factions.broadcast(name, "The privileges in rank " .. rank .. " have been added: " .. table.concat(privs, ", "))
|
|
end
|
|
|
|
function factions.on_set_rank_name(name, rank,newrank)
|
|
factions.broadcast(name, "The name of rank " .. rank .. " has been changed to " .. newrank)
|
|
end
|
|
|
|
function factions.on_delete_rank(name, rank, newrank)
|
|
factions.broadcast(name, "The rank " .. rank .. " has been deleted and replaced by " .. newrank)
|
|
end
|
|
|
|
function factions.on_set_def_rank(name, rank)
|
|
factions.broadcast(name, "The default rank given to new players has been changed to " .. rank)
|
|
end
|
|
|
|
function factions.on_reset_ranks(name)
|
|
factions.broadcast(name, "All of the faction's ranks have been reset to the default ones.")
|
|
end
|
|
|
|
function factions.on_promote(name, member)
|
|
local faction = factions.factions.get(name)
|
|
minetest.chat_send_player(member, "You have been promoted to " .. faction.players[member])
|
|
end
|
|
|
|
function factions.on_revoke_invite(name, player)
|
|
minetest.chat_send_player(player, "You are no longer invited to faction " .. name)
|
|
end
|
|
|
|
function factions.get_parcel_pos(pos)
|
|
if factions_config.protection_style == "2d" then
|
|
return math.floor(pos.x / parcel_size) * parcel_size .. "," .. math.floor(pos.z / parcel_size) * parcel_size
|
|
elseif factions_config.protection_style == "3d" then
|
|
return math.floor(pos.x / parcel_size) * parcel_size .. "," .. math.floor(pos.y / parcel_size) * parcel_size .. "," .. math.floor(pos.z / parcel_size) * parcel_size
|
|
end
|
|
end
|
|
|
|
function factions.get_player_faction(playername)
|
|
local facname = factions.players.get(playername)
|
|
if facname then
|
|
local faction = factions.factions.get(facname)
|
|
return faction, facname
|
|
end
|
|
return nil
|
|
end
|
|
|
|
function factions.get_parcel_faction(parcelpos)
|
|
local facname = factions.parcels.get(parcelpos)
|
|
if facname then
|
|
local faction = factions.factions.get(facname)
|
|
return faction, facname
|
|
end
|
|
return nil
|
|
end
|
|
|
|
function factions.get_faction(facname)
|
|
return factions.factions.get(facname)
|
|
end
|
|
|
|
function factions.get_faction_at(pos)
|
|
local y = pos.y
|
|
if factions_config.protection_depth_height_limit and (pos.y < factions_config.protection_max_depth or pos.y > factions_config.protection_max_height) then
|
|
return nil
|
|
end
|
|
local parcelpos = factions.get_parcel_pos(pos)
|
|
return factions.get_parcel_faction(parcelpos)
|
|
end
|
|
|
|
function factions.get_faction_list()
|
|
|
|
local names = {}
|
|
local directory = string.format("%s/factions/factions", minetest.get_worldpath())
|
|
local nameslist = minetest.get_dir_list(directory)
|
|
for k, v in pairs(nameslist) do
|
|
names[#names + 1] = v:sub(0, v:len() - 5)
|
|
end
|
|
|
|
return names
|
|
end
|
|
|
|
minetest.register_on_dieplayer(
|
|
function(player)
|
|
local faction, name = factions.get_player_faction(player:get_player_name())
|
|
if not faction then
|
|
return true
|
|
end
|
|
factions.decrease_power(name, factions_config.power_per_death)
|
|
return true
|
|
end
|
|
)
|
|
|
|
function factions.faction_tick()
|
|
local now = os.time()
|
|
for i, facname in pairs(factions.get_faction_list()) do
|
|
local faction = factions.factions.get(facname)
|
|
|
|
if faction ~= nil then
|
|
if factions.is_online(facname) then
|
|
if factions_config.enable_power_per_player then
|
|
local count = 0
|
|
for _ in pairs(factions.onlineplayers[facname]) do count = count + 1 end
|
|
factions.increase_power(facname, factions_config.power_per_player * count)
|
|
else
|
|
factions.increase_power(facname, factions_config.power_per_tick)
|
|
end
|
|
end
|
|
if now - faction.last_logon > factions_config.maximum_faction_inactivity or (faction.no_parcel ~= -1 and now - faction.no_parcel > factions_config.maximum_parcelless_faction_time) then
|
|
local r = ""
|
|
if now - faction.last_logon > factions_config.maximum_faction_inactivity then
|
|
r = "inactivity"
|
|
else
|
|
r = "no parcel claims"
|
|
end
|
|
factions.disband(facname, r)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
minetest.register_on_joinplayer(
|
|
function(player)
|
|
local name = player:get_player_name()
|
|
minetest.after(5, createHudfactionLand, player)
|
|
local faction, facname = factions.get_player_faction(name)
|
|
if faction then
|
|
if factions.onlineplayers[facname] == nil then
|
|
factions.onlineplayers[facname] = {}
|
|
end
|
|
|
|
factions.onlineplayers[facname][name] = true
|
|
faction.last_logon = os.time()
|
|
|
|
factions.factions.set(facname, faction)
|
|
|
|
minetest.after(5, createHudFactionName, player, facname)
|
|
minetest.after(5, createHudPower, player, faction)
|
|
|
|
if faction.no_parcel ~= -1 then
|
|
local now = os.time() - faction.no_parcel
|
|
local l = factions_config.maximum_parcelless_faction_time
|
|
minetest.chat_send_player(name, "This faction will disband in " .. l - now .. " seconds, because it has no parcels.")
|
|
end
|
|
|
|
if factions.has_permission(facname, name, "diplomacy") then
|
|
for _ in pairs(faction.request_inbox) do minetest.chat_send_player(name, "You have diplomatic requests in the inbox.") break end
|
|
end
|
|
|
|
if faction.message_of_the_day and (faction.message_of_the_day ~= "" or faction.message_of_the_day ~= " ") then
|
|
minetest.chat_send_player(name, faction.message_of_the_day)
|
|
end
|
|
end
|
|
|
|
end
|
|
)
|
|
|
|
minetest.register_on_leaveplayer(
|
|
function(player)
|
|
local name = player:get_player_name()
|
|
local faction, facname = factions.get_player_faction(name)
|
|
local id_name1 = name .. "factionLand"
|
|
|
|
if hud_ids[id_name1] then
|
|
hud_ids[id_name1] = nil
|
|
end
|
|
|
|
if faction then
|
|
factions.onlineplayers[facname][name] = nil
|
|
local id_name2 = name .. "factionName"
|
|
local id_name3 = name .. "powerWatch"
|
|
if hud_ids[id_name2] then
|
|
hud_ids[id_name2] = nil
|
|
end
|
|
if hud_ids[id_name3] then
|
|
hud_ids[id_name3] = nil
|
|
end
|
|
for k, v in pairs(factions.onlineplayers[facname]) do
|
|
return
|
|
end
|
|
factions.onlineplayers[facname] = nil
|
|
end
|
|
end
|
|
)
|
|
|
|
minetest.register_on_respawnplayer(
|
|
function(player)
|
|
local faction, facname = factions.get_player_faction(player:get_player_name())
|
|
|
|
if not faction then
|
|
return false
|
|
else
|
|
if not faction.spawn then
|
|
return false
|
|
else
|
|
player:set_pos(faction.spawn)
|
|
return true
|
|
end
|
|
end
|
|
end
|
|
)
|
|
|
|
local default_is_protected = minetest.is_protected
|
|
minetest.is_protected = function(pos, player)
|
|
local y = pos.y
|
|
|
|
if factions_config.protection_depth_height_limit and (pos.y < factions_config.protection_max_depth or pos.y > factions_config.protection_max_height) then
|
|
return false
|
|
end
|
|
|
|
local parcelpos = factions.get_parcel_pos(pos)
|
|
local parcel_faction, parcel_fac_name = factions.get_parcel_faction(parcelpos)
|
|
local player_faction, player_fac_name = factions.get_player_faction(player)
|
|
|
|
-- no faction
|
|
if not parcel_faction then
|
|
return default_is_protected(pos, player)
|
|
elseif player_faction then
|
|
if parcel_faction.name == player_faction.name then
|
|
if factions.has_permission(parcel_fac_name, player, "pain_build") then
|
|
local p = minetest.get_player_by_name(player)
|
|
p:set_hp(p:get_hp() - 0.5)
|
|
end
|
|
return not (factions.has_permission(parcel_fac_name, player, "build") or factions.has_permission(parcel_fac_name, player, "pain_build"))
|
|
elseif parcel_faction.allies[player_faction.name] then
|
|
if factions.has_permission(player_fac_name, player, "pain_build") then
|
|
local p = minetest.get_player_by_name(player)
|
|
p:set_hp(p:get_hp() - 0.5)
|
|
end
|
|
return not (factions.has_permission(player_fac_name, player, "build") or factions.has_permission(player_fac_name, player, "pain_build"))
|
|
else
|
|
return true
|
|
end
|
|
else
|
|
return true
|
|
end
|
|
end
|
|
|
|
function factionUpdate()
|
|
factions.faction_tick()
|
|
minetest.after(factions_config.tick_time, factionUpdate)
|
|
end
|