factions/factions.lua

1077 lines
32 KiB
Lua
Raw Normal View History

2016-07-30 07:38:52 +02:00
--read some basic information
2016-08-07 19:31:11 +02:00
local factions_worldid = minetest.get_worldpath()
2016-07-30 07:38:52 +02:00
2016-08-07 19:31:11 +02:00
--! @class factions
--! @brief main class for factions
factions = {}
2016-07-30 07:38:52 +02:00
--! @brief runtime data
2016-08-07 19:31:11 +02:00
factions.factions = {}
2016-08-14 15:53:40 +02:00
factions.parcels = {}
2016-08-07 19:31:11 +02:00
factions.players = {}
2016-07-30 07:38:52 +02:00
2016-08-07 02:11:17 +02:00
---------------------
--! @brief returns whether a faction can be created or not (allows for implementation of blacklists and the like)
2016-08-12 11:16:43 +02:00
--! @param name String containing the faction's name
2016-08-07 19:31:11 +02:00
factions.can_create_faction = function(name)
2018-10-18 04:04:46 +02:00
if #name > factions_config.faction_name_max_length then
2016-08-18 15:26:19 +02:00
return false
elseif factions.factions[name] then
2016-08-07 02:11:17 +02:00
return false
else
return true
2016-08-07 19:36:56 +02:00
end
2016-08-07 02:11:17 +02:00
end
factions.Faction = {
}
util = {
coords3D_string = function(coords)
return coords.x..", "..coords.y..", "..coords.z
end
}
factions.Faction.__index = factions.Faction
2016-08-12 11:16:43 +02:00
-- Faction permissions:
--
-- disband: disband the faction
2016-08-14 15:53:40 +02:00
-- claim: (un)claim parcels
2016-08-12 11:16:43 +02:00
-- playerslist: invite/kick players and open/close the faction
-- build: dig and place nodes
-- description: set the faction's description
-- ranks: create and delete ranks
-- spawn: set the faction's spawn
-- promote: set a player's rank
2018-10-15 17:24:58 +02:00
-- diplomacy: make war, or an alliance with other teams.
2016-08-12 11:16:43 +02:00
2018-10-15 17:24:58 +02:00
factions.permissions = {"disband", "claim", "playerslist", "build", "description", "ranks", "spawn", "promote"}
2018-10-18 04:04:46 +02:00
if factions_config.faction_diplomacy then
2018-10-15 17:24:58 +02:00
table.insert(factions.permissions,"diplomacy")
end
function factions.Faction:new(faction)
faction = {
2016-08-14 15:53:40 +02:00
--! @brief power of a faction (needed for parcel claiming)
2018-10-18 04:04:46 +02:00
power = factions_config.power,
2016-08-12 11:16:43 +02:00
--! @brief maximum power of a faction
2018-10-18 04:04:46 +02:00
maxpower = factions_config.maxpower,
--! @brief power currently in use
usedpower = 0.,
2016-08-12 11:16:43 +02:00
--! @brief list of player names
2016-08-07 02:11:17 +02:00
players = {},
2018-10-17 04:58:15 +02:00
--! @brief list of player names online
onlineplayers = {},
--! @brief list of player names offline
offlineplayers = {},
2016-08-12 11:16:43 +02:00
--! @brief table of ranks/permissions
2018-10-15 17:24:58 +02:00
ranks = {["leader"] = factions.permissions,
2016-08-08 16:31:11 +02:00
["moderator"] = {"claim", "playerslist", "build", "spawn"},
2016-08-07 02:11:17 +02:00
["member"] = {"build"}
},
2016-08-12 11:16:43 +02:00
--! @brief name of the leader
2016-08-07 02:11:17 +02:00
leader = nil,
2016-08-12 11:16:43 +02:00
--! @brief default joining rank for new members
2016-08-07 02:11:17 +02:00
default_rank = "member",
2016-08-12 11:16:43 +02:00
--! @brief default rank assigned to the leader
2016-08-07 02:11:17 +02:00
default_leader_rank = "leader",
2016-08-12 11:16:43 +02:00
--! @brief faction's description string
2016-08-07 02:11:17 +02:00
description = "Default faction description.",
2018-10-25 07:21:42 +02:00
--! @brief faction's message of the day.
message_of_the_day = "",
2016-08-12 11:16:43 +02:00
--! @brief list of players currently invited (can join with /f join)
2016-08-07 02:11:17 +02:00
invited_players = {},
2016-08-14 15:53:40 +02:00
--! @brief table of claimed parcels (keys are parcelpos strings)
2016-08-07 02:11:17 +02:00
land = {},
2016-08-12 11:16:43 +02:00
--! @brief table of allies
2016-08-07 02:11:17 +02:00
allies = {},
2018-10-15 17:24:58 +02:00
--
request_inbox = {},
2016-08-12 11:16:43 +02:00
--! @brief table of enemies
2016-08-07 02:11:17 +02:00
enemies = {},
2018-10-15 17:24:58 +02:00
--!
2018-10-18 07:15:26 +02:00
neutral = {},
2016-08-14 15:53:40 +02:00
--! @brief table of parcels/factions that are under attack
attacked_parcels = {},
2016-08-12 11:16:43 +02:00
--! @brief whether faction is closed or open (boolean)
join_free = true,
2016-08-17 22:28:12 +02:00
--! @brief gives certain privileges
is_admin = false,
--! @brief last time anyone logged on
last_logon = os.time(),
2018-10-15 17:24:58 +02:00
--! @brief how long this has been without parcels
no_parcel = os.time(),
} or faction
setmetatable(faction, self)
return faction
end
2016-08-07 02:11:17 +02:00
2016-08-12 11:16:43 +02:00
--! @brief create a new empty faction
function factions.new_faction(name)
local faction = factions.Faction:new(nil)
faction.name = name
2016-08-07 19:36:56 +02:00
factions.factions[name] = faction
faction:on_create()
2018-10-18 04:31:07 +02:00
minetest.after(1,
function(f)
f:on_no_parcel()
end,faction)
factions.save()
2016-08-07 02:11:17 +02:00
return faction
end
function factions.start_diplomacy(name,faction)
for i in pairs(factions.factions) do
if i ~= name and not (faction.neutral[i] or faction.allies[i] or faction.enemies[i]) then
faction:new_enemy(factions.factions[i].name)
factions.factions[i]:new_enemy(faction.name)
end
end
end
function factions.Faction.increase_power(self, power)
self.power = self.power + power
if self.power > self.maxpower - self.usedpower then
self.power = self.maxpower - self.usedpower
end
2018-10-17 04:58:15 +02:00
for i in pairs(self.onlineplayers) do
updateHudPower(minetest.get_player_by_name(i),self)
end
factions.save()
end
function factions.Faction.decrease_power(self, power)
self.power = self.power - power
2018-10-17 04:58:15 +02:00
for i in pairs(self.onlineplayers) do
updateHudPower(minetest.get_player_by_name(i),self)
end
factions.save()
end
function factions.Faction.increase_maxpower(self, power)
self.maxpower = self.maxpower + power
2018-10-17 04:58:15 +02:00
for i in pairs(self.onlineplayers) do
updateHudPower(minetest.get_player_by_name(i),self)
end
factions.save()
end
function factions.Faction.decrease_maxpower(self, power)
self.maxpower = self.maxpower - power
if self.maxpower < 0. then -- should not happen
self.maxpower = 0.
end
2018-10-17 04:58:15 +02:00
for i in pairs(self.onlineplayers) do
updateHudPower(minetest.get_player_by_name(i),self)
end
end
function factions.Faction.increase_usedpower(self, power)
self.usedpower = self.usedpower + power
2018-10-17 04:58:15 +02:00
for i in pairs(self.onlineplayers) do
updateHudPower(minetest.get_player_by_name(i),self)
end
end
function factions.Faction.decrease_usedpower(self, power)
self.usedpower = self.usedpower - power
if self.usedpower < 0. then
self.usedpower = 0.
end
2018-10-17 04:58:15 +02:00
for i in pairs(self.onlineplayers) do
updateHudPower(minetest.get_player_by_name(i),self)
end
end
-- power-per-players only.
function factions.Faction.check_power(self)
if factions_config.enable_power_per_player then
for player,unused in pairs(self.players) do
local ip = factions_ip.player_ips[player]
local notsame = true
for i,k in pairs(self.players) do
local other_ip = factions_ip.player_ips[k]
if other_ip == ip then
notsame = false
break
end
end
if notsame then
self:increase_maxpower(factions_config.powermax_per_player)
end
end
end
end
function factions.Faction.count_land(self)
local count = 0.
for k, v in pairs(self.land) do
count = count + 1
end
return count
end
2018-10-15 17:24:58 +02:00
minetest.register_on_prejoinplayer(function(name, ip)
factions_ip.player_ips[name] = ip
end)
function factions.Faction.add_player(self, player, rank)
2016-08-10 02:46:54 +02:00
self:on_player_join(player)
self.players[player] = rank or self.default_rank
factions.players[player] = self.name
self.invited_players[player] = nil
2018-10-18 04:04:46 +02:00
if factions_config.enable_power_per_player then
2018-10-15 17:24:58 +02:00
local ip = factions_ip.player_ips[player]
local notsame = true
for i,k in pairs(self.players) do
local other_ip = factions_ip.player_ips[k]
if other_ip == ip then
notsame = false
break
end
end
if notsame then
2018-10-18 04:04:46 +02:00
self:increase_maxpower(factions_config.powermax_per_player)
--self:increase_power(factions_config.power_per_player)
2018-10-15 17:24:58 +02:00
end
end
2018-10-17 04:58:15 +02:00
local pdata = minetest.get_player_by_name(player)
local ipc = pdata:is_player_connected(player)
if ipc then
createHudFactionName(pdata,self.name)
createHudPower(pdata,self)
self.offlineplayers[player] = nil
self.onlineplayers[player] = 1
else
self.offlineplayers[player] = 1
self.onlineplayers[player] = nil
end
factions.save()
end
function factions.Faction.check_players_in_faction(self)
2018-10-27 17:51:39 +02:00
for i,k in pairs(self.players) do
return true
end
2018-10-27 17:51:39 +02:00
self:disband("Zero players on faction.")
return false
end
function factions.Faction.remove_player(self, player)
self.players[player] = nil
factions.players[player] = nil
self:on_player_leave(player)
self:check_players_in_faction(self)
2018-10-18 04:04:46 +02:00
if factions_config.enable_power_per_player then
2018-10-15 17:24:58 +02:00
local ip = factions_ip.player_ips[player]
local notsame = true
for i,k in pairs(self.players) do
local other_ip = factions_ip.player_ips[k]
if other_ip == ip then
notsame = false
break
end
end
if notsame then
2018-10-18 04:04:46 +02:00
self:decrease_maxpower(factions_config.powermax_per_player)
2018-10-15 17:24:58 +02:00
end
end
2018-10-17 04:58:15 +02:00
local pdata = minetest.get_player_by_name(player)
local ipc = pdata:is_player_connected(player)
if ipc then
removeHud(pdata,"factionName")
removeHud(pdata,"powerWatch")
end
2018-10-17 04:58:15 +02:00
self.offlineplayers[player] = nil
self.onlineplayers[player] = nil
factions.save()
end
2016-08-14 15:53:40 +02:00
--! @param parcelpos position of the wanted parcel
--! @return whether this faction can claim a parcelpos
function factions.Faction.can_claim_parcel(self, parcelpos)
local fac = factions.parcels[parcelpos]
if fac then
2018-10-18 07:15:26 +02:00
if factions.factions[fac].power < 0. and self.power >= factions_config.power_per_parcel and not self.allies[factions.factions[fac].name] and not self.neutral[factions.factions[fac].name] then
return true
else
return false
end
2018-10-18 04:04:46 +02:00
elseif self.power < factions_config.power_per_parcel then
2016-08-08 16:31:11 +02:00
return false
end
return true
end
2016-08-14 15:53:40 +02:00
--! @brief claim a parcel, update power and update global parcels table
function factions.Faction.claim_parcel(self, parcelpos)
-- check if claiming over other faction's territory
2016-08-14 15:53:40 +02:00
local otherfac = factions.parcels[parcelpos]
if otherfac then
local faction = factions.factions[otherfac]
2016-08-14 15:53:40 +02:00
faction:unclaim_parcel(parcelpos)
2018-10-15 17:24:58 +02:00
faction:parcelless_check()
end
2016-08-14 15:53:40 +02:00
factions.parcels[parcelpos] = self.name
self.land[parcelpos] = true
2018-10-18 04:04:46 +02:00
self:decrease_power(factions_config.power_per_parcel)
self:increase_usedpower(factions_config.power_per_parcel)
2016-08-14 15:53:40 +02:00
self:on_claim_parcel(parcelpos)
2018-10-15 17:24:58 +02:00
self:parcelless_check()
factions.save()
end
2016-08-14 15:53:40 +02:00
--! @brief claim a parcel, update power and update global parcels table
function factions.Faction.unclaim_parcel(self, parcelpos)
factions.parcels[parcelpos] = nil
self.land[parcelpos] = nil
2018-10-18 04:04:46 +02:00
self:increase_power(factions_config.power_per_parcel)
self:decrease_usedpower(factions_config.power_per_parcel)
2016-08-14 15:53:40 +02:00
self:on_unclaim_parcel(parcelpos)
2018-10-15 17:24:58 +02:00
self:parcelless_check()
factions.save()
end
2016-08-12 11:16:43 +02:00
2018-10-15 17:24:58 +02:00
function factions.Faction.parcelless_check(self)
if self.land then
local count = 0
for index, value in pairs(self.land) do
count = count + 1
2018-10-23 06:03:11 +02:00
break
2018-10-15 17:24:58 +02:00
end
if count > 0 then
if self.no_parcel ~= -1 then
self:broadcast("Faction " .. self.name .. " will not be disbanded because it now has parcels.")
end
self.no_parcel = -1
else
self.no_parcel = os.time()
2018-10-18 04:31:07 +02:00
self:on_no_parcel()
2018-10-15 17:24:58 +02:00
end
end
end
2016-08-14 15:53:40 +02:00
--! @brief disband faction, updates global players and parcels table
function factions.Faction.disband(self, reason)
2018-10-15 17:24:58 +02:00
for i,v in pairs(factions.factions) do
if v.name ~= self.name then
if v.enemies[self.name] then
v:end_enemy(self.name)
end
if v.allies[self.name] then
v:end_alliance(self.name)
end
2018-10-18 07:15:26 +02:00
if v.neutral[self.name] then
v:end_neutral(self.name)
2018-10-15 17:24:58 +02:00
end
end
end
for k, _ in pairs(self.players) do -- remove players affiliation
factions.players[k] = nil
end
2016-08-14 15:53:40 +02:00
for k, v in pairs(self.land) do -- remove parcel claims
factions.parcels[k] = nil
end
self:on_disband(reason)
2018-10-22 21:30:11 +02:00
local playerslist = self.onlineplayers
for i,l in pairs(playerslist) do
removeHud(i,"factionName")
removeHud(i,"powerWatch")
end
2018-10-22 21:30:11 +02:00
factions.factions[self.name] = nil
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @brief change the faction leader
function factions.Faction.set_leader(self, player)
2016-08-27 22:34:09 +02:00
if self.leader then
self.players[self.leader] = self.default_rank
end
self.leader = player
self.players[player] = self.default_leader_rank
self:on_new_leader()
factions.save()
end
2016-08-12 11:16:43 +02:00
2018-10-25 07:21:42 +02:00
function factions.Faction.set_message_of_the_day(self,text)
self.message_of_the_day = text
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @brief check permissions for a given player
--! @return boolean indicating permissions. Players not in faction always receive false
function factions.Faction.has_permission(self, player, permission)
local p = self.players[player]
if not p then
return false
end
local perms = self.ranks[p]
if perms then
for i in ipairs(perms) do
if perms[i] == permission then
return true
end
end
end
return false
end
2018-10-22 21:30:11 +02:00
function factions.Faction.set_description(self, new)
self.description = new
self:on_change_description()
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @brief places player in invite list
function factions.Faction.invite_player(self, player)
self.invited_players[player] = true
self:on_player_invited(player)
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @brief removes player from invite list (can no longer join via /f join)
function factions.Faction.revoke_invite(self, player)
self.invited_players[player] = nil
self:on_revoke_invite(player)
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @brief set faction openness
function factions.Faction.toggle_join_free(self, bool)
self.join_free = bool
self:on_toggle_join_free()
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @return true if a player can use /f join, false otherwise
function factions.Faction.can_join(self, player)
return self.join_free or self.invited_players[player]
end
2016-08-12 11:16:43 +02:00
function factions.Faction.new_alliance(self, faction)
self.allies[faction] = true
self:on_new_alliance(faction)
if self.enemies[faction] then
self:end_enemy(faction)
2018-10-15 17:24:58 +02:00
end
2018-10-18 07:15:26 +02:00
if self.neutral[faction] then
self:end_neutral(faction)
end
factions.save()
end
2018-10-22 21:30:11 +02:00
function factions.Faction.end_alliance(self, faction)
self.allies[faction] = nil
self:on_end_alliance(faction)
factions.save()
end
2018-10-22 21:30:11 +02:00
2018-10-18 07:15:26 +02:00
function factions.Faction.new_neutral(self, faction)
self.neutral[faction] = true
self:on_new_neutral(faction)
2018-10-15 17:24:58 +02:00
if self.allies[faction] then
self:end_alliance(faction)
end
if self.enemies[faction] then
self:end_enemy(faction)
end
factions.save()
end
2018-10-22 21:30:11 +02:00
2018-10-18 07:15:26 +02:00
function factions.Faction.end_neutral(self, faction)
self.neutral[faction] = nil
self:on_end_neutral(faction)
2018-10-15 17:24:58 +02:00
factions.save()
end
2018-10-22 21:30:11 +02:00
function factions.Faction.new_enemy(self, faction)
self.enemies[faction] = true
self:on_new_enemy(faction)
if self.allies[faction] then
self:end_alliance(faction)
2018-10-15 17:24:58 +02:00
end
2018-10-18 07:15:26 +02:00
if self.neutral[faction] then
self:end_neutral(faction)
end
factions.save()
end
2018-10-22 21:30:11 +02:00
function factions.Faction.end_enemy(self, faction)
self.enemies[faction] = nil
self:on_end_enemy(faction)
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @brief faction's member will now spawn in a new place
function factions.Faction.set_spawn(self, pos)
2016-08-08 16:31:11 +02:00
self.spawn = {x=pos.x, y=pos.y, z=pos.z}
2016-08-15 01:50:13 +02:00
self:on_set_spawn()
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @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.Faction.add_rank(self, rank, perms)
self.ranks[rank] = perms
2016-08-08 16:31:11 +02:00
self:on_add_rank(rank)
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @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.Faction.delete_rank(self, rank, newrank)
for player, r in pairs(self.players) do
if r == rank then
self.players[player] = newrank
end
end
self.ranks[rank] = nil
self:on_delete_rank(rank, newrank)
if rank == self.default_leader_rank then
self.default_leader_rank = newrank
self:broadcast("The default leader rank has been set to "..newrank)
end
if rank == self.default_rank then
self.default_rank = newrank
self:broadcast("The default rank given to new players is set to "..newrank)
end
factions.save()
end
2016-08-12 11:16:43 +02:00
--! @brief set a player's rank
function factions.Faction.promote(self, member, rank)
self.players[member] = rank
self:on_promote(member)
end
2016-08-12 11:16:43 +02:00
--! @brief send a message to all members
function factions.Faction.broadcast(self, msg, sender)
local message = self.name.."> "..msg
if sender then
message = sender.."@"..message
end
message = "Faction<"..message
2018-10-18 04:36:07 +02:00
for k, _ in pairs(self.onlineplayers) do
minetest.chat_send_player(k, message)
end
end
2016-08-08 21:09:10 +02:00
--! @brief checks whether a faction has at least one connected player
function factions.Faction.is_online(self)
2018-10-18 04:36:07 +02:00
for playername, _ in pairs(self.onlineplayers) do
if minetest.get_player_by_name(playername) then
return true
end
end
return false
end
2016-08-15 18:31:11 +02:00
function factions.Faction.attack_parcel(self, parcelpos)
2018-10-18 04:04:46 +02:00
if factions_config.attack_parcel then
local attacked_faction = factions.get_parcel_faction(parcelpos)
if attacked_faction then
2018-10-15 17:24:58 +02:00
if not self.allies[attacked_faction.name] then
2018-10-18 04:04:46 +02:00
self.power = self.power - factions_config.power_per_attack
2018-10-15 17:24:58 +02:00
if attacked_faction.attacked_parcels[parcelpos] then
attacked_faction.attacked_parcels[parcelpos][self.name] = true
else
attacked_faction.attacked_parcels[parcelpos] = {[self.name] = true}
end
attacked_faction:broadcast("Parcel ("..parcelpos..") is being attacked by "..self.name.."!!")
if self.power < 0. then -- punish memers
minetest.chat_send_all("Faction "..self.name.." has attacked too much and has now negative power!")
end
factions.save()
else
2018-10-15 17:24:58 +02:00
self:broadcast("You can not attack that parcel because it belongs to an ally.")
end
end
end
2016-08-15 18:31:11 +02:00
end
function factions.Faction.stop_attack(self, parcelpos)
local attacked_faction = factions.parcels[parcelpos]
if attacked_faction then
attacked_faction = factions.factions[attacked_faction]
if attacked_faction.attacked_parcels[parcelpos] then
attacked_faction.attacked_parcels[parcelpos][self.name] = nil
attacked_faction:broadcast("Parcel ("..parcelpos..") is no longer under attack from "..self.name..".")
self:broadcast("Parcel ("..parcelpos..") has been reconquered by "..attacked_faction.name..".")
2016-08-15 18:31:11 +02:00
end
factions.save()
end
end
2016-08-19 10:02:12 +02:00
function factions.Faction.parcel_is_attacked_by(self, parcelpos, faction)
if self.attacked_parcels[parcelpos] then
return self.attacked_parcels[parcelpos][faction.name]
else
return false
end
end
--------------------------
-- callbacks for events --
function factions.Faction.on_create(self) --! @brief called when the faction is added to the global faction list
2016-08-10 02:46:54 +02:00
minetest.chat_send_all("Faction "..self.name.." has been created.")
end
2018-10-18 04:31:07 +02:00
function factions.Faction.on_no_parcel(self)
local now = os.time() - self.no_parcel
local l = factions_config.maximum_parcelless_faction_time
self:broadcast("This faction will disband in "..l-now.." seconds, because it has no parcels.")
end
function factions.Faction.on_player_leave(self, player)
self:broadcast(player.." has left this faction.")
end
function factions.Faction.on_player_join(self, player)
self:broadcast(player.." has joined this faction.")
end
2016-08-14 15:53:40 +02:00
function factions.Faction.on_claim_parcel(self, pos)
2016-08-14 23:27:51 +02:00
self:broadcast("Parcel ("..pos..") has been claimed.")
end
2016-08-14 15:53:40 +02:00
function factions.Faction.on_unclaim_parcel(self, pos)
2016-08-14 23:27:51 +02:00
self:broadcast("Parcel ("..pos..") has been unclaimed.")
end
function factions.Faction.on_disband(self, reason)
local msg = "Faction "..self.name.." has been disbanded."
if reason then
msg = msg.." ("..reason..")"
end
minetest.chat_send_all(msg)
end
function factions.Faction.on_new_leader(self)
self:broadcast(self.leader.." is now the leader of this faction.")
end
function factions.Faction.on_change_description(self)
self:broadcast("Faction description has been modified to: "..self.description)
end
function factions.Faction.on_player_invited(self, player)
minetest.chat_send_player(player, "You have been invited to faction "..self.name)
end
function factions.Faction.on_toggle_join_free(self, player)
2016-08-29 01:41:23 +02:00
if self.join_free then
self:broadcast("This faction is now invite-free.")
else
self:broadcast("This faction is no longer invite-free.")
end
end
function factions.Faction.on_new_alliance(self, faction)
self:broadcast("This faction is now allied with "..faction)
end
function factions.Faction.on_end_alliance(self, faction)
self:broadcast("This faction is no longer allied with "..faction.."!")
end
2018-10-18 07:15:26 +02:00
function factions.Faction.on_new_neutral(self, faction)
self:broadcast("This faction is now neutral with "..faction)
2018-10-15 17:24:58 +02:00
end
2018-10-18 07:15:26 +02:00
function factions.Faction.on_end_neutral(self, faction)
self:broadcast("This faction is no longer neutral with "..faction.."!")
2018-10-15 17:24:58 +02:00
end
function factions.Faction.on_new_enemy(self, faction)
self:broadcast("This faction is now at war with "..faction)
end
function factions.Faction.on_end_enemy(self, faction)
self:broadcast("This faction is no longer at war with "..faction.."!")
end
function factions.Faction.on_set_spawn(self)
2016-08-15 01:50:13 +02:00
self:broadcast("The faction spawn has been set to ("..util.coords3D_string(self.spawn)..").")
end
function factions.Faction.on_add_rank(self, rank)
self:broadcast("The rank "..rank.." has been created with privileges: "..table.concat(self.ranks[rank], ", "))
end
function factions.Faction.on_delete_rank(self, rank, newrank)
self:broadcast("The rank "..rank.." has been deleted and replaced by "..newrank)
end
function factions.Faction.on_promote(self, member)
minetest.chat_send_player(member, "You have been promoted to "..self.players[member])
end
function factions.Faction.on_revoke_invite(self, player)
minetest.chat_send_player(player, "You are no longer invited to faction "..self.name)
end
2016-08-14 15:53:40 +02:00
function factions.get_parcel_pos(pos)
2018-10-18 04:40:00 +02:00
if factions_config.protection_style == "2d" then
2018-10-18 04:04:46 +02:00
return math.floor(pos.x / factions_config.parcel_size)..","..math.floor(pos.z / factions_config.parcel_size)
2018-10-18 04:40:00 +02:00
elseif factions_config.protection_style == "3d" then
2018-10-18 04:04:46 +02:00
return math.floor(pos.x / factions_config.parcel_size)..","..math.floor(pos.y / factions_config.parcel_size)..","..math.floor(pos.z / factions_config.parcel_size)
2018-10-17 04:58:15 +02:00
end
2016-08-07 19:36:56 +02:00
end
2016-07-30 07:38:52 +02:00
function factions.get_player_faction(playername)
2016-08-17 22:28:12 +02:00
local facname = factions.players[playername]
if facname then
local faction = factions.factions[facname]
return faction
end
return nil
end
function factions.get_parcel_faction(parcelpos)
local facname = factions.parcels[parcelpos]
2016-08-17 22:28:12 +02:00
if facname then
local faction = factions.factions[facname]
return faction
end
return nil
end
function factions.get_faction(facname)
return factions.factions[facname]
end
function factions.get_faction_at(pos)
2018-10-17 04:58:15 +02:00
local y = pos.y
2018-10-18 04:04:46 +02:00
if factions_config.protection_depth_height_limit and (pos.y < factions_config.protection_max_depth or pos.y > factions_config.protection_max_height) then
2018-10-17 04:58:15 +02:00
return nil
end
2016-08-17 22:28:12 +02:00
local parcelpos = factions.get_parcel_pos(pos)
return factions.get_parcel_faction(parcelpos)
end
2016-07-30 07:38:52 +02:00
-------------------------------------------------------------------------------
2016-08-07 02:11:17 +02:00
-- name: add_faction(name)
2016-07-30 07:38:52 +02:00
--
2016-08-07 02:11:17 +02:00
--! @brief add a faction
2016-08-07 19:31:11 +02:00
--! @memberof factions
2016-07-30 07:38:52 +02:00
--! @public
--
2016-08-07 02:11:17 +02:00
--! @param name of faction to add
2016-07-30 07:38:52 +02:00
--!
2016-08-07 02:11:17 +02:00
--! @return faction object/false (succesfully added faction or not)
2016-07-30 07:38:52 +02:00
-------------------------------------------------------------------------------
2016-08-07 19:31:11 +02:00
function factions.add_faction(name)
2016-08-07 19:36:56 +02:00
if factions.can_create_faction(name) then
local fac = factions.new_faction(name)
fac:on_create()
return fac
else
return nil
end
2016-07-30 07:38:52 +02:00
end
-------------------------------------------------------------------------------
-- name: get_faction_list()
--
2016-08-07 19:31:11 +02:00
--! @brief get list of factions
--! @memberof factions
2016-07-30 07:38:52 +02:00
--! @public
--!
2016-08-07 19:31:11 +02:00
--! @return list of factions
2016-07-30 07:38:52 +02:00
-------------------------------------------------------------------------------
2016-08-07 19:31:11 +02:00
function factions.get_faction_list()
2016-07-30 07:38:52 +02:00
2016-08-18 15:21:59 +02:00
local retval = {}
for key,value in pairs(factions.factions) do
table.insert(retval,key)
end
return retval
2016-07-30 07:38:52 +02:00
end
-------------------------------------------------------------------------------
-- name: save()
--
--! @brief save data to file
2016-08-07 19:31:11 +02:00
--! @memberof factions
2016-07-30 07:38:52 +02:00
--! @private
-------------------------------------------------------------------------------
2016-08-07 19:31:11 +02:00
function factions.save()
2016-07-30 07:38:52 +02:00
2016-08-18 15:21:59 +02:00
--saving is done much more often than reading data to avoid delay
--due to figuring out which data to save and which is temporary only
--all data is saved here
--this implies data needs to be cleant up on load
local file,error = io.open(factions_worldid .. "/" .. "factions.conf","w")
if file ~= nil then
file:write(minetest.serialize(factions.factions))
file:close()
else
minetest.log("error","MOD factions: unable to save factions world specific data!: " .. error)
end
2018-10-15 17:24:58 +02:00
factions_ip.save()
2016-07-30 07:38:52 +02:00
end
-------------------------------------------------------------------------------
-- name: load()
--
--! @brief load data from file
2016-08-07 19:31:11 +02:00
--! @memberof factions
2016-07-30 07:38:52 +02:00
--! @private
--
--! @return true/false
-------------------------------------------------------------------------------
2016-08-07 19:31:11 +02:00
function factions.load()
local filename = "factions.conf"
local file,error = io.open(factions_worldid .. "/" .. filename,"r")
2016-08-18 15:21:59 +02:00
if file ~= nil then
local raw_data = file:read("*a")
factions.factions = minetest.deserialize(raw_data)
if factions.is_old_file(factions.factions) then
if factions.convert(filename) then
minetest.after(5,
function()
minetest.chat_send_all("Factions successfully converted.")
end)
factions.save()
end
end
2016-08-07 19:31:11 +02:00
for facname, faction in pairs(factions.factions) do
minetest.log("action", facname..","..faction.name)
for player, rank in pairs(faction.players) do
minetest.log("action", player..","..rank)
factions.players[player] = facname
2016-08-07 02:11:17 +02:00
end
2016-08-14 15:53:40 +02:00
for parcelpos, val in pairs(faction.land) do
factions.parcels[parcelpos] = facname
2016-08-07 02:11:17 +02:00
end
setmetatable(faction, factions.Faction)
2016-08-15 18:31:11 +02:00
if not faction.maxpower or faction.maxpower <= 0. then
faction.maxpower = faction.power
2016-08-15 18:31:11 +02:00
if faction.power < 0. then
faction.maxpower = 0.
end
end
if not faction.attacked_parcels then
faction.attacked_parcels = {}
end
if not faction.usedpower then
2018-10-18 04:04:46 +02:00
faction.usedpower = faction:count_land() * factions_config.power_per_parcel
end
2018-10-18 04:04:46 +02:00
if #faction.name > factions_config.faction_name_max_length then
2016-08-18 15:26:19 +02:00
faction:disband()
end
if not faction.last_logon then
faction.last_logon = os.time()
end
2018-10-18 04:55:06 +02:00
if faction.no_parcel ~= -1 then
faction.no_parcel = os.time()
end
2018-10-17 18:47:53 +02:00
for i, _ in pairs(faction.onlineplayers) do
faction.offlineplayers[i] = _
end
faction.onlineplayers = {}
2016-08-07 02:11:17 +02:00
end
2016-08-18 15:21:59 +02:00
file:close()
2016-08-07 02:11:17 +02:00
end
2018-10-15 17:24:58 +02:00
factions_ip.load()
2016-07-30 22:33:10 +02:00
end
2016-08-08 18:51:17 +02:00
function factions.is_old_file(oldfactions)
local tempfaction = factions.Faction:new(nil)
local pass = false
for facname, faction in pairs(oldfactions) do
for ni, nl in pairs(tempfaction) do
pass = false
for key, value in pairs(faction) do
if key == ni then
pass = true
break
end
end
if not pass then
tempfaction = nil
return true
end
end
-- Only check one faction to save time.
if not pass then
tempfaction = nil
return true
else
tempfaction = nil
return false
end
end
tempfaction = nil
return false
end
2016-08-08 18:51:17 +02:00
function factions.convert(filename)
local file, error = io.open(factions_worldid .. "/" .. filename, "r")
if not file then
minetest.chat_send_all("Cannot load file "..filename..". "..error)
return false
end
local raw_data = file:read("*a")
local data = minetest.deserialize(raw_data)
for facname,faction in pairs(data) do
local newfac = factions.new_faction(facname)
for oi, ol in pairs(faction) do
if newfac[oi] then
newfac[oi] = ol
end
end
for player, unused in pairs(newfac.players) do
factions.players[player] = newfac.name
end
for land, unused in pairs(newfac.land) do
factions.parcels[land] = newfac.name
end
factions.start_diplomacy(facname,newfac)
newfac:check_power()
2016-08-08 18:51:17 +02:00
end
return true
end
2016-08-18 15:21:59 +02:00
2016-08-07 02:11:17 +02:00
minetest.register_on_dieplayer(
2016-08-18 15:21:59 +02:00
function(player)
local faction = factions.get_player_faction(player:get_player_name())
if not faction then
return true
2016-08-07 02:11:17 +02:00
end
2018-10-18 04:04:46 +02:00
faction:decrease_power(factions_config.power_per_death)
2016-08-18 15:21:59 +02:00
return true
end
2016-08-07 02:11:17 +02:00
)
2018-10-17 04:58:15 +02:00
function factions.faction_tick()
local now = os.time()
for facname, faction in pairs(factions.factions) do
if faction:is_online() then
2018-10-18 04:04:46 +02:00
if factions_config.enable_power_per_player then
2018-10-17 04:58:15 +02:00
local t = faction.onlineplayers
local count = 0
for _ in pairs(t) do count = count + 1 end
2018-10-18 04:04:46 +02:00
faction:increase_power(factions_config.power_per_player*count)
2018-10-17 04:58:15 +02:00
else
2018-10-18 04:04:46 +02:00
faction:increase_power(factions_config.power_per_tick)
2018-10-15 17:24:58 +02:00
end
end
2018-10-18 04:04:46 +02:00
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
faction:disband()
end
end
end
2016-08-07 02:11:17 +02:00
minetest.register_on_joinplayer(
2016-08-18 15:21:59 +02:00
function(player)
local name = player:get_player_name()
2018-10-15 17:24:58 +02:00
createHudfactionLand(player)
local faction = factions.get_player_faction(name)
2016-08-21 01:15:41 +02:00
if faction then
faction.last_logon = os.time()
createHudFactionName(player,faction.name)
createHudPower(player,faction)
2018-10-17 04:58:15 +02:00
faction.offlineplayers[name] = nil
faction.onlineplayers[name] = 1
2018-10-18 04:31:07 +02:00
if faction.no_parcel ~= -1 then
local now = os.time() - faction.no_parcel
local l = factions_config.maximum_parcelless_faction_time
2018-10-23 05:55:19 +02:00
minetest.chat_send_player(name,"This faction will disband in "..l-now.." seconds, because it has no parcels.")
2018-10-18 04:31:07 +02:00
end
2018-10-23 05:55:19 +02:00
if faction:has_permission(name, "diplomacy") then
for _ in pairs(faction.request_inbox) do minetest.chat_send_player(name,"You have diplomatic requests in the inbox.") break end
2018-10-18 07:20:05 +02:00
end
2018-10-25 07:21:42 +02:00
if faction.message_of_the_day ~= "" or faction.message_of_the_day ~= " " then
minetest.chat_send_player(name,faction.message_of_the_day)
end
2016-08-21 01:15:41 +02:00
end
2016-08-18 15:21:59 +02:00
end
2016-08-07 02:11:17 +02:00
)
minetest.register_on_leaveplayer(
function(player)
2018-10-17 04:58:15 +02:00
local name = player:get_player_name()
local faction = factions.get_player_faction(name)
2018-10-28 02:06:48 +01:00
local id_name1 = name .. "factionLand"
if hud_ids[id_name1] then
hud_ids[id_name1] = nil
end
2018-10-17 04:58:15 +02:00
if faction then
2018-10-28 02:06:48 +01:00
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
2018-10-17 04:58:15 +02:00
faction.offlineplayers[name] = 1
faction.onlineplayers[name] = nil
end
end
)
2016-08-08 16:31:11 +02:00
minetest.register_on_respawnplayer(
function(player)
2016-08-18 15:21:59 +02:00
local faction = factions.get_player_faction(player:get_player_name())
2016-08-08 16:31:11 +02:00
if not faction then
return false
else
if not faction.spawn then
return false
else
player:setpos(faction.spawn)
return true
end
end
end
)
2016-08-04 07:21:22 +02:00
local default_is_protected = minetest.is_protected
minetest.is_protected = function(pos, player)
2018-10-17 04:58:15 +02:00
local y = pos.y
2018-10-18 04:04:46 +02:00
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
2016-08-14 15:53:40 +02:00
local parcelpos = factions.get_parcel_pos(pos)
2016-08-18 15:21:59 +02:00
local parcel_faction = factions.get_parcel_faction(parcelpos)
local player_faction = factions.get_player_faction(player)
-- no faction
2016-08-19 23:10:01 +02:00
if not parcel_faction then
return default_is_protected(pos, player)
2016-08-19 23:10:01 +02:00
elseif player_faction then
2016-08-19 03:05:44 +02:00
if parcel_faction.name == player_faction.name then
return not parcel_faction:has_permission(player, "build")
2018-10-15 17:24:58 +02:00
elseif parcel_faction.allies[player_faction.name] then
return not player_faction:has_permission(player, "build")
else
return not parcel_faction:parcel_is_attacked_by(parcelpos, player_faction)
end
2016-08-19 23:10:01 +02:00
else
return true
end
end
2018-10-17 04:58:15 +02:00
function factionUpdate()
2018-10-27 22:40:17 +02:00
factions.faction_tick()
minetest.after(factions_config.tick_time,factionUpdate)
2018-10-17 04:58:15 +02:00
end
2018-10-27 22:40:17 +02:00
minetest.after(1,hudUpdateClaimInfo)
minetest.after(1,factionUpdate)