1
0
2022-08-14 14:17:50 +03:00

384 lines
8.4 KiB
Ucode

class CVC extends Info
dependson(CVC_LocalMessage)
config(CVC);
const LatestVersion = 2;
const CfgKickProtected = class'KickProtected';
const CfgKickVote = class'KickVote';
const CfgStartWaveKickProtection = class'StartWaveKickProtection';
const CfgSkipTraderVote = class'SkipTraderVote';
const CfgPauseVote = class'PauseVote';
const CfgMapStat = class'MapStat';
const CfgMapVote = class'MapVote';
var private config int Version;
var private config E_LogLevel LogLevel;
var private KFGameInfo KFGI;
var private KFGameInfo_Survival KFGIS;
var private KFGameReplicationInfo KFGRI;
var private Array<UniqueNetId> KickProtectedPlayers;
var private Array<CVC_RepInfo> RepInfos;
public simulated function bool SafeDestroy()
{
`Log_Trace();
return (bPendingDelete || bDeleteMe || Destroy());
}
public event PreBeginPlay()
{
`Log_Trace();
if (WorldInfo.NetMode == NM_Client)
{
`Log_Fatal("NetMode == NM_Client, Destroy...");
SafeDestroy();
return;
}
Super.PreBeginPlay();
PreInit();
}
public event PostBeginPlay()
{
`Log_Trace();
if (bPendingDelete || bDeleteMe) return;
Super.PostBeginPlay();
PostInit();
}
private function PreInit()
{
`Log_Trace();
if (Version == `NO_CONFIG)
{
LogLevel = LL_Info;
SaveConfig();
}
CfgMapStat.static.InitConfig(Version, LatestVersion, LogLevel);
CfgMapVote.static.InitConfig(Version, LatestVersion, LogLevel);
CfgSkipTraderVote.static.InitConfig(Version, LatestVersion, LogLevel);
CfgPauseVote.static.InitConfig(Version, LatestVersion, LogLevel);
CfgKickVote.static.InitConfig(Version, LatestVersion, LogLevel);
CfgKickProtected.static.InitConfig(Version, LatestVersion, LogLevel);
CfgStartWaveKickProtection.static.InitConfig(Version, LatestVersion, LogLevel);
switch (Version)
{
case `NO_CONFIG:
`Log_Info("Config created");
case 1:
case MaxInt:
`Log_Info("Config updated to version"@LatestVersion);
break;
case LatestVersion:
`Log_Info("Config is up-to-date");
break;
default:
`Log_Warn("The config version is higher than the current version (are you using an old mutator?)");
`Log_Warn("Config version is" @ Version @ "but current version is" @ LatestVersion);
`Log_Warn("The config version will be changed to" @ LatestVersion);
break;
}
if (LatestVersion != Version)
{
Version = LatestVersion;
SaveConfig();
}
if (LogLevel == LL_WrongLevel)
{
LogLevel = LL_Info;
`Log_Warn("Wrong 'LogLevel', return to default value");
SaveConfig();
}
`Log_Base("LogLevel:" @ LogLevel);
CfgKickVote.static.Load(LogLevel);
CfgSkipTraderVote.static.Load(LogLevel);
CfgPauseVote.static.Load(LogLevel);
CfgMapStat.static.Load(LogLevel);
CfgMapVote.static.Load(LogLevel);
CfgStartWaveKickProtection.static.Load(LogLevel);
KickProtectedPlayers = CfgKickProtected.static.Load(LogLevel);
}
private function PostInit()
{
`Log_Trace();
if (WorldInfo == None || WorldInfo.Game == None)
{
SetTimer(1.0f, false, nameof(PostInit));
return;
}
KFGI = KFGameInfo(WorldInfo.Game);
if (KFGI == None)
{
`Log_Fatal("Incompatible gamemode:" @ WorldInfo.Game);
SafeDestroy();
return;
}
KFGIS = KFGameInfo_Survival(KFGI);
if (KFGIS == None)
{
`Log_Warn("Unknown gamemode (" $ KFGI $ "), KickProtectionStartWaves disabled");
CfgStartWaveKickProtection.default.Waves = 0;
}
if (KFGI.GameReplicationInfo == None)
{
SetTimer(1.0f, false, nameof(PostInit));
return;
}
KFGRI = KFGameReplicationInfo(KFGI.GameReplicationInfo);
if (KFGRI == None)
{
`Log_Fatal("Incompatible Replication info:" @ KFGI.GameReplicationInfo);
SafeDestroy();
return;
}
KFGRI.VoteCollectorClass = class'CVC_VoteCollector';
KFGRI.VoteCollector = new(KFGRI) KFGRI.VoteCollectorClass;
if (KFGRI.VoteCollector == None)
{
`Log_Fatal("Can't replace VoteCollector!");
SafeDestroy();
return;
}
else
{
CVC_VoteCollector(KFGRI.VoteCollector).CVC = Self;
CVC_VoteCollector(KFGRI.VoteCollector).LogLevel = LogLevel;
`Log_Info("VoteCollector replaced");
}
}
public function bool PlayerIsKickProtected(PlayerReplicationInfo PRI)
{
`Log_Trace();
return (KickProtectedPlayers.Find('Uid', PRI.UniqueId.Uid) != INDEX_NONE);
}
public function bool PlayerIsStartWaveKickProtected(KFPlayerController KFPC)
{
`Log_Trace();
return (PlayerOnStartWave(KFPC) && PlayerHasRequiredLevel(KFPC));
}
private function bool PlayerOnStartWave(KFPlayerController KFPC)
{
local CVC_RepInfo RepInfo;
`Log_Trace();
if (KFGIS != None && CfgStartWaveKickProtection.default.Waves != 0)
{
foreach RepInfos(RepInfo)
{
if (RepInfo.GetKFPC() == KFPC)
{
return (RepInfo.StartWave + CfgStartWaveKickProtection.default.Waves >= KFGIS.WaveNum);
}
}
}
return false;
}
private function bool PlayerHasRequiredLevel(KFPlayerController KFPC)
{
local KFPlayerReplicationInfo KFPRI;
`Log_Trace();
KFPRI = KFPlayerReplicationInfo(KFPC.PlayerReplicationInfo);
if (KFPRI == None)
{
return true;
}
return (KFPRI.GetActivePerkLevel() >= CfgStartWaveKickProtection.default.MinLevel);
}
public function bool PlayerCanStartKickVote(KFPlayerController KFPC, KFPlayerController KFPC_Kickee)
{
`Log_Trace();
if (KFPC_Kickee != None)
{
if (KFPC == KFPC_Kickee)
{
return false; // kickee cant vote
}
if (!PlayerHasRequiredLevel(KFPC_Kickee))
{
return true; // always can vote for players without req level
}
}
return !PlayerOnStartWave(KFPC);
}
public function BroadcastChatLocalized(E_CVC_LocalMessageType LMT, optional String HexColor, optional KFPlayerController ExceptKFPC = None, optional String String1, optional String String2)
{
local CVC_RepInfo RepInfo;
`Log_Trace();
foreach RepInfos(RepInfo)
{
if (RepInfo.GetKFPC() != ExceptKFPC)
{
RepInfo.WriteToChatLocalized(LMT, HexColor, String1, String2);
}
}
}
public function BroadcastHUDLocalized(E_CVC_LocalMessageType LMT, optional float DisplayTime = 0.0f, optional String String1, optional String String2, optional String String3)
{
local CVC_RepInfo RepInfo;
`Log_Trace();
foreach RepInfos(RepInfo)
{
if (RepInfo.GetKFPC() != None)
{
RepInfo.WriteToHUDLocalized(LMT, String1, String2, String3, DisplayTime);
}
}
}
public function BroadcastClearMessageHUD(optional float DefferedTime = 0.0f)
{
local CVC_RepInfo RepInfo;
`Log_Trace();
foreach RepInfos(RepInfo)
{
if (RepInfo.GetKFPC() != None)
{
RepInfo.DefferedClearMessageHUD(DefferedTime);
}
}
}
public function WriteToChatLocalized(KFPlayerController KFPC, E_CVC_LocalMessageType LMT, optional String HexColor, optional String String1, optional String String2)
{
local CVC_RepInfo RepInfo;
`Log_Trace();
foreach RepInfos(RepInfo)
{
if (RepInfo.GetKFPC() == KFPC)
{
RepInfo.WriteToChatLocalized(LMT, HexColor, String1, String2);
return;
}
}
}
public function WriteToHUDLocalized(KFPlayerController KFPC, E_CVC_LocalMessageType LMT, optional float DisplayTime = 0.0f, optional String String1, optional String String2, optional String String3)
{
local CVC_RepInfo RepInfo;
`Log_Trace();
foreach RepInfos(RepInfo)
{
if (RepInfo.GetKFPC() == KFPC)
{
RepInfo.WriteToHUDLocalized(LMT, String1, String2, String3, DisplayTime);
return;
}
}
}
public function NotifyLogin(Controller C)
{
`Log_Trace();
CreateRepInfo(C);
}
public function NotifyLogout(Controller C)
{
`Log_Trace();
DestroyRepInfo(C);
}
public function bool CreateRepInfo(Controller C)
{
local CVC_RepInfo RepInfo;
`Log_Trace();
if (C == None) return false;
RepInfo = Spawn(class'CVC_RepInfo', C);
if (RepInfo == None) return false;
RepInfo.CVC = Self;
RepInfo.LogLevel = LogLevel;
RepInfo.StartWave = ((KFGIS != None) ? KFGIS.WaveNum : 0);
RepInfos.AddItem(RepInfo);
return true;
}
public function bool DestroyRepInfo(Controller C)
{
local CVC_RepInfo RepInfo;
`Log_Trace();
if (C == None) return false;
foreach RepInfos(RepInfo)
{
if (RepInfo.Owner == C)
{
RepInfo.SafeDestroy();
RepInfos.RemoveItem(RepInfo);
return true;
}
}
return false;
}
DefaultProperties
{
}